部署

    • 创建项目的运行环境
    • 使用 Gunicorn 启动 flask 程序
    • 使用 supervisor 管理服务器进程
    • 使用 Nginx 做反向代理
    • 创建 Python 虚拟环境,以便隔离不同的项目
    • 安装项目依赖包

    我们在本地调试的时候经常使用命令 或者 python app.py 等启动 Flask 自带的服务器,但是,Flask 自带的服务器性能无法满足生产环境的要求,因此这里我们采用 Gunicorn 做 wsgi (Web Server Gateway Interface,Web 服务器网关接口) 容器,假设我们以 root 用户身份进行部署:

    1. (flask-todo-env)$ pip install gunicorn
    2. (flask-todo-env)$ /home/root/.virtualenvs/flask-todo-env/bin/gunicorn -w 4 -b 127.0.0.1:7345 application.app:create_app()

    上面的命令中,-w 参数指定了 worker 的数量,-b 参数绑定了地址(包含访问端口)。

    在上面,我们手动使用命令启动了 flask 程序,当程序挂掉的时候,我们又要再启动一次。另外,当我们想关闭程序的时候,我们需要找到 pid 进程号并 kill 掉。这里,我们采用一种更好的方式来管理服务器进程,我们将 supervisor 安装全局环境下,而不是在当前的虚拟环境:

    supervisor.conf 添加以下内容:

    1. [program:flask-todo-env]
    2. directory=/home/root/flask-todo-app
    3. command=/home/root/.virtualenvs/%(program_name)s/bin/gunicorn
    4. -w 4
    5. -b 127.0.0.1:7345
    6. --log-level debug
    7. --error-logfile=-
    8. --name %(program_name)s
    9. "application.app:create_app()"
    10. environment=PATH="/home/root/.virtualenvs/%(program_name)s/bin"
    11. numprocs=1
    12. user=deploy
    13. autostart=true
    14. autorestart=true
    15. redirect_stderr=true
    16. redirect_stdout=true
    17. stdout_logfile=/home/root/%(program_name)s-out.log
    18. stdout_logfile_maxbytes=100MB
    19. stdout_logfile_backups=10
    20. stderr_logfile=/home/root/%(program_name)s-err.log
    21. stderr_logfile_maxbytes=100MB
    22. stderr_logfile_backups=10

    将 Nginx 作为反向代理可以处理公共的 HTTP 请求,发送给 Gunicorn 并将响应带回给发送请求的客户端。在 ubuntu 上可以使用 sudo apt-get install nginx 安装 nginx,其他系统也类似。

    要想配置 Nginx 作为运行在 127.0.0.1:7345 的 Gunicorn 的反向代理,我们可以在 /etc/nginx/sites-enabled 下给应用创建一个文件,不妨称之为 flask-todo-app.com,nginx 的类似配置如下:

    1. server {
    2. listen 80;
    3. server_name flask-todo-app.com;
    4. # Handle all locations
    5. location / {
    6. # Pass the request to Gunicorn
    7. proxy_pass http://127.0.0.1:7345;
    8. # Set some HTTP headers so that our app knows where the request really came from
    9. proxy_set_header Host $host;
    10. proxy_set_header X-Real-IP $remote_addr;
    11. proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
    12. }

    可以看到,我们上面的部署方式,都是手动部署的,如果有多台服务器要部署上面的程序,那就会是一个恶梦,有一个自动化部署的神器 可以帮助我们解决这个问题,感兴趣的读者可以了解一下。