Skip to content

Latest commit

 

History

History
686 lines (447 loc) · 20.2 KB

README.md

File metadata and controls

686 lines (447 loc) · 20.2 KB

docker-python-flask

This project demonstrates how to Dockerize a Flask application and set it up on an AWS EC2 instance.

The goal is to first make you familiar with the manual setup of the application, providing a solid understanding of how each component works individually. Once you’re comfortable with the manual process, you’ll then learn how to streamline and simplify deployment using Docker.

This README provides comprehensive instructions for both manual setup and Dockerized deployment.

You will find detailed instructions for:

Manual Setup: Learn how to set up the Flask application manually without Docker, which provides insight into the configuration and challenges of a traditional deployment.

Dockerized Setup: Understand how to simplify and streamline deployment using Docker and Docker Compose, including detailed steps for running and managing containers.

Overview

This project provides two approaches to setting up a Flask application:

Manual Setup: Run the application and its components manually on an AWS EC2 instance.

Dockerized Setup: Use Docker and Docker Compose on an AWS EC2 instance to automate the deployment process, ensuring a consistent and reproducible environment.

Architectural Diagram

The architectural diagram below illustrates the components and their interactions in this project.

Architectural Diagram

Note: Before we begin, please fork the repository https://github.com/shivalkarrahul/docker-python-flask. The reason for forking will be explained at the end of the document.

Here are the steps to fork the repository:

  1. Log in to GitHub: Go to GitHub and sign in to your account. If you don't have an account, create one.

  2. Navigate to the Repository: Visit the repository you want to fork by clicking this link: https://github.com/shivalkarrahul/docker-python-flask.

  3. Fork the Repository: On the top-right corner of the repository page, click the Fork button.

  4. GitHub will create a copy of the repository under your account.

Understand the Application

Click to expand application details

Flask Application

The Flask application is a Python web application built using the Flask framework. Flask is a lightweight web framework that is easy to set up and scale. It is used here to create a simple web service with a few endpoints.

Files in the Flask Folder:(For the Dockerized Setup)

  • Dockerfile: This file contains the instructions to build a Docker image for the Flask application. It sets up the environment, installs dependencies, and defines how to run the Flask application.

  • main.py: This is the main application file where the Flask web server is defined. It includes the routes and logic for handling HTTP requests.

  • requirements.txt: This file lists the Python dependencies required for the Flask application. Docker uses this file to install the necessary packages inside the Docker container.

Nginx Proxy

Nginx is used as a reverse proxy server to forward requests to the Flask application. It handles incoming HTTP requests and directs them to the Flask application container.

Files in the Nginx Folder:(For the Dockerized Setup)

  • Dockerfile: This file contains the instructions to build a Docker image for the Nginx server. It sets up Nginx with the required configuration to serve as a reverse proxy.

  • conf: This folder contains the Nginx configuration files. The primary configuration file, default.conf, sets up the proxy rules, such as forwarding requests to the Flask application and handling static files.

Redis

In this setup, Redis is used solely to keep track of the application hit count. The Redis container is configured to work with the Flask application but does not have any associated files or directories in this project.

Manual Setup Instructions

To understand how the application is configured and run without Docker, you can expand this section. If you're eager to jump directly to the Dockerized setup, you can skip this part.


Before diving into the Dockerized setup of the Python Flask application, it’s valuable to understand the manual setup process. This section provides step-by-step instructions on how to set up the application from scratch without using Docker. By following these steps, you'll gain insights into the components involved and how they interact.

Create an EC2 Instance on AWS

  1. Select Ubuntu Version: 24.04 LTS
  2. Amazon Machine Image (AMI): Ubuntu Server 24.04 LTS (HVM), SSD Volume Type (ami-0e86e20dae9224db8)
  3. Volume Size: 30 GB (8 GB will also work)
  4. Instance Type: t2.medium (t2.micro is also acceptable)
  5. Subnet: Public
  6. SSH Key: workshop-docker-python-flask (Download the key on your machine - Download .ppk if you are having a Windows Machine)
  7. Security Group: Allow SSH (port 22) and HTTP (port 80) for your IP

Commands on Your Local Machine

  1. Set Permissions for Your SSH Key:

    chmod 400 ~/Downloads/workshop-docker-python-flask.pem
  2. SSH into Your EC2 Instance using Terminal:

    ssh -i ~/Downloads/workshop-docker-python-flask.pem ubuntu@<EC2-Public-IP>

or

  1. SSH into Your EC2 Instance from Windows using Putty:

Setup Application on the EC2 Instance

  1. Update and install necessary packages

    sudo apt update
    sudo apt install python3-pip nginx redis-server -y
  2. Create directories and files for the Flask app

    mkdir -p ~/my_flask_app
    cd ~/my_flask_app
    mkdir flask
    cd flask
    touch requirements.txt main.py
  3. Edit the main.py file

    vim main.py

    Add the following code:

    import redis
    from flask import Flask
    app = Flask(__name__)
    redis = redis.Redis(host='0.0.0.0', port=6379, db=0)
    
    @app.route('/')
    def hello_world():
        return 'This is a Python Flask Application with Redis accessed through Nginx'
    
    @app.route('/visitor')
    def visitor():
        redis.incr('visitor')
        visitor_num = redis.get('visitor').decode("utf-8")
        return "Visit Number = : %s" % (visitor_num)
    
    @app.route('/visitor/reset')
    def reset_visitor():
        redis.set('visitor', 0)
        visitor_num = redis.get('visitor').decode("utf-8")
        return "Visitor Count has been reset to %s" % (visitor_num)
Instructions for vim command
  1. Open the file using vim command

  2. Enter Insert Mode:

    • Press i to go into Insert Mode and start editing the file.
  3. Save and Exit:

    • Press Esc to exit Insert Mode.
    • Type :wq and press Enter to save and exit.
    • If you want to save without exiting, type :w.
    • To quit without saving changes, use :q!.
  1. Create the requirements.txt file

    vim requirements.txt

    Add the following:

    redis==3.4.1
    gunicorn>=20,<21
    Flask==2.0.3
  2. Install dependencies

    pip3 install -r requirements.txt 

    You will face an issue here; ignore it and proceed. We will use Python Virtual Environment

    sudo apt install python3-venv -y
    python3 -m venv venv
    source venv/bin/activate
    pip install -r requirements.txt
  3. Run the app using Gunicorn

    gunicorn -w 4 -b 0.0.0.0:8000 main:app

    You will face an issue here. We will fix it by adding the dependency in requirements.txt

  4. Freeze the dependencies (optional)

    pip3 freeze
  5. Update the requirements.txt if necessary

    update the requirements.txt

    vim requirements.txt
    redis==3.4.1
    gunicorn>=20,<21
    Flask==2.0.3
    Werkzeug==2.0.3
    pip3 install -r requirements.txt 
  6. Run the app using Gunicorn

    gunicorn -w 4 -b 0.0.0.0:8000 main:app

Note: Keep the Flask application running in the current session. Open a new terminal session and reconnect to the EC2 instance.

  1. Configure Nginx for proxying requests to the Flask app

    sudo vim /etc/nginx/sites-available/my_flask_app

    Add the following configuration:

    server {
        listen 80;
        server_name localhost;
    
        location / {
            proxy_pass http://127.0.0.1:8000;
        }
    
        location /hit {
            proxy_pass http://127.0.0.1:8000/visitor;
        }
    
        location /hit/reset {
            proxy_pass http://127.0.0.1:8000/visitor/reset;
        }
    }
  2. Link the Nginx configuration and restart Nginx

    sudo ln -s /etc/nginx/sites-available/my_flask_app /etc/nginx/sites-enabled/
    sudo nginx -t
    sudo rm /etc/nginx/sites-enabled/default
    sudo systemctl restart nginx
  3. Access the application

Once the Nginx, Python and Redis services are running, you can access the application through your web browser:

Replace EC2-Public-IP with your EC2 instance's actual public IP addres

  1. Stop the Applicatoin

    Stop Nginx

    sudo service nginx  stop

    Stop Redis

    sudo service redis  stop

    Stop Flask Application
    Press Ctrl + C in the terminal where gunicorn -w 4 -b 0.0.0.0:8000 main:app is running.


Hope you observed the complexity and issues we faced during the manual setup. Imagine replicating this process on multiple Development, QA, and Production machines. This is where Docker simplifies and streamlines the deployment process.

Dockerized Setup Instructions

To understand how to streamline the deployment of the application using Docker, you can expand this section.

Create an EC2 Instance on AWS

  1. Select Ubuntu Version: 24.04 LTS
  2. Amazon Machine Image (AMI): Ubuntu Server 24.04 LTS (HVM), SSD Volume Type (ami-0e86e20dae9224db8)
  3. Volume Size: 30 GB (8 GB will also work)
  4. Instance Type: t2.medium (t2.micro is also acceptable)
  5. Subnet: Public
  6. SSH Key: workshop-docker-python-flask (Download the key on your machine - Download .ppk if you are having a Windows Machine)
  7. Security Group: Allow SSH (port 22) and HTTP (port 80) for your IP

Commands on Your Local Machine

  1. Set Permissions for Your SSH Key:

    chmod 400 ~/Downloads/workshop-docker-python-flask.pem
  2. SSH into Your EC2 Instance using Terminal:

    ssh -i ~/Downloads/workshop-docker-python-flask.pem ubuntu@<EC2-Public-IP>

or

  1. SSH into Your EC2 Instance from Windows using Putty:

Setup Docker on the EC2 Instance

  1. Install Docker:

    • Update the package index and install prerequisites:

      sudo apt-get update
      sudo apt-get install ca-certificates curl -y
    • Add Docker’s GPG key:

      sudo curl -fsSL https://download.docker.com/linux/ubuntu/gpg | sudo gpg --dearmor -o /usr/share/keyrings/docker-archive-keyring.gpg
    • Add Docker’s repository:

      echo "deb [arch=$(dpkg --print-architecture) signed-by=/usr/share/keyrings/docker-archive-keyring.gpg] https://download.docker.com/linux/ubuntu $(lsb_release -cs) stable" | sudo tee /etc/apt/sources.list.d/docker.list > /dev/null
    • Install Docker:

      sudo apt-get update
      sudo apt-get install docker-ce docker-ce-cli containerd.io docker-compose-plugin -y
    • Verify Docker Installation:

      sudo docker run hello-world
    • Allow Non-Root User to Run Docker Commands:

      sudo groupadd docker
      sudo usermod -aG docker $USER
      newgrp docker
  2. Clone the Project Repository:

    cd /home/ubuntu/
    git clone https://github.com/shivalkarrahul/docker-python-flask.git
    cd docker-python-flask/

Run the Application Manually

If you prefer to start each container manually, follow these steps:

  1. Start the Redis Container:

    docker run -d --name redis redis:alpine
  2. Build and Start the Flask Application Container:

    docker build -t flask flask/
    docker run -d --name app --link redis:redis -p 8000:8000 -v app:/app flask
  3. Build and Start the Nginx Proxy Container:

    docker build -t nginx nginx/
    docker run -d --name proxy --link app:app -p 80:80 --restart always nginx
  4. Stop the Nginx Proxy, Flask and Redis Container:

    docker stop proxy app redis

Alternative: Using Docker Compose

For a simpler setup, you can use Docker Compose to manage all containers with a single command. Docker Compose can start the containers in the foreground or background, depending on your preference:

  • Start Containers in the Foreground:

    docker compose up

    This command will start all containers and keep the terminal open with their logs.

  • Stop Containers running in the Foreground:

    Press Ctrl + C in the terminal where docker compose up is running.

  • Start Containers in the Background:

    If you want to run the containers in the background, add the -d flag:

    docker compose up -d

    This command will start the containers in detached mode, allowing you to continue using the terminal.

  • Stop and Remove Containers:

    To stop and remove all containers created by Docker Compose, use:

    docker compose down

    This command will stop and clean up the containers, networks, and volumes defined in the docker-compose.yml file.

Access the Application

  • Restart Containers:

    As we have stopped the containers, start them again to access the application:

    docker compose up -d
  • Stop Containers:

    docker compose down

Once the application containers are running, you can access the application through your web browser:

Replace EC2-Public-IP with your EC2 instance's actual public IP address.

Storing and Sharing Docker Images with ECR

This section covers how to store and share your Docker images using AWS Elastic Container Registry (ECR). ECR is a fully-managed container registry that makes it easy to store, manage, and deploy Docker container images.

Step 1: Create and Attach a Role to the EC2 Instance for ECR Access

To allow an EC2 instance to push/pull images from ECR, you need to attach an appropriate IAM role with ECR permissions.

  1. Go to the IAM console, choose Create Role > AWS service > Select EC2 in Service or use case Drop Down > Select EC2 in Use Case > Click Next
  2. Search and Select AmazonEC2ContainerRegistryFullAccess, then click on the Next button.
  3. Give a name to the role (e.g. demo-ecr-role) and Click on the Create Role button.
  4. Go to your EC2 instance, choose Actions > Security > Modify IAM Role and assign the newly created role.

Now, your EC2 instance has permissions to interact with ECR, allowing you to push and pull images.

Step 2: Create an ECR Repository

To store your Docker images, you first need to create an ECR repository.

  1. Search and Open the Amazon ECR.
  2. Choose Create repository.
  3. Enter a name for your repository (e.g., my-python-flask-app).
  4. Keep the other settings to default.
  5. Click Create repository.

Step 3: Authenticate Docker to ECR

Once the repository is created, authenticate Docker to your ECR registry by running the following AWS CLI command:

Syntax:

aws ecr get-login-password --region <your-region> | docker login --username AWS --password-stdin <aws-account-id>.dkr.ecr.<your-region>.amazonaws.com

Replace your-region and aws-account-id with your specific AWS region and account ID.

The command will fail here, as we don't have aws cli on our machine. Install aws cli using the following commands.

curl "https://awscli.amazonaws.com/awscli-exe-linux-x86_64.zip" -o "awscliv2.zip"
unzip awscliv2.zip

This will also fail as we dont have unzip command on our machine. Install unzip using the following command.

sudo apt install unzip
unzip awscliv2.zip
sudo ./aws/install

Now, again try to login

Syntax:

aws ecr get-login-password --region <your-region> | docker login --username AWS --password-stdin <aws-account-id>.dkr.ecr.<your-region>.amazonaws.com

Replace your-region and aws-account-id with your specific AWS region and account ID.

Note: You should see the Login Succeeded message at the end of the logs after executing the above command.

Step 3: Tag and Push Docker Images to ECR

Tag your Docker image so that it points to your ECR repository:

First, let's list the images.

docker images

Now, let's tag the images

Syntax:

docker tag <local-image>:<tag> <aws-account-id>.dkr.ecr.<your-region>.amazonaws.com/<repository-name>:<tag>

Then, push the image to the ECR repository:

Syntax:

docker push <aws-account-id>.dkr.ecr.<your-region>.amazonaws.com/<repository-name>:<tag>

Note: In this guide, we created only one repository and pushed a single image to ECR. However, since we also have images for Nginx and Redis, you can create separate repositories for these services and push their images as well, if needed.

Cleanup

To stop the application and clean up:

  1. Terminate the EC2 Instances:

    Go to the AWS Management Console and terminate the EC2 instance to avoid incurring additional charges.

Feel free to reach out if you encounter any issues or have questions!

Kubernetes Setup Instructions (Coming Soon)

While Kubernetes setup is not covered in this guide, it's a powerful tool for orchestrating containerized applications. If you enjoyed this guide and find the Docker setup valuable, you might be interested in exploring Kubernetes for advanced features like scaling, managing, and automating containerized applications. Should you wish to dive deeper into Kubernetes, I’d be glad to include it in future updates to this repository.

Reason for Forking the Repository:

Click here to know the reasons to fork the repository


Forking the repository allows you to create a personal copy of the project under your own GitHub account. This gives you the flexibility to:

  • Experiment with the code without affecting the original repository.

  • Make changes, improvements, or adjustments to suit your own project requirements.

  • Contribute back to the original project by submitting a pull request if you would like your changes to be reviewed and potentially merged.

Let’s Connect:

LinkedInQR