django - Django-Ngix-Gunicorn 使用 Docker Compose 返回错误的媒体文件路径
问题描述
我在使用 docker compose 以生产模式部署的 Django REST Framework 应用程序时遇到问题。问题是当我的端点返回带有 ImageField 的模型时,它会返回错误的路径,特别是缺少 url 端口。静态文件与响应 url 中包含的端口一起工作正常。
例如:
当前值:http: //127.0.0.1/media/machines/034793bb-8516-45e3-a50a-4e00e7488617.png
预期: http: //127.0.0.1 :8000/media/machines/034793bb-8516-45e3-a50a-4e00e7488617.png
设置.py
BASE_DIR = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))
TEMPLATE_DIR = os.path.join(BASE_DIR, 'templates')
STATIC_DIR = os.path.join(BASE_DIR,'static')
MEDIA_ROOT = os.path.join(BASE_DIR, 'media')
MEDIA_URL = '/media/'
STATIC_URL = '/static/'
STATIC_ROOT = '/app/static/'
** 泊坞窗文件 **
FROM python:3.6
ENV PYTHONUNBUFFERED 1
RUN mkdir /app
WORKDIR /app
ADD . /app/
RUN pip install -r requirements.txt
CMD ["gunicorn", "-c", "config/gunicorn/conf.py", "--bind", ":8000", "--chdir",
"app_api.wsgi:application"]
码头工人组成
version: '3.7'
services:
web-service:
build: .
command: bash -c "pip install -r requirements.txt && python manage.py makemigrations && python
manage.py migrate && python manage.py collectstatic --noinput && python manage.py runserver
0.0.0.0:8000"
container_name: app_backend
volumes:
- static:/app/static
- media:/app/media
- .:/app
depends_on:
- db
restart: on-failure:2
db:
image: postgres
container_name: app_postgres
environment:
- POSTGRES_USER=bd_user
- POSTGRES_PASSWORD=password
- POSTGRES_DB=new_database
ports:
- "5432:5432"
volumes:
- postgres:/var/lib/postgresql/data
restart: on-failure:2
nginx:
image: nginx:latest
ports:
- 8000:80
volumes:
- ./config/nginx/conf.d:/etc/nginx/conf.d
- static:/app/static
- media:/app/media
depends_on:
- web-service
volumes:
media:
static:
postgres:
driver: local
Nginx 配置文件
upstream django_server {
server web-service:8000;
}
server {
listen 80;
server_name localhost;
location /static/ {
alias /app/static/;
}
location /media/ {
alias /app/media/;
}
location / {
proxy_pass http://django_server;
proxy_set_header X-Fowarded-For $proxy_add_x_forwarded_for;
proxy_set_header Host $host;
proxy_redirect off;
}
}
** Gunicorn 配置 **
name = 'docker_django'
loglevel = 'info'
errorlog = '-'
accesslog = '-'
workers = 2
模型
class Machine(models.Model):
name = models.CharField(max_length=100, blank=False, null=False)
description = models.CharField(max_length=250, blank=True, null=True)
image = models.ImageField(upload_to = 'machines', default='default.png')
provider = models.TextField(blank=True, null=True)
model = models.CharField(max_length=250, blank=True, null=True)
data_source = models.ForeignKey(DataSource, related_name='data_source', on_delete=models.CASCADE)
def __str__(self):
return self.name
解决方案
最简单的选择是$http_host
在 nginx 中使用,而不是$host
因为$host
不包含端口信息。但是$http_host
,某些客户端可能为空。
proxy_set_header Host $host;
由于 nginx 对端口一无所知,8000
它暴露在 docker 主机上,因此无法设置它。一种选择是Host
从请求中获取 http 标头(可能在请求中不存在,可能在极少数情况下)。
另一种选择是将端口 nginx 作为环境变量公开到 nginx 容器中,并在配置中使用此变量附加到$host
nginx 可以通过envsubst
在自定义入口点中运行来完成。
它适用于静态文件的原因 - 静态文件 url 包含在模板中(不是 api 响应),其链接相对于当前网站根目录(无论它是什么),它们不需要build_absolute_uri()
由 django 运行。
推荐阅读
- angular - 我如何在 PIE 图表上以角度显示值?我正在使用 ngx-echarts
- woocommerce - 向自定义角色显示特定订单
- graphql - ManyToOne 字段未出现在父解析器中
- javascript - 尝试在 React 中添加条件时映射功能的问题
- paypal - PayPal 订阅:ACTIVE 与 APPROVED 状态的含义
- php - 发送php邮件失败
- node.js - Electron - 导入模块作为外部文件
- ruby-on-rails - Axios POST 请求通过 react-rails 作为 GET 请求发送
- python - 拆分数据帧以训练和测试 SKlearn
- npm - 相同的 webpack 版本和不同的内部依赖?