services: traefik: image: traefik:v3.0 container_name: codetutor-traefik command: - --api.insecure=false - --providers.docker=true - --providers.docker.exposedbydefault=false - --entrypoints.web.address=:80 - --entrypoints.websecure.address=:443 - --entrypoints.web.http.redirections.entrypoint.to=websecure - --entrypoints.web.http.redirections.entrypoint.scheme=https - --certificatesresolvers.letsencrypt.acme.httpchallenge=true - --certificatesresolvers.letsencrypt.acme.httpchallenge.entrypoint=web - --certificatesresolvers.letsencrypt.acme.email=${ACME_EMAIL:?ACME_EMAIL is required} - --certificatesresolvers.letsencrypt.acme.storage=/letsencrypt/acme.json ports: - "80:80" - "443:443" volumes: - /var/run/docker.sock:ro - codetutor-letsencrypt:/letsencrypt restart: unless-stopped healthcheck: test: ["CMD", "traefik", "healthcheck"] interval: 30s timeout: 10s retries: 3 deploy: resources: limits: cpus: "0.5" memory: 256M reservations: cpus: "0.1" memory: 64M db: image: postgres:16-alpine container_name: codetutor-db environment: POSTGRES_USER: ${POSTGRES_USER:-codetutor} POSTGRES_PASSWORD: ${POSTGRES_PASSWORD:?POSTGRES_PASSWORD is required} POSTGRES_DB: ${POSTGRES_DB:-codetutor} volumes: - codetutor-db-data:/var/lib/postgresql/data healthcheck: test: ["CMD-SHELL", "pg_isready -U ${POSTGRES_USER:-codetutor}"] interval: 10s timeout: 5s retries: 5 restart: unless-stopped deploy: resources: limits: cpus: "1" memory: 512M reservations: cpus: "0.25" memory: 128M backend: build: context: ./backend dockerfile: Dockerfile container_name: codetutor-backend environment: DATABASE_URL: postgresql+asyncpg://${POSTGRES_USER:-codetutor}:${POSTGRES_PASSWORD}@db:5432/${POSTGRES_DB:-codetutor} CORS_ORIGINS: '["https://${DOMAIN:?DOMAIN is required}"]' depends_on: db: condition: service_healthy labels: - traefik.enable=true - traefik.http.routers.backend.rule=Host(`${DOMAIN}`) && PathPrefix(`/api`) - traefik.http.routers.backend.entrypoints=websecure - traefik.http.routers.backend.tls.certresolver=letsencrypt - traefik.http.services.backend.loadbalancer.server.port=8000 healthcheck: test: ["CMD", "curl", "-f", "http://localhost:8000/api/health"] interval: 30s timeout: 10s retries: 3 start_period: 10s restart: unless-stopped deploy: resources: limits: cpus: "1" memory: 512M reservations: cpus: "0.25" memory: 128M frontend: build: context: ./frontend dockerfile: Dockerfile container_name: codetutor-frontend environment: NEXT_PUBLIC_API_URL: https://${DOMAIN} depends_on: - backend labels: - traefik.enable=true - traefik.http.routers.frontend.rule=Host(`${DOMAIN}`) - traefik.http.routers.frontend.entrypoints=websecure - traefik.http.routers.frontend.tls.certresolver=letsencrypt - traefik.http.services.frontend.loadbalancer.server.port=3000 healthcheck: test: ["CMD", "curl", "-f", "http://localhost:3000"] interval: 30s timeout: 10s retries: 3 start_period: 15s restart: unless-stopped deploy: resources: limits: cpus: "1" memory: 512M reservations: cpus: "0.25" memory: 128M volumes: codetutor-db-data: codetutor-letsencrypt: