This project provides a fully functional RESTful API for managing Monkeypox cases. It supports full CRUD operations for creating, reading, updating, and deleting case records, as well as querying cases from the last 7 days. The API is containerized using Docker for easy setup and deployment.
- Features
- Used Technologies
- Overview
- Prerequisites
- Setup and Basic, Quickstart Personalization
- Running the Application
- Testing the API Using Postman
- Testing MongoDB with MongoDB Compass (Docker Setup)
- CI/CD Pipeline personalization
- Running with Docker (Advanced)
- Appendix: Creating Secrets and Access Tokens
- Notes
- CRUD Operations:
- Create: Register new Monkeypox cases with attributes like latitude, longitude, gender, and age.
- Read: Fetch all cases or filter based on criteria (e.g., cases from the last 7 days).
- Update: Modify case details.
- Delete: Remove case records.
- Recent Cases Query: Retrieve Monkeypox cases registered in the last week.
- Dockerized: The API runs inside a Docker container for easy testing and deployment.
- Docker: Containerization for scalable deployment.
- Express.js: Framework for creating RESTful routes and managing HTTP requests.
- GitHub Actions: CI/CD pipeline for automated Docker image builds and deployments.
- MongoDB: Database for storing case data, including geolocation and patient details.
- Node.js: Backend for the API.
- TypeScript: Provides type safety and code quality for better development.
Here is an explanation of the file structure within this project, detailing the purpose of each folder and file.
.github/
└── workflows/
├── publish.yml
├── publishforgit.yml
src/
├── config/
│ └── envs.plugin.ts
├── controllers/
│ └── cases/
│ └── routes.ts
├── data/
│ ├── models/
│ │ └── init.ts
├── domain/
│ ├── jobs/
│ │ └── email.job.ts
│ ├── services/
│ │ └── email.service.ts
│ ├── templates/
│ │ └── email.template.ts
├── app.ts
├── .dockerignore
├── .env.template
├── .gitignore
├── Dockerfile
├── LICENSE
├── README.md
├── docker-compose.yml
├── package-lock.json
├── package.json
└── tsconfig.json
- publish.yml: GitHub Actions workflow for building and pushing Docker images to DockerHub. This file automates the process of creating Docker images and publishing them to a registry when changes are pushed to the repository.
- publishforgit.yml: Similar to
publish.yml
, but designed for publishing Docker images to GitHub Packages instead of DockerHub.
-
config/
:- envs.plugin.ts: Contains environment variable configurations that are injected into the application at runtime.
-
controllers/
:- cases/routes.ts: Handles routing for the Monkeypox case management API. This file defines the endpoints for creating, updating, reading, and deleting cases.
-
data/
:- models/init.ts: Defines and initializes the MongoDB models (schemas) that the API uses for storing Monkeypox case data.
-
domain/
:- jobs/email.job.ts: Defines scheduled jobs for sending email notifications, which can be used for alerts and case status updates.
- services/email.service.ts: Provides the functionality for sending emails. This service interacts with an email provider (e.g., Gmail) and uses the email template.
- templates/email.template.ts: Defines the email template used when sending notifications to users. The template can be customized to include dynamic content like case details.
- app.ts: The entry point of the application, initializing the Express server, loading middleware, and setting up the API routes.
- .dockerignore: Specifies files and directories that should be ignored when building the Docker image. Similar to
.gitignore
but specific to Docker. - .env.template: A template environment file providing placeholders for all required environment variables (such as MongoDB URL, email credentials, and API keys).
- .gitignore: Specifies files and directories to be ignored by Git during version control, preventing sensitive or unnecessary files from being committed.
- Dockerfile: Defines the instructions to build a Docker image for the project. This includes specifying the base image, installing dependencies, and setting up the application environment.
- LICENSE: Contains the legal license information for the project, outlining the permissions and restrictions for using the software.
- README.md: The documentation for the project, including setup instructions, usage, and API details.
- docker-compose.yml: A Docker Compose configuration file that orchestrates the services needed to run the application (e.g., the API and MongoDB database).
- package-lock.json: Automatically generated file that locks the dependencies for the project to specific versions, ensuring consistency across different environments.
- package.json: Lists the project dependencies and scripts used by Node.js. It also defines metadata for the project, like its name, version, and entry points.
- tsconfig.json: Configuration file for TypeScript, specifying compiler options and the structure of the TypeScript project.
Basic:
- Docker Desktop: Make sure Docker Desktop is installed. You can download and install it from here.
- Postman: Install Postman to test the API endpoints. You can download it from here.
Complementary:
- MongoDB Compass: Install MongoDB Compass if you wish a visual interface to interact with MongoDB. Download MongoDB Compass.
- Node.js: Ensure Node.js is installed if you want to develop locally. Download Node.js.
git clone https://github.com/yourusername/monomap.git
cd monomap
Before running the application, you must set up your environment variables in the .env
file. This step is crucial to avoid errors during the execution of the application.
-
Create the
.env
file based on the template:- The project includes a sample
.env.template
file that stores essential environment variables. You must fill in the required values before running the application.
- The project includes a sample
-
Modify the following values with your own credentials:
PORT
: Port for the application (e.g.,3000
).MONGO_URL
: MongoDB connection string for local development.MONGO_URL_DOCKER
: MongoDB connection string when running inside Docker.MAIL_SECRET_KEY
: Your email service secret key, especially for Gmail (app-specific password).MAIL_SERVICE
: The mail service you are using (e.g., Gmail).MAIL_USER
: Your email address for sending notifications.MAPBOX_ACCESS_TOKEN
: Your Mapbox access token for geolocation services.
Example
.env
file:PORT=3000 MONGO_URL=mongodb://root:example@localhost:27017/ MONGO_URL_DOCKER=mongodb://root:example@mongo MAIL_SECRET_KEY=your_secret_key_here MAIL_SERVICE=gmail MAIL_USER=your_email@example.com MAPBOX_ACCESS_TOKEN=your_mapbox_token_here
-
Save the
.env
file and ensure all fields are correctly filled out.
After updating the environment variables, you need to restart the Docker containers to apply the changes:
- Stop the running containers with:
docker-compose down
- Start the containers again:
docker-compose up
Once you have personalized your .env
file and ensured Docker Desktop is running, you can:
-
Run the Application with Docker:
docker-compose up
This command will build and start the API inside a Docker container.
-
Stop the Docker Containers:
-
To stop the running containers, press CTRL + C in the terminal.
-
Alternatively, stop and remove the containers with:
docker-compose down
-
Once Docker is up and running, the API will be available at http://localhost:3000. You can test the API using Postman with the following routes:
-
Create a New Case:
-
Method:
POST
-
URL:
http://localhost:3000/api/cases
-
Body: (raw JSON)
{ "lat": 19.432608, "lng": -99.133209, "genre": "Male", "age": 25 }
-
-
Get All Cases:
- Method:
GET
- URL:
http://localhost:3000/api/cases
- Method:
-
Get a Case by ID:
- Method:
GET
- URL:
http://localhost:3000/api/cases/:id
- Replace
:id
with the actual case ID you want to fetch.
- Method:
-
Update a Case:
-
Method:
PUT
-
URL:
http://localhost:3000/api/cases/:id
-
Body: (raw JSON)
{ "lat": 19.432608, "lng": -99.133209, "genre": "Female", "age": 30 }
-
-
Delete a Case:
- Method:
DELETE
- URL:
http://localhost:3000/api/cases/:id
- Replace
:id
with the actual case ID you want to delete.
- Method:
-
Get Cases from the Last Week:
- Method:
GET
- URL:
http://localhost:3000/api/cases/last
- Method:
-
Open MongoDB Compass.
-
Connect to MongoDB using the connection string configured for Docker. Use the
MONGO_URL_DOCKER
value specified in your.env
file. A common example would be:mongodb://root:example@mongo:27017/
Ensure that:
root
is the username.example
is the password.mongo
is the hostname of the MongoDB service defined in yourdocker-compose.yml
file.27017
is the default MongoDB port.
-
Click "Connect": After entering the connection string, click the "Connect" button to establish a connection with your Dockerized MongoDB instance.
-
Browse the Database:
- Once connected, you will see a list of available databases inside the Dockerized MongoDB instance.
- Select the database used by your application (e.g.,
monomap
).
-
View Collections:
- Explore the collections within the database, such as those storing Monkeypox case data.
- Click on the collection name to view documents, and use the built-in query interface to filter or search specific records.
-
Inspect and Manage Data:
- View, edit, or delete documents directly from MongoDB Compass to validate that your API is correctly interacting with the database.
- For example, run a query to find all Monkeypox cases or filter data based on certain criteria, such as by
age
orlocation
.
The project uses GitHub Actions for CI/CD, automatically building and pushing Docker images to Docker Hub on every push to the main branch. This workflow is flexible and can be tailored to another credentials, docker-compose files or container registries.
Docker-compose.yml:
services:
monomap:
image: ceciliasw/monomap:latest -> Replace.
Make sure to update user and image routes (in lowercase) within the docker-compose, and GitHub Actions workflow files: .github/workflows/publish.yml
for DockerHub and .github/workflows/publishforgit.yml
for GitHub packages. Specifically, adjust the build
and push
steps as per your project needs. Additionally, ensure GITHUB_TOKEN
has read & write permissions under Settings > Actions > General in your GitHub repository.
name: Docker Build & Publish
on:
push:
branches:
- main
jobs:
build-and-push:
runs-on: ubuntu-latest
steps:
- name: Clone repository using GitHub Actions
uses: actions/checkout@v3
- name: Log in to Docker Hub
run: echo "${{ secrets.DOCKER_PAT }}" | docker login -u "${{ secrets.DOCKER_USERNAME }}" --password-stdin
- name: Build Docker Image and tag it with both the commit SHA and latest
run: docker build -t yourdockerhubusername/yourimagename:latest .
- name: Tag Latest
run: docker tag yourdockerhubusername/yourimagename:latest yourdockerhubusername/yourimagename:latest
- name: Push the image to Docker Hub
run: |
docker push yourdockerhubusername/yourimagename:latest -> A [route].
-
Clone repository using GitHub Actions:
- This step uses the
actions/checkout@v3
action to clone your repository into the workflow environment, ensuring the latest code is ready for the build process.
- This step uses the
-
Log in to Docker Hub using credentials stored in GitHub Secrets (
DOCKER_USERNAME
andDOCKER_PAT
):- This step securely logs into Docker Hub using the credentials stored in GitHub Secrets. Make sure you have added these secrets in your GitHub repository settings under Settings > Secrets and variables > Actions.
-
Build Docker image and tag it with both the commit SHA and
latest
:- This step builds the Docker image from the Dockerfile and tags it twice: once with the commit SHA (for version tracking) and once as
latest
.
- This step builds the Docker image from the Dockerfile and tags it twice: once with the commit SHA (for version tracking) and once as
-
Push the image to Docker Hub:
- This final step pushes the tagged Docker images to your Docker Hub repository, making them available for deployment.
DOCKER_USERNAME
: Your Docker Hub username.DOCKER_PAT
: Your Docker Personal Access Token (PAT).
If you're familiar with Docker and want to build and run the API manually, follow these steps:
-
Build Docker image:
docker build -t [route]:latest
-
Run Docker container:
docker run -p 3000:3000 [route]:latest
-
Access the API:
Open http://localhost:3000 in your browser or use Postman to test the endpoints.
Follow these steps to securely store secrets such as API keys and access tokens:
To use Gmail for sending emails from your API, you need to create an App Password (a secure way for Gmail API access when two-factor authentication is enabled).
- Go to your Google Account and ensure two-factor authentication (2FA) is enabled.
- Navigate to Security in the left-hand panel.
- Scroll down to the Signing in to Google section and click on App passwords.
- Sign in to your Google account if prompted.
- Under the Select the app and device you want to generate the app password for, select Mail as the app and Other for device, then name it (e.g., "Node API").
- Click Generate.
- Copy the generated password. This will serve as your
MAIL_SECRET_KEY
.
DockerHub tokens are used to authenticate and push Docker images programmatically without exposing your Docker password. You need to provide Read, Write, and Delete permissions to the token.
- Log in to DockerHub.
- Click on your profile icon in the top right and select Account Settings.
- In the left-hand menu, click Security.
- Under Access Tokens, click New Access Token.
- Name the token (e.g., "GitHub Actions CI").
- Select the scope as Read, Write, and Delete.
- Click Generate.
- Copy the token and store it in your GitHub repository secrets as
DOCKER_PAT
.
First note: The quickstart will make e-mail sending and CRUD operations functional, should a previously created database with matching names (unless personalized) exist, or the code shall automatically generate it; but the GitHub workflow won't be triggered if the cloned repository, and routes are not updated within the .yml files.
Second note: This reusable project for any disease has the potential to be further improved by incorporating additional parameters into the model, such as symptoms, hospital classification, or blood type, which would enhance its utility for research. Additionally, creating a frontend GUI or a dashboard to analyze and visualize infection trends could provide valuable insights. An extra feature to resend emails when a case is updated can also be implemented, perfecting notification processes and case monitoring.
Third note: Console logs, error messages, and the HTML e-mail template include Spanish writing.