mit einem Klick
dokploy-multi-service
// Multi-service architecture patterns for Dokploy templates including dependency chains, service communication, and complex stack design. Use when building templates with 2+ services.
// Multi-service architecture patterns for Dokploy templates including dependency chains, service communication, and complex stack design. Use when building templates with 2+ services.
Integrate Cloudflare services with Dokploy templates: R2 storage, DNS challenge for SSL, Zero Trust Access, Workers, WAF, and Tunnel. Default to CF services for external dependencies.
Generate Docker Compose files following Dokploy conventions with proper networking, volumes, and service patterns. Use when creating new Dokploy templates or converting existing compose files.
Environment variable patterns for Dokploy templates including required vs optional syntax, secrets, connection strings, and configuration organization.
Health check patterns for different service types in Dokploy templates. Covers HTTP, PostgreSQL, MongoDB, Redis, MySQL, and custom health checks.
Multi-tenancy patterns for Dokploy templates with network isolation: separate docker networks per tenant, shared infrastructure, and tenant-specific configuration.
Security best practices for Dokploy templates: secrets management, network isolation, least privilege, image security, and hardening recommendations.
| name | dokploy-multi-service |
| description | Multi-service architecture patterns for Dokploy templates including dependency chains, service communication, and complex stack design. Use when building templates with 2+ services. |
| version | 1.0.0 |
| author | Home Lab Infrastructure Team |
Structure:
┌─────────────┐ ┌─────────────┐
│ App │────▶│ Database │
│ (web UI) │ │ (internal) │
└─────────────┘ └─────────────┘
│
▼
dokploy-network
Characteristics:
service_healthyExample (Paaster + MongoDB):
services:
paaster:
image: wardpearce/paaster:3.1.7
depends_on:
mongodb:
condition: service_healthy
networks:
- paaster-net # Internal
- dokploy-network # External (Traefik)
# ... traefik labels, health check
mongodb:
image: mongo:7
networks:
- paaster-net # Internal ONLY
# ... health check, no traefik labels
networks:
paaster-net:
driver: bridge
dokploy-network:
external: true
Structure:
┌─────────────┐ ┌─────────────┐ ┌─────────────┐
│ App │────▶│ Database │ │ Cache │
│ (web UI) │ │ (internal) │◀────│ (internal) │
└─────────────┘ └─────────────┘ └─────────────┘
│ │ │
└───────────────────┴───────────────────┘
│
internal network
Dependency Logic:
services:
app:
depends_on:
database:
condition: service_healthy
cache:
condition: service_healthy
Example (Django + PostgreSQL + Redis):
services:
app:
image: myapp:1.0.0
depends_on:
postgres:
condition: service_healthy
redis:
condition: service_healthy
environment:
DATABASE_URL: postgresql://user:pass@postgres:5432/db
REDIS_URL: redis://redis:6379
networks:
- app-net
- dokploy-network
postgres:
image: postgres:16-alpine
networks:
- app-net
healthcheck:
test: ["CMD-SHELL", "pg_isready -U user -d db"]
redis:
image: redis:7-alpine
networks:
- app-net
healthcheck:
test: ["CMD", "redis-cli", "ping"]
Structure:
┌─────────────┐
│ Helper 1 │
│ (on-demand) │
└──────▲──────┘
│
┌─────────────┐ ┌──────┴──────┐ ┌─────────────┐
│ Database │◀────│ App │────▶│ Helper 2 │
│ (required) │ │ (main) │ │ (on-demand) │
└─────────────┘ └─────────────┘ └─────────────┘
│
▼
dokploy-network
Dependency Logic:
service_healthy (required at startup)service_started (called on-demand)Example (Paperless-ngx):
services:
paperless:
image: ghcr.io/paperless-ngx/paperless-ngx:2.13
depends_on:
postgres:
condition: service_healthy # Required at startup
redis:
condition: service_healthy # Required at startup
gotenberg:
condition: service_started # On-demand helper
tika:
condition: service_started # On-demand helper
environment:
PAPERLESS_TIKA_ENABLED: 1
PAPERLESS_TIKA_ENDPOINT: http://tika:9998
PAPERLESS_TIKA_GOTENBERG_ENDPOINT: http://gotenberg:3000
networks:
- paperless-net
- dokploy-network
postgres:
image: postgres:16-alpine
networks:
- paperless-net
healthcheck:
test: ["CMD-SHELL", "pg_isready -U paperless -d paperless"]
redis:
image: redis:7-alpine
networks:
- paperless-net
healthcheck:
test: ["CMD", "redis-cli", "ping"]
gotenberg:
image: gotenberg/gotenberg:8
networks:
- paperless-net
# No healthcheck - stateless converter
tika:
image: apache/tika:2.9.1.0
networks:
- paperless-net
# No healthcheck - stateless parser
Structure:
┌─────────────┐ ┌─────────────┐
│ App │────▶│ Database │
│ (main) │ │ │
└─────────────┘ └─────────────┘
│
▼
┌─────────────┐ ┌─────────────┐
│ API │────▶│ Database │
│ (secondary) │ │ (shared) │
└─────────────┘ └─────────────┘
│
▼
dokploy-network (both app and api accessible)
Example (App + API on different subdomains):
services:
app:
image: myapp-web:1.0.0
depends_on:
postgres:
condition: service_healthy
networks:
- app-net
- dokploy-network
labels:
- "traefik.http.routers.app.rule=Host(`${DOMAIN}`)"
# ...
api:
image: myapp-api:1.0.0
depends_on:
postgres:
condition: service_healthy
networks:
- app-net
- dokploy-network
labels:
- "traefik.http.routers.api.rule=Host(`api.${DOMAIN}`)"
# ...
postgres:
image: postgres:16-alpine
networks:
- app-net # Shared by both app and api
| Condition | When to Use | Example |
|---|---|---|
service_healthy | Database, cache, required services | PostgreSQL, MongoDB, Redis |
service_started | Helper services, on-demand converters | Gotenberg, Tika, sidecars |
service_completed_successfully | Init containers, migrations | DB migrations, setup scripts |
services:
forgejo:
image: codeberg.org/forgejo/forgejo:9
restart: always
depends_on:
postgres:
condition: service_healthy
volumes:
- forgejo-data:/data
environment:
FORGEJO__database__DB_TYPE: postgres
FORGEJO__database__HOST: postgres:5432
FORGEJO__database__NAME: ${POSTGRES_DB:-forgejo}
FORGEJO__database__USER: ${POSTGRES_USER:-forgejo}
FORGEJO__database__PASSWD: ${POSTGRES_PASSWORD:?Set database password}
networks:
- forgejo-net
- dokploy-network
labels:
- "traefik.enable=true"
- "traefik.http.routers.forgejo.rule=Host(`${FORGEJO_DOMAIN}`)"
- "traefik.http.routers.forgejo.entrypoints=websecure"
- "traefik.http.routers.forgejo.tls.certresolver=letsencrypt"
- "traefik.http.services.forgejo.loadbalancer.server.port=3000"
- "traefik.docker.network=dokploy-network"
healthcheck:
test: ["CMD", "curl", "-f", "http://localhost:3000/api/healthz"]
interval: 30s
timeout: 10s
retries: 3
start_period: 30s
postgres:
image: postgres:16-alpine
restart: always
volumes:
- postgres-data:/var/lib/postgresql/data
environment:
POSTGRES_DB: ${POSTGRES_DB:-forgejo}
POSTGRES_USER: ${POSTGRES_USER:-forgejo}
POSTGRES_PASSWORD: ${POSTGRES_PASSWORD:?Set database password}
networks:
- forgejo-net
healthcheck:
test: ["CMD-SHELL", "pg_isready -U ${POSTGRES_USER:-forgejo} -d ${POSTGRES_DB:-forgejo}"]
interval: 30s
timeout: 10s
retries: 3
start_period: 30s
volumes:
forgejo-data:
driver: local
postgres-data:
driver: local
networks:
forgejo-net:
driver: bridge
dokploy-network:
external: true
services:
paperless:
image: ghcr.io/paperless-ngx/paperless-ngx:2.13
restart: always
depends_on:
postgres:
condition: service_healthy
redis:
condition: service_healthy
gotenberg:
condition: service_started
tika:
condition: service_started
volumes:
- paperless-data:/usr/src/paperless/data
- paperless-media:/usr/src/paperless/media
- paperless-export:/usr/src/paperless/export
- paperless-consume:/usr/src/paperless/consume
environment:
PAPERLESS_REDIS: redis://redis:6379
PAPERLESS_DBHOST: postgres
PAPERLESS_DBNAME: ${POSTGRES_DB:-paperless}
PAPERLESS_DBUSER: ${POSTGRES_USER:-paperless}
PAPERLESS_DBPASS: ${POSTGRES_PASSWORD:?Set database password}
PAPERLESS_SECRET_KEY: ${PAPERLESS_SECRET_KEY:?Set secret key}
PAPERLESS_URL: https://${PAPERLESS_DOMAIN}
PAPERLESS_TIKA_ENABLED: 1
PAPERLESS_TIKA_ENDPOINT: http://tika:9998
PAPERLESS_TIKA_GOTENBERG_ENDPOINT: http://gotenberg:3000
networks:
- paperless-net
- dokploy-network
labels:
- "traefik.enable=true"
- "traefik.http.routers.paperless.rule=Host(`${PAPERLESS_DOMAIN}`)"
- "traefik.http.routers.paperless.entrypoints=websecure"
- "traefik.http.routers.paperless.tls.certresolver=letsencrypt"
- "traefik.http.services.paperless.loadbalancer.server.port=8000"
- "traefik.docker.network=dokploy-network"
healthcheck:
test: ["CMD", "curl", "-f", "http://localhost:8000"]
interval: 30s
timeout: 10s
retries: 3
start_period: 60s
postgres:
image: postgres:16-alpine
restart: always
volumes:
- postgres-data:/var/lib/postgresql/data
environment:
POSTGRES_DB: ${POSTGRES_DB:-paperless}
POSTGRES_USER: ${POSTGRES_USER:-paperless}
POSTGRES_PASSWORD: ${POSTGRES_PASSWORD:?Set database password}
networks:
- paperless-net
healthcheck:
test: ["CMD-SHELL", "pg_isready -U ${POSTGRES_USER:-paperless} -d ${POSTGRES_DB:-paperless}"]
interval: 30s
timeout: 10s
retries: 3
start_period: 30s
redis:
image: redis:7-alpine
restart: always
volumes:
- redis-data:/data
networks:
- paperless-net
healthcheck:
test: ["CMD", "redis-cli", "ping"]
interval: 30s
timeout: 10s
retries: 3
start_period: 10s
gotenberg:
image: gotenberg/gotenberg:8
restart: always
networks:
- paperless-net
command:
- "gotenberg"
- "--chromium-disable-javascript=true"
- "--chromium-allow-list=file:///tmp/.*"
tika:
image: apache/tika:2.9.1.0
restart: always
networks:
- paperless-net
volumes:
paperless-data:
driver: local
paperless-media:
driver: local
paperless-export:
driver: local
paperless-consume:
driver: local
postgres-data:
driver: local
redis-data:
driver: local
networks:
paperless-net:
driver: bridge
dokploy-network:
external: true
| Service | Internal URL Format |
|---|---|
| PostgreSQL | postgresql://user:pass@postgres:5432/db |
| MongoDB | mongodb://mongodb:27017/dbname |
| Redis | redis://redis:6379 |
| MySQL | mysql://user:pass@mysql:3306/db |
| HTTP APIs | http://service-name:port/path |
# Database connections
DATABASE_URL: postgresql://${DB_USER}:${DB_PASS}@postgres:5432/${DB_NAME}
MONGO_URL: mongodb://mongodb:27017/${MONGO_DB}
REDIS_URL: redis://redis:6379
# Internal HTTP services
TIKA_ENDPOINT: http://tika:9998
GOTENBERG_ENDPOINT: http://gotenberg:3000
API_ENDPOINT: http://api:8080
service_healthy for required startup dependenciesservice_started for on-demand helpersdepends_on service must existservice_healthy dependenciesIssue: Service A depends on B, B depends on A Solution: Redesign architecture, use async communication
Issue: service_healthy fails without health check
Solution: Add health check to dependency service
Issue: Connection refused errors Solution: Use exact docker-compose service name in URLs
Issue: Security vulnerability Solution: Remove dokploy-network from database services
This skill is part of the skills-first architecture - loaded during the Architecture phase to design complex service dependencies before generation begins.
dokploy-compose-structure: Base structure implementationdokploy-health-patterns: Health check configurationdokploy-environment-config: Connection strings and service URLs/dokploy-create command: Phase 2 (Architecture) - Loaded when 2+ services detectedSee: .claude/commands/dokploy-create.md for full workflow