PetSwipe is on a mission to help shelter animals find loving homes. This swipe-to-adopt platform connects prospective pet parents with adoptable animals in need. With PetSwipe, users can effortlessly browse pets, swipe right to adopt or left to pass, and manage their profiles with ease. They can also view their matches and keep track of their swipe history — making the journey to adoption simpler, faster, and more heartwarming 🐰.
Note
Inspired by Tinder UX, but for pets to find their loving humans! 🐶🐱
- About PetSwipe
- Live App
- Features
- Tech Stack & Architecture
- User Interface
- Database Schema
- Getting Started
- API Reference
- AWS Deployment
- Scripts & Utilities
- Testing
- GitHub Actions CI/CD
- Command Line Interface
- Contributing
- License
- Author
PetSwipe is a full-stack application that allows users to swipe through pets available for adoption. The app is designed to be user-friendly and visually appealing, with a focus on providing a seamless experience for both users and shelter staff.
The app is built using modern technologies, including TypeScript, Next.js, Express, and PostgreSQL. It leverages the power of AWS for storage and deployment, ensuring scalability and reliability.
The app is also designed to be modular and easy to extend, with a focus on clean code and best practices. It includes features such as user authentication, a swipe interface, personalized pet decks, and admin tools for managing pets and users.
And most importantly, it is built with the goal of helping shelter animals find their forever homes. By providing a fun and engaging way for users to browse pets, PetSwipe aims to increase adoption rates and raise awareness about the importance of pet adoption. 🐾
I hope you enjoy using PetSwipe as much as I enjoyed building it! 🐱
Tip
Please spread the word about PetSwipe to your friends and family, and help us find loving homes for as many pets as possible! 🏠❤️
PetSwipe is live on Vercel! You can now try it out and see how it works.
Tip
Link not working? Copy and paste this URL into your browser: https://petswipe.vercel.app.
Also, checkout the backend API at PetSwipe API. You can use tools like Postman or Swagger UI to explore the API endpoints.
[!IMPORTANT] > Note: Currently, most of the data is seeded with dummy data. We hope the app will be used by more real users and pet adoption shelters in the future. If you are a shelter or a pet adoption organization, please reach out to us to get all your data integrated into the app in seconds! Or you can also use the in-app manual add pet features to further enrich our pets database (only works for authenticated users).
PetSwipe is a full-stack application with the following features:
- User Authentication:
- Login, signup, password reset functionalities are all implemented
- JWT-based authentication, where tokens are stored in HTTP-only cookies for security
- Swipe Interface:
- Swipe left/right or press arrow keys/buttons to navigate through the deck of pets cards
- For each card, users can view pet details, photos, and decide to adopt or pass
- Each user is randomly assigned a sample selection of pet cards (around 90-110 cards) to review. As the app gathers more real, user-added pet data, it will improve in matching users with the most relevant pets.
- Personalized Deck:
- Deck is generated based on user preferences and past swipes
- Users will only see pets that they haven't swiped on before, and pets that are most relevant to them
- History: View all swipes & liked (adopted) pets
- Chatbot:
- A simple chatbot to answer common questions about the app and pets (e.g. breeds, adoption process, pet care tips, etc.)
- Powered by Google AI and Retrieval-Augmented Generation (RAG) for personalized responses
- Admin Tools:
- Bulk upload pets via CSV
- Export pets data
- Photo uploads to S3
- Manual match assignment
- and more!
- Responsive UI: Built with Tailwind CSS and shadcn/ui
- Fully responsive design for mobile and desktop
- Light and dark mode support
- Accessible design for all users
- Real-time Animations:
- Framer Motion for smooth transitions and animations
- Swipe animations for a more engaging experience
- Loading spinners and skeleton screens for better UX
- Analytics: Countups of swipes, matches, adoptions, etc. all are available to admins
PetSwipe is built using a modern tech stack, ensuring scalability, maintainability, and performance. The architecture is designed to be modular and easy to extend.
Layer | Technology |
---|---|
Frontend | Next.js, React, TypeScript, Tailwind CSS, shadcn/ui, Framer Motion, SWR |
Backend & API | Node.js, Express, TypeScript, TypeORM, PostgreSQL, OpenAPI (via Swagger), RabbitMQ, Redis |
Data & Storage | AWS RDS (PostgreSQL), AWS S3 |
Security & Auth | JSON Web Tokens, bcryptjs, cookie-parser |
DevOps & Deployment | Docker, AWS ECR & ECS (Fargate), Vercel, GitHub Actions |
Infrastructure | Terraform, Consul, Vault, Nomad, AWS IAM, AWS CloudWatch, AWS ALB |
AI | Google AI, Retrieval-Augmented Generation (RAG) |
Testing | Playwright (frontend), Jest (backend) |
Below is a high-level Mermaid diagram of our architecture/infrastructure:
Tip
Image not showing? Here's the link to the diagram: Mermaid Live Diagram
and more...
Entity | Column | Type | Nullable | Description | Notes / Relations |
---|---|---|---|---|---|
Match | id |
uuid |
No | Primary key | @PrimaryGeneratedColumn("uuid") |
user |
ManyToOne → AppUser |
No | Who is swiping | FK → AppUser.id , cascade on delete |
|
pet |
ManyToOne → Pet |
No | Which pet was presented | FK → Pet.id , cascade on delete |
|
matchedAt |
timestamp |
No | When it was shown | @CreateDateColumn() |
|
Pet | id |
uuid |
No | Primary key | @PrimaryGeneratedColumn("uuid") |
name |
varchar |
No | e.g. “Buddy” or “Whiskers” | @Column() |
|
type |
varchar |
No | e.g. “Dog”, “Cat” | @Column() |
|
description |
text |
Yes | Breed, color, age etc. | @Column({ type: "text", nullable: true }) |
|
photoUrl |
text |
Yes | URL to photo(s) | @Column({ type: "text", nullable: true }) |
|
shelterName |
varchar |
Yes | The shelter this pet is from | @Column({ type: "varchar", nullable: true }) |
|
shelterContact |
text |
Yes | Contact info for the shelter | @Column({ type: "text", nullable: true }) |
|
shelterAddress |
text |
Yes | Physical address of the shelter | @Column({ type: "text", nullable: true }) |
|
matches |
OneToMany → Match[] |
— | All Match records for this pet | inverse of Match.pet |
|
swipes |
OneToMany → Swipe[] |
— | All Swipe records for this pet | inverse of Swipe.pet |
|
createdAt |
timestamp |
No | When record was created | @CreateDateColumn() |
|
updatedAt |
timestamp |
No | When record was last updated | @UpdateDateColumn() |
|
Swipe | id |
uuid |
No | Primary key | @PrimaryGeneratedColumn("uuid") |
user |
ManyToOne → AppUser |
No | Who swiped | FK → AppUser.id , cascade on delete |
|
pet |
ManyToOne → Pet |
No | Which pet was swiped on | FK → Pet.id , cascade on delete |
|
liked |
boolean |
No | true = adopt, false = pass |
@Column() |
|
swipedAt |
timestamp |
No | When the swipe occurred | @CreateDateColumn() |
|
unique index | (user, pet) |
— | Prevent duplicate swipes | @Index(["user","pet"],{unique:true}) |
|
AppUser | id |
uuid |
No | Primary key | @PrimaryGeneratedColumn("uuid") |
email |
varchar |
No | Unique user email | @Column({ unique: true }) |
|
password |
varchar |
Yes | Hashed password | @Column({ nullable: true }) |
|
name |
varchar |
Yes | User’s full name | @Column({ nullable: true }) |
|
dob |
date |
Yes | Date of birth | @Column({ type: "date", nullable: true }) |
|
bio |
text |
Yes | User biography | @Column({ type: "text", nullable: true }) |
|
avatarUrl |
text |
Yes | URL to avatar image | @Column({ type: "text", nullable: true }) |
|
matches |
OneToMany → Match[] |
— | All Match records by this user | inverse of Match.user |
|
swipes |
OneToMany → Swipe[] |
— | All Swipe records by this user | inverse of Swipe.user |
|
createdAt |
timestamp |
No | When user was created | @CreateDateColumn() |
|
updatedAt |
timestamp |
No | When user was last updated | @UpdateDateColumn() |
Note
This table may not be up-to-date, as more entities and relationships could be introduced in the near future to support additional features and enhancements!
- Node.js ≥ v18
- npm ≥ v8 or Yarn
- PostgreSQL (AWS RDS recommended)
- AWS CLI & IAM credentials for S3, RDS
- Docker (optional, for local Postgres container)
- Google AI API key (for chatbot feature)
Caution
shadcn/ui
peerDeps, install frontend dependencies with:
npm install --legacy-peer-deps
# or
yarn install --ignore-engines
-
Clone & Install
git clone https://github.com/hoangsonww/PetSwipe-Match-App.git cd PetSwipe-Match-App/backend npm install
-
Environment Copy and configure:
cp .env.example .env
DATABASE_URL
: your AWS RDS Postgres connection stringJWT_SECRET
,COOKIE_SECRET
- AWS:
AWS_ACCESS_KEY_ID
,AWS_SECRET_ACCESS_KEY
,S3_BUCKET_NAME
- More in
.env.example
. Be sure that you have all required environment variables set up before running the app!
-
Seed Sample Pets (optional)
npm run seed:pets
-
Run in Development
npm run dev
The backend API is now available at
http://localhost:5001/api
.
-
Clone & Install
cd frontend npm install --legacy-peer-deps
-
Environment Create
.env.local
: (replacehttp://localhost:5001
with your backend URL)NEXT_PUBLIC_API_URL=http://localhost:5001/api
-
Run in Development
npm run dev
Frontend available at
http://localhost:3000
-
Build & Production
npm run build npm run start
Swagger docs at http://localhost:5001/api-docs.json
.
- POST
/api/auth/signup
- POST
/api/auth/login
- POST
/api/auth/logout
- POST
/api/auth/verify-email
- POST
/api/auth/reset-password
- POST
/api/matches
- GET
/api/matches
- GET
/api/matches/me
- GET
/api/pets
- POST
/api/pets
- GET
/api/pets/export
- POST
/api/pets/:petId/photo
- POST
/api/pets/upload
- POST
/api/swipes
- GET
/api/swipes/me
- GET
/api/swipes/me/liked
- GET
/api/swipes
- GET
/api/users/me
- PUT
/api/users/me
- POST
/api/users/me/avatar
- DELETE
/api/users/me/avatar
- POST
/api/chat
More endpoints may be added as the app evolves. Refer to the Swagger docs for the most up-to-date information!
Swagger UI is available at https://petswipe-backend-api.vercel.app/
.
The app is designed to be easily deployable to AWS using services like:
- RDS: PostgreSQL instance for data
- S3: Bucket for pet photo storage
- Elastic Beanstalk or ECS for backend
- ECR: Container registry for Docker images
- ECS: For hosting the backend API using Docker containers
- Fargate: Serverless compute engine for containers
- GHCR: Backup container registry for Docker images
- CloudWatch: For logging and monitoring
- IAM: For managing permissions
- Terraform: Infrastructure as Code (IaC) for managing AWS resources
- Optional: Consul for service discovery, Vault for secrets management, and Nomad for orchestration
- GitHub Actions for CI/CD
- Vercel for frontend
Tip: Use IAM roles for EC2/ECS to grant S3 & RDS access securely!
Currently, our stack is fully deployed on AWS using the following services:
- AWS RDS: Manages our PostgreSQL database.
- AWS S3: Stores pet photos and user avatars.
- Backup service: Supabase Storage.
- AWS ECS: Hosts our backend API using Docker containers.
- Backup service: Vercel (currently, due to AWS pricing issues, we have switched to Vercel for hosting the backend).
- AWS Fargate: Serverless compute engine for running containers.
- AWS ECR: Container registry for our Docker images.
- Backup service: GitHub Container Registry (GHCR).
- AWS ALB: Application Load Balancer for routing traffic to our ECS services.
- AWS IAM: Manages permissions for accessing AWS resources.
- AWS CloudWatch: For logging and monitoring.
- GitHub Actions: For CI/CD to build and deploy our Docker images to ECR.
- Vercel: Hosts our frontend application.
- HashiCorp Stack: For managing infrastructure and secrets:
- Terraform: Infrastructure as Code (IaC) for managing AWS resources.
- Consul: Service discovery and configuration management.
- Vault: Secrets management for securely storing sensitive information.
- Nomad: Orchestration for deploying and managing applications across a cluster of machines.
Note
This stack is designed to be flexible and scalable, allowing us to easily add or remove services as needed. The use of Terraform, Consul, Vault, and Nomad provides additional security and flexibility for managing our infrastructure.
To deploy the app to AWS, we use Terraform for Infrastructure as Code (IaC). This allows us to define our AWS resources in code and deploy them easily.
To get started with Terraform:
-
Install Terraform on your machine.
-
Navigate to the
infrastructure
directory:cd terraform
-
Initialize Terraform:
terraform init
-
Configure your AWS credentials in
~/.aws/credentials
or set them as environment variables. -
Run the following commands to deploy:
terraform plan # Preview the changes terraform apply # Apply the changes
This will create all the necessary AWS resources for the app, including RDS, S3, ECS, ECR, IAM roles, and more.
Caution
Make sure you have the necessary permissions to create and manage AWS resources. Review the Terraform scripts before applying them to avoid any unintended changes. For more details on how to configure and use Terraform with AWS, check out the Terraform AWS Provider documentation.
For managing secrets and service discovery, we use HashiCorp Vault, Consul, and Nomad. These tools help us securely store sensitive information, manage configurations, and orchestrate our services.
These tools provide:
- Secure secret storage & dynamic secret issuance (Vault)
- Encryption-as-a-Service & audit logging (Vault)
- Distributed service discovery & health checking (Consul)
- Centralized key/value configuration store (Consul)
- Identity-based service mesh (mTLS) for secure service-to-service communication (Consul)
- Flexible, multi-region workload orchestration & scheduling (Nomad)
- Rolling upgrades, canary deployments & autoscaling (Nomad)
- Support for containers, VMs & standalone binaries (Nomad)
These tools are optional but recommended for larger, more complex deployments. They can be set up using Terraform as well. For more information, refer to the terraform/README.md file in the terraform
directory to see how to configure and deploy these tools.
Note
If you are not familiar with these tools, you can skip this section for now. The app can run without them, but they provide additional security and flexibility for production deployments.
For managing the deployment and configuration of the app, we also use Ansible. Ansible playbooks are used to automate tasks such as:
-
Provisioning AWS infrastructure Create or update RDS instances, S3 buckets (static + uploads), ECR repositories, Application Load Balancer & Target Group, ECS clusters and services.
-
Building & deploying containers Build, tag and push backend and frontend Docker images to ECR, then trigger zero-downtime rolling updates of your Fargate service.
-
Configuring frontend hosting Sync Next.js static build to S3 and invalidate CloudFront distributions for instant cache busting.
-
Managing application configuration Template out and distribute environment variables, secrets (via AWS Secrets Manager or HashiCorp Vault), and config files to running services.
-
Database migrations & backups Run schema migrations on PostgreSQL (via TypeORM) and schedule or trigger automated backups and snapshots.
-
Lifecycle & housekeeping Apply S3 lifecycle rules (e.g. purge old uploads), rotate credentials, and clean up unused resources.
-
Service discovery & secrets bootstrap (If using HashiCorp stack) Bootstrap or update Consul/Nomad clusters, deploy Vault auto-unseal configuration, and distribute ACL tokens.
-
OS & user management Install OS packages, manage system users, SSH keys, and security hardening on any EC2 hosts you might run.
-
CI/CD integration Hook into GitHub Actions workflows to run Ansible playbooks on push or merge, ensuring every change is tested and deployed automatically.
Caution
Remember to set up your AWS credentials and permissions correctly before running any Ansible playbooks. Ensure that the IAM user/role has the necessary permissions to create and manage the resources defined in your playbooks.
Tip
For more information on how to use Ansible with AWS, check out the ansible/README.md file and the Ansible AWS documentation.
Warning
NEVER store sensitive information (like AWS credentials, database passwords, etc.) directly in your Ansible playbooks or inventory files. Use Ansible Vault or environment variables to securely manage secrets. Also, NEVER commit your .env
files or any sensitive configuration files to version control!
The app also comes with several scripts and utilities to help with development and deployment:
- Seed Pets:
npm run seed:pets
- Assign Pets:
npm run assign:user
- TypeORM CLI:
npm run typeorm <command>
- Swagger: Auto-generated OpenAPI spec
Additionally, a Makefile
is included for common tasks:
make install # Install dependencies
make start # Start the backend server
make dev # Start the backend server in development mode
make up # Start the backend server with Docker
make test # Run tests
make lint # Lint the codebase
make deploy # Deploy to AWS
make clean # Clean up Docker containers
# and more...
To start the entire app with Docker, run:
docker-compose up --build
This will pull the images from ECR and start the backend and database containers locally on your machine.
Additionally, there are also Shell scripts that help you run Docker commands easily:
pull_and_run.sh
: Pulls the latest Docker image from ECR and runs it.upload_to_ghcr.sh
: Builds the Docker image and uploads it to GitHub Container Registry (remember to set up your GitHub PAT and username in the script/export them in your shell).
PetSwipe also includes a Docker Compute Terminal for running commands inside the Docker container. You can use it to run database migrations, seed data, or any other commands you need.
To hop into the Docker Compute Terminal, run:
docker-compose exec compute /bin/zsh
This will give you a shell inside the Docker container where you can run commands as if you were on a regular terminal, plus you can access all the installed dependencies and tools of the application.
PetSwipe includes a comprehensive testing suite to ensure the application works as expected. The tests are organized into three main categories:
- Playwright: End-to-end tests for the frontend UI.
- Jest: Unit and integration tests for the backend API and frontend API helper functions.
- Chai & Mocha: Additional tests for specific features and functionalities.
- Commitlint: Ensures commit messages follow the conventional commit format.
Playwright is used for end-to-end testing of the frontend application. It simulates user interactions and verifies that the UI behaves as expected.
To run Playwright tests, navigate to the frontend
directory and run:
cd frontend
npm run test:e2e
Jest is used for unit and integration tests in the backend API and some frontend components. It provides a fast and reliable testing framework.
To run Jest tests in the backend, navigate to the backend
directory and run:
cd backend
npm run test
This will run all tests in the backend API, including unit tests for controllers, services, and integration tests for the database.
To run Jest tests in the frontend, navigate to the frontend
directory and run:
cd frontend
npm run test
Chai and Mocha are used for additional tests in the frontend. To run Chai & Mocha tests, navigate to the frontend
directory and run:
cd frontend
npm run test:mocha
PetSwipe uses GitHub Actions for Continuous Integration and Continuous Deployment (CI/CD). The workflow is defined in .github/workflows/workflow.yml
.
The CI/CD pipeline includes the following steps:
- Checkout Code: Pulls the latest code from the repository.
- Set Up Node.js: Installs the specified Node.js version.
- Install Dependencies: Installs the necessary dependencies for both backend and frontend.
- Run Linting: Runs ESLint and Prettier to ensure code quality and formatting.
- Run Tests: Executes Jest tests for the backend and Playwright tests for the frontend.
- Build Frontend: Builds the Next.js frontend application.
- Build Docker Images: Builds Docker images for both backend and frontend.
- Push Docker Images: Pushes the built images to GitHub Container Registry (GHCR).
- Deploy to AWS: Deploys the backend API to AWS ECS and the frontend to Vercel.
- Notify on Failure: Sends notifications via email if any step fails.
- Commitlint: Ensures commit messages follow the conventional commit format.
This setup ensures that every change pushed to the repository is automatically tested, built, and deployed, providing a robust and reliable development workflow.
The app also includes a CLI for managing pets and users. You can run the CLI commands from the root directory:
petswipe <command> [options]
petswipe dev
: Start backend & frontend in development mode.petswipe build
: Build both backend & frontend applications.petswipe docker:build
: Build & push Docker images to GitHub Container Registry.petswipe up
: Pull Docker images and start the stack.petswipe down
: Stop the Docker Compose stack.petswipe clean
: Remove build artifacts.petswipe lint
: Run linters in both projects.petswipe test
: Run tests in both projects.
This CLI is designed to make it easier to manage the application and perform common tasks without having to navigate through multiple directories or run multiple commands.
- Fork the repo & clone
- Create a feature branch
- Code, lint, test
- IMPORTANT: Run
npm run format
in the root directory to format all files before committing!
- IMPORTANT: Run
- Open a Pull Request. We'll review and merge it if it is meaningful and useful!
Please follow the existing code style (ESLint, Prettier, TypeScript). This helps maintain a clean and consistent codebase!
© 2025 Son Nguyen. I hope this code is useful for you and your projects. Feel free to use it, modify it, and share it with others.
Licensed under the MIT License. See LICENSE for details.
Important
In short, you may use this code for personal or educational purposes, but please do not use it for commercial purposes without permission. If you do use this code, please give proper credit to the original author.
This application is built with ❤️ by Son Nguyen in 2025:
Feel free to reach out for any questions, suggestions, or even collaborations!
❤️ Thank you for helping pets find their forever homes! 🐶🐱