Dokploy

Frappe HR

Feature rich HR & Payroll software. 100% FOSS and customizable.

Frappe HR logo

Configuration

x-custom-image: &custom_image
  image: ${IMAGE_NAME:-ghcr.io/frappe/hrms}:${VERSION:-version-15}
  pull_policy: ${PULL_POLICY:-always}
  deploy:
    restart_policy:
      condition: always

services:
  backend:
    <<: *custom_image
    volumes:
      - sites:/home/frappe/frappe-bench/sites
    networks:
      - bench-network
    healthcheck:
      test:
        - CMD
        - wait-for-it
        - '0.0.0.0:8000'
      interval: 2s
      timeout: 10s
      retries: 30

  frontend:
    <<: *custom_image
    command:
      - nginx-entrypoint.sh
    depends_on:
      backend:
        condition: service_started
        required: true
      websocket:
        condition: service_started
        required: true
    environment:
      BACKEND: backend:8000
      FRAPPE_SITE_NAME_HEADER: ${FRAPPE_SITE_NAME_HEADER:-$$host}
      SOCKETIO: websocket:9000
      UPSTREAM_REAL_IP_ADDRESS: 127.0.0.1
      UPSTREAM_REAL_IP_HEADER: X-Forwarded-For
      UPSTREAM_REAL_IP_RECURSIVE: "off"
    volumes:
      - sites:/home/frappe/frappe-bench/sites

    networks:
      - bench-network

    healthcheck:
      test:
        - CMD
        - wait-for-it
        - '0.0.0.0:8080'
      interval: 2s
      timeout: 30s
      retries: 30

  queue-default:
    <<: *custom_image
    command:
      - bench
      - worker
      - --queue
      - default
    volumes:
      - sites:/home/frappe/frappe-bench/sites
    networks:
      - bench-network
    healthcheck:
      test:
        - CMD
        - wait-for-it
        - 'redis-queue:6379'
      interval: 2s
      timeout: 10s
      retries: 30
    depends_on:
      configurator:
        condition: service_completed_successfully
        required: true

  queue-long:
    <<: *custom_image
    command:
      - bench
      - worker
      - --queue
      - long
    volumes:
      - sites:/home/frappe/frappe-bench/sites
    networks:
      - bench-network
    healthcheck:
      test:
        - CMD
        - wait-for-it
        - 'redis-queue:6379'
      interval: 2s
      timeout: 10s
      retries: 30
    depends_on:
      configurator:
        condition: service_completed_successfully
        required: true

  queue-short:
    <<: *custom_image
    command:
      - bench
      - worker
      - --queue
      - short
    volumes:
      - sites:/home/frappe/frappe-bench/sites
    networks:
      - bench-network
    healthcheck:
      test:
        - CMD
        - wait-for-it
        - 'redis-queue:6379'
      interval: 2s
      timeout: 10s
      retries: 30
    depends_on:
      configurator:
        condition: service_completed_successfully
        required: true

  scheduler:
    <<: *custom_image
    healthcheck:
      test:
        - CMD
        - wait-for-it
        - 'redis-queue:6379'
      interval: 2s
      timeout: 10s
      retries: 30
    command:
      - bench
      - schedule
    depends_on:
      configurator:
        condition: service_completed_successfully
        required: true
    volumes:
      - sites:/home/frappe/frappe-bench/sites
    networks:
      - bench-network

  websocket:
    <<: *custom_image
    healthcheck:
      test:
        - CMD
        - wait-for-it
        - '0.0.0.0:9000'
      interval: 2s
      timeout: 10s
      retries: 30
    command:
      - node
      - /home/frappe/frappe-bench/apps/frappe/socketio.js
    depends_on:
      configurator:
        condition: service_completed_successfully
        required: true
    volumes:
      - sites:/home/frappe/frappe-bench/sites
    networks:
      - bench-network

  configurator:
    <<: *custom_image
    deploy:
      mode: replicated
      replicas: ${CONFIGURE:-0}
      restart_policy:
        condition: none
    entrypoint: ["bash", "-c"]
    command:
      - >
        [[ $${REGENERATE_APPS_TXT} == "1" ]] && ls -1 apps > sites/apps.txt;
        [[ -n `grep -hs ^ sites/common_site_config.json | jq -r ".db_host // empty"` ]] && exit 0;
        bench set-config -g db_host $$DB_HOST;
        bench set-config -gp db_port $$DB_PORT;
        bench set-config -g redis_cache "redis://$$REDIS_CACHE";
        bench set-config -g redis_queue "redis://$$REDIS_QUEUE";
        bench set-config -g redis_socketio "redis://$$REDIS_QUEUE";
        bench set-config -gp socketio_port $$SOCKETIO_PORT;
    environment:
      DB_HOST: "${DB_HOST:-db}"
      DB_PORT: "3306"
      REDIS_CACHE: redis-cache:6379
      REDIS_QUEUE: redis-queue:6379
      SOCKETIO_PORT: "9000"
      REGENERATE_APPS_TXT: "${REGENERATE_APPS_TXT:-0}"
    volumes:
      - sites:/home/frappe/frappe-bench/sites
    networks:
      - bench-network

  create-site:
    <<: *custom_image
    deploy:
      mode: replicated
      replicas: ${CREATE_SITE:-0}
      restart_policy:
        condition: none
    entrypoint: ["bash", "-c"]
    command:
      - >
        wait-for-it -t 120 $$DB_HOST:$$DB_PORT;
        wait-for-it -t 120 redis-cache:6379;
        wait-for-it -t 120 redis-queue:6379;
        export start=`date +%s`;
        until [[ -n `grep -hs ^ sites/common_site_config.json | jq -r ".db_host // empty"` ]] && \
          [[ -n `grep -hs ^ sites/common_site_config.json | jq -r ".redis_cache // empty"` ]] && \
          [[ -n `grep -hs ^ sites/common_site_config.json | jq -r ".redis_queue // empty"` ]];
        do
          echo "Waiting for sites/common_site_config.json to be created";
          sleep 5;
          if (( `date +%s`-start > 120 )); then
            echo "could not find sites/common_site_config.json with required keys";
            exit 1
          fi
        done;
        echo "sites/common_site_config.json found";
        [[ -d "sites/${SITE_NAME}" ]] && echo "${SITE_NAME} already exists" && exit 0;
        bench new-site --mariadb-user-host-login-scope='%' --admin-password=$${ADMIN_PASSWORD} --db-root-username=root --db-root-password=$${DB_ROOT_PASSWORD} $${INSTALL_APP_ARGS} $${SITE_NAME};
    volumes:
      - sites:/home/frappe/frappe-bench/sites
    environment:
      SITE_NAME: ${SITE_NAME}
      ADMIN_PASSWORD: ${ADMIN_PASSWORD}
      DB_HOST: ${DB_HOST:-db}
      DB_PORT: "${DB_PORT:-3306}"
      DB_ROOT_PASSWORD: ${DB_ROOT_PASSWORD}
      INSTALL_APP_ARGS: ${INSTALL_APP_ARGS}
    networks:
      - bench-network

  migration:
    <<: *custom_image
    deploy:
      mode: replicated
      replicas: ${MIGRATE:-0}
      restart_policy:
        condition: none
    entrypoint: ["bash", "-c"]
    command:
      - >
        curl -f http://${SITE_NAME}:8080/api/method/ping || echo "Site busy" && exit 0;
        bench --site all set-config -p maintenance_mode 1;
        bench --site all set-config -p pause_scheduler 1;
        bench --site all migrate;
        bench --site all set-config -p maintenance_mode 0;
        bench --site all set-config -p pause_scheduler 0;
    volumes:
      - sites:/home/frappe/frappe-bench/sites
    networks:
      - bench-network

  db:
    image: mariadb:10.6
    deploy:
      mode: replicated
      replicas: ${ENABLE_DB:-0}
      restart_policy:
        condition: always
    healthcheck:
      test: mysqladmin ping -h localhost --password=${DB_ROOT_PASSWORD}
      interval: 1s
      retries: 20
    command:
      - --character-set-server=utf8mb4
      - --collation-server=utf8mb4_unicode_ci
      - --skip-character-set-client-handshake
      - --skip-innodb-read-only-compressed
    environment:
      - MYSQL_ROOT_PASSWORD=${DB_ROOT_PASSWORD}
      - MARIADB_ROOT_PASSWORD=${DB_ROOT_PASSWORD}
    volumes:
      - db-data:/var/lib/mysql
    networks:
      - bench-network

  redis-cache:
    deploy:
      restart_policy:
        condition: always
    image: redis:6.2-alpine
    volumes:
      - redis-cache-data:/data
    networks:
      - bench-network
    healthcheck:
      test:
        - CMD
        - redis-cli
        - ping
      interval: 5s
      timeout: 5s
      retries: 3

  redis-queue:
    deploy:
      restart_policy:
        condition: always
    image: redis:6.2-alpine
    volumes:
      - redis-queue-data:/data
    networks:
      - bench-network
    healthcheck:
      test:
        - CMD
        - redis-cli
        - ping
      interval: 5s
      timeout: 5s
      retries: 3

  redis-socketio:
    deploy:
      restart_policy:
        condition: always
    image: redis:6.2-alpine
    volumes:
      - redis-socketio-data:/data
    networks:
      - bench-network
    healthcheck:
      test:
        - CMD
        - redis-cli
        - ping
      interval: 5s
      timeout: 5s
      retries: 3

volumes:
  db-data:
  redis-cache-data:
  redis-queue-data:
  redis-socketio-data:
  sites:
    driver_opts:
      type: "${SITE_VOLUME_TYPE}"
      o: "${SITE_VOLUME_OPTS}"
      device: "${SITE_VOLUME_DEV}"

networks:
  bench-network:
[variables]
main_domain = "${domain}"
db_root_password = "${password:32}"
admin_password = "${password:32}"

[config]
env = [
  "SITE_NAME=${main_domain}",
  "ADMIN_PASSWORD=${admin_password}",
  "DB_ROOT_PASSWORD=${db_root_password}",
  "MIGRATE=1",
  "ENABLE_DB=1",
  "DB_HOST=db",
  "CREATE_SITE=1",
  "CONFIGURE=1",
  "REGENERATE_APPS_TXT=1",
  "INSTALL_APP_ARGS=--install-app hrms",
  "IMAGE_NAME=ghcr.io/frappe/hrms",
  "VERSION=version-15",
  "FRAPPE_SITE_NAME_HEADER=",
]
mounts = []

[[config.domains]]
serviceName = "frontend"
port = 8_080
host = "${main_domain}"

Base64

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

{
  "compose": "x-custom-image: &custom_image\n  image: ${IMAGE_NAME:-ghcr.io/frappe/hrms}:${VERSION:-version-15}\n  pull_policy: ${PULL_POLICY:-always}\n  deploy:\n    restart_policy:\n      condition: always\n\nservices:\n  backend:\n    <<: *custom_image\n    volumes:\n      - sites:/home/frappe/frappe-bench/sites\n    networks:\n      - bench-network\n    healthcheck:\n      test:\n        - CMD\n        - wait-for-it\n        - '0.0.0.0:8000'\n      interval: 2s\n      timeout: 10s\n      retries: 30\n\n  frontend:\n    <<: *custom_image\n    command:\n      - nginx-entrypoint.sh\n    depends_on:\n      backend:\n        condition: service_started\n        required: true\n      websocket:\n        condition: service_started\n        required: true\n    environment:\n      BACKEND: backend:8000\n      FRAPPE_SITE_NAME_HEADER: ${FRAPPE_SITE_NAME_HEADER:-$$host}\n      SOCKETIO: websocket:9000\n      UPSTREAM_REAL_IP_ADDRESS: 127.0.0.1\n      UPSTREAM_REAL_IP_HEADER: X-Forwarded-For\n      UPSTREAM_REAL_IP_RECURSIVE: \"off\"\n    volumes:\n      - sites:/home/frappe/frappe-bench/sites\n\n    networks:\n      - bench-network\n    \n    healthcheck:\n      test:\n        - CMD\n        - wait-for-it\n        - '0.0.0.0:8080'\n      interval: 2s\n      timeout: 30s\n      retries: 30\n\n  queue-default:\n    <<: *custom_image\n    command:\n      - bench\n      - worker\n      - --queue\n      - default\n    volumes:\n      - sites:/home/frappe/frappe-bench/sites\n    networks:\n      - bench-network\n    healthcheck:\n      test:\n        - CMD\n        - wait-for-it\n        - 'redis-queue:6379'\n      interval: 2s\n      timeout: 10s\n      retries: 30\n    depends_on:\n      configurator:\n        condition: service_completed_successfully\n        required: true\n\n  queue-long:\n    <<: *custom_image\n    command:\n      - bench\n      - worker\n      - --queue\n      - long\n    volumes:\n      - sites:/home/frappe/frappe-bench/sites\n    networks:\n      - bench-network\n    healthcheck:\n      test:\n        - CMD\n        - wait-for-it\n        - 'redis-queue:6379'\n      interval: 2s\n      timeout: 10s\n      retries: 30\n    depends_on:\n      configurator:\n        condition: service_completed_successfully\n        required: true\n\n  queue-short:\n    <<: *custom_image\n    command:\n      - bench\n      - worker\n      - --queue\n      - short\n    volumes:\n      - sites:/home/frappe/frappe-bench/sites\n    networks:\n      - bench-network\n    healthcheck:\n      test:\n        - CMD\n        - wait-for-it\n        - 'redis-queue:6379'\n      interval: 2s\n      timeout: 10s\n      retries: 30\n    depends_on:\n      configurator:\n        condition: service_completed_successfully\n        required: true\n\n  scheduler:\n    <<: *custom_image\n    healthcheck:\n      test:\n        - CMD\n        - wait-for-it\n        - 'redis-queue:6379'\n      interval: 2s\n      timeout: 10s\n      retries: 30\n    command:\n      - bench\n      - schedule\n    depends_on:\n      configurator:\n        condition: service_completed_successfully\n        required: true\n    volumes:\n      - sites:/home/frappe/frappe-bench/sites\n    networks:\n      - bench-network\n\n  websocket:\n    <<: *custom_image\n    healthcheck:\n      test:\n        - CMD\n        - wait-for-it\n        - '0.0.0.0:9000'\n      interval: 2s\n      timeout: 10s\n      retries: 30\n    command:\n      - node\n      - /home/frappe/frappe-bench/apps/frappe/socketio.js\n    depends_on:\n      configurator:\n        condition: service_completed_successfully\n        required: true\n    volumes:\n      - sites:/home/frappe/frappe-bench/sites\n    networks:\n      - bench-network\n\n  configurator:\n    <<: *custom_image\n    deploy:\n      mode: replicated\n      replicas: ${CONFIGURE:-0}\n      restart_policy:\n        condition: none\n    entrypoint: [\"bash\", \"-c\"]\n    command:\n      - >\n        [[ $${REGENERATE_APPS_TXT} == \"1\" ]] && ls -1 apps > sites/apps.txt;\n        [[ -n `grep -hs ^ sites/common_site_config.json | jq -r \".db_host // empty\"` ]] && exit 0;\n        bench set-config -g db_host $$DB_HOST;\n        bench set-config -gp db_port $$DB_PORT;\n        bench set-config -g redis_cache \"redis://$$REDIS_CACHE\";\n        bench set-config -g redis_queue \"redis://$$REDIS_QUEUE\";\n        bench set-config -g redis_socketio \"redis://$$REDIS_QUEUE\";\n        bench set-config -gp socketio_port $$SOCKETIO_PORT;\n    environment:\n      DB_HOST: \"${DB_HOST:-db}\"\n      DB_PORT: \"3306\"\n      REDIS_CACHE: redis-cache:6379\n      REDIS_QUEUE: redis-queue:6379\n      SOCKETIO_PORT: \"9000\"\n      REGENERATE_APPS_TXT: \"${REGENERATE_APPS_TXT:-0}\"\n    volumes:\n      - sites:/home/frappe/frappe-bench/sites\n    networks:\n      - bench-network\n\n  create-site:\n    <<: *custom_image\n    deploy:\n      mode: replicated\n      replicas: ${CREATE_SITE:-0}\n      restart_policy:\n        condition: none\n    entrypoint: [\"bash\", \"-c\"]\n    command:\n      - >\n        wait-for-it -t 120 $$DB_HOST:$$DB_PORT;\n        wait-for-it -t 120 redis-cache:6379;\n        wait-for-it -t 120 redis-queue:6379;\n        export start=`date +%s`;\n        until [[ -n `grep -hs ^ sites/common_site_config.json | jq -r \".db_host // empty\"` ]] && \\\n          [[ -n `grep -hs ^ sites/common_site_config.json | jq -r \".redis_cache // empty\"` ]] && \\\n          [[ -n `grep -hs ^ sites/common_site_config.json | jq -r \".redis_queue // empty\"` ]];\n        do\n          echo \"Waiting for sites/common_site_config.json to be created\";\n          sleep 5;\n          if (( `date +%s`-start > 120 )); then\n            echo \"could not find sites/common_site_config.json with required keys\";\n            exit 1\n          fi\n        done;\n        echo \"sites/common_site_config.json found\";\n        [[ -d \"sites/${SITE_NAME}\" ]] && echo \"${SITE_NAME} already exists\" && exit 0;\n        bench new-site --mariadb-user-host-login-scope='%' --admin-password=$${ADMIN_PASSWORD} --db-root-username=root --db-root-password=$${DB_ROOT_PASSWORD} $${INSTALL_APP_ARGS} $${SITE_NAME};\n    volumes:\n      - sites:/home/frappe/frappe-bench/sites\n    environment:\n      SITE_NAME: ${SITE_NAME}\n      ADMIN_PASSWORD: ${ADMIN_PASSWORD}\n      DB_HOST: ${DB_HOST:-db}\n      DB_PORT: \"${DB_PORT:-3306}\"\n      DB_ROOT_PASSWORD: ${DB_ROOT_PASSWORD}\n      INSTALL_APP_ARGS: ${INSTALL_APP_ARGS}\n    networks:\n      - bench-network\n\n  migration:\n    <<: *custom_image\n    deploy:\n      mode: replicated\n      replicas: ${MIGRATE:-0}\n      restart_policy:\n        condition: none\n    entrypoint: [\"bash\", \"-c\"]\n    command:\n      - >\n        curl -f http://${SITE_NAME}:8080/api/method/ping || echo \"Site busy\" && exit 0;\n        bench --site all set-config -p maintenance_mode 1;\n        bench --site all set-config -p pause_scheduler 1;\n        bench --site all migrate;\n        bench --site all set-config -p maintenance_mode 0;\n        bench --site all set-config -p pause_scheduler 0;\n    volumes:\n      - sites:/home/frappe/frappe-bench/sites\n    networks:\n      - bench-network\n\n  db:\n    image: mariadb:10.6\n    deploy:\n      mode: replicated\n      replicas: ${ENABLE_DB:-0}\n      restart_policy:\n        condition: always\n    healthcheck:\n      test: mysqladmin ping -h localhost --password=${DB_ROOT_PASSWORD}\n      interval: 1s\n      retries: 20\n    command:\n      - --character-set-server=utf8mb4\n      - --collation-server=utf8mb4_unicode_ci\n      - --skip-character-set-client-handshake\n      - --skip-innodb-read-only-compressed\n    environment:\n      - MYSQL_ROOT_PASSWORD=${DB_ROOT_PASSWORD}\n      - MARIADB_ROOT_PASSWORD=${DB_ROOT_PASSWORD}\n    volumes:\n      - db-data:/var/lib/mysql\n    networks:\n      - bench-network\n\n  redis-cache:\n    deploy:\n      restart_policy:\n        condition: always\n    image: redis:6.2-alpine\n    volumes:\n      - redis-cache-data:/data\n    networks:\n      - bench-network\n    healthcheck:\n      test:\n        - CMD\n        - redis-cli\n        - ping\n      interval: 5s\n      timeout: 5s\n      retries: 3\n\n  redis-queue:\n    deploy:\n      restart_policy:\n        condition: always\n    image: redis:6.2-alpine\n    volumes:\n      - redis-queue-data:/data\n    networks:\n      - bench-network\n    healthcheck:\n      test:\n        - CMD\n        - redis-cli\n        - ping\n      interval: 5s\n      timeout: 5s\n      retries: 3\n\n  redis-socketio:\n    deploy:\n      restart_policy:\n        condition: always\n    image: redis:6.2-alpine\n    volumes:\n      - redis-socketio-data:/data\n    networks:\n      - bench-network\n    healthcheck:\n      test:\n        - CMD\n        - redis-cli\n        - ping\n      interval: 5s\n      timeout: 5s\n      retries: 3\n\nvolumes:\n  db-data:\n  redis-cache-data:\n  redis-queue-data:\n  redis-socketio-data:\n  sites:\n    driver_opts:\n      type: \"${SITE_VOLUME_TYPE}\"\n      o: \"${SITE_VOLUME_OPTS}\"\n      device: \"${SITE_VOLUME_DEV}\"\n\nnetworks:\n  bench-network:",
  "config": "[variables]\nmain_domain = \"${domain}\"\ndb_root_password = \"${password:32}\"\nadmin_password = \"${password:32}\"\n\n[config]\nenv = [\n  \"SITE_NAME=${main_domain}\",\n  \"ADMIN_PASSWORD=${admin_password}\",\n  \"DB_ROOT_PASSWORD=${db_root_password}\",\n  \"MIGRATE=1\",\n  \"ENABLE_DB=1\",\n  \"DB_HOST=db\",\n  \"CREATE_SITE=1\",\n  \"CONFIGURE=1\",\n  \"REGENERATE_APPS_TXT=1\",\n  \"INSTALL_APP_ARGS=--install-app hrms\",\n  \"IMAGE_NAME=ghcr.io/frappe/hrms\",\n  \"VERSION=version-15\",\n  \"FRAPPE_SITE_NAME_HEADER=\",\n]\nmounts = []\n\n[[config.domains]]\nserviceName = \"frontend\"\nport = 8_080\nhost = \"${main_domain}\"\n"
}

Tags

hrms, payroll, leaves, expenses, attendance, performace


Version: version-15

On this page