mirror of
https://github.com/kevinveenbirkenbach/computer-playbook.git
synced 2025-10-09 18:28:10 +02:00
EspoCRM role: custom image + single data volume + runtime flag setter
• Build a custom image and replace upstream entrypoint with docker-entrypoint-custom.sh (strict fail on flag script). • Introduce set_flags.php and wire via ESPOCRM_SET_FLAGS_SCRIPT; apply flags at container start; clear cache afterwards. • Keep exactly one Docker volume (data:/var/www/html/); drop separate custom/extensions mounts. • Compose: use custom image, add healthchecks & depends_on for daemon/websocket; keep service healthy gating. • Ansible: deploy scripts, build & up via handlers; patch siteUrl as www-data; run upgrade non-fatal; always run flag setter. • Vars/Env: add ESPO_INIT_* toggles and ESPOCRM_SET_FLAGS_SCRIPT; refactor variables for scripts & custom image paths. Conversation context: https://chatgpt.com/share/68dd1992-020c-800f-bcf5-2db60cb4aab2
This commit is contained in:
@@ -51,4 +51,5 @@ docker:
|
||||
mem_limit: 0.5g
|
||||
pids_limit: 384
|
||||
volumes:
|
||||
data: espocrm_data
|
||||
data: espocrm_data
|
||||
maintenance_mode: false
|
60
roles/web-app-espocrm/files/docker-entrypoint-custom.sh
Normal file
60
roles/web-app-espocrm/files/docker-entrypoint-custom.sh
Normal file
@@ -0,0 +1,60 @@
|
||||
#!/bin/sh
|
||||
set -euo pipefail
|
||||
|
||||
log() { printf '%s %s\n' "[entrypoint]" "$*" >&2; }
|
||||
|
||||
# --- Simple boolean normalization --------------------------------------------
|
||||
bool_norm () {
|
||||
v="$(printf '%s' "${1:-}" | tr '[:upper:]' '[:lower:]')"
|
||||
case "$v" in
|
||||
1|true|yes|on) echo "true" ;;
|
||||
0|false|no|off|"") echo "false" ;;
|
||||
*) echo "false" ;;
|
||||
esac
|
||||
}
|
||||
|
||||
# Expected ENV (from env.j2)
|
||||
MAINTENANCE="$(bool_norm "${ESPO_INIT_MAINTENANCE_MODE:-false}")"
|
||||
CRON_DISABLED="$(bool_norm "${ESPO_INIT_CRON_DISABLED:-false}")"
|
||||
USE_CACHE="$(bool_norm "${ESPO_INIT_USE_CACHE:-true}")"
|
||||
|
||||
APP_DIR="/var/www/html"
|
||||
SET_FLAGS_SCRIPT="${ESPOCRM_SET_FLAGS_SCRIPT}"
|
||||
|
||||
# --- Wait for bootstrap.php (max 60s, e.g. fresh volume) ----------------------
|
||||
log "Waiting for ${APP_DIR}/bootstrap.php..."
|
||||
for i in $(seq 1 60); do
|
||||
[ -f "${APP_DIR}/bootstrap.php" ] && break
|
||||
sleep 1
|
||||
done
|
||||
if [ ! -f "${APP_DIR}/bootstrap.php" ]; then
|
||||
log "ERROR: bootstrap.php missing after 60s"; exit 1
|
||||
fi
|
||||
|
||||
# --- Apply config flags via set_flags.php ------------------------------------
|
||||
log "Applying runtime flags via set_flags.php..."
|
||||
php "${SET_FLAGS_SCRIPT}"
|
||||
|
||||
# --- Clear cache (safe) -------------------------------------------------------
|
||||
php "${APP_DIR}/clear_cache.php" || true
|
||||
|
||||
# --- Hand off to CMD ----------------------------------------------------------
|
||||
if [ "$#" -gt 0 ]; then
|
||||
log "Exec CMD: $*"
|
||||
exec "$@"
|
||||
fi
|
||||
|
||||
# Try common server commands
|
||||
for cmd in apache2-foreground httpd-foreground php-fpm php-fpm8.3 php-fpm8.2 supervisord; do
|
||||
if command -v "$cmd" >/dev/null 2>&1; then
|
||||
log "Starting: $cmd"
|
||||
case "$cmd" in
|
||||
php-fpm|php-fpm8.*) exec "$cmd" -F ;;
|
||||
supervisord) exec "$cmd" -n ;;
|
||||
*) exec "$cmd" ;;
|
||||
esac
|
||||
fi
|
||||
done
|
||||
|
||||
log "No known server command found; tailing to keep container alive."
|
||||
exec tail -f /dev/null
|
33
roles/web-app-espocrm/files/set_flags.php
Normal file
33
roles/web-app-espocrm/files/set_flags.php
Normal file
@@ -0,0 +1,33 @@
|
||||
<?php
|
||||
/**
|
||||
* set_flags.php – Ensure EspoCRM runtime flags are set idempotently.
|
||||
*/
|
||||
|
||||
require "/var/www/html/bootstrap.php";
|
||||
|
||||
$app = new \Espo\Core\Application();
|
||||
$c = $app->getContainer();
|
||||
$cfg = $c->get("config");
|
||||
$w = $c->get("injectableFactory")->create("\Espo\Core\Utils\Config\ConfigWriter");
|
||||
|
||||
// Read from ENV
|
||||
$flags = [
|
||||
"maintenanceMode" => in_array(strtolower(getenv("ESPO_INIT_MAINTENANCE_MODE") ?: "false"), ["1","true","yes","on"]),
|
||||
"cronDisabled" => in_array(strtolower(getenv("ESPO_INIT_CRON_DISABLED") ?: "false"), ["1","true","yes","on"]),
|
||||
"useCache" => in_array(strtolower(getenv("ESPO_INIT_USE_CACHE") ?: "true"), ["1","true","yes","on"])
|
||||
];
|
||||
|
||||
$changed = false;
|
||||
foreach ($flags as $k => $v) {
|
||||
if ($cfg->get($k) !== $v) {
|
||||
$w->set($k, $v);
|
||||
$changed = true;
|
||||
}
|
||||
}
|
||||
|
||||
if ($changed) {
|
||||
$w->save();
|
||||
echo "CHANGED\n";
|
||||
} else {
|
||||
echo "UNCHANGED\n";
|
||||
}
|
@@ -37,7 +37,7 @@
|
||||
|
||||
- name: Ensure siteUrl matches canonical domain
|
||||
ansible.builtin.shell: |
|
||||
{{ docker_compose_command_exec }} -T {{ ESPOCRM_SERVICE }} php -r '
|
||||
{{ docker_compose_command_exec }} -T --user {{ ESPOCRM_USER }} {{ ESPOCRM_SERVICE }} php -r '
|
||||
require "/var/www/html/bootstrap.php";
|
||||
$app = new \Espo\Core\Application();
|
||||
$c = $app->getContainer();
|
||||
@@ -54,57 +54,3 @@
|
||||
chdir: "{{ docker_compose.directories.instance }}"
|
||||
register: siteurl_set
|
||||
changed_when: "'CHANGED' in siteurl_set.stdout"
|
||||
|
||||
- name: Ensure maintenance off, cron on, cache on (idempotent via ConfigWriter)
|
||||
block:
|
||||
- name: Apply config via ConfigWriter as app user
|
||||
command: >
|
||||
docker exec --user {{ ESPOCRM_USER }} {{ ESPOCRM_CONTAINER }}
|
||||
php -r '
|
||||
require "/var/www/html/bootstrap.php";
|
||||
$app = new \Espo\Core\Application();
|
||||
$c = $app->getContainer();
|
||||
$cfg = $c->get("config");
|
||||
$w = $c->get("injectableFactory")->create("\Espo\Core\Utils\Config\ConfigWriter");
|
||||
$pairs = [
|
||||
"maintenanceMode" => false,
|
||||
"cronDisabled" => false,
|
||||
"useCache" => true
|
||||
];
|
||||
$changed = false;
|
||||
foreach ($pairs as $k => $v) {
|
||||
if ($cfg->get($k) !== $v) { $w->set($k, $v); $changed = true; }
|
||||
}
|
||||
if ($changed) { $w->save(); echo "CHANGED"; }
|
||||
'
|
||||
register: cfg_set
|
||||
changed_when: "'CHANGED' in cfg_set.stdout"
|
||||
|
||||
rescue:
|
||||
- name: Apply config via ConfigWriter as root (fallback)
|
||||
command: >
|
||||
docker exec --user root {{ ESPOCRM_CONTAINER }}
|
||||
php -r '
|
||||
require "/var/www/html/bootstrap.php";
|
||||
$app = new \Espo\Core\Application();
|
||||
$c = $app->getContainer();
|
||||
$cfg = $c->get("config");
|
||||
$w = $c->get("injectableFactory")->create("\Espo\Core\Utils\Config\ConfigWriter");
|
||||
$pairs = [
|
||||
"maintenanceMode" => false,
|
||||
"cronDisabled" => false,
|
||||
"useCache" => true
|
||||
];
|
||||
$changed = false;
|
||||
foreach ($pairs as $k => $v) {
|
||||
if ($cfg->get($k) !== $v) { $w->set($k, $v); $changed = true; }
|
||||
}
|
||||
if ($changed) { $w->save(); echo "CHANGED"; }
|
||||
'
|
||||
register: cfg_set
|
||||
changed_when: "'CHANGED' in cfg_set.stdout"
|
||||
|
||||
- name: Clear EspoCRM cache (only when config changed and we are updating)
|
||||
command: >
|
||||
docker exec --user {{ ESPOCRM_USER }} {{ ESPOCRM_CONTAINER }} php clear_cache.php
|
||||
when: "'CHANGED' in cfg_set.stdout and MODE_UPDATE | bool"
|
||||
|
@@ -3,17 +3,53 @@
|
||||
include_role:
|
||||
name: sys-stk-full-stateful
|
||||
vars:
|
||||
docker_compose_flush_handlers: true
|
||||
docker_compose_flush_handlers: false
|
||||
|
||||
- name: "Deploy '{{ ESPOCRM_ENTRYPOINT_SCRIPT_HOST_ABS }}'"
|
||||
copy:
|
||||
src: "{{ ESPOCRM_ENTRYPOINT_SCRIPT_FILE }}"
|
||||
dest: "{{ ESPOCRM_ENTRYPOINT_SCRIPT_HOST_ABS }}"
|
||||
notify:
|
||||
- docker compose up
|
||||
- docker compose build
|
||||
|
||||
- name: "Deploy '{{ ESPOCRM_SET_FLAG_SCRIPT_HOST_ABS }}'"
|
||||
copy:
|
||||
src: "{{ ESPOCRM_SET_FLAG_SCRIPT_FILE }}"
|
||||
dest: "{{ ESPOCRM_SET_FLAG_SCRIPT_HOST_ABS }}"
|
||||
notify:
|
||||
- docker compose up
|
||||
- docker compose build
|
||||
|
||||
|
||||
- name: "Docker Compose Up for '{{ application_id }}'"
|
||||
meta: flush_handlers
|
||||
|
||||
- name: Check if config.php exists in EspoCRM
|
||||
command: docker exec --user root {{ ESPOCRM_CONTAINER }} test -f {{ ESPOCRM_CONFIG_FILE_PRIVATE }}
|
||||
register: config_file_exists
|
||||
changed_when: false
|
||||
failed_when: false
|
||||
failed_when: false
|
||||
|
||||
- name: Patch EspoCRM config.php
|
||||
include_tasks: 01_patch_config.yml
|
||||
when: config_file_exists.rc == 0
|
||||
|
||||
- name: Run EspoCRM upgrade (only when MODE_UPDATE is true)
|
||||
command: >
|
||||
docker exec --user {{ ESPOCRM_USER }} {{ ESPOCRM_CONTAINER }}
|
||||
php command.php upgrade -y
|
||||
register: espocrm_upgrade
|
||||
changed_when: "'Upgrading' in espocrm_upgrade.stdout or 'successfully' in espocrm_upgrade.stdout"
|
||||
failed_when: false
|
||||
when: MODE_UPDATE | bool
|
||||
|
||||
- name: Run flag setter as root (fallback)
|
||||
command: >
|
||||
docker exec --user root {{ ESPOCRM_CONTAINER }}
|
||||
php {{ ESPOCRM_SET_FLAG_SCRIPT_DOCKER }}
|
||||
register: flags_result_root
|
||||
changed_when: "'CHANGED' in flags_result_root.stdout"
|
||||
|
||||
- name: Flush handlers to make DB available before password reset
|
||||
meta: flush_handlers
|
||||
|
9
roles/web-app-espocrm/templates/Dockerfile.j2
Normal file
9
roles/web-app-espocrm/templates/Dockerfile.j2
Normal file
@@ -0,0 +1,9 @@
|
||||
FROM "{{ ESPOCRM_IMAGE }}:{{ ESPOCRM_VERSION }}"
|
||||
|
||||
COPY {{ ESPOCRM_SET_FLAG_SCRIPT_HOST_REL }} {{ ESPOCRM_SET_FLAG_SCRIPT_DOCKER }}
|
||||
RUN chmod +x {{ ESPOCRM_SET_FLAG_SCRIPT_DOCKER }}
|
||||
|
||||
COPY {{ ESPOCRM_ENTRYPOINT_SCRIPT_HOST_REL }} {{ ESPOCRM_ENTRYPOINT_SCRIPT_DOCKER }}
|
||||
RUN chmod +x {{ ESPOCRM_ENTRYPOINT_SCRIPT_DOCKER }}
|
||||
|
||||
ENTRYPOINT ["{{ ESPOCRM_ENTRYPOINT_SCRIPT_DOCKER }}"]
|
@@ -2,8 +2,9 @@
|
||||
|
||||
{% set service_name = ESPOCRM_SERVICE %}
|
||||
{{ service_name }}:
|
||||
{{ lookup('template', 'roles/docker-container/templates/build.yml.j2') | indent(4) }}
|
||||
container_name: {{ ESPOCRM_CONTAINER }}
|
||||
image: "{{ ESPOCRM_IMAGE }}:{{ ESPOCRM_VERSION }}"
|
||||
image: "{{ ESPOCRM_CUSTOM_IMAGE }}"
|
||||
init: true
|
||||
stop_signal: SIGTERM
|
||||
stop_grace_period: 30s
|
||||
@@ -14,12 +15,13 @@
|
||||
{% include 'roles/docker-container/templates/depends_on/dmbs_excl.yml.j2' %}
|
||||
{% include 'roles/docker-container/templates/networks.yml.j2' %}
|
||||
volumes:
|
||||
- data:/var/www/html
|
||||
- data:/var/www/html/
|
||||
|
||||
{% set service_name = 'daemon' %}
|
||||
{{ service_name }}:
|
||||
image: "{{ ESPOCRM_IMAGE }}:{{ ESPOCRM_VERSION }}"
|
||||
image: "{{ ESPOCRM_CUSTOM_IMAGE }}"
|
||||
container_name: {{ ESPOCRM_CONTAINER }}_{{ service_name }}
|
||||
pull_policy: never
|
||||
init: true
|
||||
stop_signal: SIGTERM
|
||||
stop_grace_period: 30s
|
||||
@@ -27,12 +29,22 @@
|
||||
entrypoint: docker-daemon.sh
|
||||
{% include 'roles/docker-container/templates/networks.yml.j2' %}
|
||||
volumes:
|
||||
- data:/var/www/html
|
||||
- data:/var/www/html/
|
||||
depends_on:
|
||||
{{ ESPOCRM_SERVICE }}:
|
||||
condition: service_healthy
|
||||
healthcheck:
|
||||
test: ["CMD", "php", "-r", "require '/var/www/html/bootstrap.php'; echo 'OK';"]
|
||||
interval: 30s
|
||||
timeout: 5s
|
||||
retries: 5
|
||||
start_period: 40s
|
||||
|
||||
{% set service_name = 'websocket' %}
|
||||
{{ service_name }}:
|
||||
image: "{{ ESPOCRM_IMAGE }}:{{ ESPOCRM_VERSION }}"
|
||||
image: "{{ ESPOCRM_CUSTOM_IMAGE }}"
|
||||
container_name: {{ ESPOCRM_CONTAINER }}_{{ service_name }}
|
||||
pull_policy: never
|
||||
init: true
|
||||
stop_signal: SIGTERM
|
||||
stop_grace_period: 30s
|
||||
@@ -46,12 +58,21 @@
|
||||
{% include 'roles/docker-container/templates/depends_on/dmbs_excl.yml.j2' %}
|
||||
{% include 'roles/docker-container/templates/networks.yml.j2' %}
|
||||
volumes:
|
||||
- data:/var/www/html
|
||||
- data:/var/www/html/
|
||||
ports:
|
||||
- "127.0.0.1:{{ ports.localhost.websocket[application_id] }}:8080"
|
||||
depends_on:
|
||||
{{ ESPOCRM_SERVICE }}:
|
||||
condition: service_healthy
|
||||
healthcheck:
|
||||
test: ["CMD", "sh", "-c", "exec 3<>/dev/tcp/127.0.0.1/8080 && echo 'OK' && exec 3<&- 3>&-"]
|
||||
interval: 30s
|
||||
timeout: 5s
|
||||
retries: 5
|
||||
start_period: 40s
|
||||
|
||||
{% include 'roles/docker-compose/templates/volumes.yml.j2' %}
|
||||
data:
|
||||
name: {{ ESPOCRM_VOLUME }}
|
||||
name: {{ ESPOCRM_DATA_VOLUME }}
|
||||
|
||||
{% include 'roles/docker-compose/templates/networks.yml.j2' %}
|
@@ -103,3 +103,9 @@ ESPOCRM_CONFIG_OIDC_USERNAME_CLAIM={{ OIDC.ATTRIBUTES.USERNAME }}
|
||||
# ESPOCRM_CONFIG_OIDC_SYNC_TEAMS=true
|
||||
# ESPOCRM_CONFIG_OIDC_GROUP_CLAIM=group
|
||||
{% endif %}
|
||||
|
||||
# --- Espo init toggles controlled at container start (used by custom entrypoint)
|
||||
ESPO_INIT_MAINTENANCE_MODE=false # false = disable maintenance (recommended)
|
||||
ESPO_INIT_CRON_DISABLED=false # false = enable cron jobs in config
|
||||
ESPO_INIT_USE_CACHE=true # true = enable caching
|
||||
ESPOCRM_SET_FLAGS_SCRIPT={{ ESPOCRM_SET_FLAG_SCRIPT_DOCKER }}
|
||||
|
@@ -1,24 +1,47 @@
|
||||
# General
|
||||
application_id: "web-app-espocrm"
|
||||
entity_name: "{{ application_id | get_entity_name }}"
|
||||
application_id: "web-app-espocrm"
|
||||
entity_name: "{{ application_id | get_entity_name }}"
|
||||
|
||||
# Database
|
||||
database_type: "mariadb"
|
||||
database_type: "mariadb"
|
||||
|
||||
# Webserver
|
||||
location_ws: "/ws"
|
||||
ws_port: "{{ ports.localhost.websocket[application_id] }}"
|
||||
client_max_body_size: "100m"
|
||||
vhost_flavour: "ws_generic"
|
||||
location_ws: "/ws"
|
||||
ws_port: "{{ ports.localhost.websocket[application_id] }}"
|
||||
client_max_body_size: "100m"
|
||||
vhost_flavour: "ws_generic"
|
||||
|
||||
# Espocrm
|
||||
ESPOCRM_VERSION: "{{ applications | get_app_conf(application_id, 'docker.services.espocrm.version') }}"
|
||||
ESPOCRM_IMAGE: "{{ applications | get_app_conf(application_id, 'docker.services.espocrm.image') }}"
|
||||
ESPOCRM_CONTAINER: "{{ applications | get_app_conf(application_id, 'docker.services.espocrm.name') }}"
|
||||
ESPOCRM_VOLUME: "{{ applications | get_app_conf(application_id, 'docker.volumes.data') }}"
|
||||
ESPOCRM_SERVICE: "{{ entity_name }}"
|
||||
|
||||
ESPOCRM_CONFIG_FILE_PRIVATE: "/var/www/html/data/config-internal.php"
|
||||
ESPOCRM_URL: "{{ domains | get_url(application_id, WEB_PROTOCOL) }}"
|
||||
ESPOCRM_OIDC_ENABLED: "{{ applications | get_app_conf(application_id, 'features.oidc') }}"
|
||||
ESPOCRM_USER: "www-data"
|
||||
## Container
|
||||
ESPOCRM_VERSION: "{{ applications | get_app_conf(application_id, 'docker.services.' ~ entity_name ~'.version') }}"
|
||||
ESPOCRM_IMAGE: "{{ applications | get_app_conf(application_id, 'docker.services.' ~ entity_name ~'.image') }}"
|
||||
ESPOCRM_CUSTOM_IMAGE: "custom_espocrm"
|
||||
ESPOCRM_CONTAINER: "{{ applications | get_app_conf(application_id, 'docker.services.' ~ entity_name ~'.name') }}"
|
||||
ESPOCRM_SERVICE: "{{ entity_name }}"
|
||||
|
||||
## Volumes
|
||||
ESPOCRM_DATA_VOLUME: "{{ applications | get_app_conf(application_id, 'docker.volumes.data') }}"
|
||||
|
||||
## Scripts
|
||||
|
||||
### Entrypoint
|
||||
ESPOCRM_ENTRYPOINT_SCRIPT_FILE: "docker-entrypoint-custom.sh"
|
||||
ESPOCRM_ENTRYPOINT_SCRIPT_HOST_ABS: "{{ [ docker_compose.directories.volumes, ESPOCRM_ENTRYPOINT_SCRIPT_FILE ] | path_join }}"
|
||||
ESPOCRM_ENTRYPOINT_SCRIPT_HOST_REL: "volumes/{{ ESPOCRM_ENTRYPOINT_SCRIPT_FILE }}"
|
||||
ESPOCRM_ENTRYPOINT_SCRIPT_DOCKER: "{{ [ '/usr/local/bin/', ESPOCRM_ENTRYPOINT_SCRIPT_FILE ] | path_join }}"
|
||||
|
||||
### Set Flag
|
||||
ESPOCRM_SET_FLAG_SCRIPT_FILE: "set_flags.php"
|
||||
ESPOCRM_SET_FLAG_SCRIPT_HOST_ABS: "{{ [ docker_compose.directories.volumes, ESPOCRM_SET_FLAG_SCRIPT_FILE ] | path_join }}"
|
||||
ESPOCRM_SET_FLAG_SCRIPT_HOST_REL: "volumes/{{ ESPOCRM_SET_FLAG_SCRIPT_FILE }}"
|
||||
ESPOCRM_SET_FLAG_SCRIPT_DOCKER: "{{ [ '/usr/local/bin/', ESPOCRM_SET_FLAG_SCRIPT_FILE ] | path_join }}"
|
||||
|
||||
ESPOCRM_CONFIG_FILE_PRIVATE: "/var/www/html/data/config-internal.php"
|
||||
ESPOCRM_URL: "{{ domains | get_url(application_id, WEB_PROTOCOL) }}"
|
||||
ESPOCRM_OIDC_ENABLED: "{{ applications | get_app_conf(application_id, 'features.oidc') }}"
|
||||
ESPOCRM_USER: "www-data"
|
||||
|
||||
ESPO_INIT_MAINTENANCE_MODE: "{{ applications | get_app_conf(application_id, 'maintenance_mode') | default(false) }}"
|
||||
ESPO_INIT_CRON_DISABLED: "{{ ESPO_INIT_MAINTENANCE_MODE }}" # disable cron only when in maintenance
|
||||
ESPO_INIT_USE_CACHE: "{{ not ESPO_INIT_MAINTENANCE_MODE }}" # enable cache when NOT in maintenance
|
Reference in New Issue
Block a user