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.
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 → Advanced → Base64 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"
}Links
Tags
analytics
Version: latest
Openinary
Openinary is a self-hosted Cloudinary alternative.
OpenResty Manager
The easiest using, powerful and beautiful OpenResty Manager (Nginx Enhanced Version) , open source alternative to OpenResty Edge, which can enable you to easily reverse proxy your websites with security running at home or internet, including Access Control, HTTP Flood Protection, Free SSL, without having to know too much about OpenResty or Let's Encrypt.