From aa8721a612bff6cf2f8094cb7d763d6bb7729b8d Mon Sep 17 00:00:00 2001 From: Nick Satterly Date: Sun, 7 Apr 2024 20:14:09 +0200 Subject: [PATCH] fix: switch to gunicorn WSGI server --- Dockerfile | 10 +++---- README.md | 4 +-- config/templates/app/gunicorn.conf.py.j2 | 14 ++++++++++ config/templates/app/nginx.conf.j2 | 11 ++++---- config/templates/app/supervisord.conf.j2 | 6 +++-- config/templates/app/uwsgi.ini.j2 | 34 ------------------------ contrib/kubernetes/backend/Dockerfile | 2 +- docker-entrypoint.sh | 13 ++++----- requirements-docker.txt | 2 +- 9 files changed, 37 insertions(+), 59 deletions(-) create mode 100644 config/templates/app/gunicorn.conf.py.j2 delete mode 100644 config/templates/app/uwsgi.ini.j2 diff --git a/Dockerfile b/Dockerfile index 05619ebf..e6ca2d5e 100644 --- a/Dockerfile +++ b/Dockerfile @@ -1,4 +1,4 @@ -FROM python:3.9-slim-buster +FROM python:3.10-slim-buster ENV PYTHONUNBUFFERED 1 ENV PIP_DISABLE_PIP_VERSION_CHECK=1 @@ -15,11 +15,7 @@ ENV WEBUI_VERSION=8.7.0 ENV NGINX_WORKER_PROCESSES=1 ENV NGINX_WORKER_CONNECTIONS=1024 -ENV UWSGI_PROCESSES=5 -ENV UWSGI_LISTEN=100 -ENV UWSGI_BUFFER_SIZE=8192 -ENV UWSGI_MAX_WORKER_LIFETIME=30 -ENV UWSGI_WORKER_LIFETIME_DELTA=3 +ENV GUNICORN_WORKERS=5 ENV HEARTBEAT_SEVERITY=major ENV HK_EXPIRED_DELETE_HRS=2 @@ -77,7 +73,7 @@ COPY requirements*.txt /app/ # hadolint ignore=DL3013 RUN pip install --no-cache-dir pip virtualenv jinja2 && \ python3 -m venv /venv && \ - /venv/bin/pip install --no-cache-dir --upgrade setuptools && \ + /venv/bin/pip install --no-cache-dir --upgrade setuptools wheel && \ /venv/bin/pip install --no-cache-dir --requirement /app/requirements.txt && \ /venv/bin/pip install --no-cache-dir --requirement /app/requirements-docker.txt ENV PATH $PATH:/venv/bin diff --git a/README.md b/README.md index f6ceb7fa..232be71e 100644 --- a/README.md +++ b/README.md @@ -126,8 +126,8 @@ API to ease deployment more generally: `NGINX_WORKER_CONNECTIONS` - maximum number of simultaneous connections that can be opened by a worker process (default:`1024`) -`UWSGI_PROCESSES` - - number of processes for uWSGI (default:`5`) +`GUNICORN_WORKERS` + - number of worker processes for Gunicorn (default:`5`) `UWSGI_LISTEN` - max number of concurrent connections (default:`100`) diff --git a/config/templates/app/gunicorn.conf.py.j2 b/config/templates/app/gunicorn.conf.py.j2 new file mode 100644 index 00000000..8d35639c --- /dev/null +++ b/config/templates/app/gunicorn.conf.py.j2 @@ -0,0 +1,14 @@ + +bind = '127.0.0.1:29000' + +chdir = '/app' +daemon = False +raw_env = [ + 'SCRIPT_NAME=/api', +] +worker_tmp_dir = '/dev/shm' +workers = {{ env.GUNICORN_WORKERS }} + +{%- if env.DEBUG %} +loglevel = 'debug' +{%- endif %} diff --git a/config/templates/app/nginx.conf.j2 b/config/templates/app/nginx.conf.j2 index 76772ab7..4327eb2f 100644 --- a/config/templates/app/nginx.conf.j2 +++ b/config/templates/app/nginx.conf.j2 @@ -47,13 +47,12 @@ http { access_log /dev/stdout main; location /api { - include /etc/nginx/uwsgi_params; - uwsgi_pass backend; + proxy_pass http://backend; - uwsgi_param Host $host; - uwsgi_param X-Real-IP $remote_addr; - uwsgi_param X-Forwarded-For $proxy_add_x_forwarded_for; - uwsgi_param X-Forwarded-Proto $http_x_forwarded_proto; + 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-Proto $http_x_forwarded_proto; } root /web; diff --git a/config/templates/app/supervisord.conf.j2 b/config/templates/app/supervisord.conf.j2 index b5d4919e..e01f8dea 100644 --- a/config/templates/app/supervisord.conf.j2 +++ b/config/templates/app/supervisord.conf.j2 @@ -4,8 +4,10 @@ logfile=/tmp/supervisord.log loglevel={{ env.SUPERVISORD_LOG_LEVEL|lower or 'debug' }} pidfile=/tmp/supervisord.pid -[program:uwsgi] -command=/venv/bin/uwsgi --ini /app/uwsgi.ini +[program:gunicorn] +command=/venv/bin/gunicorn wsgi:app -c /app/gunicorn.conf.py +autostart=true +autorestart=true redirect_stderr=true [program:nginx] diff --git a/config/templates/app/uwsgi.ini.j2 b/config/templates/app/uwsgi.ini.j2 deleted file mode 100644 index f40eb73e..00000000 --- a/config/templates/app/uwsgi.ini.j2 +++ /dev/null @@ -1,34 +0,0 @@ -[uwsgi] -chdir = /app -module = wsgi -manage-script-name = true -mount = /api=wsgi:app -master = true -processes = {{ env.UWSGI_PROCESSES }} -listen = {{ env.UWSGI_LISTEN }} -{%- if env.UWSGI_MAX_WORKER_LIFETIME %} -max-worker-lifetime = {{ env.UWSGI_MAX_WORKER_LIFETIME }} -max-worker-lifetime-delta = {{ env.UWSGI_WORKER_LIFETIME_DELTA }} -{%- endif %} - -{%- if env.UWSGI_THREADS %} -threads = {{ env.UWSGI_THREADS }} -enable-threads = True -{%- endif %} - -socket = 127.0.0.1:29000 -buffer-size = {{ env.UWSGI_BUFFER_SIZE }} -chmod-socket = 664 -uid = alerta -gid = root -vacuum = true - -die-on-term = true - -{%- if env.DEBUG %} -show-config -stats = :1717 -stats-http -{%- else %} -disable-logging = True -{%- endif %} diff --git a/contrib/kubernetes/backend/Dockerfile b/contrib/kubernetes/backend/Dockerfile index 7298f808..e11baa56 100644 --- a/contrib/kubernetes/backend/Dockerfile +++ b/contrib/kubernetes/backend/Dockerfile @@ -10,7 +10,7 @@ RUN apt-get update && apt-get install -y \ RUN pip install --no-cache-dir virtualenv && \ virtualenv --python=python3 /venv && \ - /venv/bin/pip install uwsgi alerta alerta-server==$VERSION + /venv/bin/pip install gunicorn alerta alerta-server==$VERSION ENV PATH $PATH:/venv/bin diff --git a/docker-entrypoint.sh b/docker-entrypoint.sh index 1f3cd8a6..bb30e812 100755 --- a/docker-entrypoint.sh +++ b/docker-entrypoint.sh @@ -7,7 +7,7 @@ ALERTA_CONF_FILE=${ALERTA_CONF_FILE:-/app/alerta.conf} ALERTA_SVR_CONF_FILE=${ALERTA_SVR_CONF_FILE:-/app/alertad.conf} ALERTA_WEB_CONF_FILE=${ALERTA_WEB_CONF_FILE:-/web/config.json} NGINX_CONF_FILE=/app/nginx.conf -UWSGI_CONF_FILE=/app/uwsgi.ini +GUNICORN_CONF_FILE=/app/gunicorn.conf.py SUPERVISORD_CONF_FILE=/app/supervisord.conf ADMIN_USER=${ADMIN_USERS%%,*} @@ -67,12 +67,13 @@ if [ ! -f "${NGINX_CONF_FILE}" ]; then echo "# Create nginx configuration file." python3 -c "${JINJA2}" < ${NGINX_CONF_FILE}.j2 >${NGINX_CONF_FILE} fi +cat ${NGINX_CONF_FILE} nginx -t -c ${NGINX_CONF_FILE} -# Generate uWSGI config, if not supplied. -if [ ! -f "${UWSGI_CONF_FILE}" ]; then - echo "# Create uWSGI configuration file." - python3 -c "${JINJA2}" < ${UWSGI_CONF_FILE}.j2 >${UWSGI_CONF_FILE} +# Generate Gunicorn config, if not supplied. +if [ ! -f "${GUNICORN_CONF_FILE}" ]; then + echo "# Create Gunicorn configuration file." + python3 -c "${JINJA2}" < ${GUNICORN_CONF_FILE}.j2 >${GUNICORN_CONF_FILE} fi # Generate web config, if not supplied. @@ -88,7 +89,7 @@ echo Alerta Client ${CLIENT_VERSION} echo Alerta WebUI ${WEBUI_VERSION} nginx -v -echo uwsgi $(uwsgi --version) +gunicorn --version mongo --version | grep MongoDB psql --version python3 --version diff --git a/requirements-docker.txt b/requirements-docker.txt index 4ebc1e78..8436af7e 100644 --- a/requirements-docker.txt +++ b/requirements-docker.txt @@ -1,4 +1,4 @@ lxml==4.9.2 pysaml2==7.2.1 python-ldap==3.4.3 -uWSGI==2.0.21 +gunicorn==21.2.0