services: bot: build: context: ../.. dockerfile: apps/bot/Dockerfile environment: NODE_ENV: production PORT: ${BOT_PORT:?8080} LOG_LEVEL: ${LOG_LEVEL:-info} DATABASE_URL: ${DATABASE_URL:?} DB_SCHEMA: ${DB_SCHEMA:-public} TELEGRAM_BOT_TOKEN: ${TELEGRAM_BOT_TOKEN:?} TELEGRAM_WEBHOOK_SECRET: ${TELEGRAM_WEBHOOK_SECRET:?} TELEGRAM_WEBHOOK_PATH: ${TELEGRAM_WEBHOOK_PATH:-/webhook/telegram} MINI_APP_URL: ${MINI_APP_URL:?} MINI_APP_ALLOWED_ORIGINS: ${MINI_APP_ALLOWED_ORIGINS:?} OPENAI_API_KEY: ${OPENAI_API_KEY:-} PURCHASE_PARSER_MODEL: ${PURCHASE_PARSER_MODEL:-gpt-4o-mini} ASSISTANT_MODEL: ${ASSISTANT_MODEL:-gpt-4o-mini} TOPIC_PROCESSOR_MODEL: ${TOPIC_PROCESSOR_MODEL:-gpt-4o-mini} TOPIC_PROCESSOR_TIMEOUT_MS: ${TOPIC_PROCESSOR_TIMEOUT_MS:-10000} ASSISTANT_TIMEOUT_MS: ${ASSISTANT_TIMEOUT_MS:-20000} ASSISTANT_MEMORY_MAX_TURNS: ${ASSISTANT_MEMORY_MAX_TURNS:-12} ASSISTANT_RATE_LIMIT_BURST: ${ASSISTANT_RATE_LIMIT_BURST:-5} ASSISTANT_RATE_LIMIT_BURST_WINDOW_MS: ${ASSISTANT_RATE_LIMIT_BURST_WINDOW_MS:-60000} ASSISTANT_RATE_LIMIT_ROLLING: ${ASSISTANT_RATE_LIMIT_ROLLING:-50} ASSISTANT_RATE_LIMIT_ROLLING_WINDOW_MS: ${ASSISTANT_RATE_LIMIT_ROLLING_WINDOW_MS:-86400000} SCHEDULER_SHARED_SECRET: ${SCHEDULER_SHARED_SECRET:?} SCHEDULED_DISPATCH_PROVIDER: ${SCHEDULED_DISPATCH_PROVIDER:?self-hosted} SCHEDULER_OIDC_ALLOWED_EMAILS: ${SCHEDULER_OIDC_ALLOWED_EMAILS:-} SCHEDULED_DISPATCH_PUBLIC_BASE_URL: ${SCHEDULED_DISPATCH_PUBLIC_BASE_URL:-} GCP_SCHEDULED_DISPATCH_PROJECT_ID: ${GCP_SCHEDULED_DISPATCH_PROJECT_ID:-} GCP_SCHEDULED_DISPATCH_LOCATION: ${GCP_SCHEDULED_DISPATCH_LOCATION:-} GCP_SCHEDULED_DISPATCH_QUEUE: ${GCP_SCHEDULED_DISPATCH_QUEUE:-} AWS_SCHEDULED_DISPATCH_REGION: ${AWS_SCHEDULED_DISPATCH_REGION:-} AWS_SCHEDULED_DISPATCH_TARGET_LAMBDA_ARN: ${AWS_SCHEDULED_DISPATCH_TARGET_LAMBDA_ARN:-} AWS_SCHEDULED_DISPATCH_ROLE_ARN: ${AWS_SCHEDULED_DISPATCH_ROLE_ARN:-} AWS_SCHEDULED_DISPATCH_GROUP_NAME: ${AWS_SCHEDULED_DISPATCH_GROUP_NAME:-} command: - /bin/sh - -lc - bun packages/db/dist/migrate.js && exec bun apps/bot/dist/index.js healthcheck: test: - CMD - bun - -e - "fetch('http://127.0.0.1:' + (process.env.PORT ?? '8080') + '/healthz').then((res) => process.exit(res.ok ? 0 : 1)).catch(() => process.exit(1))" interval: 30s timeout: 5s retries: 3 start_period: 15s restart: unless-stopped miniapp: build: context: ../.. dockerfile: apps/miniapp/Dockerfile environment: BOT_API_URL: ${BOT_API_URL:?} depends_on: bot: condition: service_healthy restart: unless-stopped scheduler: build: context: ../.. dockerfile: apps/bot/Dockerfile command: - bun - apps/bot/dist/scheduler-runner.js environment: NODE_ENV: production LOG_LEVEL: ${LOG_LEVEL:-info} DATABASE_URL: ${DATABASE_URL:?} DB_SCHEMA: ${DB_SCHEMA:-public} TELEGRAM_BOT_TOKEN: ${TELEGRAM_BOT_TOKEN:?} TELEGRAM_WEBHOOK_SECRET: ${TELEGRAM_WEBHOOK_SECRET:?} TELEGRAM_WEBHOOK_PATH: ${TELEGRAM_WEBHOOK_PATH:-/webhook/telegram} MINI_APP_URL: ${MINI_APP_URL:?} MINI_APP_ALLOWED_ORIGINS: ${MINI_APP_ALLOWED_ORIGINS:?} OPENAI_API_KEY: ${OPENAI_API_KEY:-} PURCHASE_PARSER_MODEL: ${PURCHASE_PARSER_MODEL:-gpt-4o-mini} ASSISTANT_MODEL: ${ASSISTANT_MODEL:-gpt-4o-mini} TOPIC_PROCESSOR_MODEL: ${TOPIC_PROCESSOR_MODEL:-gpt-4o-mini} TOPIC_PROCESSOR_TIMEOUT_MS: ${TOPIC_PROCESSOR_TIMEOUT_MS:-10000} ASSISTANT_TIMEOUT_MS: ${ASSISTANT_TIMEOUT_MS:-20000} ASSISTANT_MEMORY_MAX_TURNS: ${ASSISTANT_MEMORY_MAX_TURNS:-12} ASSISTANT_RATE_LIMIT_BURST: ${ASSISTANT_RATE_LIMIT_BURST:-5} ASSISTANT_RATE_LIMIT_BURST_WINDOW_MS: ${ASSISTANT_RATE_LIMIT_BURST_WINDOW_MS:-60000} ASSISTANT_RATE_LIMIT_ROLLING: ${ASSISTANT_RATE_LIMIT_ROLLING:-50} ASSISTANT_RATE_LIMIT_ROLLING_WINDOW_MS: ${ASSISTANT_RATE_LIMIT_ROLLING_WINDOW_MS:-86400000} SCHEDULER_SHARED_SECRET: ${SCHEDULER_SHARED_SECRET:?} SCHEDULED_DISPATCH_PROVIDER: ${SCHEDULED_DISPATCH_PROVIDER:?self-hosted} BOT_INTERNAL_BASE_URL: http://bot:8080 SCHEDULER_POLL_INTERVAL_MS: ${SCHEDULER_POLL_INTERVAL_MS:-60000} SCHEDULER_DUE_SCAN_LIMIT: ${SCHEDULER_DUE_SCAN_LIMIT:-25} depends_on: bot: condition: service_healthy restart: unless-stopped