Fix production deployment: use pre-built images instead of local builds + handle git conflicts

This commit is contained in:
Stiftung Development
2025-09-15 23:21:05 +02:00
parent f4f7d26d36
commit 5ff7b7a3ed
2 changed files with 94 additions and 143 deletions

View File

@@ -195,9 +195,15 @@ jobs:
script: | script: |
cd /opt/stiftung cd /opt/stiftung
# Stash any local changes to avoid conflicts
git stash push -m "Auto-stash before deployment $(date)"
# Pull latest changes using Personal Access Token # Pull latest changes using Personal Access Token
git pull https://$DEPLOY_TOKEN@github.com/remmerinio/stiftung-management-system.git main git pull https://$DEPLOY_TOKEN@github.com/remmerinio/stiftung-management-system.git main
# Backup current compose.yml
cp compose.yml compose.yml.backup
# Copy production docker compose file to the active compose.yml # Copy production docker compose file to the active compose.yml
cp deploy-production/docker-compose.prod.yml compose.yml cp deploy-production/docker-compose.prod.yml compose.yml

View File

@@ -1,194 +1,139 @@
# Production Docker Compose Configuration
# This file is used for production deployment via GitHub Actions
# For local development, use: docker-compose -f compose.dev.yml up
#
# IMPORTANT: This configuration requires ALL environment variables to be
# provided via the production server's .env file. No fallback values are
# included for security reasons.
services: services:
db: db:
image: postgres:15-alpine image: postgres:16-alpine
restart: unless-stopped restart: unless-stopped
environme beat: environment:
build:
context: ../app
dockerfile: Dockerfile
POSTGRES_DB: ${POSTGRES_DB} POSTGRES_DB: ${POSTGRES_DB}
POSTGRES_USER: ${POSTGRES_USER} POSTGRES_USER: ${POSTGRES_USER}
POSTGRES_PASSWORD: ${POSTGRES_PASSWORD} POSTGRES_PASSWORD: ${POSTGRES_PASSWORD}
volumes: volumes:
- postgres_data:/var/lib/postgresql/data - dbdata:/var/lib/postgresql/data
- ../app/backups:/backups
deploy:
resources:
limits:
memory: 1G
cpus: '1.0'
healthcheck: healthcheck:
test: ["CMD-SHELL", "pg_isready -U ${POSTGRES_USER} -d ${POSTGRES_DB}"] test: ["CMD-SHELL", "pg_isready -U $${POSTGRES_USER} -d $${POSTGRES_DB}"]
interval: 30s interval: 10s
timeout: 10s timeout: 5s
retries: 5 retries: 5
redis: redis:
image: redis:7-alpine image: redis:7-alpine
restart: unless-stopped restart: unless-stopped
command: redis-server --maxmemory 256mb --maxmemory-policy allkeys-lru
volumes:
- redis_data:/data
deploy:
resources:
limits:
memory: 512M
cpus: '0.5'
web: web:
image: ghcr.io/remmerinio/stiftung-management-system:main image: ghcr.io/remmerinio/stiftung-management-system:latest
restart: unless-stopped restart: unless-stopped
environment:
- DEBUG=False
- ALLOWED_HOSTS=${ALLOWED_HOSTS}
- SECRET_KEY=${SECRET_KEY}
- DATABASE_URL=postgresql://${POSTGRES_USER}:${POSTGRES_PASSWORD}@db:5432/${POSTGRES_DB}
- REDIS_URL=redis://redis:6379/0
- CELERY_BROKER_URL=redis://redis:6379/0
- CELERY_RESULT_BACKEND=redis://redis:6379/0
volumes:
- ../app/media:/app/media
- ../app/static:/app/static
- ../app/backups:/app/backups
depends_on: depends_on:
db: db:
condition: service_healthy condition: service_healthy
redis: redis:
condition: service_started condition: service_started
deploy: environment:
resources: - POSTGRES_DB=${POSTGRES_DB}
limits: - POSTGRES_USER=${POSTGRES_USER}
memory: 1G - POSTGRES_PASSWORD=${POSTGRES_PASSWORD}
cpus: '1.0' - DB_HOST=${DB_HOST}
- DB_PORT=${DB_PORT}
- DJANGO_SECRET_KEY=${DJANGO_SECRET_KEY}
- DJANGO_DEBUG=${DJANGO_DEBUG}
- DJANGO_ALLOWED_HOSTS=${DJANGO_ALLOWED_HOSTS}
- LANGUAGE_CODE=${LANGUAGE_CODE}
- TIME_ZONE=${TIME_ZONE}
- REDIS_URL=${REDIS_URL}
- PAPERLESS_API_URL=${PAPERLESS_API_URL}
- PAPERLESS_API_TOKEN=${PAPERLESS_API_TOKEN}
ports: ports:
- "127.0.0.1:8000:8000" - "8081:8000"
volumes:
- media_files:/app/media
command: ["gunicorn", "core.wsgi:application", "--bind", "0.0.0.0:8000", "--workers", "3"]
worker: worker:
image: ghcr.io/remmerinio/stiftung-management-system:main image: ghcr.io/remmerinio/stiftung-management-system:latest
restart: unless-stopped restart: unless-stopped
command: celery -A core worker -l info --concurrency=2
environment: environment:
- DEBUG=False - POSTGRES_DB=${POSTGRES_DB}
- SECRET_KEY=${SECRET_KEY} - POSTGRES_USER=${POSTGRES_USER}
- DATABASE_URL=postgresql://${POSTGRES_USER}:${POSTGRES_PASSWORD}@db:5432/${POSTGRES_DB} - POSTGRES_PASSWORD=${POSTGRES_PASSWORD}
- REDIS_URL=redis://redis:6379/0 - DB_HOST=${DB_HOST}
- CELERY_BROKER_URL=redis://redis:6379/0 - DB_PORT=${DB_PORT}
- CELERY_RESULT_BACKEND=redis://redis:6379/0 - DJANGO_SECRET_KEY=${DJANGO_SECRET_KEY}
volumes: - DJANGO_DEBUG=${DJANGO_DEBUG}
- ./app/media:/app/media - REDIS_URL=${REDIS_URL}
- ./backups:/app/backups
depends_on: depends_on:
db: - redis
condition: service_healthy - db
redis: command: ["celery", "-A", "core", "worker", "-l", "info"]
condition: service_started
deploy:
resources:
limits:
memory: 512M
cpus: '0.5'
beat: beat:
image: ghcr.io/remmerinio/stiftung-management-system:main image: ghcr.io/remmerinio/stiftung-management-system:latest
restart: unless-stopped restart: unless-stopped
command: celery -A core beat -l info --scheduler django_celery_beat.schedulers:DatabaseScheduler
environment: environment:
- DEBUG=False - POSTGRES_DB=${POSTGRES_DB}
- SECRET_KEY=${SECRET_KEY} - POSTGRES_USER=${POSTGRES_USER}
- DATABASE_URL=postgresql://${POSTGRES_USER}:${POSTGRES_PASSWORD}@db:5432/${POSTGRES_DB} - POSTGRES_PASSWORD=${POSTGRES_PASSWORD}
- REDIS_URL=redis://redis:6379/0 - DB_HOST=${DB_HOST}
- CELERY_BROKER_URL=redis://redis:6379/0 - DB_PORT=${DB_PORT}
- CELERY_RESULT_BACKEND=redis://redis:6379/0 - DJANGO_SECRET_KEY=${DJANGO_SECRET_KEY}
volumes: - DJANGO_DEBUG=${DJANGO_DEBUG}
- ./app/media:/app/media - REDIS_URL=${REDIS_URL}
- ./backups:/app/backups
depends_on: depends_on:
db: - redis
condition: service_healthy - db
redis: command: ["celery", "-A", "core", "beat", "-l", "info"]
condition: service_started
deploy:
resources:
limits:
memory: 256M
cpus: '0.25'
grampsweb: grampsweb:
image: ghcr.io/gramps-project/grampsweb:latest image: ghcr.io/gramps-project/grampsweb:latest
restart: unless-stopped
environment:
GRAMPSWEB_TREE: ${GRAMPSWEB_TREE:-Stiftung}
GRAMPSWEB_SECRET_KEY: ${GRAMPSWEB_SECRET_KEY}
GRAMPSWEB_ADMIN_EMAIL: ${GRAMPSWEB_ADMIN_EMAIL}
GRAMPSWEB_ADMIN_PASSWORD: ${GRAMPSWEB_ADMIN_PASSWORD}
volumes:
- gramps_users:/app/users
- gramps_index:/app/indexdir
- gramps_thumb_cache:/app/thumbnail_cache
- gramps_cache:/app/cache
- gramps_secret:/app/secret
- ./media:/app/media
ports: ports:
- "127.0.0.1:5000:5000" - "8090:80"
deploy: environment:
resources: - GRAMPSWEB_SECRET_KEY=${GRAMPSWEB_SECRET_KEY}
limits: - GRAMPSWEB_ADMIN_EMAIL=${GRAMPSWEB_ADMIN_EMAIL}
memory: 512M - GRAMPSWEB_ADMIN_PASSWORD=${GRAMPSWEB_ADMIN_PASSWORD}
cpus: '0.5' volumes:
- gramps_data:/app/data
paperless: paperless:
image: ghcr.io/paperless-ngx/paperless-ngx:latest image: ghcr.io/paperless-ngx/paperless-ngx:latest
restart: unless-stopped
depends_on:
- db
- redis
ports: ports:
- "127.0.0.1:8080:8000" - "8080:8000"
healthcheck: environment:
test: ["CMD", "curl", "-fs", "-S", "--max-time", "2", "http://localhost:8000"] - PAPERLESS_REDIS=redis://redis:6379
interval: 30s - PAPERLESS_DBHOST=db
timeout: 10s - PAPERLESS_DBPORT=5432
retries: 5 - PAPERLESS_DBNAME=${POSTGRES_DB}
- PAPERLESS_DBUSER=${POSTGRES_USER}
- PAPERLESS_DBPASS=${POSTGRES_PASSWORD}
- PAPERLESS_SECRET_KEY=${PAPERLESS_SECRET_KEY}
- PAPERLESS_URL=https://vhtv-stiftung.de
- PAPERLESS_ALLOWED_HOSTS=vhtv-stiftung.de,localhost
- PAPERLESS_CORS_ALLOWED_HOSTS=https://vhtv-stiftung.de
- PAPERLESS_FORCE_SCRIPT_NAME=/paperless
- PAPERLESS_STATIC_URL=/paperless/static/
- PAPERLESS_LOGIN_REDIRECT_URL=/paperless/
- PAPERLESS_LOGOUT_REDIRECT_URL=/paperless/
- PAPERLESS_ADMIN_USER=${PAPERLESS_ADMIN_USER}
- PAPERLESS_ADMIN_PASSWORD=${PAPERLESS_ADMIN_PASSWORD}
- PAPERLESS_ADMIN_MAIL=${PAPERLESS_ADMIN_MAIL}
volumes: volumes:
- paperless_data:/usr/src/paperless/data - paperless_data:/usr/src/paperless/data
- paperless_media:/usr/src/paperless/media - paperless_media:/usr/src/paperless/media
- paperless_export:/usr/src/paperless/export - paperless_export:/usr/src/paperless/export
- paperless_consume:/usr/src/paperless/consume - paperless_consume:/usr/src/paperless/consume
environment: depends_on:
PAPERLESS_REDIS: redis://redis:6379 - db
PAPERLESS_DBHOST: db - redis
PAPERLESS_DBNAME: ${PAPERLESS_DB:-paperless}
PAPERLESS_DBUSER: ${PAPERLESS_USER:-paperless}
PAPERLESS_DBPASS: ${PAPERLESS_PASSWORD:-paperless}
PAPERLESS_ADMIN_USER: ${PAPERLESS_ADMIN_USER:-admin}
PAPERLESS_ADMIN_PASSWORD: ${PAPERLESS_ADMIN_PASSWORD:-admin}
PAPERLESS_ADMIN_MAIL: ${PAPERLESS_ADMIN_MAIL:-admin@localhost}
PAPERLESS_SECRET_KEY: ${PAPERLESS_SECRET_KEY}
PAPERLESS_URL: https://vhtv-stiftung.de/paperless
PAPERLESS_ALLOWED_HOSTS: vhtv-stiftung.de,www.vhtv-stiftung.de
PAPERLESS_CORS_ALLOWED_HOSTS: https://vhtv-stiftung.de,https://www.vhtv-stiftung.de
PAPERLESS_TRUSTED_PROXIES: 172.16.0.0/12,10.0.0.0/8,192.168.0.0/16
PAPERLESS_FORCE_SCRIPT_NAME: /paperless
PAPERLESS_STATIC_URL: /paperless/static/
deploy:
resources:
limits:
memory: 2G
cpus: '1.0'
volumes: volumes:
postgres_data: dbdata:
redis_data: gramps_data:
gramps_users:
gramps_index:
gramps_thumb_cache:
gramps_cache:
gramps_secret:
paperless_data: paperless_data:
paperless_media: paperless_media:
paperless_export: paperless_export:
paperless_consume: paperless_consume:
networks:
default:
driver: bridge