Spaces:
Running
Running
| set -euo pipefail | |
| # Tighten default file permissions for any files created by this process | |
| 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}" | |
| # Namespace-based Proxy Configuration (n8n at root internally) | |
| 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}" | |
| # Disable noisy or unnecessary services | |
| 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}" | |
| # n8n v2 uses built-in user management. | |
| 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 | |
| # Stop the background sync loop gracefully | |
| if [ -n "${SYNC_PID:-}" ]; then | |
| kill "$SYNC_PID" 2>/dev/null || true | |
| wait "$SYNC_PID" 2>/dev/null || true | |
| fi | |
| # Wait for n8n to finish its graceful shutdown to ensure DB state is flushed | |
| 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=$! | |
| # Readiness probe | |
| 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" | |