- 原文地址:How to write Dockerfiles for Python Web Apps
- 原文做者:Praveen Durairaj
- 译文出自:掘金翻译计划
- 本文永久连接:github.com/xitu/gold-m…
- 译者:lsvih
- 校对者:Starriers, steinliber
本文涵盖了从建立简单的 Dockerfile 到生产环境多级构建 Python 应用的例子。如下为本指南的内容摘要:css
gunicorn
进行热加载。pip install
。flask
的 static 及 template 目录部署静态文件(好比 React、Vue、Angular 生成的 bundle)。alpine
进行生产环境下的多级构建,减小最终镜像文件的大小。--reload
与 --reload_extra_files
监视文件(包括 html、css 及 js)的修改。若是你须要以上步骤的代码,请参考 GitHub repo.html
假设咱们有一个名为 python-app 的应用,为其准备一个简单的目录结构。在顶级目录下,包含 Dockerfile
以及 src
文件夹。前端
python app 的源码就存放在 src
目录中,app 的依赖关系保存在 requirements.txt
里。为了简洁起见,咱们假设 server.py 定义了一个运行于 8080 端口的 flask 服务。python
python-app
├── Dockerfile
└── src
└── server.py
└── requirements.txt
复制代码
FROM python:3.6
# 建立 app 目录
WORKDIR /app
# 安装 app 依赖
COPY src/requirements.txt ./
RUN pip install -r requirements.txt
# 打包 app 源码
COPY src /app
EXPOSE 8080
CMD [ "python", "server.py" ] 复制代码
咱们将使用最新版本的 python:3.6
做为基础镜像。android
在构建镜像时,docker 会获取全部位于 context
目录下的文件。为了提升 docker 构建的速度,能够在 context 目录中添加 .dockerignore
文件来排除不须要的文件与目录。ios
一般,你的 .dockerignore
文件件应该以下所示:git
.git
__pycache__
*.pyc
*.pyo
*.pyd
.Python
env
复制代码
构建并运行此镜像:github
$ cd python-docker
$ docker build -t python-docker-dev .
$ docker run --rm -it -p 8080:8080 python-docker-dev
复制代码
你将能在 [http://localhost:8080](http://localhost:8080.)
访问此 app。使用 Ctrl+C
组合键能够退出程序。web
如今,假设你但愿在每次修改代码(好比在本地部署时)时都运行以上代码,那么你须要在启停 python 服务时将代码源文件挂载到容器中。docker
$ docker run --rm -it -p 8080:8080 -v $(pwd):/app \
python-docker-dev bash
root@id:/app# python src/server.py
复制代码
gunicorn 是一款运行于 Unix 下的 Python WSGI HTTP server,使用的是 pre-fork worker 模型(注,Arbiter 是 gunicorn 的 master,所以称 gunicorn 为 pre-fork worker)。你可使用各类各样的选项来配置 gunicorn。向 gunicorn 命令中传入 --reload
或是将 reload
写入配置文件,就可让 gunicorn 在有文件发生变化时自动重启 python 服务。
FROM python:3.6
# 建立 app 目录
WORKDIR /app
# 安装 app 依赖
COPY gunicorn_app/requirements.txt ./
RUN pip install -r requirements.txt
# 打包 app 源码
COPY gunicorn_app /app
EXPOSE 8080
复制代码
咱们将构建镜像并运行 gunicorn,以便在 app
目录下文件发生变更时对代码进行 rebuild。
$ cd python-docker
$ docker build -t python-hot-reload-docker .
$ docker run --rm -it -p 8080:8080 -v $(pwd):/app \
python-hot-reload-docker bash
root@id:/app# gunicorn --config ./gunicorn_app/conf/gunicorn_config.py gunicorn_app:app
复制代码
一切在 app
目录下 python 文件的更改都会触发 rebuild,发生的变化都能在 [http://localhost:8080](http://localhost:8080.)
上实时展现。请注意,咱们已经将文件挂载到了容器中,所以 gunicorn 才能正常工做。
其它格式的文件怎么办? 若是你但愿 gunicorn 在监视代码变更的时候也监视其它类型的文件(如 template、view 之类的文件),能够在 reload_extra_files
参数中进行指定。此参数接受数组形式的多个文件名。
你能够经过 docker run,使用 python 镜像来简单地运行 python 单文件脚本。
docker run -it --rm --name single-python-script -v "$PWD":/app -w /app python:3 python your-daemon-or-script.py
复制代码
你也能够给脚本传递一些参数。在上面的例子中,咱们就已经挂载了当前工做目录,也就是说能够将目录中的文件当作参数传递。
上面的 Dockerfile 假定了你是使用 Python 运行一个 API 服务器。若是你想用 Python 为 React.js、Vue.js、Angular.js app 提供服务,可使用 Flask。Flask 为渲染静态文件提供了一种便捷的方式:html 文件放在 templates
目录中,css、js 及图片放在 static
目录中。
请在此 repo 中查看简单的 hello world 静态 app 的目录结构。
FROM python:3.6
# 建立 app 目录
WORKDIR /app
# 安装 app 依赖
COPY static_app/requirements.txt ./
RUN pip install -r requirements.txt
# 打包 app 源码
COPY static_app /app
EXPOSE 8080
CMD ["python","server.py"] 复制代码
In your server.py,
if __name__ == '__main__':
app.run(host='0.0.0.0')
复制代码
请注意,host 须要设置为 0.0.0.0
- 这样可让你的服务在容器外被访问。若是不设置此参数,host 会默认设为 localhost
。
FROM python:3.6
# 建立 app 目录
WORKDIR /app
# 安装 app 依赖
COPY gunicorn_app/requirements.txt ./
RUN pip install -r requirements.txt
# 打包 app 源码
COPY . /app
EXPOSE 8080
CMD ["gunicorn", "--config", "./gunicorn_app/conf/gunicorn_config.py", "gunicorn_app:app"] 复制代码
构建并运行这个一体化镜像:
$ cd python-docker
$ docker build -t python-docker-prod .
$ docker run --rm -it -p 8080:8080 python-docker-prod
复制代码
因为底层为 Debian,构建完成后镜像约为 700MB(具体数值取决于你的源码)。下面探讨如何减少这个文件的大小。
使用多级构建时,将在 Dockerfile 中使用多个 FROM
语句,但最后仅会使用最终阶段构建的文件。这样,获得的镜像将仅包含生产服务器中所需的依赖,理想状况下文件将很是小。
当你须要使用依赖于系统的模块或须要编译的模块时,这种构建模式十分有用。好比 pycrypto
和 numpy
就很适合这种方法。
# ---- 基础 python 镜像 ----
FROM python:3.6 AS base
# 建立 app 目录
WORKDIR /app
# ---- 依赖 ----
FROM base AS dependencies
COPY gunicorn_app/requirements.txt ./ # 安装 app 依赖
RUN pip install -r requirements.txt
# ---- 复制文件并 build ----
FROM dependencies AS build
WORKDIR /app COPY . /app # 在须要时进行 Build 或 Compile
# --- 使用 Alpine 发布 ----
FROM python:3.6-alpine3.7 AS release
# 建立 app 目录
WORKDIR /app
COPY --from=dependencies /app/requirements.txt ./ COPY --from=dependencies /root/.cache /root/.cache
# 安装 app 依赖
RUN pip install -r requirements.txt COPY --from=build /app/ ./ CMD ["gunicorn", "--config", "./gunicorn_app/conf/gunicorn_config.py", "gunicorn_app:app"] 复制代码
使用上面的方法,用 Alpine 构建的镜像文件大小约 90MB,比以前少了 8 倍。使用 alpine
版本进行构建能有效减少镜像的大小。
注意: 上面的 Dockerfiles 是为 python 3
编写的,你能够只作少数修改就能将其改成 python 2
版本。若是你要部署的是 django
应用,也应该能经过少数改动就作出可部署于生产环境的 Dockerfiles。
若是你对前面的方法有任何建议,或但愿看到别的用例,请告知做者。
欢迎加入 Reddit 或 HackerNews 参与讨论 :)
此外,你是否试过将 python web app 部署在 Hasura 上呢?这实际上是将 python 应用部署于 HTTPS 域名的最快的方法(仅需使用 git push)。尝试使用 hasura.io/hub/project… 的模板快速入门吧!Hasura 中全部的项目模板都带有 Dockerfile 与 Kubernetes 标准文件,你能够自由进行定义。
掘金翻译计划 是一个翻译优质互联网技术文章的社区,文章来源为 掘金 上的英文分享文章。内容覆盖 Android、iOS、前端、后端、区块链、产品、设计、人工智能等领域,想要查看更多优质译文请持续关注 掘金翻译计划、官方微博、知乎专栏。