Dokploy

OpenPanel

An open-source web and product analytics platform that combines the power of Mixpanel with the ease of Plausible and one of the best Google Analytics replacements.

OpenPanel logo

Configuration

x-common: &x-common
  NODE_ENV: production
  SELF_HOSTED: "true"
  API_URL: ${API_URL}
  DASHBOARD_URL: ${DASHBOARD_URL}
  DATABASE_URL: postgres://${POSTGRES_USER}:${POSTGRES_PASSWORD}@op-db:5432/${POSTGRES_DB}?schema=public
  DATABASE_URL_DIRECT: postgres://${POSTGRES_USER}:${POSTGRES_PASSWORD}@op-db:5432/${POSTGRES_DB}?schema=public
  REDIS_URL: redis://default:${REDIS_PASSWORD}@op-kv:6379
  CLICKHOUSE_URL: http://op-ch:8123/openpanel

services:
  op-db:
    image: postgres:14-alpine
    restart: always
    volumes:
      - op-db-data:/var/lib/postgresql/data
    healthcheck:
      test: ['CMD-SHELL', 'pg_isready -U $${POSTGRES_USER} -d $${POSTGRES_DB}']
      interval: 10s
      timeout: 5s
      retries: 5
    environment:
      - POSTGRES_DB=${POSTGRES_DB}
      - POSTGRES_USER=${POSTGRES_USER}
      - POSTGRES_PASSWORD=${POSTGRES_PASSWORD}

  op-kv:
    image: redis:7.2.5-alpine
    restart: always
    volumes:
      - op-kv-data:/data
    command: redis-server --requirepass ${REDIS_PASSWORD} --maxmemory-policy noeviction
    healthcheck:
      test: ['CMD', 'redis-cli', '-a', '${REDIS_PASSWORD}', 'ping']
      interval: 10s
      timeout: 5s
      retries: 5

  op-ch:
    image: clickhouse/clickhouse-server:25.10.2.65
    restart: always
    volumes:
      - op-ch-data:/var/lib/clickhouse
      - op-ch-logs:/var/log/clickhouse-server
      - ../files/clickhouse_config:/etc/clickhouse-server/config.d
      - ../files/clickhouse_users:/etc/clickhouse-server/users.d
      - ../files/clickhouse_init:/docker-entrypoint-initdb.d
    environment:
      - CLICKHOUSE_SKIP_USER_SETUP=1
    healthcheck:
      test: ['CMD-SHELL', 'clickhouse-client --query "SELECT 1" -d openpanel']
      interval: 10s
      timeout: 5s
      retries: 5

  op-api:
    image: lindesvard/openpanel-api:2
    restart: always
    command: >
      sh -c "
        echo 'Waiting for PostgreSQL to be ready...'
        while ! nc -z op-db 5432; do
          sleep 1
        done
        echo 'PostgreSQL is ready'

        echo 'Waiting for ClickHouse to be ready...'
        while ! nc -z op-ch 8123; do
          sleep 1
        done
        echo 'ClickHouse is ready'

        echo 'Running migrations...'

        CI=true pnpm -r run migrate:deploy

        pnpm start
      "
    environment:
      COOKIE_SECRET: ${COOKIE_SECRET}
      ALLOW_REGISTRATION: ${ALLOW_REGISTRATION}
      ALLOW_INVITATION: ${ALLOW_INVITATION}
      EMAIL_SENDER: ${EMAIL_SENDER}
      RESEND_API_KEY: ${RESEND_API_KEY}
      <<: *x-common
    healthcheck:
      test: ['CMD-SHELL', 'curl -f http://localhost:3000/healthcheck || exit 1']
      interval: 10s
      timeout: 5s
      retries: 5
    depends_on:
      op-db:
        condition: service_healthy
      op-ch:
        condition: service_healthy
      op-kv:
        condition: service_healthy

  op-dashboard:
    image: lindesvard/openpanel-dashboard:2
    restart: always
    depends_on:
      op-api:
        condition: service_healthy
    environment:
      <<: *x-common
    healthcheck:
      test: ['CMD-SHELL', 'curl -f http://localhost:3000/api/healthcheck || exit 1']
      interval: 10s
      timeout: 5s
      retries: 5

  op-worker:
    image: lindesvard/openpanel-worker:2
    restart: always
    depends_on:
      op-api:
        condition: service_healthy
    environment:
      <<: *x-common
    healthcheck:
      test: ['CMD-SHELL', 'curl -f http://localhost:3000/healthcheck || exit 1']
      interval: 10s
      timeout: 5s
      retries: 5

volumes:
  op-db-data:
  op-kv-data:
  op-ch-data:
  op-ch-logs:
[variables]
main_domain = "${domain}"
db_password = "${password:32}"
cookie_secret = "${base64:32}"
redis_password = "${password:32}"

[config]
# ClickHouse config files - mounted as directories
[[config.mounts]]
filePath = "./clickhouse_config/op-config.xml"
content = """
<clickhouse>
    <logger>
        <level>warning</level>
        <console>true</console>
    </logger>
    <keep_alive_timeout>10</keep_alive_timeout>
    <!-- Stop all the unnecessary logging -->
    <query_thread_log remove="remove"/>
    <query_log remove="remove"/>
    <text_log remove="remove"/>
    <trace_log remove="remove"/>
    <metric_log remove="remove"/>
    <asynchronous_metric_log remove="remove"/>
    <session_log remove="remove"/>
    <part_log remove="remove"/>
    <listen_host>0.0.0.0</listen_host>
    <interserver_listen_host>0.0.0.0</interserver_listen_host>
    <interserver_http_host>opch</interserver_http_host>
    <!-- Disable cgroup memory observer -->
    <cgroups_memory_usage_observer_wait_time>0</cgroups_memory_usage_observer_wait_time>
    <!-- Not used anymore, but kept for backwards compatibility -->
    <macros>
        <shard>1</shard>
        <replica>replica1</replica>
        <cluster>openpanel_cluster</cluster>
    </macros>
</clickhouse>
"""

[[config.mounts]]
filePath = "./clickhouse_users/op-user-config.xml"
content = """
<clickhouse>
    <profiles>
        <default>
            <log_queries>0</log_queries>
            <log_query_threads>0</log_query_threads>
        </default>
    </profiles>
</clickhouse>
"""

[[config.mounts]]
filePath = "./clickhouse_init/1_init-db.sql"
content = """
CREATE DATABASE IF NOT EXISTS openpanel;
"""

[[config.domains]]
serviceName = "op-dashboard"
port = 3000
host = "${main_domain}"

[[config.domains]]
serviceName = "op-api"
port = 3000
host = "${main_domain}"
path = "/api"
stripPath = true

[config.env]
DASHBOARD_URL = "http://${main_domain}"
API_URL = "http://${main_domain}/api"

# Database configuration
POSTGRES_DB = "openpanel"
POSTGRES_USER = "openpanel"
POSTGRES_PASSWORD = "${db_password}"
REDIS_PASSWORD = "${redis_password}"

# Security
COOKIE_SECRET = "${cookie_secret}"

# Registration settings
ALLOW_REGISTRATION = "true"
ALLOW_INVITATION = "true"

# Email configuration (optional - configure for email notifications)
EMAIL_SENDER = ""
RESEND_API_KEY = ""

Base64

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

{
  "compose": "x-common: &x-common\n  NODE_ENV: production\n  SELF_HOSTED: \"true\"\n  API_URL: ${API_URL}\n  DASHBOARD_URL: ${DASHBOARD_URL}\n  DATABASE_URL: postgres://${POSTGRES_USER}:${POSTGRES_PASSWORD}@op-db:5432/${POSTGRES_DB}?schema=public\n  DATABASE_URL_DIRECT: postgres://${POSTGRES_USER}:${POSTGRES_PASSWORD}@op-db:5432/${POSTGRES_DB}?schema=public\n  REDIS_URL: redis://default:${REDIS_PASSWORD}@op-kv:6379\n  CLICKHOUSE_URL: http://op-ch:8123/openpanel\n\nservices:\n  op-db:\n    image: postgres:14-alpine\n    restart: always\n    volumes:\n      - op-db-data:/var/lib/postgresql/data\n    healthcheck:\n      test: ['CMD-SHELL', 'pg_isready -U $${POSTGRES_USER} -d $${POSTGRES_DB}']\n      interval: 10s\n      timeout: 5s\n      retries: 5\n    environment:\n      - POSTGRES_DB=${POSTGRES_DB}\n      - POSTGRES_USER=${POSTGRES_USER}\n      - POSTGRES_PASSWORD=${POSTGRES_PASSWORD}\n\n  op-kv:\n    image: redis:7.2.5-alpine\n    restart: always\n    volumes:\n      - op-kv-data:/data\n    command: redis-server --requirepass ${REDIS_PASSWORD} --maxmemory-policy noeviction\n    healthcheck:\n      test: ['CMD', 'redis-cli', '-a', '${REDIS_PASSWORD}', 'ping']\n      interval: 10s\n      timeout: 5s\n      retries: 5\n\n  op-ch:\n    image: clickhouse/clickhouse-server:25.10.2.65\n    restart: always\n    volumes:\n      - op-ch-data:/var/lib/clickhouse\n      - op-ch-logs:/var/log/clickhouse-server\n      - ../files/clickhouse_config:/etc/clickhouse-server/config.d\n      - ../files/clickhouse_users:/etc/clickhouse-server/users.d\n      - ../files/clickhouse_init:/docker-entrypoint-initdb.d\n    environment:\n      - CLICKHOUSE_SKIP_USER_SETUP=1\n    healthcheck:\n      test: ['CMD-SHELL', 'clickhouse-client --query \"SELECT 1\" -d openpanel']\n      interval: 10s\n      timeout: 5s\n      retries: 5\n\n  op-api:\n    image: lindesvard/openpanel-api:2\n    restart: always\n    command: >\n      sh -c \"\n        echo 'Waiting for PostgreSQL to be ready...'\n        while ! nc -z op-db 5432; do\n          sleep 1\n        done\n        echo 'PostgreSQL is ready'\n\n        echo 'Waiting for ClickHouse to be ready...'\n        while ! nc -z op-ch 8123; do\n          sleep 1\n        done\n        echo 'ClickHouse is ready'\n\n        echo 'Running migrations...'\n\n        CI=true pnpm -r run migrate:deploy\n\n        pnpm start\n      \"\n    environment:\n      COOKIE_SECRET: ${COOKIE_SECRET}\n      ALLOW_REGISTRATION: ${ALLOW_REGISTRATION}\n      ALLOW_INVITATION: ${ALLOW_INVITATION}\n      EMAIL_SENDER: ${EMAIL_SENDER}\n      RESEND_API_KEY: ${RESEND_API_KEY}\n      <<: *x-common\n    healthcheck:\n      test: ['CMD-SHELL', 'curl -f http://localhost:3000/healthcheck || exit 1']\n      interval: 10s\n      timeout: 5s\n      retries: 5\n    depends_on:\n      op-db:\n        condition: service_healthy\n      op-ch:\n        condition: service_healthy\n      op-kv:\n        condition: service_healthy\n\n  op-dashboard:\n    image: lindesvard/openpanel-dashboard:2\n    restart: always\n    depends_on:\n      op-api:\n        condition: service_healthy\n    environment:\n      <<: *x-common\n    healthcheck:\n      test: ['CMD-SHELL', 'curl -f http://localhost:3000/api/healthcheck || exit 1']\n      interval: 10s\n      timeout: 5s\n      retries: 5\n\n  op-worker:\n    image: lindesvard/openpanel-worker:2\n    restart: always\n    depends_on:\n      op-api:\n        condition: service_healthy\n    environment:\n      <<: *x-common\n    healthcheck:\n      test: ['CMD-SHELL', 'curl -f http://localhost:3000/healthcheck || exit 1']\n      interval: 10s\n      timeout: 5s\n      retries: 5\n\nvolumes:\n  op-db-data:\n  op-kv-data:\n  op-ch-data:\n  op-ch-logs:\n",
  "config": "[variables]\nmain_domain = \"${domain}\"\ndb_password = \"${password:32}\"\ncookie_secret = \"${base64:32}\"\nredis_password = \"${password:32}\"\n\n[config]\n# ClickHouse config files - mounted as directories\n[[config.mounts]]\nfilePath = \"./clickhouse_config/op-config.xml\"\ncontent = \"\"\"\n<clickhouse>\n    <logger>\n        <level>warning</level>\n        <console>true</console>\n    </logger>\n    <keep_alive_timeout>10</keep_alive_timeout>\n    <!-- Stop all the unnecessary logging -->\n    <query_thread_log remove=\"remove\"/>\n    <query_log remove=\"remove\"/>\n    <text_log remove=\"remove\"/>\n    <trace_log remove=\"remove\"/>\n    <metric_log remove=\"remove\"/>\n    <asynchronous_metric_log remove=\"remove\"/>\n    <session_log remove=\"remove\"/>\n    <part_log remove=\"remove\"/>\n    <listen_host>0.0.0.0</listen_host>\n    <interserver_listen_host>0.0.0.0</interserver_listen_host>\n    <interserver_http_host>opch</interserver_http_host>\n    <!-- Disable cgroup memory observer -->\n    <cgroups_memory_usage_observer_wait_time>0</cgroups_memory_usage_observer_wait_time>\n    <!-- Not used anymore, but kept for backwards compatibility -->\n    <macros>\n        <shard>1</shard>\n        <replica>replica1</replica>\n        <cluster>openpanel_cluster</cluster>\n    </macros>\n</clickhouse>\n\"\"\"\n\n[[config.mounts]]\nfilePath = \"./clickhouse_users/op-user-config.xml\"\ncontent = \"\"\"\n<clickhouse>\n    <profiles>\n        <default>\n            <log_queries>0</log_queries>\n            <log_query_threads>0</log_query_threads>\n        </default>\n    </profiles>\n</clickhouse>\n\"\"\"\n\n[[config.mounts]]\nfilePath = \"./clickhouse_init/1_init-db.sql\"\ncontent = \"\"\"\nCREATE DATABASE IF NOT EXISTS openpanel;\n\"\"\"\n\n[[config.domains]]\nserviceName = \"op-dashboard\"\nport = 3000\nhost = \"${main_domain}\"\n\n[[config.domains]]\nserviceName = \"op-api\"\nport = 3000\nhost = \"${main_domain}\"\npath = \"/api\"\nstripPath = true\n\n[config.env]\nDASHBOARD_URL = \"http://${main_domain}\"\nAPI_URL = \"http://${main_domain}/api\"\n\n# Database configuration\nPOSTGRES_DB = \"openpanel\"\nPOSTGRES_USER = \"openpanel\"\nPOSTGRES_PASSWORD = \"${db_password}\"\nREDIS_PASSWORD = \"${redis_password}\"\n\n# Security\nCOOKIE_SECRET = \"${cookie_secret}\"\n\n# Registration settings\nALLOW_REGISTRATION = \"true\"\nALLOW_INVITATION = \"true\"\n\n# Email configuration (optional - configure for email notifications)\nEMAIL_SENDER = \"\"\nRESEND_API_KEY = \"\"\n"
}

Tags

analytics


Version: latest

On this page