Files
breakpilot-core/docker-compose.yml
Benjamin Boenisch eb43b40dd0
All checks were successful
CI / go-lint (push) Has been skipped
CI / python-lint (push) Has been skipped
CI / nodejs-lint (push) Has been skipped
CI / test-go-consent (push) Successful in 29s
CI / test-python-voice (push) Successful in 31s
CI / test-bqas (push) Successful in 29s
feat: voice-service hinzugefuegt, nginx upstreams aktualisiert
- voice-service in docker-compose.yml hinzugefuegt (bp-core-voice-service)
- nginx: voice-service upstream von bp-lehrer auf bp-core geaendert
- nginx: edu-search upstream von breakpilot-edu-search auf bp-lehrer-edu-search geaendert
- extra_hosts fuer edu-search entfernt (jetzt containerisiert in lehrer)
- health-aggregator: voice-service zu CHECK_SERVICES hinzugefuegt

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-02-15 18:24:32 +01:00

1189 lines
37 KiB
YAML

# =========================================================
# BreakPilot Core — Shared Infrastructure
# =========================================================
# Start: docker compose up -d
# Health: http://macmini:8099/health
# =========================================================
networks:
breakpilot-network:
driver: bridge
name: breakpilot-network
volumes:
# Infrastructure
vault_data:
vault_agent_config:
vault_certs:
breakpilot_db_data:
valkey_data:
qdrant_data:
minio_data:
# Communication
synapse_data:
synapse_db_data:
jitsi_web_config:
jitsi_web_crontabs:
jitsi_transcripts:
jitsi_prosody_config:
jitsi_prosody_plugins:
jitsi_jicofo_config:
jitsi_jvb_config:
jibri_recordings:
# CI/CD
gitea_data:
gitea_config:
gitea_runner_data:
woodpecker_data:
# ERP
erpnext_db_data:
erpnext_redis_queue_data:
erpnext_redis_cache_data:
erpnext_sites:
erpnext_logs:
# Services
voice_session_data:
embedding_models:
services:
# =========================================================
# REVERSE PROXY
# =========================================================
nginx:
image: nginx:alpine
container_name: bp-core-nginx
ports:
- "443:443"
- "80:80"
- "3000:3000" # Website (Lehrer)
- "3002:3002" # Admin Lehrer
- "3006:3006" # Developer Portal (Compliance)
- "3007:3007" # Admin Compliance (NEU)
- "8000:8000" # Backend Core
- "8001:8001" # Backend Lehrer (NEU)
- "8002:8002" # Backend Compliance (NEU)
- "8086:8086" # Klausur Service
- "8087:8087" # Embedding Service
- "8089:8089" # Edu-Search
- "8091:8091" # Voice Service (WSS)
- "8093:8093" # AI Compliance SDK
- "8097:8097" # RAG Service (NEU)
- "8443:8443" # Jitsi Meet
- "3008:3008" # Admin Core
- "3010:3010" # Portal Dashboard
volumes:
- ./nginx/conf.d:/etc/nginx/conf.d:ro
- vault_certs:/etc/nginx/certs:ro
- ./nginx/html:/usr/share/nginx/html/portal:ro
depends_on:
vault-agent:
condition: service_started
restart: unless-stopped
networks:
- breakpilot-network
# =========================================================
# SECRETS MANAGEMENT
# =========================================================
vault:
image: hashicorp/vault:1.15
entrypoint: ["vault"]
command: server -config=/vault/config/config.hcl
container_name: bp-core-vault
ports:
- "8200:8200"
volumes:
- vault_data:/vault/data
- ./vault/config.hcl:/vault/config/config.hcl:ro
cap_add:
- IPC_LOCK
environment:
VAULT_ADDR: "http://127.0.0.1:8200"
healthcheck:
test: ["CMD-SHELL", "vault status; test $? -le 2"]
interval: 10s
timeout: 5s
retries: 3
restart: unless-stopped
networks:
- breakpilot-network
vault-init:
image: hashicorp/vault:1.15
container_name: bp-core-vault-init
volumes:
- ./vault/init-vault.sh:/vault/scripts/init-vault.sh:ro
- ./vault/init-pki.sh:/vault/scripts/init-pki.sh:ro
- ./vault/init-secrets.sh:/vault/scripts/init-secrets.sh:ro
- vault_data:/vault/data
- vault_agent_config:/vault/agent/data
- vault_certs:/vault/certs
environment:
VAULT_ADDR: "http://vault:8200"
entrypoint: /bin/sh
command: /vault/scripts/init-vault.sh
depends_on:
vault:
condition: service_healthy
restart: "no"
networks:
- breakpilot-network
vault-agent:
image: hashicorp/vault:1.15
container_name: bp-core-vault-agent
volumes:
- ./vault/agent/config.hcl:/vault/agent/config.hcl:ro
- ./vault/agent/templates:/vault/agent/templates:ro
- ./vault/agent/split-certs.sh:/vault/agent/split-certs.sh:ro
- vault_agent_config:/vault/agent/data
- vault_certs:/vault/certs
environment:
VAULT_ADDR: "http://vault:8200"
entrypoint: /bin/sh
command: -c "vault agent -config=/vault/agent/config.hcl"
depends_on:
vault:
condition: service_healthy
vault-init:
condition: service_completed_successfully
restart: unless-stopped
networks:
- breakpilot-network
# =========================================================
# DATABASES
# =========================================================
postgres:
image: postgis/postgis:16-3.4-alpine
container_name: bp-core-postgres
ports:
- "5432:5432"
volumes:
- breakpilot_db_data:/var/lib/postgresql/data
- ./scripts/init-schemas.sql:/docker-entrypoint-initdb.d/20-init-schemas.sql:ro
environment:
POSTGRES_USER: ${POSTGRES_USER:-breakpilot}
POSTGRES_PASSWORD: ${POSTGRES_PASSWORD:-breakpilot123}
POSTGRES_DB: ${POSTGRES_DB:-breakpilot_db}
healthcheck:
test: ["CMD-SHELL", "pg_isready -U breakpilot -d breakpilot_db"]
interval: 5s
timeout: 5s
retries: 5
restart: unless-stopped
networks:
- breakpilot-network
valkey:
image: valkey/valkey:8-alpine
container_name: bp-core-valkey
ports:
- "6379:6379"
volumes:
- valkey_data:/data
command: valkey-server --appendonly yes --maxmemory 256mb --maxmemory-policy allkeys-lru
healthcheck:
test: ["CMD", "valkey-cli", "ping"]
interval: 5s
timeout: 3s
retries: 5
restart: unless-stopped
networks:
- breakpilot-network
synapse-db:
image: postgres:16-alpine
container_name: bp-core-synapse-db
profiles: [chat]
environment:
POSTGRES_USER: synapse
POSTGRES_PASSWORD: ${SYNAPSE_DB_PASSWORD:-synapse_secret}
POSTGRES_DB: synapse
POSTGRES_INITDB_ARGS: "--encoding=UTF-8 --lc-collate=C --lc-ctype=C"
volumes:
- synapse_db_data:/var/lib/postgresql/data
healthcheck:
test: ["CMD-SHELL", "pg_isready -U synapse"]
interval: 5s
timeout: 5s
retries: 5
restart: unless-stopped
networks:
- breakpilot-network
# =========================================================
# VECTOR DB & OBJECT STORAGE
# =========================================================
qdrant:
image: qdrant/qdrant:v1.12.1
container_name: bp-core-qdrant
ports:
- "6333:6333"
- "6334:6334"
volumes:
- qdrant_data:/qdrant/storage
environment:
QDRANT__SERVICE__GRPC_PORT: 6334
healthcheck:
test: ["CMD-SHELL", "bash -c 'echo > /dev/tcp/127.0.0.1/6333'"]
interval: 10s
timeout: 5s
retries: 3
restart: unless-stopped
networks:
- breakpilot-network
minio:
image: minio/minio:latest
container_name: bp-core-minio
ports:
- "9000:9000"
- "9001:9001"
volumes:
- minio_data:/data
environment:
MINIO_ROOT_USER: ${MINIO_ROOT_USER:-breakpilot}
MINIO_ROOT_PASSWORD: ${MINIO_ROOT_PASSWORD:-breakpilot123}
command: server /data --console-address ":9001"
healthcheck:
test: ["CMD", "mc", "ready", "local"]
interval: 10s
timeout: 5s
retries: 3
restart: unless-stopped
networks:
- breakpilot-network
# =========================================================
# SHARED SERVICES
# =========================================================
backend-core:
build:
context: ./backend-core
dockerfile: Dockerfile
container_name: bp-core-backend
platform: linux/arm64
expose:
- "8000"
volumes:
- /var/run/docker.sock:/var/run/docker.sock:ro
environment:
DATABASE_URL: postgresql+asyncpg://${POSTGRES_USER:-breakpilot}:${POSTGRES_PASSWORD:-breakpilot123}@postgres:5432/${POSTGRES_DB:-breakpilot_db}?options=-csearch_path%3Dcore,public
JWT_SECRET: ${JWT_SECRET:-your-super-secret-jwt-key-change-in-production}
ENVIRONMENT: ${ENVIRONMENT:-development}
VALKEY_URL: redis://valkey:6379/0
SESSION_TTL_HOURS: ${SESSION_TTL_HOURS:-24}
CONSENT_SERVICE_URL: http://consent-service:8081
VAULT_ADDR: http://vault:8200
VAULT_TOKEN: ${VAULT_TOKEN:-breakpilot-dev-token}
USE_VAULT_SECRETS: ${USE_VAULT_SECRETS:-false}
SMTP_HOST: ${SMTP_HOST:-mailpit}
SMTP_PORT: ${SMTP_PORT:-1025}
SMTP_USERNAME: ${SMTP_USERNAME:-}
SMTP_PASSWORD: ${SMTP_PASSWORD:-}
SMTP_FROM_NAME: ${SMTP_FROM_NAME:-BreakPilot}
SMTP_FROM_ADDR: ${SMTP_FROM_ADDR:-noreply@breakpilot.app}
extra_hosts:
- "host.docker.internal:host-gateway"
depends_on:
postgres:
condition: service_healthy
valkey:
condition: service_healthy
consent-service:
condition: service_started
mailpit:
condition: service_started
restart: unless-stopped
networks:
- breakpilot-network
consent-service:
build:
context: ./consent-service
dockerfile: Dockerfile
container_name: bp-core-consent-service
platform: linux/arm64
ports:
- "8081:8081"
environment:
DATABASE_URL: postgres://${POSTGRES_USER:-breakpilot}:${POSTGRES_PASSWORD:-breakpilot123}@postgres:5432/${POSTGRES_DB:-breakpilot_db}
JWT_SECRET: ${JWT_SECRET:-your-super-secret-jwt-key-change-in-production}
JWT_REFRESH_SECRET: ${JWT_REFRESH_SECRET:-your-refresh-secret}
PORT: 8081
ENVIRONMENT: ${ENVIRONMENT:-development}
ALLOWED_ORIGINS: "*"
VALKEY_URL: redis://valkey:6379/0
SESSION_TTL_HOURS: ${SESSION_TTL_HOURS:-24}
SMTP_HOST: ${SMTP_HOST:-mailpit}
SMTP_PORT: ${SMTP_PORT:-1025}
SMTP_USERNAME: ${SMTP_USERNAME:-}
SMTP_PASSWORD: ${SMTP_PASSWORD:-}
SMTP_FROM_NAME: ${SMTP_FROM_NAME:-BreakPilot}
SMTP_FROM_ADDR: ${SMTP_FROM_ADDR:-noreply@breakpilot.app}
FRONTEND_URL: ${FRONTEND_URL:-https://macmini}
depends_on:
postgres:
condition: service_healthy
valkey:
condition: service_healthy
mailpit:
condition: service_started
restart: unless-stopped
networks:
- breakpilot-network
billing-service:
build:
context: ./billing-service
dockerfile: Dockerfile
container_name: bp-core-billing-service
platform: linux/arm64
ports:
- "8083:8083"
environment:
DATABASE_URL: postgres://${POSTGRES_USER:-breakpilot}:${POSTGRES_PASSWORD:-breakpilot123}@postgres:5432/${POSTGRES_DB:-breakpilot_db}
JWT_SECRET: ${JWT_SECRET:-your-super-secret-jwt-key-change-in-production}
PORT: 8083
ENVIRONMENT: ${ENVIRONMENT:-development}
ALLOWED_ORIGINS: "*"
STRIPE_SECRET_KEY: ${STRIPE_SECRET_KEY:-}
STRIPE_WEBHOOK_SECRET: ${STRIPE_WEBHOOK_SECRET:-}
STRIPE_PUBLISHABLE_KEY: ${STRIPE_PUBLISHABLE_KEY:-}
BILLING_SUCCESS_URL: ${BILLING_SUCCESS_URL:-https://macmini/billing/success}
BILLING_CANCEL_URL: ${BILLING_CANCEL_URL:-https://macmini/billing/cancel}
FRONTEND_URL: ${FRONTEND_URL:-https://macmini}
TRIAL_PERIOD_DAYS: ${TRIAL_PERIOD_DAYS:-14}
INTERNAL_API_KEY: ${INTERNAL_API_KEY:-internal-key}
depends_on:
postgres:
condition: service_healthy
restart: unless-stopped
networks:
- breakpilot-network
# =========================================================
# RAG SERVICE (NEU — extrahiert aus klausur-service)
# =========================================================
rag-service:
build:
context: ./rag-service
dockerfile: Dockerfile
container_name: bp-core-rag-service
platform: linux/arm64
expose:
- "8097"
environment:
PORT: 8097
QDRANT_URL: http://qdrant:6333
MINIO_ENDPOINT: minio:9000
MINIO_ACCESS_KEY: ${MINIO_ROOT_USER:-breakpilot}
MINIO_SECRET_KEY: ${MINIO_ROOT_PASSWORD:-breakpilot123}
MINIO_BUCKET: ${MINIO_BUCKET:-breakpilot-rag}
MINIO_SECURE: "false"
EMBEDDING_SERVICE_URL: http://embedding-service:8087
JWT_SECRET: ${JWT_SECRET:-your-super-secret-jwt-key-change-in-production}
ENVIRONMENT: ${ENVIRONMENT:-development}
depends_on:
qdrant:
condition: service_healthy
minio:
condition: service_healthy
embedding-service:
condition: service_healthy
healthcheck:
test: ["CMD", "curl", "-f", "http://127.0.0.1:8097/health"]
interval: 30s
timeout: 10s
retries: 3
start_period: 15s
restart: unless-stopped
networks:
- breakpilot-network
embedding-service:
build:
context: ./embedding-service
dockerfile: Dockerfile
container_name: bp-core-embedding-service
platform: linux/arm64
volumes:
- embedding_models:/root/.cache/huggingface
environment:
EMBEDDING_BACKEND: ${EMBEDDING_BACKEND:-local}
LOCAL_EMBEDDING_MODEL: ${LOCAL_EMBEDDING_MODEL:-sentence-transformers/all-MiniLM-L6-v2}
LOCAL_RERANKER_MODEL: ${LOCAL_RERANKER_MODEL:-cross-encoder/ms-marco-MiniLM-L-6-v2}
PDF_EXTRACTION_BACKEND: ${PDF_EXTRACTION_BACKEND:-pymupdf}
OPENAI_API_KEY: ${OPENAI_API_KEY:-}
COHERE_API_KEY: ${COHERE_API_KEY:-}
LOG_LEVEL: ${LOG_LEVEL:-INFO}
deploy:
resources:
limits:
memory: 4G
healthcheck:
test: ["CMD", "python", "-c", "import httpx; r=httpx.get('http://127.0.0.1:8087/health'); r.raise_for_status()"]
interval: 30s
timeout: 10s
start_period: 120s
retries: 3
restart: unless-stopped
networks:
- breakpilot-network
# =========================================================
# HEALTH AGGREGATOR (NEU)
# =========================================================
health-aggregator:
build:
context: ./scripts
dockerfile: Dockerfile.health
container_name: bp-core-health
platform: linux/arm64
ports:
- "8099:8099"
environment:
PORT: 8099
CHECK_SERVICES: "postgres:5432,valkey:6379,qdrant:6333,minio:9000,backend-core:8000,rag-service:8097,embedding-service:8087,voice-service:8091"
depends_on:
postgres:
condition: service_healthy
healthcheck:
test: ["CMD", "curl", "-f", "http://127.0.0.1:8099/health"]
interval: 30s
timeout: 10s
retries: 3
restart: unless-stopped
networks:
- breakpilot-network
# =========================================================
# COMMUNICATION
# =========================================================
synapse:
image: matrixdotorg/synapse:latest
container_name: bp-core-synapse
profiles: [chat]
ports:
- "8008:8008"
- "8448:8448"
volumes:
- synapse_data:/data
environment:
SYNAPSE_SERVER_NAME: ${SYNAPSE_SERVER_NAME:-macmini}
SYNAPSE_REPORT_STATS: "no"
SYNAPSE_NO_TLS: "true"
SYNAPSE_ENABLE_REGISTRATION: ${SYNAPSE_ENABLE_REGISTRATION:-true}
SYNAPSE_LOG_LEVEL: ${SYNAPSE_LOG_LEVEL:-WARNING}
UID: "1000"
GID: "1000"
healthcheck:
test: ["CMD", "curl", "-f", "http://127.0.0.1:8008/health"]
interval: 30s
timeout: 10s
start_period: 30s
retries: 3
depends_on:
synapse-db:
condition: service_healthy
restart: unless-stopped
networks:
- breakpilot-network
jitsi-web:
image: jitsi/web:stable-9823
container_name: bp-core-jitsi-web
expose:
- "80"
volumes:
- jitsi_web_config:/config
- jitsi_web_crontabs:/var/spool/cron/crontabs
- jitsi_transcripts:/usr/share/jitsi-meet/transcripts
environment:
ENABLE_XMPP_WEBSOCKET: "true"
ENABLE_COLIBRI_WEBSOCKET: "true"
XMPP_DOMAIN: ${XMPP_DOMAIN:-meet.jitsi}
XMPP_BOSH_URL_BASE: http://jitsi-xmpp:5280
XMPP_MUC_DOMAIN: ${XMPP_MUC_DOMAIN:-muc.meet.jitsi}
XMPP_GUEST_DOMAIN: ${XMPP_GUEST_DOMAIN:-guest.meet.jitsi}
TZ: ${TZ:-Europe/Berlin}
PUBLIC_URL: ${JITSI_PUBLIC_URL:-https://macmini:8443}
JICOFO_AUTH_USER: focus
ENABLE_AUTH: ${JITSI_ENABLE_AUTH:-false}
ENABLE_GUESTS: "true"
ENABLE_RECORDING: "true"
ENABLE_LIVESTREAMING: "false"
DISABLE_HTTPS: "true"
APP_NAME: "BreakPilot Meet"
NATIVE_APP_NAME: "BreakPilot Meet"
PROVIDER_NAME: "BreakPilot"
depends_on:
- jitsi-xmpp
networks:
breakpilot-network:
aliases:
- meet.jitsi
jitsi-xmpp:
image: jitsi/prosody:stable-9823
container_name: bp-core-jitsi-xmpp
volumes:
- jitsi_prosody_config:/config
- jitsi_prosody_plugins:/prosody-plugins-custom
environment:
XMPP_DOMAIN: ${XMPP_DOMAIN:-meet.jitsi}
XMPP_AUTH_DOMAIN: ${XMPP_AUTH_DOMAIN:-auth.meet.jitsi}
XMPP_MUC_DOMAIN: ${XMPP_MUC_DOMAIN:-muc.meet.jitsi}
XMPP_INTERNAL_MUC_DOMAIN: ${XMPP_INTERNAL_MUC_DOMAIN:-internal-muc.meet.jitsi}
XMPP_GUEST_DOMAIN: ${XMPP_GUEST_DOMAIN:-guest.meet.jitsi}
XMPP_RECORDER_DOMAIN: ${XMPP_RECORDER_DOMAIN:-recorder.meet.jitsi}
XMPP_CROSS_DOMAIN: "true"
TZ: ${TZ:-Europe/Berlin}
JICOFO_AUTH_USER: focus
JICOFO_AUTH_PASSWORD: ${JICOFO_AUTH_PASSWORD:-jicofo_secret}
JVB_AUTH_USER: jvb
JVB_AUTH_PASSWORD: ${JVB_AUTH_PASSWORD:-jvb_secret}
JIBRI_XMPP_USER: jibri
JIBRI_XMPP_PASSWORD: ${JIBRI_XMPP_PASSWORD:-jibri_secret}
JIBRI_RECORDER_USER: recorder
JIBRI_RECORDER_PASSWORD: ${JIBRI_RECORDER_PASSWORD:-recorder_secret}
LOG_LEVEL: ${XMPP_LOG_LEVEL:-warn}
PUBLIC_URL: ${JITSI_PUBLIC_URL:-https://macmini:8443}
ENABLE_AUTH: ${JITSI_ENABLE_AUTH:-false}
ENABLE_GUESTS: "true"
restart: unless-stopped
networks:
breakpilot-network:
aliases:
- xmpp.meet.jitsi
jitsi-jicofo:
image: jitsi/jicofo:stable-9823
container_name: bp-core-jitsi-jicofo
volumes:
- jitsi_jicofo_config:/config
environment:
XMPP_DOMAIN: ${XMPP_DOMAIN:-meet.jitsi}
XMPP_AUTH_DOMAIN: ${XMPP_AUTH_DOMAIN:-auth.meet.jitsi}
XMPP_MUC_DOMAIN: ${XMPP_MUC_DOMAIN:-muc.meet.jitsi}
XMPP_INTERNAL_MUC_DOMAIN: ${XMPP_INTERNAL_MUC_DOMAIN:-internal-muc.meet.jitsi}
XMPP_SERVER: jitsi-xmpp
JICOFO_AUTH_USER: focus
JICOFO_AUTH_PASSWORD: ${JICOFO_AUTH_PASSWORD:-jicofo_secret}
TZ: ${TZ:-Europe/Berlin}
ENABLE_AUTH: ${JITSI_ENABLE_AUTH:-false}
AUTH_TYPE: internal
ENABLE_AUTO_OWNER: "true"
depends_on:
- jitsi-xmpp
restart: unless-stopped
networks:
- breakpilot-network
jitsi-jvb:
image: jitsi/jvb:stable-9823
container_name: bp-core-jitsi-jvb
ports:
- "10000:10000/udp"
- "8080:8080"
volumes:
- jitsi_jvb_config:/config
environment:
XMPP_DOMAIN: ${XMPP_DOMAIN:-meet.jitsi}
XMPP_AUTH_DOMAIN: ${XMPP_AUTH_DOMAIN:-auth.meet.jitsi}
XMPP_INTERNAL_MUC_DOMAIN: ${XMPP_INTERNAL_MUC_DOMAIN:-internal-muc.meet.jitsi}
XMPP_SERVER: jitsi-xmpp
JVB_AUTH_USER: jvb
JVB_AUTH_PASSWORD: ${JVB_AUTH_PASSWORD:-jvb_secret}
JVB_PORT: 10000
JVB_STUN_SERVERS: ${JVB_STUN_SERVERS:-stun.l.google.com:19302}
TZ: ${TZ:-Europe/Berlin}
PUBLIC_URL: ${JITSI_PUBLIC_URL:-https://macmini:8443}
COLIBRI_REST_ENABLED: "true"
ENABLE_COLIBRI_WEBSOCKET: "true"
depends_on:
- jitsi-xmpp
restart: unless-stopped
networks:
- breakpilot-network
jibri:
build:
context: ./docker/jibri
dockerfile: Dockerfile
container_name: bp-core-jibri
volumes:
- jibri_recordings:/recordings
- /dev/shm:/dev/shm
shm_size: 2gb
cap_add:
- SYS_ADMIN
- NET_BIND_SERVICE
environment:
XMPP_DOMAIN: ${XMPP_DOMAIN:-meet.jitsi}
XMPP_AUTH_DOMAIN: ${XMPP_AUTH_DOMAIN:-auth.meet.jitsi}
XMPP_INTERNAL_MUC_DOMAIN: ${XMPP_INTERNAL_MUC_DOMAIN:-internal-muc.meet.jitsi}
XMPP_RECORDER_DOMAIN: ${XMPP_RECORDER_DOMAIN:-recorder.meet.jitsi}
XMPP_SERVER: jitsi-xmpp
XMPP_MUC_DOMAIN: ${XMPP_MUC_DOMAIN:-muc.meet.jitsi}
JIBRI_XMPP_USER: jibri
JIBRI_XMPP_PASSWORD: ${JIBRI_XMPP_PASSWORD:-jibri_secret}
JIBRI_RECORDER_USER: recorder
JIBRI_RECORDER_PASSWORD: ${JIBRI_RECORDER_PASSWORD:-recorder_secret}
JIBRI_BREWERY_MUC: JibriBrewery
JIBRI_RECORDING_DIR: /recordings
JIBRI_FINALIZE_SCRIPT: /finalize.sh
TZ: ${TZ:-Europe/Berlin}
DISPLAY: ":0"
RESOLUTION: "1920x1080"
MINIO_ENDPOINT: minio:9000
MINIO_ACCESS_KEY: ${MINIO_ROOT_USER:-breakpilot}
MINIO_SECRET_KEY: ${MINIO_ROOT_PASSWORD:-breakpilot123}
MINIO_BUCKET: ${MINIO_BUCKET:-breakpilot-recordings}
BACKEND_WEBHOOK_URL: http://backend-core:8000/api/recordings/webhook
depends_on:
- jitsi-xmpp
- minio
profiles:
- recording
restart: unless-stopped
networks:
- breakpilot-network
# =========================================================
# DEVOPS & CI/CD
# =========================================================
gitea:
image: gitea/gitea:1.22-rootless
container_name: bp-core-gitea
ports:
- "3003:3003"
- "2222:2222"
volumes:
- gitea_data:/var/lib/gitea
- gitea_config:/etc/gitea
- /etc/timezone:/etc/timezone:ro
- /etc/localtime:/etc/localtime:ro
environment:
USER_UID: "1000"
USER_GID: "1000"
GITEA__database__DB_TYPE: postgres
GITEA__database__HOST: postgres:5432
GITEA__database__NAME: ${POSTGRES_DB:-breakpilot_db}
GITEA__database__USER: ${POSTGRES_USER:-breakpilot}
GITEA__database__PASSWD: ${POSTGRES_PASSWORD:-breakpilot123}
GITEA__server__DOMAIN: macmini
GITEA__server__SSH_DOMAIN: macmini
GITEA__server__ROOT_URL: http://macmini:3003/
GITEA__server__HTTP_PORT: "3003"
GITEA__server__SSH_PORT: "2222"
GITEA__server__SSH_LISTEN_PORT: "2222"
GITEA__actions__ENABLED: "true"
GITEA__actions__DEFAULT_ACTIONS_URL: "https://github.com"
GITEA__service__DISABLE_REGISTRATION: "true"
GITEA__service__REQUIRE_SIGNIN_VIEW: "true"
GITEA__repository__DEFAULT_BRANCH: main
GITEA__log__LEVEL: Warn
GITEA__security__INSTALL_LOCK: "true"
GITEA__webhook__ALLOWED_HOST_LIST: "*"
extra_hosts:
- "macmini:192.168.178.100"
depends_on:
postgres:
condition: service_healthy
healthcheck:
test: ["CMD", "curl", "-f", "http://127.0.0.1:3003/api/healthz"]
interval: 30s
timeout: 10s
start_period: 30s
retries: 3
restart: unless-stopped
networks:
- breakpilot-network
gitea-runner:
image: gitea/act_runner:latest
container_name: bp-core-gitea-runner
volumes:
- gitea_runner_data:/data
- ./gitea/runner-config.yaml:/config/config.yaml:ro
- /var/run/docker.sock:/var/run/docker.sock
environment:
CONFIG_FILE: /config/config.yaml
GITEA_INSTANCE_URL: http://gitea:3003
GITEA_RUNNER_REGISTRATION_TOKEN: ${GITEA_RUNNER_TOKEN:-}
GITEA_RUNNER_NAME: breakpilot-runner
GITEA_RUNNER_LABELS: "ubuntu-latest:docker://node:20-bullseye,ubuntu-22.04:docker://node:20-bullseye"
depends_on:
gitea:
condition: service_healthy
restart: unless-stopped
networks:
- breakpilot-network
woodpecker-server:
image: woodpeckerci/woodpecker-server:v3
container_name: bp-core-woodpecker-server
ports:
- "8090:8000"
volumes:
- woodpecker_data:/var/lib/woodpecker
environment:
WOODPECKER_OPEN: "true"
WOODPECKER_HOST: ${WOODPECKER_HOST:-http://macmini:8090}
WOODPECKER_ADMIN: ${WOODPECKER_ADMIN:-pilotadmin}
WOODPECKER_GITEA: "true"
WOODPECKER_GITEA_URL: http://macmini:3003
WOODPECKER_GITEA_CLIENT: ${WOODPECKER_GITEA_CLIENT:-}
WOODPECKER_GITEA_SECRET: ${WOODPECKER_GITEA_SECRET:-}
WOODPECKER_AGENT_SECRET: ${WOODPECKER_AGENT_SECRET:-woodpecker-secret}
WOODPECKER_DATABASE_DRIVER: sqlite3
WOODPECKER_DATABASE_DATASOURCE: /var/lib/woodpecker/woodpecker.sqlite
WOODPECKER_LOG_LEVEL: warn
WOODPECKER_PLUGINS_PRIVILEGED: "plugins/docker"
WOODPECKER_PLUGINS_TRUSTED_CLONE: "true"
extra_hosts:
- "macmini:192.168.178.100"
depends_on:
gitea:
condition: service_healthy
restart: unless-stopped
networks:
- breakpilot-network
woodpecker-agent:
image: woodpeckerci/woodpecker-agent:v3
container_name: bp-core-woodpecker-agent
volumes:
- /var/run/docker.sock:/var/run/docker.sock
environment:
WOODPECKER_SERVER: woodpecker-server:9000
WOODPECKER_AGENT_SECRET: ${WOODPECKER_AGENT_SECRET:-woodpecker-secret}
WOODPECKER_MAX_WORKFLOWS: "2"
WOODPECKER_LOG_LEVEL: warn
WOODPECKER_BACKEND: docker
DOCKER_HOST: unix:///var/run/docker.sock
WOODPECKER_BACKEND_DOCKER_EXTRA_HOSTS: "macmini:192.168.178.100"
WOODPECKER_BACKEND_DOCKER_NETWORK: breakpilot-network
depends_on:
- woodpecker-server
restart: unless-stopped
networks:
- breakpilot-network
# =========================================================
# WORKFLOW ENGINE
# =========================================================
camunda:
image: camunda/camunda-bpm-platform:7.21.0
container_name: bp-core-camunda
ports:
- "8089:8080"
environment:
DB_DRIVER: org.postgresql.Driver
DB_URL: jdbc:postgresql://postgres:5432/${POSTGRES_DB:-breakpilot_db}
DB_USERNAME: ${POSTGRES_USER:-breakpilot}
DB_PASSWORD: ${POSTGRES_PASSWORD:-breakpilot123}
DB_VALIDATE_ON_BORROW: "true"
WAIT_FOR: postgres:5432
CAMUNDA_BPM_ADMIN_USER_ID: ${CAMUNDA_ADMIN_USER:-admin}
CAMUNDA_BPM_ADMIN_USER_PASSWORD: ${CAMUNDA_ADMIN_PASSWORD:-admin}
depends_on:
postgres:
condition: service_healthy
healthcheck:
test: ["CMD", "curl", "-f", "http://127.0.0.1:8080/camunda/api/engine"]
interval: 30s
timeout: 10s
start_period: 60s
retries: 5
profiles:
- bpmn
restart: unless-stopped
networks:
- breakpilot-network
# =========================================================
# DOCUMENTATION & UTILITIES
# =========================================================
docs:
build:
context: .
dockerfile: docs-src/Dockerfile
container_name: bp-core-docs
profiles: [docs]
platform: linux/arm64
ports:
- "8009:80"
healthcheck:
test: ["CMD", "wget", "-q", "--spider", "http://127.0.0.1:80/"]
interval: 30s
timeout: 10s
retries: 3
restart: unless-stopped
networks:
- breakpilot-network
mailpit:
image: axllent/mailpit:latest
container_name: bp-core-mailpit
ports:
- "8025:8025"
- "1025:1025"
environment:
MP_SMTP_AUTH_ACCEPT_ANY: "true"
MP_SMTP_AUTH_ALLOW_INSECURE: "true"
restart: unless-stopped
networks:
- breakpilot-network
# =========================================================
# VOICE SERVICE
# =========================================================
voice-service:
build:
context: ./voice-service
dockerfile: Dockerfile
container_name: bp-core-voice-service
platform: linux/arm64
expose:
- "8091"
volumes:
- voice_session_data:/app/data/sessions
environment:
PORT: 8091
DATABASE_URL: postgresql://${POSTGRES_USER:-breakpilot}:${POSTGRES_PASSWORD:-breakpilot123}@postgres:5432/${POSTGRES_DB:-breakpilot_db}
VALKEY_URL: redis://valkey:6379/0
KLAUSUR_SERVICE_URL: http://bp-lehrer-klausur-service:8086
OLLAMA_BASE_URL: ${OLLAMA_BASE_URL:-http://host.docker.internal:11434}
OLLAMA_VOICE_MODEL: ${OLLAMA_VOICE_MODEL:-llama3.2}
ENVIRONMENT: ${ENVIRONMENT:-development}
JWT_SECRET: ${JWT_SECRET:-your-super-secret-jwt-key-change-in-production}
extra_hosts:
- "host.docker.internal:host-gateway"
depends_on:
postgres:
condition: service_healthy
valkey:
condition: service_started
healthcheck:
test: ["CMD", "curl", "-f", "http://127.0.0.1:8091/health"]
interval: 30s
timeout: 10s
start_period: 60s
retries: 3
restart: unless-stopped
networks:
- breakpilot-network
# =========================================================
# NIGHT SCHEDULER
# =========================================================
night-scheduler:
build:
context: ./night-scheduler
dockerfile: Dockerfile
container_name: bp-core-night-scheduler
ports:
- "8096:8096"
volumes:
- /var/run/docker.sock:/var/run/docker.sock
- ./night-scheduler/config:/config
environment:
COMPOSE_PROJECT_NAME: breakpilot-core
CONTAINER_PATTERN: "bp-*"
EXCLUDED_CONTAINERS: "bp-core-night-scheduler,bp-core-nginx,bp-core-postgres,bp-core-valkey"
healthcheck:
test: ["CMD", "curl", "-f", "http://127.0.0.1:8096/health"]
interval: 30s
timeout: 10s
start_period: 10s
retries: 3
restart: unless-stopped
networks:
- breakpilot-network
# =========================================================
# ADMIN CORE
# =========================================================
admin-core:
build:
context: ./admin-core
dockerfile: Dockerfile
args:
NEXT_PUBLIC_API_URL: ${NEXT_PUBLIC_API_URL:-https://macmini:8000}
container_name: bp-core-admin
platform: linux/arm64
expose:
- "3000"
environment:
NODE_ENV: production
BACKEND_URL: http://backend-core:8000
WOODPECKER_URL: http://bp-core-woodpecker-server:8000
WOODPECKER_TOKEN: ${WOODPECKER_TOKEN:-}
OLLAMA_URL: ${OLLAMA_URL:-http://host.docker.internal:11434}
extra_hosts:
- "host.docker.internal:host-gateway"
volumes:
- /var/run/docker.sock:/var/run/docker.sock:ro
restart: unless-stopped
networks:
- breakpilot-network
# =========================================================
# ERP (ERPNext)
# =========================================================
erpnext-db:
image: mariadb:10.6
container_name: bp-core-erpnext-db
profiles: [erp]
environment:
MYSQL_ROOT_PASSWORD: ${ERPNEXT_DB_ROOT_PASSWORD:-erpnext_root}
MYSQL_DATABASE: erpnext
MYSQL_USER: erpnext
MYSQL_PASSWORD: ${ERPNEXT_DB_PASSWORD:-erpnext_secret}
volumes:
- erpnext_db_data:/var/lib/mysql
healthcheck:
test: ["CMD", "mysqladmin", "ping", "-h", "127.0.0.1"]
interval: 5s
timeout: 5s
retries: 10
restart: unless-stopped
networks:
- breakpilot-network
erpnext-redis-queue:
image: redis:alpine
profiles: [erp]
container_name: bp-core-erpnext-redis-queue
volumes:
- erpnext_redis_queue_data:/data
restart: unless-stopped
networks:
- breakpilot-network
erpnext-redis-cache:
profiles: [erp]
image: redis:alpine
container_name: bp-core-erpnext-redis-cache
volumes:
- erpnext_redis_cache_data:/data
restart: unless-stopped
networks:
- breakpilot-network
erpnext-create-site:
image: frappe/erpnext:latest
container_name: bp-core-erpnext-create-site
profiles: [erp]
volumes:
- erpnext_sites:/home/frappe/frappe-bench/sites
- erpnext_logs:/home/frappe/frappe-bench/logs
environment:
DB_HOST: erpnext-db
DB_PORT: "3306"
REDIS_CACHE: redis://erpnext-redis-cache:6379/0
REDIS_QUEUE: redis://erpnext-redis-queue:6379/0
SOCKETIO_PORT: "9000"
entrypoint: bash -c "ls sites/erpnext.local/site_config.json 2>/dev/null && echo 'Site exists' || bench new-site erpnext.local --mariadb-root-password=${ERPNEXT_DB_ROOT_PASSWORD:-erpnext_root} --admin-password=${ERPNEXT_ADMIN_PASSWORD:-admin} --install-app erpnext --set-default"
depends_on:
erpnext-db:
condition: service_healthy
erpnext-redis-cache:
condition: service_started
erpnext-redis-queue:
condition: service_started
restart: "no"
networks:
- breakpilot-network
erpnext-backend:
image: frappe/erpnext:latest
container_name: bp-core-erpnext-backend
profiles: [erp]
volumes:
- erpnext_sites:/home/frappe/frappe-bench/sites
- erpnext_logs:/home/frappe/frappe-bench/logs
environment:
DB_HOST: erpnext-db
DB_PORT: "3306"
REDIS_CACHE: redis://erpnext-redis-cache:6379/0
REDIS_QUEUE: redis://erpnext-redis-queue:6379/0
SOCKETIO_PORT: "9000"
depends_on:
erpnext-db:
condition: service_healthy
erpnext-create-site:
condition: service_completed_successfully
restart: unless-stopped
networks:
- breakpilot-network
erpnext-websocket:
image: frappe/erpnext:latest
container_name: bp-core-erpnext-websocket
profiles: [erp]
command: ["node", "/home/frappe/frappe-bench/apps/frappe/socketio.js"]
volumes:
- erpnext_sites:/home/frappe/frappe-bench/sites
- erpnext_logs:/home/frappe/frappe-bench/logs
environment:
DB_HOST: erpnext-db
DB_PORT: "3306"
REDIS_CACHE: redis://erpnext-redis-cache:6379/0
REDIS_QUEUE: redis://erpnext-redis-queue:6379/0
SOCKETIO_PORT: "9000"
depends_on:
erpnext-db:
condition: service_healthy
erpnext-create-site:
condition: service_completed_successfully
restart: unless-stopped
networks:
- breakpilot-network
erpnext-scheduler:
image: frappe/erpnext:latest
container_name: bp-core-erpnext-scheduler
profiles: [erp]
command: ["bench", "schedule"]
volumes:
- erpnext_sites:/home/frappe/frappe-bench/sites
- erpnext_logs:/home/frappe/frappe-bench/logs
environment:
DB_HOST: erpnext-db
DB_PORT: "3306"
REDIS_CACHE: redis://erpnext-redis-cache:6379/0
REDIS_QUEUE: redis://erpnext-redis-queue:6379/0
SOCKETIO_PORT: "9000"
depends_on:
erpnext-db:
condition: service_healthy
erpnext-create-site:
condition: service_completed_successfully
restart: unless-stopped
networks:
- breakpilot-network
erpnext-worker-long:
image: frappe/erpnext:latest
container_name: bp-core-erpnext-worker-long
profiles: [erp]
command: ["bench", "worker", "--queue", "long"]
volumes:
- erpnext_sites:/home/frappe/frappe-bench/sites
- erpnext_logs:/home/frappe/frappe-bench/logs
environment:
DB_HOST: erpnext-db
DB_PORT: "3306"
REDIS_CACHE: redis://erpnext-redis-cache:6379/0
REDIS_QUEUE: redis://erpnext-redis-queue:6379/0
SOCKETIO_PORT: "9000"
depends_on:
erpnext-db:
condition: service_healthy
erpnext-create-site:
condition: service_completed_successfully
restart: unless-stopped
networks:
- breakpilot-network
erpnext-worker-short:
image: frappe/erpnext:latest
container_name: bp-core-erpnext-worker-short
profiles: [erp]
command: ["bench", "worker", "--queue", "short"]
volumes:
- erpnext_sites:/home/frappe/frappe-bench/sites
- erpnext_logs:/home/frappe/frappe-bench/logs
environment:
DB_HOST: erpnext-db
DB_PORT: "3306"
REDIS_CACHE: redis://erpnext-redis-cache:6379/0
REDIS_QUEUE: redis://erpnext-redis-queue:6379/0
SOCKETIO_PORT: "9000"
depends_on:
erpnext-db:
condition: service_healthy
erpnext-create-site:
condition: service_completed_successfully
restart: unless-stopped
networks:
- breakpilot-network
erpnext-frontend:
image: frappe/erpnext:latest
container_name: bp-core-erpnext-frontend
profiles: [erp]
command: ["nginx-entrypoint.sh"]
ports:
- "8092:8080"
volumes:
- erpnext_sites:/home/frappe/frappe-bench/sites
- erpnext_logs:/home/frappe/frappe-bench/logs
environment:
BACKEND: erpnext-backend:8000
SOCKETIO: erpnext-websocket:9000
UPSTREAM_REAL_IP_ADDRESS: "127.0.0.1"
UPSTREAM_REAL_IP_HEADER: X-Forwarded-For
UPSTREAM_REAL_IP_RECURSIVE: "off"
FRAPPE_SITE_NAME_HEADER: erpnext.local
depends_on:
- erpnext-backend
- erpnext-websocket
restart: unless-stopped
networks:
- breakpilot-network
# =========================================================
# BACKUP (Profile)
# =========================================================
backup:
image: postgres:16-alpine
container_name: bp-core-backup
volumes:
- ./backups:/backups
environment:
PGHOST: postgres
PGUSER: ${POSTGRES_USER:-breakpilot}
PGPASSWORD: ${POSTGRES_PASSWORD:-breakpilot123}
PGDATABASE: ${POSTGRES_DB:-breakpilot_db}
entrypoint: /bin/sh
command: -c "pg_dump -Fc > /backups/breakpilot_db_$(date +%Y%m%d_%H%M%S).backup && echo 'Backup complete'"
profiles:
- backup
depends_on:
postgres:
condition: service_healthy
networks:
- breakpilot-network
# =========================================================
# PITCH DECK - Investor Presentation
# =========================================================
pitch-deck:
build:
context: ./pitch-deck
dockerfile: Dockerfile
container_name: bp-core-pitch-deck
platform: linux/arm64
ports:
- "3012:3000"
environment:
NODE_ENV: production
DATABASE_URL: postgres://${POSTGRES_USER:-breakpilot}:${POSTGRES_PASSWORD:-breakpilot123}@postgres:5432/${POSTGRES_DB:-breakpilot_db}
OLLAMA_URL: ${OLLAMA_URL:-http://host.docker.internal:11434}
OLLAMA_MODEL: ${OLLAMA_MODEL:-qwen3:30b-a3b}
extra_hosts:
- "host.docker.internal:host-gateway"
depends_on:
postgres:
condition: service_healthy
restart: unless-stopped
networks:
- breakpilot-network