使用 nginx、django、daphne 部署到 docker

Posted

技术标签:

【中文标题】使用 nginx、django、daphne 部署到 docker【英文标题】:Deploy to docker with nginx, django, daphne 【发布时间】:2017-07-14 01:29:55 【问题描述】:

我想将我的服务部署到 docker。

我的服务是使用 python+django 和 django-channels 开发的

── myproject ├── myproject │ ├── settings.py │ ├── urls.py │ ├── asgi.py │ ├── ... ├── collected_static │ ├── js │ ├── css │ ├── ... ├── nginx │ ├── Dockerfile │ ├── service.conf ├── requirements.txt ├── manage.py ├── Dockerfile └── docker-compose.yml

我的项目/Dockerfile:

FROM python
ENV PYTHONUNBURRERED 1

RUN mkdir -p /opt/myproject
WORKDIR /opt/myproject
ADD . /opt/myproject

RUN pip install -r requirements.txt
RUN python manage.py migrate

myproject/docker-compose.yml:

version: '2'
services:
  nginx:
    build: ./nginx
    networks:
      - front
      - back
    ports:
      - "80:80"
    depends_on:
      - daphne
  redis:
    image: redis
    networks:
      - "back"
    ports:
      - "6379:6379"
  worker:
    build: .
    working_dir: /opt/myproject
    command: bash -c "python manage.py runworker"
    environment:
      - REDIS_HOST=redis
    networks:
      - front
      - back
    depends_on:
      - redis
    links:
      - redis
  daphne:
    build: .
    working_dir: /opt/myproject
    command: bash -c "daphne -b 0.0.0.0 -p 8000 myproject.asgi:channel_layer"
    ports:
      - "8000:8000"
    environment:
      - REDIS_HOST=redis
    networks:
      - front
      - back
     depends_on:
      - redis
     links:
      - redis
  networks:
    front:
    back:

我的项目/nginx/Dockerfile

FROM nginx
COPY service.conf /etc/nginx/sites-enabled/

myproject/nginx/service.conf

server 
  listen 80;
  server_name example.com #i just want to hide domain name..
  charset utf-8;
  client_max_body_size 20M;

  location /static/ 
    alias /opt/myproject/collected_static/;
  

  location / 
    proxy_pass http://0.0.0.0:8000;
    proxy_http_version 1.1;
    proxy_set_header Upgrade $http_upgrade;
    proxy_set_header Connection "upgrade";

    proxy_redirect off;
    proxy_set_header Host $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-Host $server_name;
  

我写了一个命令docker-compose up -d,nginx和daphne运行良好。

但是当我连接到 example.com:80 时,我只能看到 nginx 默认页面。

当我连接到 example.com:8000 时,我只能看到 myproject 的服务页面。 (但看不到静态文件)

我要链接nginx和daphne服务!我该怎么办?请帮帮我。

当我在没有 docker 的情况下使用 nginx+daphne+django 进行部署时,我的服务运行良好。

【问题讨论】:

我创建了一个github示例,如何在docker-compose中设置nginx、daphne、gunicorn github.com/pplonski/simple-tasks 【参考方案1】:

TLDR;

Nginx 配置不正确,而且你的 docker-compose 需要一些修正:

Nginx

Nginx website 提供了一些有用的 Docker 部署技巧,您应该阅读这些技巧,包括一个示例,非常简单的 Dockerfile:

FROM nginx
RUN rm /etc/nginx/conf.d/default.conf
RUN rm /etc/nginx/conf.d/example_ssl.conf
COPY content /usr/share/nginx/html
COPY conf /etc/nginx

这表明您需要进行一些改进(有关 Docker 的进一步帮助,请参阅 Docker Compose 部分)。

考虑到我们将在下面进行的部署更新,您还需要更改 Nginx 配置:

重命名service.conf -> service.template 更改listen $NGINX_PORT; 更改server_name $NGINX_HOST; 更改proxy_pass http://$DAPHNE_HOST:$DAPHNE_PORT;

Docker 编写

现在你的 Nginx 配置是正确的,你需要正确设置 docker compose 指令,幸运的是,Docker Hub Nginx page 有一个 docker compose 示例:

这是一个使用 docker-compose.yml 的示例:

web:
  image: nginx
  volumes:
   - ./mysite.template:/etc/nginx/conf.d/mysite.template
  ports:
   - "8080:80"
  environment:
   - NGINX_HOST=foobar.com
   - NGINX_PORT=80
  command: /bin/bash -c "envsubst < /etc/nginx/conf.d/mysite.template > /etc/nginx/conf.d/default.conf && nginx -g 'daemon off;'"

mysite.template 文件可能包含如下变量引用:

listen $NGINX_PORT;

来自 r00m 的answer

您可以进行所有这些改进,事实上,如果不共享卷,您的静态文件将无法正确提供。

为项目创建图像并重复使用它 添加卷引用以允许共享静态文件 可选:您还应该遵循有关收集静态文件的建议,但您的项目结构有点表明您已经这样做了。

把所有东西放在一起

最后,我们可以合并这三个改进,为我们提供以下设置:

我的项目/Dockerfile:

FROM python
ENV PYTHONUNBUFFERED 1

RUN mkdir -p /opt/myproject
WORKDIR /opt/myproject
ADD . /opt/myproject

RUN pip install -r requirements.txt
RUN python manage.py migrate # Can this be done during build? i.e. no link to the DB?

VOLUME ["/opt/myproject/collected_static"]

myproject/docker-compose.yml:

version: '2'
services:
  nginx:
    build: ./nginx
    networks:
      - front
      - back
    ports:
      - "80:80"
    volumes_from:
      - "daphne"
    environment:
      - NGINX_HOST=example.com
      - NGINX_PORT=80
      - DAPHNE_HOST=daphne
      - DAPHEN_PORT=8000
    depends_on:
      - daphne
    links:
      - daphne
    command: /bin/bash -c "envsubst < /etc/nginx/conf.d/service.template > /etc/nginx/conf.d/default.conf && nginx -g 'daemon off;'"
  redis:
    image: redis
    networks:
      - "back"
    ports:
      - "6379:6379"
  daphne:
    build: .
    image: "myproject:latest"
    working_dir: /opt/myproject
    command: bash -c "daphne -b 0.0.0.0 -p 8000 myproject.asgi:channel_layer"
    ports:
      - "8000:8000"
    environment:
      - REDIS_HOST=redis
    networks:
      - front
      - back
     depends_on:
      - redis
     links:
      - redis
  worker:
    image: "myproject:latest"
    working_dir: /opt/myproject
    command: bash -c "python manage.py runworker"
    environment:
      - REDIS_HOST=redis
    networks:
      - front
      - back
    depends_on:
      - redis
    links:
      - redis
  networks:
    front:
    back:

我的项目/nginx/Dockerfile

FROM nginx
RUN rm /etc/nginx/conf.d/default.conf
RUN rm /etc/nginx/conf.d/example_ssl.conf
COPY service.template /etc/nginx/conf.d

myproject/nginx/service.template

server 
  listen $NGINX_PORT;
  server_name $NGINX_HOST
  charset utf-8;
  client_max_body_size 20M;

  location /static/ 
    alias /opt/myproject/collected_static/;
  

  location / 
    proxy_pass http://$DAPHNE_HOST:$DAPHNE_PORT;
    proxy_http_version 1.1;
    proxy_set_header Upgrade $http_upgrade;
    proxy_set_header Connection "upgrade";

    proxy_redirect off;
    proxy_set_header Host $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-Host $server_name;
  

最后的想法

我不确定你想用你的网络指令实现什么,但它几乎可以肯定没有实现它,例如 nginx 不应该连接到你的后端网络(我认为...)。 您需要考虑“迁移”是应该在构建时还是运行时完成。 您需要能够轻松更改您的 nginx 配置吗?如果是这样,您应该从 nginx 构建中删除 COPY 并从 Docker Compose 部分添加卷指令。

【讨论】:

【参考方案2】:

你错误地配置了 NGINX。试试proxy_pass http://127.0.0.1:8000;

至于静态文件,那是因为你还没有让容器可以使用这些文件。我建议进行以下修改:

我的项目/Dockerfile:

[...]
ADD . /opt/myproject
VOLUME ["/opt/myproject/collected_static"]
[..]
# may I also suggest automatic static file collection?
RUN python manage.py collectstatic --noinput

myproject/docker-compose.yml:

[...]
build: ./nginx
volumes_from:
  - "worker" # or daphne

我还会考虑将 image 选项添加到 daphne 和 worker 服务中。这将标记图像并允许重复使用它,因此它只会构建一次(而不是两次)。

myproject:
  build: .
  image: "myproject:latest"
[..]
worker:
  image: "myproject:latest"
[..]
daphne:
  image: "myproject:latest"

【讨论】:

究竟是什么不起作用?您收到什么错误消息? 我看不到任何错误消息。它只显示 Nginx 主页。

以上是关于使用 nginx、django、daphne 部署到 docker的主要内容,如果未能解决你的问题,请参考以下文章

使用 SSL 使用 Daphne + NGINX 部署 Django 通道

使用 Daphne 和 Nginx 部署 django 频道的问题

使用daphne部署django channles websocket 项目

Django 通道 websocket 连接和断开连接(Nginx + Daphne + Django + Channels)

django3,websocket,asgi,daphne,nginx 结合使用示例

在 nginx 后面使用 Daphne