使用docker部署nginx+flask+gunicorn+mysql项目

使用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 #启动服务
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

友情链接更多精彩内容