使用docker部署nginx+flask+gunicorn+mysql项目
WSGI服务配合nginx介绍https://www.cnblogs.com/vh-pg/p/11731637.html
目录结构
[root@k8s-master ~]# tree lab_project/ -L 2
lab_project/
├── docker-compose.yml
├── flask
│ ├── Dockerfile
│ └── lab_app
├── mysql
│ ├── Dockerfile
│ └── laboratory_web.sql
└── nginx
├── Dockerfile
└── nginx.conf
4 directories, 6 files
创建 mysql 镜像+容器
准备初始化数据库的 sql 脚本(mysqldump -uroot -p*** LABORATORY_WEB > laboratory_web.sql),包括建库,以及表的DDL和DML,这里为 laboratory_web.sql
以mysql5.7.20为基础镜像,Dockerfile如下:
FROM mysql:5.7.20
COPY ./laboratory_web.sql /docker-entrypoint-initdb.d
将 laboratory_web.sql 初始化数据库脚本复制进镜像的 /docker-entrypoint-initdb.d, 再启动容器时会自动执行脚本建库
使用build构建后,run 启动容器
docker build -t lab_project_mysql .
docker run -itd --name mysql -p 3306:3306 -e MYSQL_ROOT_PASSWORD=*** lab_project_mysql
启动容器时要指定容器的名字以便再容器之间通信使用,还需要指定mysql root用户的密码。
接下来可以进入容器查看应用需要的数据库是否构建成功
docker exec -it mysql env LANG=C.UTF-8 /bin/bash
mysql -uroot -p***
这里需要设置环境变量 utf-8 防止乱码
创建 flask 镜像+容器
准备 flask 项目,这里是 lab_app, 项目中需要包含 requirement.txt 来安装 flask 项目所需要的依赖。
这里的 requirement.txt 中包含了 gunicorn :
Flask==1.0.2
Flask-Script==2.0.6
Flask-SQLAlchemy==2.3.2
PyMySQL==0.9.3
gunicorn==19.7.1
目前需要将版本号去掉
以 python:3.8.0-alpine 为基础镜像,Dockerfile 如下:
FROM python:3.8.0-alpine
ADD ./lab_app/requirements.txt /root/requirements.txt
WORKDIR /root/lab_app
RUN apk add build-base && pip3 install -r /root/requirements.txt
CMD ["gunicorn", "-w", "5", "-b", "0.0.0.0:5000", "manage:app"]
gunicorn此时不能加 -D 参数,CMD是容器启动时才会执行的命令,如果加入 -D 参数即后台运行,相当于命令已经执行完,把Docker看成是一个线程的话,相当于这个线程已经结束,该容器在启动完后会自动退出。
容器内的 gunicorn 如果使用 127.0.0.1:5000 这个地址,那么在容器外是无法访问到的,因为端口5000只监听在127.0.0.1这个地址上,这个地址是没有映射到主机的,所以从宿主机访问不了,因此改为 0.0.0.0 监听5000端口即可
关于 flask 连接 mysql,这里有两种方式,一种是使用 docker0 桥接网卡来进行通信,一种是使用容器之间的 link
使用 docker0 网卡
由于 mysql 容器监听主机的3306端口, 因此访问宿主机 ip : 3306 即可访问 mysql 容器, flask 项目中的连接数据库的地址需要改为 docker 的地址,使用 ifconfig 查看 docker0 网桥的地址,docker是通过这个网桥来使主机和容器之间互相通信,因此容器内部要访问宿主机,地址应该为 docker0 的地址+数据库端口 172.27.0.1:3306
所以flask 的配置为
app.config['SQLALCHEMY_DATABASE_URI'] = 'mysql+pymysql://root:***@172.27.0.1:3306/LABORATORY_WEB?charset=utf8'
使用容器之间的 link
需要在启动flask容器时指定 --link参数,来连接两个容器,此时指定的是 mysql 的容器名,--link mysql:mysql 【--link 容器名:容器别名 参数】
所以flask 的配置为
app.config['SQLALCHEMY_DATABASE_URI'] = 'mysql+pymysql://root:***@mysql:3306/LABORATORY_WEB?charset=utf8'
使用build构建后,run 启动容器
docker build -t lab_project_flask .
docker run -itd --name flask -p 5000:5000 --link mysql:mysql lab_project_flask
此时可以通过 curl 127.0.0.1:5000 来访问到 flask 项目
创建 nginx 镜像+容器
准备 nginx.conf 配置文件
nginx.conf
user nginx;
worker_processes auto;
error_log /var/log/nginx/error.log;
pid /run/nginx.pid;
include /usr/share/nginx/modules/*.conf;
events {
worker_connections 1024;
}
http {
log_format main '$remote_addr - $remote_user [$time_local] "$request" '
'$status $body_bytes_sent "$http_referer" '
'"$http_user_agent" "$http_x_forwarded_for"';
access_log /var/log/nginx/access.log main;
sendfile on;
tcp_nopush on;
tcp_nodelay on;
keepalive_timeout 65;
types_hash_max_size 2048;
include /etc/nginx/mime.types;
default_type application/octet-stream;
include /etc/nginx/conf.d/*.conf;
server {
listen 80 default_server;
listen [::]:80 default_server;
server_name _;
access_log /var/log/nginx/cccc_access.log;
error_log /var/log/nginx/cccc_error.log;
root /usr/share/nginx/html;
include /etc/nginx/default.d/*.conf;
location / {
proxy_pass http://flask:5000;
proxy_redirect off;
proxy_set_header Host $http_host;
proxy_set_header X-Real-IP $remote_addr;
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
proxy_set_header X-Forwarded-Proto $scheme;
}
location ~ .*\.(gif|jpg|jpeg|png|bmp|swf|ico)$ {
proxy_pass http://flask:5000;
expires 30d;
}
location ~ .*\.(js|css)?$ {
proxy_pass http://flask:5000;
expires 15d;
}
error_page 404 /404.html;
location = /40x.html {
}
error_page 500 502 503 504 /50x.html;
location = /50x.html {
}
}
}
nginx 容器也通过 link 参数连接 flask 容器,因此配置文件中 location 的代理地址 写 flask:5000 即可
以 nginx:1.16.1 为基础镜像,Dockerfile 如下:
FROM nginx:1.16.1
COPY ./nginx.conf /etc/nginx/nginx.conf
使用build构建后,run 启动容器
docker build -t lab_project_nginx .
docker run -itd --name nginx -p 80:80 --link flask lab_project_nginx
接下来就可直接访问主机 ip 地址访问到 flask 项目了
docker-compose,yml
version: '3'
services:
mysql:
build: ./mysql
ports:
- "3306:3306"
environment:
- MYSQL_ROOT_PASSWORD=wxb2018.LZU
restart: always
flask:
build: ./flask
links:
- mysql:mysql
restart: always
volumes:
- "./flask/lab_app:/root/lab_app"
nginx:
build: ./nginx
ports:
- "80:80"
links:
- flask
cd /root/lab_project
docker-compose up -d #启动服务
# 如果需要更新代码,将lab_app更新后进行打包,然后重启服务
zip -q -r lab_app-2020-3-22.zip lab_app/
docker-compose down -v #停止服务
docker-compose up -d #启动服务