Flask is a popular web framework for building web applications in Python. It is considered a microframework because it provides only the bare essentials for web development, focusing on simplicity and extensibility.
Flask is one of the most popular frameworks for developing REST applications in Python. Flask also has a server to do local deployments and test the application, but no one would go with it to production. So we need another server to deploy the application in production, and one option is to use a WSGI (Web Server Gateway Interface) server like Gunicorn.
In this tutorial, you will learn how to create a Flask application, configure it to Gunicorn, and containerize it.
The application
Let's create a simple Hello World application using Flask as a web framework.
You’ll need to install Python 3 and pip
.
Setup
Create a new file with the name requirements.txt
with the dependencies required for the project:
Flask
Gunicorn
Then update the pip
dependencies and install the dependencies by running the following command:
pip3 install --upgrade pip
pip3 install -r requirements.txt
Application
Create a new file named app.py
containing the Flask REST endpoint:
from flask import Flask
app = Flask(__name__)
@app.route('/')
def hello_world():
return 'Hello, World!'
Running
Let's test the application locally:
flask --app app run
The output will be similar to the following:
* Serving Flask app 'app' * Debug mode: off WARNING: This is a development server. Do not use it in a production deployment. Use a production WSGI server instead. * Running on http://127.0.0.1:5000 Press CTRL+C to quit
If you send a request to the defined endpoint, you'll get the predefined output:
curl localhost:5000/ Hello, World!
So far, so good, but as the log message says, we need another way to use a WSGI server for going to production. So, let's do it.
WSGI
WSGI stands for Web Server Gateway Interface. It is a specification for a standardized interface between web servers and web applications or frameworks written in Python. WSGI defines a contract that allows web servers to communicate with Python web applications, enabling them to work together seamlessly.
This separation of concerns allows web servers to focus on handling low-level networking tasks, such as accepting incoming requests and managing connections. In contrast, web applications can focus on handling the application logic and generating responses.
You can switch between server implementations with minimal application changes. In this case, we'll use Gunicorn.
First, we configure the Gunicorn server by creating a new file named gunicorn_config.py, where we'll configure things like the number of workers and threads and open port or timeout.
import os
workers = int(os.environ.get('GUNICORN_PROCESSES', '2'))
threads = int(os.environ.get('GUNICORN_THREADS', '4'))
# timeout = int(os.environ.get('GUNICORN_TIMEOUT', '120'))
bind = os.environ.get('GUNICORN_BIND', '0.0.0.0:8080')
forwarded_allow_ips = '*'
secure_scheme_headers = { 'X-Forwarded-Proto': 'https' }
Now, the last thing to do is start the Gunicorn server with the application deployed.
Let's run the following command in the terminal:
gunicorn --config gunicorn_config.py app:app
The first app
is the name of the Python file containing the Flask application.
The output shows the application up and running in Gunicorn:
[2023-07-17 22:55:05 +0200] [15903] [INFO] Starting gunicorn 20.1.0 [2023-07-17 22:55:05 +0200] [15903] [INFO] Listening at: http://0.0.0.0:8080 (15903) [2023-07-17 22:55:05 +0200] [15903] [INFO] Using worker: gthread [2023-07-17 22:55:05 +0200] [15904] [INFO] Booting worker with pid: 15904 [2023-07-17 22:55:05 +0200] [15905] [INFO] Booting worker with pid: 15905
If you re-access the service using port 8080, you'll get the Hello World again.
curl localhost:8080/
Hello, World!
Now, with the application up and running locally, it's time to containerize it.
Containerization
Generally speaking, to containerize an application, you create a Dockerfile with all the instructions and dependencies required to run the application.
Dockerfile
The base image contains Python, so we only copy the application files, install the requirements to run the application, and use Gunicorn to start it.
FROM python:3.7.3-slim
COPY requirements.txt /
RUN pip3 install --upgrade pip
RUN pip3 install -r /requirements.txt
COPY . /app
WORKDIR /app
EXPOSE 8080
CMD ["gunicorn","--config", "gunicorn_config.py", "app:app"]
Building the container
We are going to use Podman Desktop to build the container. With Podman (or Docker) running, let's build the container running the following command:
podman build -t hello-flask:1.0.0 .
The output shows the building steps executed:
STEP 1/8: FROM python:3.7.3-slim STEP 2/8: COPY requirements.txt / --> 7721820f5f2e STEP 3/8: RUN pip3 install --upgrade pip … STEP 8/8: CMD ["gunicorn","--config", "gunicorn_config.py", "app:app"] COMMIT hello-flask:1.0.0 --> 7bf50a7a7936 Successfully tagged localhost/hello-flask:1.0.0 7bf50a7a7936d9b87af74b7d49a94c69d217712901d78ba170acb1f28f800d51
[ Learn more: What is Podman Desktop? A developer's introduction ]
Running the container
To run the container, use Podman (or Docker):
podman run -it --rm -p 8080:8080 hello-flask:1.0.0
[2023-07-18 07:35:29 +0000] [1] [INFO] Starting gunicorn 21.0.1
[2023-07-18 07:35:29 +0000] [1] [INFO] Listening at: http://0.0.0.0:8080 (1)
[2023-07-18 07:35:29 +0000] [1] [INFO] Using worker: gthread
[2023-07-18 07:35:29 +0000] [4] [INFO] Booting worker with pid: 4
[2023-07-18 07:35:29 +0000] [5] [INFO] Booting worker with pid: 5
Kubernetes
If you want to try the application in Kubernetes, use the following YAML file:
apiVersion: apps/v1
kind: Deployment
metadata:
labels:
app: hello-service
name: hello-service
spec:
replicas: 1
selector:
matchLabels:
app: hello-service
template:
metadata:
labels:
app: hello-service
spec:
containers:
- name: hello-service
image: quay.io/lordofthejars/hello-flask:1.0.0
ports:
- containerPort: 8080
---
apiVersion: v1
kind: Service
metadata:
name: hello-service
labels:
app: hello-service
spec:
ports:
- name: http
port: 8080
selector:
app: hello-service
type: LoadBalancer
Use any Kubernetes cluster (minikube, kind, etc.) or a cloud Kubernetes implementation like OpenShift for free in the Developer Sandbox for Red Hat OpenShift.
Conclusion
Flask is one of the most popular frameworks in Python to develop web applications, but when it comes to moving to production, you need a server like Gunicorn to deploy it.
Containerizing the application is a simple process of putting these steps into a Dockerfile and choosing the right base image.
Last updated: September 19, 2023