| #!/bin/bash |
| set -euo pipefail |
|
|
| |
| umask 0077 |
|
|
| APP_DIR="/home/node/app" |
| N8N_HOME="/home/node/.n8n" |
| N8N_PORT="${N8N_PORT:-5678}" |
| PUBLIC_PORT="${PUBLIC_PORT:-7861}" |
| SYNC_INTERVAL="${SYNC_INTERVAL:-180}" |
| N8N_STARTUP_TIMEOUT="${N8N_STARTUP_TIMEOUT:-180}" |
|
|
| mkdir -p "$N8N_HOME" |
|
|
| SPACE_HOST_DETECTED="${SPACE_HOST_OVERRIDE:-${SPACE_HOST:-}}" |
| if [ -n "$SPACE_HOST_DETECTED" ]; then |
| export N8N_HOST="${N8N_HOST:-$SPACE_HOST_DETECTED}" |
| |
| export N8N_PATH="/" |
| export N8N_PROTOCOL="https" |
| export N8N_HOST="${SPACE_HOST_DETECTED}" |
| export WEBHOOK_URL="https://${SPACE_HOST_DETECTED}/" |
| export N8N_EDITOR_BASE_URL="https://${SPACE_HOST_DETECTED}/" |
| fi |
|
|
| export N8N_PORT |
| export N8N_PROTOCOL="${N8N_PROTOCOL:-https}" |
| export N8N_PROXY_HOPS="${N8N_PROXY_HOPS:-1}" |
| export N8N_LISTEN_ADDRESS="${N8N_LISTEN_ADDRESS:-0.0.0.0}" |
| if [ -z "${N8N_SECURE_COOKIE:-}" ]; then |
| if [ "${N8N_PROTOCOL}" = "https" ]; then |
| export N8N_SECURE_COOKIE="true" |
| else |
| export N8N_SECURE_COOKIE="false" |
| fi |
| else |
| export N8N_SECURE_COOKIE |
| fi |
| export N8N_DIAGNOSTICS_ENABLED="${N8N_DIAGNOSTICS_ENABLED:-false}" |
| export N8N_PERSONALIZATION_ENABLED="${N8N_PERSONALIZATION_ENABLED:-false}" |
| export N8N_USER_FOLDER="$N8N_HOME" |
| export N8N_ENFORCE_SETTINGS_FILE_PERMISSIONS="${N8N_ENFORCE_SETTINGS_FILE_PERMISSIONS:-true}" |
| export GENERIC_TIMEZONE="${GENERIC_TIMEZONE:-${TZ:-UTC}}" |
| export TZ="${TZ:-$GENERIC_TIMEZONE}" |
|
|
|
|
|
|
| |
| export N8N_PYTHON_NODES_ENABLED="${N8N_PYTHON_NODES_ENABLED:-false}" |
| export N8N_TASK_RUNNERS_ENABLED="${N8N_TASK_RUNNERS_ENABLED:-false}" |
| export N8N_LICENSE_AUTO_RENEW_ENABLED="${N8N_LICENSE_AUTO_RENEW_ENABLED:-false}" |
| export N8N_LOG_LEVEL="${N8N_LOG_LEVEL:-error}" |
|
|
| |
|
|
| echo "" |
| echo " ββββββββββββββββββββββββββββββββββββββ" |
| echo " β Hugging8n β" |
| echo " ββββββββββββββββββββββββββββββββββββββ" |
| echo "" |
| echo "Public host : ${SPACE_HOST_DETECTED:-not detected}" |
| echo "n8n port : ${N8N_PORT}" |
| echo "Public port : ${PUBLIC_PORT}" |
| echo "Timezone : ${GENERIC_TIMEZONE}" |
| echo "Sync every : ${SYNC_INTERVAL}s" |
| echo "Startup wait: ${N8N_STARTUP_TIMEOUT}s" |
|
|
| if [ -n "${HF_TOKEN:-}" ]; then |
| echo "Restoring persisted n8n state from HF Dataset..." |
| python3 "$APP_DIR/n8n-sync.py" restore || true |
| else |
| echo "HF_TOKEN is not set. Running without dataset persistence." |
| fi |
|
|
| CLOUDFLARE_WORKERS_TOKEN="${CLOUDFLARE_WORKERS_TOKEN:-${CLOUDFLARE_API_TOKEN:-}}" |
| export CLOUDFLARE_WORKERS_TOKEN |
| CF_PROXY_ENV_FILE="/tmp/hugging8n-cloudflare-proxy.env" |
| if [ -n "${CLOUDFLARE_WORKERS_TOKEN:-}" ] || [ -n "${CLOUDFLARE_PROXY_URL:-}" ]; then |
| export CLOUDFLARE_PROXY_DEBUG="${CLOUDFLARE_PROXY_DEBUG:-false}" |
| echo "Preparing Cloudflare outbound proxy..." |
| python3 "$APP_DIR/cloudflare-proxy-setup.py" || true |
| if [ -f "$CF_PROXY_ENV_FILE" ]; then |
| . "$CF_PROXY_ENV_FILE" |
| echo " Proxy environment loaded: ${CLOUDFLARE_PROXY_URL:-none}" |
| fi |
| fi |
|
|
| cleanup() { |
| echo "Stopping Hugging8n..." |
| [ -n "${PROXY_PID:-}" ] && kill "$PROXY_PID" 2>/dev/null || true |
|
|
| |
| if [ -n "${SYNC_PID:-}" ]; then |
| kill "$SYNC_PID" 2>/dev/null || true |
| wait "$SYNC_PID" 2>/dev/null || true |
| fi |
|
|
| |
| if [ -n "${N8N_PID:-}" ]; then |
| kill -TERM "$N8N_PID" 2>/dev/null || true |
| wait "$N8N_PID" 2>/dev/null || true |
| fi |
|
|
| if [ -n "${HF_TOKEN:-}" ]; then |
| echo "Running final backup pass..." |
| python3 "$APP_DIR/n8n-sync.py" sync-once || true |
| fi |
| } |
|
|
| trap cleanup EXIT INT TERM |
|
|
| if [ -n "${HF_TOKEN:-}" ]; then |
| python3 "$APP_DIR/n8n-sync.py" loop & |
| SYNC_PID=$! |
| fi |
|
|
| node "$APP_DIR/health-server.js" & |
| PROXY_PID=$! |
|
|
| if [ -n "${CLOUDFLARE_WORKERS_TOKEN:-}" ]; then |
| echo "Setting up Cloudflare KeepAlive monitor..." |
| python3 "$APP_DIR/cloudflare-keepalive-setup.py" || true |
| fi |
|
|
| n8n start & |
| N8N_PID=$! |
|
|
| |
| echo "Waiting for n8n to be ready on port ${N8N_PORT}..." |
| start_ts="$(date +%s)" |
| until curl -sf "http://127.0.0.1:${N8N_PORT}/healthz" > /dev/null 2>&1; do |
| now_ts="$(date +%s)" |
| elapsed="$((now_ts - start_ts))" |
| if [ "$elapsed" -ge "$N8N_STARTUP_TIMEOUT" ]; then |
| echo "n8n did not become ready within ${N8N_STARTUP_TIMEOUT}s. Exiting." |
| kill -TERM "$N8N_PID" 2>/dev/null || true |
| wait "$N8N_PID" 2>/dev/null || true |
| exit 1 |
| fi |
|
|
| if ! kill -0 "$N8N_PID" 2>/dev/null; then |
| echo "n8n process exited before readiness check passed. Exiting." |
| exit 1 |
| fi |
|
|
| sleep 1 |
| done |
| echo "n8n is ready!" |
|
|
| wait "$N8N_PID" |
|
|