Dokploy

App Flowy

AppFlowy is an open-source alternative to Notion. You are in charge of your data and customizations.

App Flowy logo

Configuration

services:
  nginx:
    restart: on-failure
    image: nginx
    volumes:
      - ../files/nginx/nginx.conf:/etc/nginx/nginx.conf

  minio:
    restart: on-failure
    image: minio/minio
    environment:
      - MINIO_BROWSER_REDIRECT_URL=${APPFLOWY_BASE_URL}/minio
      - MINIO_ROOT_USER=${APPFLOWY_S3_ACCESS_KEY:-minioadmin}
      - MINIO_ROOT_PASSWORD=${APPFLOWY_S3_SECRET_KEY:-minioadmin}
    command: server /data --console-address ":9001"
    volumes:
      - minio_data:/data

  postgres:
    restart: on-failure
    image: pgvector/pgvector:pg16
    environment:
      - POSTGRES_USER=${POSTGRES_USER:-postgres}
      - POSTGRES_DB=${POSTGRES_DB:-postgres}
      - POSTGRES_PASSWORD=${POSTGRES_PASSWORD:-password}
      - POSTGRES_HOST=${POSTGRES_HOST:-postgres}
    healthcheck:
      test: ["CMD", "pg_isready", "-U", "${POSTGRES_USER}", "-d", "${POSTGRES_DB}"]
      interval: 5s
      timeout: 5s
      retries: 12
    volumes:
      - postgres_data:/var/lib/postgresql/data
      - ./files/volumes/postgres/:/docker-entrypoint-initdb.d/

  redis:
    restart: on-failure
    image: redis

  gotrue:
    restart: on-failure
    image: appflowyinc/gotrue:${GOTRUE_VERSION:-latest}
    depends_on:
      postgres:
        condition: service_healthy
    healthcheck:
      test: ["CMD", "curl", "--fail", "http://127.0.0.1:9999/health"]
      interval: 5s
      timeout: 5s
      retries: 12
    environment:
      # Admin Configuration
      - GOTRUE_ADMIN_EMAIL=${GOTRUE_ADMIN_EMAIL}
      - GOTRUE_ADMIN_PASSWORD=${GOTRUE_ADMIN_PASSWORD}
      - GOTRUE_DISABLE_SIGNUP=${GOTRUE_DISABLE_SIGNUP:-false}

      # Site Configuration
      - GOTRUE_SITE_URL=appflowy-flutter://
      - GOTRUE_URI_ALLOW_LIST=${GOTRUE_URI_ALLOW_LIST:-**}
      - API_EXTERNAL_URL=${API_EXTERNAL_URL}

      # JWT Configuration
      - GOTRUE_JWT_SECRET=${GOTRUE_JWT_SECRET}
      - GOTRUE_JWT_EXP=${GOTRUE_JWT_EXP:-7200}
      - GOTRUE_JWT_ADMIN_GROUP_NAME=supabase_admin

      # Database Configuration
      - GOTRUE_DB_DRIVER=postgres
      - DATABASE_URL=${GOTRUE_DATABASE_URL}
      - PORT=9999

      # Email Configuration
      - GOTRUE_SMTP_HOST=${GOTRUE_SMTP_HOST}
      - GOTRUE_SMTP_PORT=${GOTRUE_SMTP_PORT}
      - GOTRUE_SMTP_USER=${GOTRUE_SMTP_USER}
      - GOTRUE_SMTP_PASS=${GOTRUE_SMTP_PASS}
      - GOTRUE_SMTP_ADMIN_EMAIL=${GOTRUE_SMTP_ADMIN_EMAIL}
      - GOTRUE_SMTP_MAX_FREQUENCY=${GOTRUE_SMTP_MAX_FREQUENCY:-1ns}
      - GOTRUE_RATE_LIMIT_EMAIL_SENT=${GOTRUE_RATE_LIMIT_EMAIL_SENT:-100}
      - GOTRUE_MAILER_AUTOCONFIRM=${GOTRUE_MAILER_AUTOCONFIRM:-true}

      # Email Templates
      - GOTRUE_MAILER_URLPATHS_CONFIRMATION=/gotrue/verify
      - GOTRUE_MAILER_URLPATHS_INVITE=/gotrue/verify
      - GOTRUE_MAILER_URLPATHS_RECOVERY=/gotrue/verify
      - GOTRUE_MAILER_URLPATHS_EMAIL_CHANGE=/gotrue/verify
      - GOTRUE_MAILER_TEMPLATES_MAGIC_LINK=${GOTRUE_MAILER_TEMPLATES_MAGIC_LINK}

      # OAuth Providers
      - GOTRUE_EXTERNAL_GOOGLE_ENABLED=${GOTRUE_EXTERNAL_GOOGLE_ENABLED:-false}
      - GOTRUE_EXTERNAL_GOOGLE_CLIENT_ID=${GOTRUE_EXTERNAL_GOOGLE_CLIENT_ID}
      - GOTRUE_EXTERNAL_GOOGLE_SECRET=${GOTRUE_EXTERNAL_GOOGLE_SECRET}
      - GOTRUE_EXTERNAL_GOOGLE_REDIRECT_URI=${GOTRUE_EXTERNAL_GOOGLE_REDIRECT_URI}

      - GOTRUE_EXTERNAL_GITHUB_ENABLED=${GOTRUE_EXTERNAL_GITHUB_ENABLED:-false}
      - GOTRUE_EXTERNAL_GITHUB_CLIENT_ID=${GOTRUE_EXTERNAL_GITHUB_CLIENT_ID}
      - GOTRUE_EXTERNAL_GITHUB_SECRET=${GOTRUE_EXTERNAL_GITHUB_SECRET}
      - GOTRUE_EXTERNAL_GITHUB_REDIRECT_URI=${GOTRUE_EXTERNAL_GITHUB_REDIRECT_URI}

      - GOTRUE_EXTERNAL_DISCORD_ENABLED=${GOTRUE_EXTERNAL_DISCORD_ENABLED:-false}
      - GOTRUE_EXTERNAL_DISCORD_CLIENT_ID=${GOTRUE_EXTERNAL_DISCORD_CLIENT_ID}
      - GOTRUE_EXTERNAL_DISCORD_SECRET=${GOTRUE_EXTERNAL_DISCORD_SECRET}
      - GOTRUE_EXTERNAL_DISCORD_REDIRECT_URI=${GOTRUE_EXTERNAL_DISCORD_REDIRECT_URI}

      # SAML Configuration
      - GOTRUE_SAML_ENABLED=${GOTRUE_SAML_ENABLED:-false}
      - GOTRUE_SAML_PRIVATE_KEY=${GOTRUE_SAML_PRIVATE_KEY}

  appflowy_cloud:
    restart: on-failure
    image: appflowyinc/appflowy_cloud:${APPFLOWY_CLOUD_VERSION:-latest}
    depends_on:
      gotrue:
        condition: service_healthy
    environment:
      # Core Configuration
      - RUST_LOG=${RUST_LOG:-info}
      - APPFLOWY_ENVIRONMENT=production
      - APPFLOWY_DATABASE_URL=${APPFLOWY_DATABASE_URL}
      - APPFLOWY_REDIS_URI=${APPFLOWY_REDIS_URI}
      - APPFLOWY_WEB_URL=${APPFLOWY_WEB_URL}

      # Authentication Configuration
      - APPFLOWY_GOTRUE_JWT_SECRET=${GOTRUE_JWT_SECRET}
      - APPFLOWY_GOTRUE_JWT_EXP=${GOTRUE_JWT_EXP:-7200}
      - APPFLOWY_GOTRUE_BASE_URL=${APPFLOWY_GOTRUE_BASE_URL}

      # File Storage Configuration
      - APPFLOWY_S3_CREATE_BUCKET=${APPFLOWY_S3_CREATE_BUCKET:-true}
      - APPFLOWY_S3_USE_MINIO=${APPFLOWY_S3_USE_MINIO:-true}
      - APPFLOWY_S3_MINIO_URL=${APPFLOWY_S3_MINIO_URL}
      - APPFLOWY_S3_ACCESS_KEY=${APPFLOWY_S3_ACCESS_KEY}
      - APPFLOWY_S3_SECRET_KEY=${APPFLOWY_S3_SECRET_KEY}
      - APPFLOWY_S3_BUCKET=${APPFLOWY_S3_BUCKET:-appflowy}
      - APPFLOWY_S3_REGION=${APPFLOWY_S3_REGION:-us-east-1}
      - APPFLOWY_S3_PRESIGNED_URL_ENDPOINT=${APPFLOWY_S3_PRESIGNED_URL_ENDPOINT}

      # Email Configuration
      - APPFLOWY_MAILER_SMTP_HOST=${APPFLOWY_MAILER_SMTP_HOST}
      - APPFLOWY_MAILER_SMTP_PORT=${APPFLOWY_MAILER_SMTP_PORT}
      - APPFLOWY_MAILER_SMTP_USERNAME=${APPFLOWY_MAILER_SMTP_USERNAME}
      - APPFLOWY_MAILER_SMTP_EMAIL=${APPFLOWY_MAILER_SMTP_EMAIL}
      - APPFLOWY_MAILER_SMTP_PASSWORD=${APPFLOWY_MAILER_SMTP_PASSWORD}
      - APPFLOWY_MAILER_SMTP_TLS_KIND=${APPFLOWY_MAILER_SMTP_TLS_KIND:-wrapper}

      # Access Control and Performance
      - APPFLOWY_ACCESS_CONTROL=${APPFLOWY_ACCESS_CONTROL:-true}
      - APPFLOWY_DATABASE_MAX_CONNECTIONS=${APPFLOWY_DATABASE_MAX_CONNECTIONS:-40}
      - APPFLOWY_WEBSOCKET_MAILBOX_SIZE=${APPFLOWY_WEBSOCKET_MAILBOX_SIZE:-6000}

      # AI Configuration
      - AI_SERVER_HOST=${AI_SERVER_HOST:-ai}
      - AI_SERVER_PORT=${AI_SERVER_PORT:-5001}
      - AI_OPENAI_API_KEY=${AI_OPENAI_API_KEY}

  admin_frontend:
    restart: on-failure
    image: appflowyinc/admin_frontend:${APPFLOWY_ADMIN_FRONTEND_VERSION:-latest}
    depends_on:
      gotrue:
        condition: service_healthy
      appflowy_cloud:
        condition: service_started
    environment:
      - RUST_LOG=${RUST_LOG:-info}
      - ADMIN_FRONTEND_REDIS_URL=${ADMIN_FRONTEND_REDIS_URL}
      - ADMIN_FRONTEND_GOTRUE_URL=${ADMIN_FRONTEND_GOTRUE_URL}
      - ADMIN_FRONTEND_APPFLOWY_CLOUD_URL=${ADMIN_FRONTEND_APPFLOWY_CLOUD_URL}
      - ADMIN_FRONTEND_PATH_PREFIX=${ADMIN_FRONTEND_PATH_PREFIX:-/console}

  ai:
    restart: on-failure
    image: appflowyinc/appflowy_ai:${APPFLOWY_AI_VERSION:-latest}
    depends_on:
      postgres:
        condition: service_healthy
      appflowy_cloud:
        condition: service_started
    healthcheck:
      test: ["CMD", "curl", "-f", "http://localhost:5001/health"]
      interval: 30s
      timeout: 10s
      retries: 3
      start_period: 40s
    environment:
      # Core AI Configuration
      - AI_SERVER_PORT=${AI_SERVER_PORT:-5001}
      - OPENAI_API_KEY=${AI_OPENAI_API_KEY}
      - DEFAULT_AI_MODEL=${DEFAULT_AI_MODEL:-gpt-4o-mini}
      - DEFAULT_AI_COMPLETION_MODEL=${DEFAULT_AI_COMPLETION_MODEL:-gpt-4o-mini}

      # Azure OpenAI (optional)
      - AZURE_OPENAI_API_KEY=${AZURE_OPENAI_API_KEY}
      - AZURE_OPENAI_ENDPOINT=${AZURE_OPENAI_ENDPOINT}
      - AZURE_OPENAI_API_VERSION=${AZURE_OPENAI_API_VERSION}

      # Database and Cache
      - AI_DATABASE_URL=${APPFLOWY_DATABASE_URL}
      - AI_REDIS_URL=${APPFLOWY_REDIS_URI}

      # File Storage for AI
      - APPFLOWY_S3_ACCESS_KEY=${APPFLOWY_S3_ACCESS_KEY}
      - APPFLOWY_S3_SECRET_KEY=${APPFLOWY_S3_SECRET_KEY}
      - APPFLOWY_S3_BUCKET=${APPFLOWY_S3_BUCKET:-appflowy}
      - APPFLOWY_S3_REGION=${APPFLOWY_S3_REGION:-us-east-1}
      - AI_USE_MINIO=${APPFLOWY_S3_USE_MINIO:-true}
      - AI_MINIO_URL=${APPFLOWY_S3_MINIO_URL}

      # Integration
      - AI_APPFLOWY_HOST=${APPFLOWY_BASE_URL}
      - APPFLOWY_GOTRUE_JWT_SECRET=${GOTRUE_JWT_SECRET}

  appflowy_worker:
    restart: on-failure
    image: appflowyinc/appflowy_worker:${APPFLOWY_WORKER_VERSION:-latest}
    depends_on:
      postgres:
        condition: service_healthy
    environment:
      # Core Configuration
      - RUST_LOG=${RUST_LOG:-info}
      - APPFLOWY_ENVIRONMENT=production
      - APPFLOWY_WORKER_REDIS_URL=${APPFLOWY_WORKER_REDIS_URL}
      - APPFLOWY_WORKER_ENVIRONMENT=production
      - APPFLOWY_WORKER_DATABASE_URL=${APPFLOWY_WORKER_DATABASE_URL}
      - APPFLOWY_WORKER_DATABASE_NAME=${APPFLOWY_WORKER_DATABASE_NAME}
      - APPFLOWY_WORKER_IMPORT_TICK_INTERVAL=${APPFLOWY_WORKER_IMPORT_TICK_INTERVAL:-30}

      # File Storage Configuration
      - APPFLOWY_S3_USE_MINIO=${APPFLOWY_S3_USE_MINIO:-true}
      - APPFLOWY_S3_MINIO_URL=${APPFLOWY_S3_MINIO_URL}
      - APPFLOWY_S3_ACCESS_KEY=${APPFLOWY_S3_ACCESS_KEY}
      - APPFLOWY_S3_SECRET_KEY=${APPFLOWY_S3_SECRET_KEY}
      - APPFLOWY_S3_BUCKET=${APPFLOWY_S3_BUCKET:-appflowy}
      - APPFLOWY_S3_REGION=${APPFLOWY_S3_REGION:-us-east-1}

      # Email Configuration
      - APPFLOWY_MAILER_SMTP_HOST=${APPFLOWY_MAILER_SMTP_HOST}
      - APPFLOWY_MAILER_SMTP_PORT=${APPFLOWY_MAILER_SMTP_PORT}
      - APPFLOWY_MAILER_SMTP_USERNAME=${APPFLOWY_MAILER_SMTP_USERNAME}
      - APPFLOWY_MAILER_SMTP_EMAIL=${APPFLOWY_MAILER_SMTP_EMAIL}
      - APPFLOWY_MAILER_SMTP_PASSWORD=${APPFLOWY_MAILER_SMTP_PASSWORD}
      - APPFLOWY_MAILER_SMTP_TLS_KIND=${APPFLOWY_MAILER_SMTP_TLS_KIND:-wrapper}

  appflowy_web:
    restart: on-failure
    image: appflowyinc/appflowy_web:${APPFLOWY_WEB_VERSION:-latest}
    depends_on:
      - appflowy_cloud
    environment:
      - AF_BASE_URL=${APPFLOWY_BASE_URL}
      - AF_GOTRUE_URL=${APPFLOWY_BASE_URL}/gotrue
      - AF_WS_V2_URL=${APPFLOWY_WEBSOCKET_BASE_URL}

volumes:
  postgres_data:
  minio_data:
[variables]
main_domain = "${domain}"

[config]
env = [
  # =============================================================================
  # 🌐 CORE DOMAIN CONFIGURATION
  # =============================================================================
  "FQDN=${main_domain}",
  "SCHEME=https",
  "WS_SCHEME=wss",
  "APPFLOWY_BASE_URL=https://${main_domain}",
  "APPFLOWY_WEBSOCKET_BASE_URL=wss://${main_domain}/ws/v2",
  "APPFLOWY_WEB_URL=https://${main_domain}",
  "API_EXTERNAL_URL=https://${main_domain}/gotrue",
  "TZ=UTC",

  # Admin Configuration
  "GOTRUE_ADMIN_EMAIL=${email}",
  "GOTRUE_ADMIN_PASSWORD=${password:16}",
  "GOTRUE_DISABLE_SIGNUP=false",

  # =============================================================================
  # 🗄️ DATABASE & CACHE CONFIGURATION
  # =============================================================================
  "POSTGRES_HOST=postgres",
  "POSTGRES_USER=appflowy",
  "POSTGRES_PASSWORD=${password:64}",
  "POSTGRES_PORT=5432",
  "POSTGRES_DB=appflowy",
  "REDIS_HOST=redis",
  "REDIS_PORT=6379",

  # =============================================================================
  # 🔐 GOTRUE AUTHENTICATION CONFIGURATION
  # =============================================================================

  # JWT Configuration
  "GOTRUE_JWT_SECRET=${password:64}",
  "GOTRUE_JWT_EXP=7200",
  "GOTRUE_JWT_ADMIN_GROUP_NAME=supabase_admin",

  # Database Configuration
  "GOTRUE_DB_DRIVER=postgres",
  "GOTRUE_DATABASE_URL=postgres://appflowy:${POSTGRES_PASSWORD}@postgres:5432/appflowy?search_path=auth",
  "PORT=9999",

  # Site Configuration
  "GOTRUE_SITE_URL=appflowy-flutter://",
  "GOTRUE_URI_ALLOW_LIST=**",

  # Email Configuration (SMTP - Configure for production)
  "GOTRUE_SMTP_HOST=",
  "GOTRUE_SMTP_PORT=465",
  "GOTRUE_SMTP_USER=",
  "GOTRUE_SMTP_PASS=",
  "GOTRUE_SMTP_ADMIN_EMAIL=${GOTRUE_ADMIN_EMAIL}",
  "GOTRUE_SMTP_MAX_FREQUENCY=1ns",
  "GOTRUE_RATE_LIMIT_EMAIL_SENT=100",
  "GOTRUE_MAILER_AUTOCONFIRM=true",

  # Email Templates
  "GOTRUE_MAILER_URLPATHS_CONFIRMATION=/gotrue/verify",
  "GOTRUE_MAILER_URLPATHS_INVITE=/gotrue/verify",
  "GOTRUE_MAILER_URLPATHS_RECOVERY=/gotrue/verify",
  "GOTRUE_MAILER_URLPATHS_EMAIL_CHANGE=/gotrue/verify",
  "GOTRUE_MAILER_TEMPLATES_MAGIC_LINK=",

  # OAuth Providers (Configure as needed)
  "GOTRUE_EXTERNAL_GOOGLE_ENABLED=false",
  "GOTRUE_EXTERNAL_GOOGLE_CLIENT_ID=",
  "GOTRUE_EXTERNAL_GOOGLE_SECRET=",
  "GOTRUE_EXTERNAL_GOOGLE_REDIRECT_URI=https://${main_domain}/gotrue/callback",

  "GOTRUE_EXTERNAL_GITHUB_ENABLED=false",
  "GOTRUE_EXTERNAL_GITHUB_CLIENT_ID=",
  "GOTRUE_EXTERNAL_GITHUB_SECRET=",
  "GOTRUE_EXTERNAL_GITHUB_REDIRECT_URI=https://${main_domain}/gotrue/callback",

  "GOTRUE_EXTERNAL_DISCORD_ENABLED=false",
  "GOTRUE_EXTERNAL_DISCORD_CLIENT_ID=",
  "GOTRUE_EXTERNAL_DISCORD_SECRET=",
  "GOTRUE_EXTERNAL_DISCORD_REDIRECT_URI=https://${main_domain}/gotrue/callback",

  # SAML Configuration
  "GOTRUE_SAML_ENABLED=false",
  "GOTRUE_SAML_PRIVATE_KEY=",

  # =============================================================================
  # ☁️ APPFLOWY CLOUD SERVICE CONFIGURATION
  # =============================================================================
  # Core Configuration
  "RUST_LOG=info",
  "APPFLOWY_ENVIRONMENT=production",
  "APPFLOWY_DATABASE_URL=postgres://appflowy:${POSTGRES_PASSWORD}@postgres:5432/appflowy",
  "APPFLOWY_REDIS_URI=redis://redis:6379",

  # Authentication Integration
  "APPFLOWY_GOTRUE_JWT_SECRET=${GOTRUE_JWT_SECRET}",
  "APPFLOWY_GOTRUE_JWT_EXP=7200",
  "APPFLOWY_GOTRUE_BASE_URL=http://gotrue:9999",

  # Access Control and Performance
  "APPFLOWY_ACCESS_CONTROL=true",
  "APPFLOWY_DATABASE_MAX_CONNECTIONS=40",
  "APPFLOWY_WEBSOCKET_MAILBOX_SIZE=6000",

  # Email Configuration (SMTP)
  "APPFLOWY_MAILER_SMTP_HOST=",
  "APPFLOWY_MAILER_SMTP_PORT=465",
  "APPFLOWY_MAILER_SMTP_USERNAME=",
  "APPFLOWY_MAILER_SMTP_EMAIL=",
  "APPFLOWY_MAILER_SMTP_PASSWORD=",
  "APPFLOWY_MAILER_SMTP_TLS_KIND=wrapper",

  # =============================================================================
  # 💾 FILE STORAGE CONFIGURATION (MinIO/S3)
  # =============================================================================
  # MinIO Configuration
  "MINIO_HOST=minio",
  "MINIO_PORT=9000",
  "APPFLOWY_S3_USE_MINIO=true",
  "APPFLOWY_S3_CREATE_BUCKET=true",
  "APPFLOWY_S3_MINIO_URL=http://minio:9000",

  # Storage Credentials
  "APPFLOWY_S3_ACCESS_KEY=${password:16}",
  "APPFLOWY_S3_SECRET_KEY=${password:32}",

  # Storage Configuration
  "APPFLOWY_S3_BUCKET=appflowy",
  "APPFLOWY_S3_REGION=us-east-1",
  "APPFLOWY_S3_PRESIGNED_URL_ENDPOINT=https://${main_domain}/minio-api",

  # AWS S3 Configuration (Alternative to MinIO)
  # "APPFLOWY_S3_USE_MINIO=false",
  # "APPFLOWY_S3_REGION=us-east-1",

  # =============================================================================
  # 🎛️ ADMIN FRONTEND CONFIGURATION
  # =============================================================================
  "ADMIN_FRONTEND_REDIS_URL=redis://redis:6379",
  "ADMIN_FRONTEND_GOTRUE_URL=http://gotrue:9999",
  "ADMIN_FRONTEND_APPFLOWY_CLOUD_URL=http://appflowy_cloud:8000",
  "ADMIN_FRONTEND_PATH_PREFIX=/console",

  # =============================================================================
  # 🤖 AI FEATURES CONFIGURATION (Optional)
  # =============================================================================
  # OpenAI Configuration
  "AI_OPENAI_API_KEY=",
  "DEFAULT_AI_MODEL=gpt-4o-mini",
  "DEFAULT_AI_COMPLETION_MODEL=gpt-4o-mini",

  # Azure OpenAI (Alternative)
  "AZURE_OPENAI_API_KEY=",
  "AZURE_OPENAI_ENDPOINT=",
  "AZURE_OPENAI_API_VERSION=",

  # AI Service Configuration
  "AI_SERVER_HOST=ai",
  "AI_SERVER_PORT=5001",
  "AI_DATABASE_URL=postgresql+psycopg://appflowy:${POSTGRES_PASSWORD}@postgres:5432/appflowy",
  "AI_REDIS_URL=redis://redis:6379",
  "AI_USE_MINIO=true",
  "AI_MINIO_URL=http://minio:9000",
  "AI_APPFLOWY_HOST=https://${main_domain}",

  # Embedding Configuration
  "APPFLOWY_EMBEDDING_CHUNK_SIZE=2000",
  "APPFLOWY_EMBEDDING_CHUNK_OVERLAP=200",

  # =============================================================================
  # ⚙️ WORKER SERVICES CONFIGURATION
  # =============================================================================
  # AppFlowy Worker
  "APPFLOWY_WORKER_REDIS_URL=redis://redis:6379",
  "APPFLOWY_WORKER_ENVIRONMENT=production",
  "APPFLOWY_WORKER_DATABASE_URL=postgres://appflowy:${POSTGRES_PASSWORD}@postgres:5432/appflowy",
  "APPFLOWY_WORKER_DATABASE_NAME=appflowy",
  "APPFLOWY_WORKER_IMPORT_TICK_INTERVAL=30",

  # Indexer Configuration
  "APPFLOWY_INDEXER_ENABLED=true",
  "APPFLOWY_INDEXER_DATABASE_URL=postgres://appflowy:${POSTGRES_PASSWORD}@postgres:5432/appflowy",
  "APPFLOWY_INDEXER_REDIS_URL=redis://redis:6379",
  "APPFLOWY_INDEXER_EMBEDDING_BUFFER_SIZE=5000",

  # Collaboration Service
  "APPFLOWY_COLLABORATE_MULTI_THREAD=false",
  "APPFLOWY_COLLABORATE_REMOVE_BATCH_SIZE=100",

  # =============================================================================
  # 🌐 NGINX CONFIGURATION
  # =============================================================================
  "NGINX_PORT=80",
  "NGINX_TLS_PORT=443",

  # =============================================================================
  # 🛠️ VERSION TAGS (Easily Configurable)
  # =============================================================================
  "GOTRUE_VERSION=latest",
  "APPFLOWY_CLOUD_VERSION=latest",
  "APPFLOWY_ADMIN_FRONTEND_VERSION=latest",
  "APPFLOWY_AI_VERSION=latest",
  "APPFLOWY_WORKER_VERSION=latest",
  "APPFLOWY_WEB_VERSION=latest",
]

[[config.domains]]
serviceName = "nginx"
port = 80
host = "${main_domain}"


[[config.mounts]]
filePath = "/nginx/nginx.conf"
content = """# Minimal nginx configuration for AppFlowy-Cloud
# Self Hosted AppFlowy Cloud user should alter this file to suit their needs,
# or use the appflowy.site.conf in external_proxy_config/nginx if they are using
# an external proxy.

events {
    worker_connections 1024;
}

http {
    # docker dns resolver
    resolver 127.0.0.11 valid=10s;
    #error_log /var/log/nginx/error.log debug;

    map $http_upgrade $connection_upgrade {
        default upgrade;
        '' close;
    }

    map $http_origin $cors_origin {
        # AppFlowy Web origin
        "~^http://localhost:3000$" $http_origin;
        default "null";
    }

    server {
        listen 8080;

        # https://github.com/nginxinc/nginx-prometheus-exporter
        location = /stub_status {
            stub_status;
        }
    }


    server {

        listen 80;
        client_max_body_size 10M;

        underscores_in_headers on;
        set $appflowy_cloud_backend "http://appflowy_cloud:8000";
        set $gotrue_backend "http://gotrue:9999";
        set $admin_frontend_backend "http://admin_frontend:3000";
        set $appflowy_web_backend "http://appflowy_web:80";
        set $minio_backend "http://minio:9001";
        set $minio_api_backend "http://minio:9000";
        # Host name for minio, used internally within docker compose
        set $minio_internal_host "minio:9000";
        set $pgadmin_backend "http://pgadmin:80";

        # GoTrue
        location /gotrue/ {
            if ($request_method = 'OPTIONS') {
                add_header 'Access-Control-Allow-Origin' $cors_origin always;
                add_header 'Access-Control-Allow-Credentials' 'true' always;
                add_header 'Access-Control-Allow-Headers' '*' always;
                add_header 'Access-Control-Allow-Methods' 'GET, POST, PUT, DELETE, PATCH, OPTIONS' always;
                add_header 'Access-Control-Max-Age' 3600 always;
                add_header 'Content-Type' 'text/plain charset=UTF-8' always;
                add_header 'Content-Length' 0 always;
                return 204;
            }

            proxy_pass $gotrue_backend;

            rewrite ^/gotrue(/.*)$ $1 break;

            # Allow headers like redirect_to to be handed over to the gotrue
            # for correct redirecting
            proxy_set_header Host $http_host;
            proxy_pass_request_headers on;
        }

        # WebSocket
        location /ws {
            proxy_pass $appflowy_cloud_backend;

            proxy_http_version 1.1;
            proxy_set_header Upgrade $http_upgrade;
            proxy_set_header Connection "Upgrade";
            proxy_set_header Host $host;
            proxy_read_timeout 86400s;
        }

        location /api {
            proxy_pass $appflowy_cloud_backend;
            proxy_set_header X-Request-Id $request_id;
            proxy_set_header Host $http_host;

            # Set CORS headers for other requests
            if ($request_method = 'OPTIONS') {
                add_header 'Access-Control-Allow-Origin' $cors_origin always;
                add_header 'Access-Control-Allow-Methods' 'GET, POST, PUT, DELETE, PATCH, OPTIONS' always;
                add_header 'Access-Control-Allow-Headers' 'Content-Type, Authorization, Accept, Client-Version, Device-Id' always;
                add_header 'Access-Control-Max-Age' 3600 always;
                return 204;
            }

            add_header 'Access-Control-Allow-Origin' $cors_origin always;
            add_header 'Access-Control-Allow-Methods' 'GET, POST, PUT, DELETE, PATCH, OPTIONS' always;
            add_header 'Access-Control-Allow-Headers' 'Content-Type, Authorization, Accept, Client-Version, Device-Id' always;
            add_header 'Access-Control-Max-Age' 3600 always;

            location ~* ^/api/workspace/([a-zA-Z0-9_-]+)/publish$ {
                proxy_pass $appflowy_cloud_backend;
                proxy_request_buffering off;
                client_max_body_size 256M;
                if ($request_method = 'OPTIONS') {
                    add_header 'Access-Control-Allow-Origin' $cors_origin always;
                    add_header 'Access-Control-Allow-Methods' 'GET, POST, PUT, DELETE, PATCH, OPTIONS' always;
                    add_header 'Access-Control-Allow-Headers' 'Content-Type, Authorization, Accept, Client-Version, Device-Id' always;
                    add_header 'Access-Control-Max-Age' 3600 always;
                    return 204;
                }

                add_header 'Access-Control-Allow-Origin' $cors_origin always;
                add_header 'Access-Control-Allow-Methods' 'GET, POST, PUT, DELETE, PATCH, OPTIONS' always;
                add_header 'Access-Control-Allow-Headers' 'Content-Type, Authorization, Accept, Client-Version, Device-Id' always;
                add_header 'Access-Control-Max-Age' 3600 always;
            }

            # AppFlowy-Cloud
            location /api/chat {
                proxy_pass $appflowy_cloud_backend;

                proxy_http_version 1.1;
                proxy_set_header Connection "";
                chunked_transfer_encoding on;
                proxy_buffering off;
                proxy_cache off;

                proxy_read_timeout 600s;
                proxy_connect_timeout 600s;
                proxy_send_timeout 600s;
            }

            location /api/import {
                proxy_pass $appflowy_cloud_backend;

                # Set headers
                proxy_set_header X-Request-Id $request_id;
                proxy_set_header Host $http_host;

                # Handle CORS
                add_header 'Access-Control-Allow-Origin' $cors_origin always;
                add_header 'Access-Control-Allow-Methods' 'GET, POST, PUT, DELETE, PATCH, OPTIONS' always;
                add_header 'Access-Control-Allow-Headers' 'Content-Type, Authorization, Accept, Device-Id' always;
                add_header 'Access-Control-Max-Age' 3600 always;

                # Timeouts
                proxy_read_timeout 600s;
                proxy_connect_timeout 600s;
                proxy_send_timeout 600s;

                # Disable buffering for large file uploads
                proxy_request_buffering off;
                proxy_buffering off;
                proxy_cache off;
                client_max_body_size 2G;
            }
        }

        # Minio Web UI
        # Derive from: https://min.io/docs/minio/linux/integrations/setup-nginx-proxy-with-minio.html
        # Optional Module, comment this section if you did not deploy minio in docker-compose.yml
        # This endpoint is meant to be used for the MinIO Web UI, accessible via the admin portal
        location /minio/ {
            proxy_pass $minio_backend;

            rewrite ^/minio/(.*) /$1 break;
            proxy_set_header Host $http_host;
            proxy_set_header X-Real-IP $remote_addr;
            proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
            proxy_set_header X-Forwarded-Proto $scheme;
            proxy_set_header X-NginX-Proxy true;

            ## This is necessary to pass the correct IP to be hashed
            real_ip_header X-Real-IP;

            proxy_connect_timeout 300s;

            ## To support websockets in MinIO versions released after January 2023
            proxy_http_version 1.1;
            proxy_set_header Upgrade $http_upgrade;
            proxy_set_header Connection "upgrade";
            # Some environments may encounter CORS errors (Kubernetes + Nginx Ingress)
            # Uncomment the following line to set the Origin request to an empty string
            # proxy_set_header Origin '';

            chunked_transfer_encoding off;
        }

        # Optional Module, comment this section if you did not deploy minio in docker-compose.yml
        # This is used for presigned url, which is needs to be exposed to the AppFlowy client application.
        location /minio-api/ {
            proxy_pass $minio_api_backend;

            # Set the host to internal host because the presigned url was signed against the internal host
            proxy_set_header Host $minio_internal_host;
            proxy_set_header X-Real-IP $remote_addr;
            proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
            proxy_set_header X-Forwarded-Proto $scheme;

            rewrite ^/minio-api/(.*) /$1 break;

            proxy_connect_timeout 300s;
            # Default is HTTP/1, keepalive is only enabled in HTTP/1.1
            proxy_http_version 1.1;
            proxy_set_header Connection "";
            chunked_transfer_encoding off;
        }

        # PgAdmin
        # Optional Module, comment this section if you did not deploy pgadmin in docker-compose.yml
        location /pgadmin/ {
            set $pgadmin pgadmin;
            proxy_pass $pgadmin_backend;

            proxy_set_header X-Script-Name /pgadmin;
            proxy_set_header X-Scheme $scheme;
            proxy_set_header Host $host;
            proxy_redirect off;
        }

        # Admin Frontend
        # Optional Module, comment this section if you did not deploy admin_frontend in docker-compose.yml
        location /console {
            proxy_pass $admin_frontend_backend;

            proxy_set_header X-Scheme $scheme;
            proxy_set_header Host $host;
        }

        # AppFlowy Web
        location / {
            proxy_pass $appflowy_web_backend;
            proxy_set_header X-Scheme $scheme;
            proxy_set_header Host $host;
        }
    }

}
"""

Base64

To import this template in Dokploy: create a Compose service → AdvancedBase64 import and paste the content below:

{
  "compose": "services:\n  nginx:\n    restart: on-failure\n    image: nginx\n    volumes:\n      - ../files/nginx/nginx.conf:/etc/nginx/nginx.conf\n\n  minio:\n    restart: on-failure\n    image: minio/minio\n    environment:\n      - MINIO_BROWSER_REDIRECT_URL=${APPFLOWY_BASE_URL}/minio\n      - MINIO_ROOT_USER=${APPFLOWY_S3_ACCESS_KEY:-minioadmin}\n      - MINIO_ROOT_PASSWORD=${APPFLOWY_S3_SECRET_KEY:-minioadmin}\n    command: server /data --console-address \":9001\"\n    volumes:\n      - minio_data:/data\n\n  postgres:\n    restart: on-failure\n    image: pgvector/pgvector:pg16\n    environment:\n      - POSTGRES_USER=${POSTGRES_USER:-postgres}\n      - POSTGRES_DB=${POSTGRES_DB:-postgres}\n      - POSTGRES_PASSWORD=${POSTGRES_PASSWORD:-password}\n      - POSTGRES_HOST=${POSTGRES_HOST:-postgres}\n    healthcheck:\n      test: [\"CMD\", \"pg_isready\", \"-U\", \"${POSTGRES_USER}\", \"-d\", \"${POSTGRES_DB}\"]\n      interval: 5s\n      timeout: 5s\n      retries: 12\n    volumes:\n      - postgres_data:/var/lib/postgresql/data\n      - ./files/volumes/postgres/:/docker-entrypoint-initdb.d/\n\n  redis:\n    restart: on-failure\n    image: redis\n\n  gotrue:\n    restart: on-failure\n    image: appflowyinc/gotrue:${GOTRUE_VERSION:-latest}\n    depends_on:\n      postgres:\n        condition: service_healthy\n    healthcheck:\n      test: [\"CMD\", \"curl\", \"--fail\", \"http://127.0.0.1:9999/health\"]\n      interval: 5s\n      timeout: 5s\n      retries: 12\n    environment:\n      # Admin Configuration\n      - GOTRUE_ADMIN_EMAIL=${GOTRUE_ADMIN_EMAIL}\n      - GOTRUE_ADMIN_PASSWORD=${GOTRUE_ADMIN_PASSWORD}\n      - GOTRUE_DISABLE_SIGNUP=${GOTRUE_DISABLE_SIGNUP:-false}\n      \n      # Site Configuration\n      - GOTRUE_SITE_URL=appflowy-flutter://\n      - GOTRUE_URI_ALLOW_LIST=${GOTRUE_URI_ALLOW_LIST:-**}\n      - API_EXTERNAL_URL=${API_EXTERNAL_URL}\n      \n      # JWT Configuration\n      - GOTRUE_JWT_SECRET=${GOTRUE_JWT_SECRET}\n      - GOTRUE_JWT_EXP=${GOTRUE_JWT_EXP:-7200}\n      - GOTRUE_JWT_ADMIN_GROUP_NAME=supabase_admin\n      \n      # Database Configuration\n      - GOTRUE_DB_DRIVER=postgres\n      - DATABASE_URL=${GOTRUE_DATABASE_URL}\n      - PORT=9999\n      \n      # Email Configuration\n      - GOTRUE_SMTP_HOST=${GOTRUE_SMTP_HOST}\n      - GOTRUE_SMTP_PORT=${GOTRUE_SMTP_PORT}\n      - GOTRUE_SMTP_USER=${GOTRUE_SMTP_USER}\n      - GOTRUE_SMTP_PASS=${GOTRUE_SMTP_PASS}\n      - GOTRUE_SMTP_ADMIN_EMAIL=${GOTRUE_SMTP_ADMIN_EMAIL}\n      - GOTRUE_SMTP_MAX_FREQUENCY=${GOTRUE_SMTP_MAX_FREQUENCY:-1ns}\n      - GOTRUE_RATE_LIMIT_EMAIL_SENT=${GOTRUE_RATE_LIMIT_EMAIL_SENT:-100}\n      - GOTRUE_MAILER_AUTOCONFIRM=${GOTRUE_MAILER_AUTOCONFIRM:-true}\n      \n      # Email Templates\n      - GOTRUE_MAILER_URLPATHS_CONFIRMATION=/gotrue/verify\n      - GOTRUE_MAILER_URLPATHS_INVITE=/gotrue/verify\n      - GOTRUE_MAILER_URLPATHS_RECOVERY=/gotrue/verify\n      - GOTRUE_MAILER_URLPATHS_EMAIL_CHANGE=/gotrue/verify\n      - GOTRUE_MAILER_TEMPLATES_MAGIC_LINK=${GOTRUE_MAILER_TEMPLATES_MAGIC_LINK}\n      \n      # OAuth Providers\n      - GOTRUE_EXTERNAL_GOOGLE_ENABLED=${GOTRUE_EXTERNAL_GOOGLE_ENABLED:-false}\n      - GOTRUE_EXTERNAL_GOOGLE_CLIENT_ID=${GOTRUE_EXTERNAL_GOOGLE_CLIENT_ID}\n      - GOTRUE_EXTERNAL_GOOGLE_SECRET=${GOTRUE_EXTERNAL_GOOGLE_SECRET}\n      - GOTRUE_EXTERNAL_GOOGLE_REDIRECT_URI=${GOTRUE_EXTERNAL_GOOGLE_REDIRECT_URI}\n      \n      - GOTRUE_EXTERNAL_GITHUB_ENABLED=${GOTRUE_EXTERNAL_GITHUB_ENABLED:-false}\n      - GOTRUE_EXTERNAL_GITHUB_CLIENT_ID=${GOTRUE_EXTERNAL_GITHUB_CLIENT_ID}\n      - GOTRUE_EXTERNAL_GITHUB_SECRET=${GOTRUE_EXTERNAL_GITHUB_SECRET}\n      - GOTRUE_EXTERNAL_GITHUB_REDIRECT_URI=${GOTRUE_EXTERNAL_GITHUB_REDIRECT_URI}\n      \n      - GOTRUE_EXTERNAL_DISCORD_ENABLED=${GOTRUE_EXTERNAL_DISCORD_ENABLED:-false}\n      - GOTRUE_EXTERNAL_DISCORD_CLIENT_ID=${GOTRUE_EXTERNAL_DISCORD_CLIENT_ID}\n      - GOTRUE_EXTERNAL_DISCORD_SECRET=${GOTRUE_EXTERNAL_DISCORD_SECRET}\n      - GOTRUE_EXTERNAL_DISCORD_REDIRECT_URI=${GOTRUE_EXTERNAL_DISCORD_REDIRECT_URI}\n      \n      # SAML Configuration\n      - GOTRUE_SAML_ENABLED=${GOTRUE_SAML_ENABLED:-false}\n      - GOTRUE_SAML_PRIVATE_KEY=${GOTRUE_SAML_PRIVATE_KEY}\n\n  appflowy_cloud:\n    restart: on-failure\n    image: appflowyinc/appflowy_cloud:${APPFLOWY_CLOUD_VERSION:-latest}\n    depends_on:\n      gotrue:\n        condition: service_healthy\n    environment:\n      # Core Configuration\n      - RUST_LOG=${RUST_LOG:-info}\n      - APPFLOWY_ENVIRONMENT=production\n      - APPFLOWY_DATABASE_URL=${APPFLOWY_DATABASE_URL}\n      - APPFLOWY_REDIS_URI=${APPFLOWY_REDIS_URI}\n      - APPFLOWY_WEB_URL=${APPFLOWY_WEB_URL}\n      \n      # Authentication Configuration\n      - APPFLOWY_GOTRUE_JWT_SECRET=${GOTRUE_JWT_SECRET}\n      - APPFLOWY_GOTRUE_JWT_EXP=${GOTRUE_JWT_EXP:-7200}\n      - APPFLOWY_GOTRUE_BASE_URL=${APPFLOWY_GOTRUE_BASE_URL}\n      \n      # File Storage Configuration\n      - APPFLOWY_S3_CREATE_BUCKET=${APPFLOWY_S3_CREATE_BUCKET:-true}\n      - APPFLOWY_S3_USE_MINIO=${APPFLOWY_S3_USE_MINIO:-true}\n      - APPFLOWY_S3_MINIO_URL=${APPFLOWY_S3_MINIO_URL}\n      - APPFLOWY_S3_ACCESS_KEY=${APPFLOWY_S3_ACCESS_KEY}\n      - APPFLOWY_S3_SECRET_KEY=${APPFLOWY_S3_SECRET_KEY}\n      - APPFLOWY_S3_BUCKET=${APPFLOWY_S3_BUCKET:-appflowy}\n      - APPFLOWY_S3_REGION=${APPFLOWY_S3_REGION:-us-east-1}\n      - APPFLOWY_S3_PRESIGNED_URL_ENDPOINT=${APPFLOWY_S3_PRESIGNED_URL_ENDPOINT}\n      \n      # Email Configuration\n      - APPFLOWY_MAILER_SMTP_HOST=${APPFLOWY_MAILER_SMTP_HOST}\n      - APPFLOWY_MAILER_SMTP_PORT=${APPFLOWY_MAILER_SMTP_PORT}\n      - APPFLOWY_MAILER_SMTP_USERNAME=${APPFLOWY_MAILER_SMTP_USERNAME}\n      - APPFLOWY_MAILER_SMTP_EMAIL=${APPFLOWY_MAILER_SMTP_EMAIL}\n      - APPFLOWY_MAILER_SMTP_PASSWORD=${APPFLOWY_MAILER_SMTP_PASSWORD}\n      - APPFLOWY_MAILER_SMTP_TLS_KIND=${APPFLOWY_MAILER_SMTP_TLS_KIND:-wrapper}\n      \n      # Access Control and Performance\n      - APPFLOWY_ACCESS_CONTROL=${APPFLOWY_ACCESS_CONTROL:-true}\n      - APPFLOWY_DATABASE_MAX_CONNECTIONS=${APPFLOWY_DATABASE_MAX_CONNECTIONS:-40}\n      - APPFLOWY_WEBSOCKET_MAILBOX_SIZE=${APPFLOWY_WEBSOCKET_MAILBOX_SIZE:-6000}\n      \n      # AI Configuration\n      - AI_SERVER_HOST=${AI_SERVER_HOST:-ai}\n      - AI_SERVER_PORT=${AI_SERVER_PORT:-5001}\n      - AI_OPENAI_API_KEY=${AI_OPENAI_API_KEY}\n\n  admin_frontend:\n    restart: on-failure\n    image: appflowyinc/admin_frontend:${APPFLOWY_ADMIN_FRONTEND_VERSION:-latest}\n    depends_on:\n      gotrue:\n        condition: service_healthy\n      appflowy_cloud:\n        condition: service_started\n    environment:\n      - RUST_LOG=${RUST_LOG:-info}\n      - ADMIN_FRONTEND_REDIS_URL=${ADMIN_FRONTEND_REDIS_URL}\n      - ADMIN_FRONTEND_GOTRUE_URL=${ADMIN_FRONTEND_GOTRUE_URL}\n      - ADMIN_FRONTEND_APPFLOWY_CLOUD_URL=${ADMIN_FRONTEND_APPFLOWY_CLOUD_URL}\n      - ADMIN_FRONTEND_PATH_PREFIX=${ADMIN_FRONTEND_PATH_PREFIX:-/console}\n\n  ai:\n    restart: on-failure\n    image: appflowyinc/appflowy_ai:${APPFLOWY_AI_VERSION:-latest}\n    depends_on:\n      postgres:\n        condition: service_healthy\n      appflowy_cloud:\n        condition: service_started\n    healthcheck:\n      test: [\"CMD\", \"curl\", \"-f\", \"http://localhost:5001/health\"]\n      interval: 30s\n      timeout: 10s\n      retries: 3\n      start_period: 40s\n    environment:\n      # Core AI Configuration\n      - AI_SERVER_PORT=${AI_SERVER_PORT:-5001}\n      - OPENAI_API_KEY=${AI_OPENAI_API_KEY}\n      - DEFAULT_AI_MODEL=${DEFAULT_AI_MODEL:-gpt-4o-mini}\n      - DEFAULT_AI_COMPLETION_MODEL=${DEFAULT_AI_COMPLETION_MODEL:-gpt-4o-mini}\n      \n      # Azure OpenAI (optional)\n      - AZURE_OPENAI_API_KEY=${AZURE_OPENAI_API_KEY}\n      - AZURE_OPENAI_ENDPOINT=${AZURE_OPENAI_ENDPOINT}\n      - AZURE_OPENAI_API_VERSION=${AZURE_OPENAI_API_VERSION}\n      \n      # Database and Cache\n      - AI_DATABASE_URL=${APPFLOWY_DATABASE_URL}\n      - AI_REDIS_URL=${APPFLOWY_REDIS_URI}\n      \n      # File Storage for AI\n      - APPFLOWY_S3_ACCESS_KEY=${APPFLOWY_S3_ACCESS_KEY}\n      - APPFLOWY_S3_SECRET_KEY=${APPFLOWY_S3_SECRET_KEY}\n      - APPFLOWY_S3_BUCKET=${APPFLOWY_S3_BUCKET:-appflowy}\n      - APPFLOWY_S3_REGION=${APPFLOWY_S3_REGION:-us-east-1}\n      - AI_USE_MINIO=${APPFLOWY_S3_USE_MINIO:-true}\n      - AI_MINIO_URL=${APPFLOWY_S3_MINIO_URL}\n      \n      # Integration\n      - AI_APPFLOWY_HOST=${APPFLOWY_BASE_URL}\n      - APPFLOWY_GOTRUE_JWT_SECRET=${GOTRUE_JWT_SECRET}\n\n  appflowy_worker:\n    restart: on-failure\n    image: appflowyinc/appflowy_worker:${APPFLOWY_WORKER_VERSION:-latest}\n    depends_on:\n      postgres:\n        condition: service_healthy\n    environment:\n      # Core Configuration\n      - RUST_LOG=${RUST_LOG:-info}\n      - APPFLOWY_ENVIRONMENT=production\n      - APPFLOWY_WORKER_REDIS_URL=${APPFLOWY_WORKER_REDIS_URL}\n      - APPFLOWY_WORKER_ENVIRONMENT=production\n      - APPFLOWY_WORKER_DATABASE_URL=${APPFLOWY_WORKER_DATABASE_URL}\n      - APPFLOWY_WORKER_DATABASE_NAME=${APPFLOWY_WORKER_DATABASE_NAME}\n      - APPFLOWY_WORKER_IMPORT_TICK_INTERVAL=${APPFLOWY_WORKER_IMPORT_TICK_INTERVAL:-30}\n      \n      # File Storage Configuration\n      - APPFLOWY_S3_USE_MINIO=${APPFLOWY_S3_USE_MINIO:-true}\n      - APPFLOWY_S3_MINIO_URL=${APPFLOWY_S3_MINIO_URL}\n      - APPFLOWY_S3_ACCESS_KEY=${APPFLOWY_S3_ACCESS_KEY}\n      - APPFLOWY_S3_SECRET_KEY=${APPFLOWY_S3_SECRET_KEY}\n      - APPFLOWY_S3_BUCKET=${APPFLOWY_S3_BUCKET:-appflowy}\n      - APPFLOWY_S3_REGION=${APPFLOWY_S3_REGION:-us-east-1}\n      \n      # Email Configuration\n      - APPFLOWY_MAILER_SMTP_HOST=${APPFLOWY_MAILER_SMTP_HOST}\n      - APPFLOWY_MAILER_SMTP_PORT=${APPFLOWY_MAILER_SMTP_PORT}\n      - APPFLOWY_MAILER_SMTP_USERNAME=${APPFLOWY_MAILER_SMTP_USERNAME}\n      - APPFLOWY_MAILER_SMTP_EMAIL=${APPFLOWY_MAILER_SMTP_EMAIL}\n      - APPFLOWY_MAILER_SMTP_PASSWORD=${APPFLOWY_MAILER_SMTP_PASSWORD}\n      - APPFLOWY_MAILER_SMTP_TLS_KIND=${APPFLOWY_MAILER_SMTP_TLS_KIND:-wrapper}\n\n  appflowy_web:\n    restart: on-failure\n    image: appflowyinc/appflowy_web:${APPFLOWY_WEB_VERSION:-latest}\n    depends_on:\n      - appflowy_cloud\n    environment:\n      - AF_BASE_URL=${APPFLOWY_BASE_URL}\n      - AF_GOTRUE_URL=${APPFLOWY_BASE_URL}/gotrue\n      - AF_WS_V2_URL=${APPFLOWY_WEBSOCKET_BASE_URL}\n\nvolumes:\n  postgres_data:\n  minio_data:\n",
  "config": "[variables]\nmain_domain = \"${domain}\"\n\n[config]\nenv = [\n  # =============================================================================\n  # 🌐 CORE DOMAIN CONFIGURATION\n  # =============================================================================\n  \"FQDN=${main_domain}\",\n  \"SCHEME=https\",\n  \"WS_SCHEME=wss\",\n  \"APPFLOWY_BASE_URL=https://${main_domain}\",\n  \"APPFLOWY_WEBSOCKET_BASE_URL=wss://${main_domain}/ws/v2\",\n  \"APPFLOWY_WEB_URL=https://${main_domain}\",\n  \"API_EXTERNAL_URL=https://${main_domain}/gotrue\",\n  \"TZ=UTC\",\n\n  # Admin Configuration\n  \"GOTRUE_ADMIN_EMAIL=${email}\",\n  \"GOTRUE_ADMIN_PASSWORD=${password:16}\",\n  \"GOTRUE_DISABLE_SIGNUP=false\",\n\n  # =============================================================================\n  # 🗄️ DATABASE & CACHE CONFIGURATION\n  # =============================================================================\n  \"POSTGRES_HOST=postgres\",\n  \"POSTGRES_USER=appflowy\",\n  \"POSTGRES_PASSWORD=${password:64}\",\n  \"POSTGRES_PORT=5432\",\n  \"POSTGRES_DB=appflowy\",\n  \"REDIS_HOST=redis\",\n  \"REDIS_PORT=6379\",\n\n  # =============================================================================\n  # 🔐 GOTRUE AUTHENTICATION CONFIGURATION\n  # =============================================================================\n\n  # JWT Configuration\n  \"GOTRUE_JWT_SECRET=${password:64}\",\n  \"GOTRUE_JWT_EXP=7200\",\n  \"GOTRUE_JWT_ADMIN_GROUP_NAME=supabase_admin\",\n\n  # Database Configuration\n  \"GOTRUE_DB_DRIVER=postgres\",\n  \"GOTRUE_DATABASE_URL=postgres://appflowy:${POSTGRES_PASSWORD}@postgres:5432/appflowy?search_path=auth\",\n  \"PORT=9999\",\n\n  # Site Configuration\n  \"GOTRUE_SITE_URL=appflowy-flutter://\",\n  \"GOTRUE_URI_ALLOW_LIST=**\",\n\n  # Email Configuration (SMTP - Configure for production)\n  \"GOTRUE_SMTP_HOST=\",\n  \"GOTRUE_SMTP_PORT=465\",\n  \"GOTRUE_SMTP_USER=\",\n  \"GOTRUE_SMTP_PASS=\",\n  \"GOTRUE_SMTP_ADMIN_EMAIL=${GOTRUE_ADMIN_EMAIL}\",\n  \"GOTRUE_SMTP_MAX_FREQUENCY=1ns\",\n  \"GOTRUE_RATE_LIMIT_EMAIL_SENT=100\",\n  \"GOTRUE_MAILER_AUTOCONFIRM=true\",\n\n  # Email Templates\n  \"GOTRUE_MAILER_URLPATHS_CONFIRMATION=/gotrue/verify\",\n  \"GOTRUE_MAILER_URLPATHS_INVITE=/gotrue/verify\",\n  \"GOTRUE_MAILER_URLPATHS_RECOVERY=/gotrue/verify\",\n  \"GOTRUE_MAILER_URLPATHS_EMAIL_CHANGE=/gotrue/verify\",\n  \"GOTRUE_MAILER_TEMPLATES_MAGIC_LINK=\",\n\n  # OAuth Providers (Configure as needed)\n  \"GOTRUE_EXTERNAL_GOOGLE_ENABLED=false\",\n  \"GOTRUE_EXTERNAL_GOOGLE_CLIENT_ID=\",\n  \"GOTRUE_EXTERNAL_GOOGLE_SECRET=\",\n  \"GOTRUE_EXTERNAL_GOOGLE_REDIRECT_URI=https://${main_domain}/gotrue/callback\",\n\n  \"GOTRUE_EXTERNAL_GITHUB_ENABLED=false\",\n  \"GOTRUE_EXTERNAL_GITHUB_CLIENT_ID=\",\n  \"GOTRUE_EXTERNAL_GITHUB_SECRET=\",\n  \"GOTRUE_EXTERNAL_GITHUB_REDIRECT_URI=https://${main_domain}/gotrue/callback\",\n\n  \"GOTRUE_EXTERNAL_DISCORD_ENABLED=false\",\n  \"GOTRUE_EXTERNAL_DISCORD_CLIENT_ID=\",\n  \"GOTRUE_EXTERNAL_DISCORD_SECRET=\",\n  \"GOTRUE_EXTERNAL_DISCORD_REDIRECT_URI=https://${main_domain}/gotrue/callback\",\n\n  # SAML Configuration\n  \"GOTRUE_SAML_ENABLED=false\",\n  \"GOTRUE_SAML_PRIVATE_KEY=\",\n\n  # =============================================================================\n  # ☁️ APPFLOWY CLOUD SERVICE CONFIGURATION\n  # =============================================================================\n  # Core Configuration\n  \"RUST_LOG=info\",\n  \"APPFLOWY_ENVIRONMENT=production\",\n  \"APPFLOWY_DATABASE_URL=postgres://appflowy:${POSTGRES_PASSWORD}@postgres:5432/appflowy\",\n  \"APPFLOWY_REDIS_URI=redis://redis:6379\",\n\n  # Authentication Integration\n  \"APPFLOWY_GOTRUE_JWT_SECRET=${GOTRUE_JWT_SECRET}\",\n  \"APPFLOWY_GOTRUE_JWT_EXP=7200\",\n  \"APPFLOWY_GOTRUE_BASE_URL=http://gotrue:9999\",\n\n  # Access Control and Performance\n  \"APPFLOWY_ACCESS_CONTROL=true\",\n  \"APPFLOWY_DATABASE_MAX_CONNECTIONS=40\",\n  \"APPFLOWY_WEBSOCKET_MAILBOX_SIZE=6000\",\n\n  # Email Configuration (SMTP)\n  \"APPFLOWY_MAILER_SMTP_HOST=\",\n  \"APPFLOWY_MAILER_SMTP_PORT=465\",\n  \"APPFLOWY_MAILER_SMTP_USERNAME=\",\n  \"APPFLOWY_MAILER_SMTP_EMAIL=\",\n  \"APPFLOWY_MAILER_SMTP_PASSWORD=\",\n  \"APPFLOWY_MAILER_SMTP_TLS_KIND=wrapper\",\n\n  # =============================================================================\n  # 💾 FILE STORAGE CONFIGURATION (MinIO/S3)\n  # =============================================================================\n  # MinIO Configuration\n  \"MINIO_HOST=minio\",\n  \"MINIO_PORT=9000\",\n  \"APPFLOWY_S3_USE_MINIO=true\",\n  \"APPFLOWY_S3_CREATE_BUCKET=true\",\n  \"APPFLOWY_S3_MINIO_URL=http://minio:9000\",\n\n  # Storage Credentials\n  \"APPFLOWY_S3_ACCESS_KEY=${password:16}\",\n  \"APPFLOWY_S3_SECRET_KEY=${password:32}\",\n\n  # Storage Configuration\n  \"APPFLOWY_S3_BUCKET=appflowy\",\n  \"APPFLOWY_S3_REGION=us-east-1\",\n  \"APPFLOWY_S3_PRESIGNED_URL_ENDPOINT=https://${main_domain}/minio-api\",\n\n  # AWS S3 Configuration (Alternative to MinIO)\n  # \"APPFLOWY_S3_USE_MINIO=false\",\n  # \"APPFLOWY_S3_REGION=us-east-1\",\n\n  # =============================================================================\n  # 🎛️ ADMIN FRONTEND CONFIGURATION\n  # =============================================================================\n  \"ADMIN_FRONTEND_REDIS_URL=redis://redis:6379\",\n  \"ADMIN_FRONTEND_GOTRUE_URL=http://gotrue:9999\",\n  \"ADMIN_FRONTEND_APPFLOWY_CLOUD_URL=http://appflowy_cloud:8000\",\n  \"ADMIN_FRONTEND_PATH_PREFIX=/console\",\n\n  # =============================================================================\n  # 🤖 AI FEATURES CONFIGURATION (Optional)\n  # =============================================================================\n  # OpenAI Configuration\n  \"AI_OPENAI_API_KEY=\",\n  \"DEFAULT_AI_MODEL=gpt-4o-mini\",\n  \"DEFAULT_AI_COMPLETION_MODEL=gpt-4o-mini\",\n\n  # Azure OpenAI (Alternative)\n  \"AZURE_OPENAI_API_KEY=\",\n  \"AZURE_OPENAI_ENDPOINT=\",\n  \"AZURE_OPENAI_API_VERSION=\",\n\n  # AI Service Configuration\n  \"AI_SERVER_HOST=ai\",\n  \"AI_SERVER_PORT=5001\",\n  \"AI_DATABASE_URL=postgresql+psycopg://appflowy:${POSTGRES_PASSWORD}@postgres:5432/appflowy\",\n  \"AI_REDIS_URL=redis://redis:6379\",\n  \"AI_USE_MINIO=true\",\n  \"AI_MINIO_URL=http://minio:9000\",\n  \"AI_APPFLOWY_HOST=https://${main_domain}\",\n\n  # Embedding Configuration\n  \"APPFLOWY_EMBEDDING_CHUNK_SIZE=2000\",\n  \"APPFLOWY_EMBEDDING_CHUNK_OVERLAP=200\",\n\n  # =============================================================================\n  # ⚙️ WORKER SERVICES CONFIGURATION\n  # =============================================================================\n  # AppFlowy Worker\n  \"APPFLOWY_WORKER_REDIS_URL=redis://redis:6379\",\n  \"APPFLOWY_WORKER_ENVIRONMENT=production\",\n  \"APPFLOWY_WORKER_DATABASE_URL=postgres://appflowy:${POSTGRES_PASSWORD}@postgres:5432/appflowy\",\n  \"APPFLOWY_WORKER_DATABASE_NAME=appflowy\",\n  \"APPFLOWY_WORKER_IMPORT_TICK_INTERVAL=30\",\n\n  # Indexer Configuration\n  \"APPFLOWY_INDEXER_ENABLED=true\",\n  \"APPFLOWY_INDEXER_DATABASE_URL=postgres://appflowy:${POSTGRES_PASSWORD}@postgres:5432/appflowy\",\n  \"APPFLOWY_INDEXER_REDIS_URL=redis://redis:6379\",\n  \"APPFLOWY_INDEXER_EMBEDDING_BUFFER_SIZE=5000\",\n\n  # Collaboration Service\n  \"APPFLOWY_COLLABORATE_MULTI_THREAD=false\",\n  \"APPFLOWY_COLLABORATE_REMOVE_BATCH_SIZE=100\",\n\n  # =============================================================================\n  # 🌐 NGINX CONFIGURATION\n  # =============================================================================\n  \"NGINX_PORT=80\",\n  \"NGINX_TLS_PORT=443\",\n\n  # =============================================================================\n  # 🛠️ VERSION TAGS (Easily Configurable)\n  # =============================================================================\n  \"GOTRUE_VERSION=latest\",\n  \"APPFLOWY_CLOUD_VERSION=latest\",\n  \"APPFLOWY_ADMIN_FRONTEND_VERSION=latest\",\n  \"APPFLOWY_AI_VERSION=latest\",\n  \"APPFLOWY_WORKER_VERSION=latest\",\n  \"APPFLOWY_WEB_VERSION=latest\",\n]\n\n[[config.domains]]\nserviceName = \"nginx\"\nport = 80\nhost = \"${main_domain}\"\n\n\n[[config.mounts]]\nfilePath = \"/nginx/nginx.conf\"\ncontent = \"\"\"# Minimal nginx configuration for AppFlowy-Cloud\n# Self Hosted AppFlowy Cloud user should alter this file to suit their needs,\n# or use the appflowy.site.conf in external_proxy_config/nginx if they are using\n# an external proxy.\n\nevents {\n    worker_connections 1024;\n}\n\nhttp {\n    # docker dns resolver\n    resolver 127.0.0.11 valid=10s;\n    #error_log /var/log/nginx/error.log debug;\n\n    map $http_upgrade $connection_upgrade {\n        default upgrade;\n        '' close;\n    }\n\n    map $http_origin $cors_origin {\n        # AppFlowy Web origin\n        \"~^http://localhost:3000$\" $http_origin;\n        default \"null\";\n    }\n\n    server {\n        listen 8080;\n\n        # https://github.com/nginxinc/nginx-prometheus-exporter\n        location = /stub_status {\n            stub_status;\n        }\n    }\n\n\n    server {\n\n        listen 80;\n        client_max_body_size 10M;\n\n        underscores_in_headers on;\n        set $appflowy_cloud_backend \"http://appflowy_cloud:8000\";\n        set $gotrue_backend \"http://gotrue:9999\";\n        set $admin_frontend_backend \"http://admin_frontend:3000\";\n        set $appflowy_web_backend \"http://appflowy_web:80\";\n        set $minio_backend \"http://minio:9001\";\n        set $minio_api_backend \"http://minio:9000\";\n        # Host name for minio, used internally within docker compose\n        set $minio_internal_host \"minio:9000\";\n        set $pgadmin_backend \"http://pgadmin:80\";\n\n        # GoTrue\n        location /gotrue/ {\n            if ($request_method = 'OPTIONS') {\n                add_header 'Access-Control-Allow-Origin' $cors_origin always;\n                add_header 'Access-Control-Allow-Credentials' 'true' always;\n                add_header 'Access-Control-Allow-Headers' '*' always;\n                add_header 'Access-Control-Allow-Methods' 'GET, POST, PUT, DELETE, PATCH, OPTIONS' always;\n                add_header 'Access-Control-Max-Age' 3600 always;\n                add_header 'Content-Type' 'text/plain charset=UTF-8' always;\n                add_header 'Content-Length' 0 always;\n                return 204;\n            }\n\n            proxy_pass $gotrue_backend;\n\n            rewrite ^/gotrue(/.*)$ $1 break;\n\n            # Allow headers like redirect_to to be handed over to the gotrue\n            # for correct redirecting\n            proxy_set_header Host $http_host;\n            proxy_pass_request_headers on;\n        }\n\n        # WebSocket\n        location /ws {\n            proxy_pass $appflowy_cloud_backend;\n\n            proxy_http_version 1.1;\n            proxy_set_header Upgrade $http_upgrade;\n            proxy_set_header Connection \"Upgrade\";\n            proxy_set_header Host $host;\n            proxy_read_timeout 86400s;\n        }\n\n        location /api {\n            proxy_pass $appflowy_cloud_backend;\n            proxy_set_header X-Request-Id $request_id;\n            proxy_set_header Host $http_host;\n\n            # Set CORS headers for other requests\n            if ($request_method = 'OPTIONS') {\n                add_header 'Access-Control-Allow-Origin' $cors_origin always;\n                add_header 'Access-Control-Allow-Methods' 'GET, POST, PUT, DELETE, PATCH, OPTIONS' always;\n                add_header 'Access-Control-Allow-Headers' 'Content-Type, Authorization, Accept, Client-Version, Device-Id' always;\n                add_header 'Access-Control-Max-Age' 3600 always;\n                return 204;\n            }\n\n            add_header 'Access-Control-Allow-Origin' $cors_origin always;\n            add_header 'Access-Control-Allow-Methods' 'GET, POST, PUT, DELETE, PATCH, OPTIONS' always;\n            add_header 'Access-Control-Allow-Headers' 'Content-Type, Authorization, Accept, Client-Version, Device-Id' always;\n            add_header 'Access-Control-Max-Age' 3600 always;\n\n            location ~* ^/api/workspace/([a-zA-Z0-9_-]+)/publish$ {\n                proxy_pass $appflowy_cloud_backend;\n                proxy_request_buffering off;\n                client_max_body_size 256M;\n                if ($request_method = 'OPTIONS') {\n                    add_header 'Access-Control-Allow-Origin' $cors_origin always;\n                    add_header 'Access-Control-Allow-Methods' 'GET, POST, PUT, DELETE, PATCH, OPTIONS' always;\n                    add_header 'Access-Control-Allow-Headers' 'Content-Type, Authorization, Accept, Client-Version, Device-Id' always;\n                    add_header 'Access-Control-Max-Age' 3600 always;\n                    return 204;\n                }\n\n                add_header 'Access-Control-Allow-Origin' $cors_origin always;\n                add_header 'Access-Control-Allow-Methods' 'GET, POST, PUT, DELETE, PATCH, OPTIONS' always;\n                add_header 'Access-Control-Allow-Headers' 'Content-Type, Authorization, Accept, Client-Version, Device-Id' always;\n                add_header 'Access-Control-Max-Age' 3600 always;\n            }\n\n            # AppFlowy-Cloud\n            location /api/chat {\n                proxy_pass $appflowy_cloud_backend;\n\n                proxy_http_version 1.1;\n                proxy_set_header Connection \"\";\n                chunked_transfer_encoding on;\n                proxy_buffering off;\n                proxy_cache off;\n\n                proxy_read_timeout 600s;\n                proxy_connect_timeout 600s;\n                proxy_send_timeout 600s;\n            }\n\n            location /api/import {\n                proxy_pass $appflowy_cloud_backend;\n\n                # Set headers\n                proxy_set_header X-Request-Id $request_id;\n                proxy_set_header Host $http_host;\n\n                # Handle CORS\n                add_header 'Access-Control-Allow-Origin' $cors_origin always;\n                add_header 'Access-Control-Allow-Methods' 'GET, POST, PUT, DELETE, PATCH, OPTIONS' always;\n                add_header 'Access-Control-Allow-Headers' 'Content-Type, Authorization, Accept, Device-Id' always;\n                add_header 'Access-Control-Max-Age' 3600 always;\n\n                # Timeouts\n                proxy_read_timeout 600s;\n                proxy_connect_timeout 600s;\n                proxy_send_timeout 600s;\n\n                # Disable buffering for large file uploads\n                proxy_request_buffering off;\n                proxy_buffering off;\n                proxy_cache off;\n                client_max_body_size 2G;\n            }\n        }\n\n        # Minio Web UI\n        # Derive from: https://min.io/docs/minio/linux/integrations/setup-nginx-proxy-with-minio.html\n        # Optional Module, comment this section if you did not deploy minio in docker-compose.yml\n        # This endpoint is meant to be used for the MinIO Web UI, accessible via the admin portal\n        location /minio/ {\n            proxy_pass $minio_backend;\n\n            rewrite ^/minio/(.*) /$1 break;\n            proxy_set_header Host $http_host;\n            proxy_set_header X-Real-IP $remote_addr;\n            proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;\n            proxy_set_header X-Forwarded-Proto $scheme;\n            proxy_set_header X-NginX-Proxy true;\n\n            ## This is necessary to pass the correct IP to be hashed\n            real_ip_header X-Real-IP;\n\n            proxy_connect_timeout 300s;\n\n            ## To support websockets in MinIO versions released after January 2023\n            proxy_http_version 1.1;\n            proxy_set_header Upgrade $http_upgrade;\n            proxy_set_header Connection \"upgrade\";\n            # Some environments may encounter CORS errors (Kubernetes + Nginx Ingress)\n            # Uncomment the following line to set the Origin request to an empty string\n            # proxy_set_header Origin '';\n\n            chunked_transfer_encoding off;\n        }\n\n        # Optional Module, comment this section if you did not deploy minio in docker-compose.yml\n        # This is used for presigned url, which is needs to be exposed to the AppFlowy client application.\n        location /minio-api/ {\n            proxy_pass $minio_api_backend;\n\n            # Set the host to internal host because the presigned url was signed against the internal host\n            proxy_set_header Host $minio_internal_host;\n            proxy_set_header X-Real-IP $remote_addr;\n            proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;\n            proxy_set_header X-Forwarded-Proto $scheme;\n\n            rewrite ^/minio-api/(.*) /$1 break;\n\n            proxy_connect_timeout 300s;\n            # Default is HTTP/1, keepalive is only enabled in HTTP/1.1\n            proxy_http_version 1.1;\n            proxy_set_header Connection \"\";\n            chunked_transfer_encoding off;\n        }\n\n        # PgAdmin\n        # Optional Module, comment this section if you did not deploy pgadmin in docker-compose.yml\n        location /pgadmin/ {\n            set $pgadmin pgadmin;\n            proxy_pass $pgadmin_backend;\n\n            proxy_set_header X-Script-Name /pgadmin;\n            proxy_set_header X-Scheme $scheme;\n            proxy_set_header Host $host;\n            proxy_redirect off;\n        }\n\n        # Admin Frontend\n        # Optional Module, comment this section if you did not deploy admin_frontend in docker-compose.yml\n        location /console {\n            proxy_pass $admin_frontend_backend;\n\n            proxy_set_header X-Scheme $scheme;\n            proxy_set_header Host $host;\n        }\n\n        # AppFlowy Web\n        location / {\n            proxy_pass $appflowy_web_backend;\n            proxy_set_header X-Scheme $scheme;\n            proxy_set_header Host $host;\n        }\n    }\n\n}\n\"\"\"\n"
}

Tags

productivity, self-hosted, notes, knowledge-base, notion-alternative


Version: 0.9.3

On this page