我无法通过docker容器通过本地主机上NGINX上的Unix套接字连接在ASGI + Gunicorn上运行的应用程序。
提供我在docker容器中并运行NGINX的信息:
/usr/sbin/nginx
我可以打开http://localhost/api/v1/items并从NGINX获取404,这意味着它至少正在运行。
执行正在运行的docker服务,我可以使用以下命令启动Gunicorn:
gunicorn app.main:app --name asgi --workers 3 --user=root --group=root --bind=unix:///tmp/asgi.sock --log-level=debug --log-file=- -k uvicorn.workers.UvicornWorker -c /gunicorn_conf.py
Gunicorn正确启动,并且可以与另一个执行程序一起使用,这样可以卷曲已绑定到的UNIX套接字并收到200响应。
curl --unix-socket ///tmp/asgi.sock http://localhost/api/v1/items
我认为这意味着我在NGINX将流量定向到http://localhost/api/v1/items的配置中存在一些差异。
nginx.conf
daemon off;
user nginx;
worker_processes 1;
pid /var/run/nginx.pid;
events {
worker_connections 1024;
}
http {
access_log /dev/stdout;
upstream asgi {
server unix:/tmp/asgi.sock fail_timeout=0;
}
server {
listen 80;
server_name localhost;
client_max_body_size 4G;
location / {
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
proxy_set_header Host $http_host;
proxy_redirect off;
proxy_pass http://asgi;
}
}
}
gunicorn_conf.py
import json
import multiprocessing
import os
workers_per_core_str = os.getenv("WORKERS_PER_CORE", "1")
web_concurrency_str = os.getenv("WEB_CONCURRENCY", None)
host = os.getenv("HOST", "unix")
port = os.getenv("PORT", "///tmp/asgi.sock")
bind_env = os.getenv("BIND", None)
use_loglevel = os.getenv("LOG_LEVEL", "info")
if bind_env:
use_bind = bind_env
else:
use_bind = f"{host}:{port}"
cores = multiprocessing.cpu_count()
workers_per_core = float(workers_per_core_str)
default_web_concurrency = workers_per_core * cores
if web_concurrency_str:
web_concurrency = int(web_concurrency_str)
assert web_concurrency > 0
else:
web_concurrency = max(int(default_web_concurrency), 2)
# Gunicorn config variables
loglevel = use_loglevel
workers = web_concurrency
bind = use_bind
keepalive = 120
errorlog = "-"
# For debugging and testing
log_data = {
"loglevel": loglevel,
"workers": workers,
"bind": bind,
# Additional, non-gunicorn variables
"workers_per_core": workers_per_core,
"host": host,
"port": port,
}
print(json.dumps(log_data))
更新:
但是,我能够解决我的问题,在此留有注释。在我的原始帖子中,我的NGINX配置没有在http块中包含服务器,并且我更新了日志以写入控制台,尽管假设文件位置存在,这没有什么区别。
我也正在通过Supervisord启动NGINX和Gunicorn,这最初没有说明,因为我认为这超出了问题的范围。但是,我现在可以使用Supervisord来启动这两个过程,但是我必须daemon off;
进入NGINX配置才能使它起作用。没有它,该过程将表明该端口已在使用中。
我使用最新版本的配置(包括supervisor.ini)更新了帖子
supervisor.ini
[supervisord]
nodaemon=true
[program:asgi]
command=gunicorn app.main:app --name asgi --workers 3 --user=root --group=root --bind=unix:/tmp/asgi.sock --log-level=debug --log-file=- -k uvicorn.workers.UvicornWorker -c /gunicorn_conf.py
user = root ; User to run as
autostart=true
autorestart=true
stdout_logfile=/dev/stdout ; Where to write log messages
stdout_logfile_maxbytes=0
stderr_logfile=/dev/stderr
stderr_logfile_maxbytes=0
environment=LANG=en_US.UTF-8,LC_ALL=en_US.UTF-8 ; Set UTF-8 as default encoding
[program:nginx]
command=/usr/sbin/nginx -c /etc/nginx/conf.d/nginx.conf
stdout_logfile=/dev/stdout
stdout_logfile_maxbytes=0
stderr_logfile=/dev/stderr
stderr_logfile_maxbytes=0
# Graceful stop, see http://nginx.org/en/docs/control.html
stopsignal=QUIT
要运行监督者:/usr/bin/supervisord -c /etc/supervisor.d/supervisord.ini
要亲自尝试,请参阅我为此application做的github存储库。
答案 0 :(得分:0)
解决了我的问题,并使用更改的文件更新了原始问题。主要问题是使用守护程序关闭后台程序以允许NGINX和Gunicorn都运行,并将NGINX的服务器配置放置在html块中。