@better-openclaw/core 1.0.16 → 1.0.18
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +27 -0
- package/dist/bare-metal-partition.cjs +53 -0
- package/dist/bare-metal-partition.cjs.map +1 -0
- package/dist/bare-metal-partition.d.cts +26 -0
- package/dist/bare-metal-partition.d.cts.map +1 -0
- package/dist/bare-metal-partition.test.cjs +57 -0
- package/dist/bare-metal-partition.test.cjs.map +1 -0
- package/dist/bare-metal-partition.test.d.cts +1 -0
- package/dist/bare-metal-partition.test.mjs +1 -1
- package/dist/chunk-C0xms8kb.cjs +34 -0
- package/dist/compose-validation.test.cjs +67 -0
- package/dist/compose-validation.test.cjs.map +1 -0
- package/dist/compose-validation.test.d.cts +1 -0
- package/dist/compose-validation.test.d.mts +1 -0
- package/dist/compose-validation.test.mjs +66 -0
- package/dist/compose-validation.test.mjs.map +1 -0
- package/dist/composer.cjs +462 -0
- package/dist/composer.cjs.map +1 -0
- package/dist/composer.d.cts +21 -0
- package/dist/composer.d.cts.map +1 -0
- package/dist/composer.d.mts.map +1 -1
- package/dist/composer.mjs +187 -27
- package/dist/composer.mjs.map +1 -1
- package/dist/composer.snapshot.test.cjs +91 -0
- package/dist/composer.snapshot.test.cjs.map +1 -0
- package/dist/composer.snapshot.test.d.cts +1 -0
- package/dist/composer.snapshot.test.mjs +1 -1
- package/dist/composer.test.cjs +166 -0
- package/dist/composer.test.cjs.map +1 -0
- package/dist/composer.test.d.cts +1 -0
- package/dist/composer.test.mjs +2 -2
- package/dist/composer.test.mjs.map +1 -1
- package/dist/deployers/coolify.cjs +174 -0
- package/dist/deployers/coolify.cjs.map +1 -0
- package/dist/deployers/coolify.d.cts +25 -0
- package/dist/deployers/coolify.d.cts.map +1 -0
- package/dist/deployers/coolify.d.mts +25 -0
- package/dist/deployers/coolify.d.mts.map +1 -0
- package/dist/deployers/coolify.mjs +172 -0
- package/dist/deployers/coolify.mjs.map +1 -0
- package/dist/deployers/dokploy.cjs +141 -0
- package/dist/deployers/dokploy.cjs.map +1 -0
- package/dist/deployers/dokploy.d.cts +24 -0
- package/dist/deployers/dokploy.d.cts.map +1 -0
- package/dist/deployers/dokploy.d.mts +24 -0
- package/dist/deployers/dokploy.d.mts.map +1 -0
- package/dist/deployers/dokploy.mjs +139 -0
- package/dist/deployers/dokploy.mjs.map +1 -0
- package/dist/deployers/index.cjs +26 -0
- package/dist/deployers/index.cjs.map +1 -0
- package/dist/deployers/index.d.cts +14 -0
- package/dist/deployers/index.d.cts.map +1 -0
- package/dist/deployers/index.d.mts +14 -0
- package/dist/deployers/index.d.mts.map +1 -0
- package/dist/deployers/index.mjs +21 -0
- package/dist/deployers/index.mjs.map +1 -0
- package/dist/deployers/types.cjs +0 -0
- package/dist/deployers/types.d.cts +72 -0
- package/dist/deployers/types.d.cts.map +1 -0
- package/dist/deployers/types.d.mts +72 -0
- package/dist/deployers/types.d.mts.map +1 -0
- package/dist/deployers/types.mjs +1 -0
- package/dist/errors.cjs +27 -0
- package/dist/errors.cjs.map +1 -0
- package/dist/errors.d.cts +17 -0
- package/dist/errors.d.cts.map +1 -0
- package/dist/generate.cjs +225 -0
- package/dist/generate.cjs.map +1 -0
- package/dist/generate.d.cts +12 -0
- package/dist/generate.d.cts.map +1 -0
- package/dist/generate.d.mts.map +1 -1
- package/dist/generate.mjs +34 -5
- package/dist/generate.mjs.map +1 -1
- package/dist/generate.test.cjs +283 -0
- package/dist/generate.test.cjs.map +1 -0
- package/dist/generate.test.d.cts +1 -0
- package/dist/generate.test.mjs +1 -1
- package/dist/generators/bare-metal-install.cjs +226 -0
- package/dist/generators/bare-metal-install.cjs.map +1 -0
- package/dist/generators/bare-metal-install.d.cts +17 -0
- package/dist/generators/bare-metal-install.d.cts.map +1 -0
- package/dist/generators/bare-metal-install.test.cjs +47 -0
- package/dist/generators/bare-metal-install.test.cjs.map +1 -0
- package/dist/generators/bare-metal-install.test.d.cts +1 -0
- package/dist/generators/bare-metal-install.test.mjs +1 -1
- package/dist/generators/caddy.cjs +79 -0
- package/dist/generators/caddy.cjs.map +1 -0
- package/dist/generators/caddy.d.cts +17 -0
- package/dist/generators/caddy.d.cts.map +1 -0
- package/dist/generators/caddy.test.cjs +44 -0
- package/dist/generators/caddy.test.cjs.map +1 -0
- package/dist/generators/caddy.test.d.cts +1 -0
- package/dist/generators/caddy.test.mjs +1 -1
- package/dist/generators/cloud-init.cjs +132 -0
- package/dist/generators/cloud-init.cjs.map +1 -0
- package/dist/generators/cloud-init.d.cts +25 -0
- package/dist/generators/cloud-init.d.cts.map +1 -0
- package/dist/generators/cloud-init.d.mts +25 -0
- package/dist/generators/cloud-init.d.mts.map +1 -0
- package/dist/generators/cloud-init.mjs +130 -0
- package/dist/generators/cloud-init.mjs.map +1 -0
- package/dist/generators/env.cjs +325 -0
- package/dist/generators/env.cjs.map +1 -0
- package/dist/generators/env.d.cts +52 -0
- package/dist/generators/env.d.cts.map +1 -0
- package/dist/generators/env.d.mts +6 -0
- package/dist/generators/env.d.mts.map +1 -1
- package/dist/generators/env.mjs +58 -6
- package/dist/generators/env.mjs.map +1 -1
- package/dist/generators/env.test.cjs +59 -0
- package/dist/generators/env.test.cjs.map +1 -0
- package/dist/generators/env.test.d.cts +1 -0
- package/dist/generators/env.test.mjs +1 -1
- package/dist/generators/get-shit-done.cjs +40 -0
- package/dist/generators/get-shit-done.cjs.map +1 -0
- package/dist/generators/get-shit-done.d.cts +10 -0
- package/dist/generators/get-shit-done.d.cts.map +1 -0
- package/dist/generators/grafana.cjs +296 -0
- package/dist/generators/grafana.cjs.map +1 -0
- package/dist/generators/grafana.d.cts +22 -0
- package/dist/generators/grafana.d.cts.map +1 -0
- package/dist/generators/health-check.cjs +707 -0
- package/dist/generators/health-check.cjs.map +1 -0
- package/dist/generators/health-check.d.cts +18 -0
- package/dist/generators/health-check.d.cts.map +1 -0
- package/dist/generators/health-check.d.mts +2 -2
- package/dist/generators/health-check.d.mts.map +1 -1
- package/dist/generators/health-check.mjs.map +1 -1
- package/dist/generators/health-check.test.cjs +84 -0
- package/dist/generators/health-check.test.cjs.map +1 -0
- package/dist/generators/health-check.test.d.cts +1 -0
- package/dist/generators/health-check.test.mjs +1 -1
- package/dist/generators/n8n-workflows.cjs +77 -0
- package/dist/generators/n8n-workflows.cjs.map +1 -0
- package/dist/generators/n8n-workflows.d.cts +11 -0
- package/dist/generators/n8n-workflows.d.cts.map +1 -0
- package/dist/generators/native-services.cjs +80 -0
- package/dist/generators/native-services.cjs.map +1 -0
- package/dist/generators/native-services.d.cts +17 -0
- package/dist/generators/native-services.d.cts.map +1 -0
- package/dist/generators/openclaw-install-script.cjs +68 -0
- package/dist/generators/openclaw-install-script.cjs.map +1 -0
- package/dist/generators/openclaw-install-script.d.cts +12 -0
- package/dist/generators/openclaw-install-script.d.cts.map +1 -0
- package/dist/generators/openclaw-install-script.d.mts +12 -0
- package/dist/generators/openclaw-install-script.d.mts.map +1 -0
- package/dist/generators/openclaw-install-script.mjs +66 -0
- package/dist/generators/openclaw-install-script.mjs.map +1 -0
- package/dist/generators/openclaw-json.cjs +540 -0
- package/dist/generators/openclaw-json.cjs.map +1 -0
- package/dist/generators/openclaw-json.d.cts +16 -0
- package/dist/generators/openclaw-json.d.cts.map +1 -0
- package/dist/generators/openclaw-json.d.mts +8 -3
- package/dist/generators/openclaw-json.d.mts.map +1 -1
- package/dist/generators/openclaw-json.mjs +214 -86
- package/dist/generators/openclaw-json.mjs.map +1 -1
- package/dist/generators/postgres-init.cjs +189 -0
- package/dist/generators/postgres-init.cjs.map +1 -0
- package/dist/generators/postgres-init.d.cts +30 -0
- package/dist/generators/postgres-init.d.cts.map +1 -0
- package/dist/generators/postgres-init.d.mts.map +1 -1
- package/dist/generators/postgres-init.mjs +35 -0
- package/dist/generators/postgres-init.mjs.map +1 -1
- package/dist/generators/prometheus.cjs +109 -0
- package/dist/generators/prometheus.cjs.map +1 -0
- package/dist/generators/prometheus.d.cts +14 -0
- package/dist/generators/prometheus.d.cts.map +1 -0
- package/dist/generators/readme.cjs +286 -0
- package/dist/generators/readme.cjs.map +1 -0
- package/dist/generators/readme.d.cts +27 -0
- package/dist/generators/readme.d.cts.map +1 -0
- package/dist/generators/readme.d.mts +3 -1
- package/dist/generators/readme.d.mts.map +1 -1
- package/dist/generators/readme.mjs +88 -3
- package/dist/generators/readme.mjs.map +1 -1
- package/dist/generators/scripts.cjs +353 -0
- package/dist/generators/scripts.cjs.map +1 -0
- package/dist/generators/scripts.d.cts +10 -0
- package/dist/generators/scripts.d.cts.map +1 -0
- package/dist/generators/scripts.test.cjs +51 -0
- package/dist/generators/scripts.test.cjs.map +1 -0
- package/dist/generators/scripts.test.d.cts +1 -0
- package/dist/generators/scripts.test.mjs +1 -1
- package/dist/generators/skills.cjs +590 -0
- package/dist/generators/skills.cjs.map +1 -0
- package/dist/generators/skills.d.cts +14 -0
- package/dist/generators/skills.d.cts.map +1 -0
- package/dist/generators/stack-manifest.cjs +70 -0
- package/dist/generators/stack-manifest.cjs.map +1 -0
- package/dist/generators/stack-manifest.d.cts +51 -0
- package/dist/generators/stack-manifest.d.cts.map +1 -0
- package/dist/generators/stack-manifest.d.mts +51 -0
- package/dist/generators/stack-manifest.d.mts.map +1 -0
- package/dist/generators/stack-manifest.mjs +68 -0
- package/dist/generators/stack-manifest.mjs.map +1 -0
- package/dist/generators/traefik.cjs +88 -0
- package/dist/generators/traefik.cjs.map +1 -0
- package/dist/generators/traefik.d.cts +19 -0
- package/dist/generators/traefik.d.cts.map +1 -0
- package/dist/generators/traefik.test.cjs +68 -0
- package/dist/generators/traefik.test.cjs.map +1 -0
- package/dist/generators/traefik.test.d.cts +1 -0
- package/dist/generators/traefik.test.mjs +1 -1
- package/dist/index.cjs +118 -0
- package/dist/index.d.cts +31 -0
- package/dist/index.d.mts +9 -3
- package/dist/index.mjs +8 -3
- package/dist/magic-string.es-D2agHT3I.cjs +1015 -0
- package/dist/magic-string.es-D2agHT3I.cjs.map +1 -0
- package/dist/manifest-4uLbISXV.cjs +1190 -0
- package/dist/manifest-4uLbISXV.cjs.map +1 -0
- package/dist/migrations.cjs +37 -0
- package/dist/migrations.cjs.map +1 -0
- package/dist/migrations.d.cts +14 -0
- package/dist/migrations.d.cts.map +1 -0
- package/dist/migrations.test.cjs +41 -0
- package/dist/migrations.test.cjs.map +1 -0
- package/dist/migrations.test.d.cts +1 -0
- package/dist/migrations.test.mjs +1 -1
- package/dist/port-scanner.cjs +125 -0
- package/dist/port-scanner.cjs.map +1 -0
- package/dist/port-scanner.d.cts +27 -0
- package/dist/port-scanner.d.cts.map +1 -0
- package/dist/port-scanner.d.mts +27 -0
- package/dist/port-scanner.d.mts.map +1 -0
- package/dist/port-scanner.mjs +122 -0
- package/dist/port-scanner.mjs.map +1 -0
- package/dist/presets/presets.test.cjs +65 -0
- package/dist/presets/presets.test.cjs.map +1 -0
- package/dist/presets/presets.test.d.cts +1 -0
- package/dist/presets/presets.test.d.mts +1 -0
- package/dist/presets/presets.test.mjs +64 -0
- package/dist/presets/presets.test.mjs.map +1 -0
- package/dist/presets/registry.cjs +193 -0
- package/dist/presets/registry.cjs.map +1 -0
- package/dist/presets/registry.d.cts +9 -0
- package/dist/presets/registry.d.cts.map +1 -0
- package/dist/presets/registry.d.mts.map +1 -1
- package/dist/presets/registry.mjs +38 -8
- package/dist/presets/registry.mjs.map +1 -1
- package/dist/presets/registry.test.cjs +36 -0
- package/dist/presets/registry.test.cjs.map +1 -0
- package/dist/presets/registry.test.d.cts +1 -0
- package/dist/presets/registry.test.mjs +5 -2
- package/dist/presets/registry.test.mjs.map +1 -1
- package/dist/resolver.cjs +217 -0
- package/dist/resolver.cjs.map +1 -0
- package/dist/resolver.d.cts +26 -0
- package/dist/resolver.d.cts.map +1 -0
- package/dist/resolver.mjs +2 -0
- package/dist/resolver.mjs.map +1 -1
- package/dist/resolver.test.cjs +236 -0
- package/dist/resolver.test.cjs.map +1 -0
- package/dist/resolver.test.d.cts +1 -0
- package/dist/resolver.test.mjs +1 -1
- package/dist/schema.cjs +354 -0
- package/dist/schema.cjs.map +1 -0
- package/dist/schema.d.cts +854 -0
- package/dist/schema.d.cts.map +1 -0
- package/dist/schema.d.mts +56 -1
- package/dist/schema.d.mts.map +1 -1
- package/dist/schema.mjs +31 -4
- package/dist/schema.mjs.map +1 -1
- package/dist/schema.test.cjs +263 -0
- package/dist/schema.test.cjs.map +1 -0
- package/dist/schema.test.d.cts +1 -0
- package/dist/schema.test.mjs +1 -1
- package/dist/services/definitions/anything-llm.cjs +46 -0
- package/dist/services/definitions/anything-llm.cjs.map +1 -0
- package/dist/services/definitions/anything-llm.d.cts +7 -0
- package/dist/services/definitions/anything-llm.d.cts.map +1 -0
- package/dist/services/definitions/anything-llm.mjs +2 -2
- package/dist/services/definitions/anything-llm.mjs.map +1 -1
- package/dist/services/definitions/appflowy.cjs +47 -0
- package/dist/services/definitions/appflowy.cjs.map +1 -0
- package/dist/services/definitions/appflowy.d.cts +7 -0
- package/dist/services/definitions/appflowy.d.cts.map +1 -0
- package/dist/services/definitions/appflowy.mjs +1 -1
- package/dist/services/definitions/appflowy.mjs.map +1 -1
- package/dist/services/definitions/authentik.cjs +107 -0
- package/dist/services/definitions/authentik.cjs.map +1 -0
- package/dist/services/definitions/authentik.d.cts +7 -0
- package/dist/services/definitions/authentik.d.cts.map +1 -0
- package/dist/services/definitions/authentik.mjs +5 -5
- package/dist/services/definitions/authentik.mjs.map +1 -1
- package/dist/services/definitions/beszel.cjs +47 -0
- package/dist/services/definitions/beszel.cjs.map +1 -0
- package/dist/services/definitions/beszel.d.cts +7 -0
- package/dist/services/definitions/beszel.d.cts.map +1 -0
- package/dist/services/definitions/beszel.mjs +1 -1
- package/dist/services/definitions/beszel.mjs.map +1 -1
- package/dist/services/definitions/browserless.cjs +87 -0
- package/dist/services/definitions/browserless.cjs.map +1 -0
- package/dist/services/definitions/browserless.d.cts +7 -0
- package/dist/services/definitions/browserless.d.cts.map +1 -0
- package/dist/services/definitions/browserless.mjs +1 -1
- package/dist/services/definitions/browserless.mjs.map +1 -1
- package/dist/services/definitions/caddy.cjs +73 -0
- package/dist/services/definitions/caddy.cjs.map +1 -0
- package/dist/services/definitions/caddy.d.cts +7 -0
- package/dist/services/definitions/caddy.d.cts.map +1 -0
- package/dist/services/definitions/caddy.mjs +2 -2
- package/dist/services/definitions/caddy.mjs.map +1 -1
- package/dist/services/definitions/cal-com.cjs +90 -0
- package/dist/services/definitions/cal-com.cjs.map +1 -0
- package/dist/services/definitions/cal-com.d.cts +7 -0
- package/dist/services/definitions/cal-com.d.cts.map +1 -0
- package/dist/services/definitions/cal-com.mjs +2 -2
- package/dist/services/definitions/cal-com.mjs.map +1 -1
- package/dist/services/definitions/chromadb.cjs +65 -0
- package/dist/services/definitions/chromadb.cjs.map +1 -0
- package/dist/services/definitions/chromadb.d.cts +7 -0
- package/dist/services/definitions/chromadb.d.cts.map +1 -0
- package/dist/services/definitions/chromadb.mjs +3 -3
- package/dist/services/definitions/chromadb.mjs.map +1 -1
- package/dist/services/definitions/claude-code.cjs +54 -0
- package/dist/services/definitions/claude-code.cjs.map +1 -0
- package/dist/services/definitions/claude-code.d.cts +7 -0
- package/dist/services/definitions/claude-code.d.cts.map +1 -0
- package/dist/services/definitions/claude-code.mjs +1 -1
- package/dist/services/definitions/claude-code.mjs.map +1 -1
- package/dist/services/definitions/code-server.cjs +68 -0
- package/dist/services/definitions/code-server.cjs.map +1 -0
- package/dist/services/definitions/code-server.d.cts +7 -0
- package/dist/services/definitions/code-server.d.cts.map +1 -0
- package/dist/services/definitions/code-server.mjs +1 -1
- package/dist/services/definitions/code-server.mjs.map +1 -1
- package/dist/services/definitions/codex.cjs +47 -0
- package/dist/services/definitions/codex.cjs.map +1 -0
- package/dist/services/definitions/codex.d.cts +7 -0
- package/dist/services/definitions/codex.d.cts.map +1 -0
- package/dist/services/definitions/codex.mjs +1 -1
- package/dist/services/definitions/codex.mjs.map +1 -1
- package/dist/services/definitions/comfyui.cjs +85 -0
- package/dist/services/definitions/comfyui.cjs.map +1 -0
- package/dist/services/definitions/comfyui.d.cts +7 -0
- package/dist/services/definitions/comfyui.d.cts.map +1 -0
- package/dist/services/definitions/comfyui.mjs +1 -1
- package/dist/services/definitions/comfyui.mjs.map +1 -1
- package/dist/services/definitions/convex-dashboard.cjs +48 -0
- package/dist/services/definitions/convex-dashboard.cjs.map +1 -0
- package/dist/services/definitions/convex-dashboard.d.cts +7 -0
- package/dist/services/definitions/convex-dashboard.d.cts.map +1 -0
- package/dist/services/definitions/convex.cjs +120 -0
- package/dist/services/definitions/convex.cjs.map +1 -0
- package/dist/services/definitions/convex.d.cts +7 -0
- package/dist/services/definitions/convex.d.cts.map +1 -0
- package/dist/services/definitions/coolify.cjs +51 -0
- package/dist/services/definitions/coolify.cjs.map +1 -0
- package/dist/services/definitions/coolify.d.cts +7 -0
- package/dist/services/definitions/coolify.d.cts.map +1 -0
- package/dist/services/definitions/coolify.mjs +1 -1
- package/dist/services/definitions/coolify.mjs.map +1 -1
- package/dist/services/definitions/crowdsec.cjs +64 -0
- package/dist/services/definitions/crowdsec.cjs.map +1 -0
- package/dist/services/definitions/crowdsec.d.cts +7 -0
- package/dist/services/definitions/crowdsec.d.cts.map +1 -0
- package/dist/services/definitions/crowdsec.mjs +2 -2
- package/dist/services/definitions/crowdsec.mjs.map +1 -1
- package/dist/services/definitions/desktop-environment.cjs +155 -0
- package/dist/services/definitions/desktop-environment.cjs.map +1 -0
- package/dist/services/definitions/desktop-environment.d.cts +7 -0
- package/dist/services/definitions/desktop-environment.d.cts.map +1 -0
- package/dist/services/definitions/desktop-environment.mjs +1 -1
- package/dist/services/definitions/desktop-environment.mjs.map +1 -1
- package/dist/services/definitions/dify.cjs +83 -0
- package/dist/services/definitions/dify.cjs.map +1 -0
- package/dist/services/definitions/dify.d.cts +7 -0
- package/dist/services/definitions/dify.d.cts.map +1 -0
- package/dist/services/definitions/dify.mjs +1 -1
- package/dist/services/definitions/dify.mjs.map +1 -1
- package/dist/services/definitions/docsgpt.cjs +47 -0
- package/dist/services/definitions/docsgpt.cjs.map +1 -0
- package/dist/services/definitions/docsgpt.d.cts +7 -0
- package/dist/services/definitions/docsgpt.d.cts.map +1 -0
- package/dist/services/definitions/docsgpt.mjs +1 -1
- package/dist/services/definitions/docsgpt.mjs.map +1 -1
- package/dist/services/definitions/dokploy.cjs +51 -0
- package/dist/services/definitions/dokploy.cjs.map +1 -0
- package/dist/services/definitions/dokploy.d.cts +7 -0
- package/dist/services/definitions/dokploy.d.cts.map +1 -0
- package/dist/services/definitions/dokploy.mjs +2 -2
- package/dist/services/definitions/dokploy.mjs.map +1 -1
- package/dist/services/definitions/dozzle.cjs +43 -0
- package/dist/services/definitions/dozzle.cjs.map +1 -0
- package/dist/services/definitions/dozzle.d.cts +7 -0
- package/dist/services/definitions/dozzle.d.cts.map +1 -0
- package/dist/services/definitions/dozzle.mjs +1 -1
- package/dist/services/definitions/dozzle.mjs.map +1 -1
- package/dist/services/definitions/ffmpeg.cjs +68 -0
- package/dist/services/definitions/ffmpeg.cjs.map +1 -0
- package/dist/services/definitions/ffmpeg.d.cts +7 -0
- package/dist/services/definitions/ffmpeg.d.cts.map +1 -0
- package/dist/services/definitions/ffmpeg.mjs +2 -2
- package/dist/services/definitions/ffmpeg.mjs.map +1 -1
- package/dist/services/definitions/flowise.cjs +47 -0
- package/dist/services/definitions/flowise.cjs.map +1 -0
- package/dist/services/definitions/flowise.d.cts +7 -0
- package/dist/services/definitions/flowise.d.cts.map +1 -0
- package/dist/services/definitions/flowise.mjs +1 -1
- package/dist/services/definitions/flowise.mjs.map +1 -1
- package/dist/services/definitions/gemini-cli.cjs +54 -0
- package/dist/services/definitions/gemini-cli.cjs.map +1 -0
- package/dist/services/definitions/gemini-cli.d.cts +7 -0
- package/dist/services/definitions/gemini-cli.d.cts.map +1 -0
- package/dist/services/definitions/gemini-cli.mjs +7 -1
- package/dist/services/definitions/gemini-cli.mjs.map +1 -1
- package/dist/services/definitions/ghost.cjs +60 -0
- package/dist/services/definitions/ghost.cjs.map +1 -0
- package/dist/services/definitions/ghost.d.cts +7 -0
- package/dist/services/definitions/ghost.d.cts.map +1 -0
- package/dist/services/definitions/ghost.mjs +1 -1
- package/dist/services/definitions/ghost.mjs.map +1 -1
- package/dist/services/definitions/gitea.cjs +51 -0
- package/dist/services/definitions/gitea.cjs.map +1 -0
- package/dist/services/definitions/gitea.d.cts +7 -0
- package/dist/services/definitions/gitea.d.cts.map +1 -0
- package/dist/services/definitions/gitea.mjs +1 -1
- package/dist/services/definitions/gitea.mjs.map +1 -1
- package/dist/services/definitions/gotify.cjs +64 -0
- package/dist/services/definitions/gotify.cjs.map +1 -0
- package/dist/services/definitions/gotify.d.cts +7 -0
- package/dist/services/definitions/gotify.d.cts.map +1 -0
- package/dist/services/definitions/gotify.mjs +3 -3
- package/dist/services/definitions/gotify.mjs.map +1 -1
- package/dist/services/definitions/grafana.cjs +77 -0
- package/dist/services/definitions/grafana.cjs.map +1 -0
- package/dist/services/definitions/grafana.d.cts +7 -0
- package/dist/services/definitions/grafana.d.cts.map +1 -0
- package/dist/services/definitions/grafana.mjs +2 -2
- package/dist/services/definitions/grafana.mjs.map +1 -1
- package/dist/services/definitions/headscale.cjs +73 -0
- package/dist/services/definitions/headscale.cjs.map +1 -0
- package/dist/services/definitions/headscale.d.cts +7 -0
- package/dist/services/definitions/headscale.d.cts.map +1 -0
- package/dist/services/definitions/headscale.mjs +2 -2
- package/dist/services/definitions/headscale.mjs.map +1 -1
- package/dist/services/definitions/hexstrike.cjs +46 -0
- package/dist/services/definitions/hexstrike.cjs.map +1 -0
- package/dist/services/definitions/hexstrike.d.cts +7 -0
- package/dist/services/definitions/hexstrike.d.cts.map +1 -0
- package/dist/services/definitions/hexstrike.mjs +1 -1
- package/dist/services/definitions/hexstrike.mjs.map +1 -1
- package/dist/services/definitions/homeassistant.cjs +63 -0
- package/dist/services/definitions/homeassistant.cjs.map +1 -0
- package/dist/services/definitions/homeassistant.d.cts +7 -0
- package/dist/services/definitions/homeassistant.d.cts.map +1 -0
- package/dist/services/definitions/homeassistant.mjs +1 -1
- package/dist/services/definitions/homeassistant.mjs.map +1 -1
- package/dist/services/definitions/immich.cjs +94 -0
- package/dist/services/definitions/immich.cjs.map +1 -0
- package/dist/services/definitions/immich.d.cts +7 -0
- package/dist/services/definitions/immich.d.cts.map +1 -0
- package/dist/services/definitions/immich.mjs +6 -6
- package/dist/services/definitions/immich.mjs.map +1 -1
- package/dist/services/definitions/index.cjs +291 -0
- package/dist/services/definitions/index.cjs.map +1 -0
- package/dist/services/definitions/index.d.cts +101 -0
- package/dist/services/definitions/index.d.cts.map +1 -0
- package/dist/services/definitions/jellyfin.cjs +64 -0
- package/dist/services/definitions/jellyfin.cjs.map +1 -0
- package/dist/services/definitions/jellyfin.d.cts +7 -0
- package/dist/services/definitions/jellyfin.d.cts.map +1 -0
- package/dist/services/definitions/jellyfin.mjs +1 -1
- package/dist/services/definitions/jellyfin.mjs.map +1 -1
- package/dist/services/definitions/jenkins.cjs +64 -0
- package/dist/services/definitions/jenkins.cjs.map +1 -0
- package/dist/services/definitions/jenkins.d.cts +7 -0
- package/dist/services/definitions/jenkins.d.cts.map +1 -0
- package/dist/services/definitions/jenkins.mjs +1 -1
- package/dist/services/definitions/jenkins.mjs.map +1 -1
- package/dist/services/definitions/kimi.cjs +48 -0
- package/dist/services/definitions/kimi.cjs.map +1 -0
- package/dist/services/definitions/kimi.d.cts +7 -0
- package/dist/services/definitions/kimi.d.cts.map +1 -0
- package/dist/services/definitions/kimi.mjs +1 -1
- package/dist/services/definitions/kimi.mjs.map +1 -1
- package/dist/services/definitions/lasuite-meet-agents.cjs +38 -0
- package/dist/services/definitions/lasuite-meet-agents.cjs.map +1 -0
- package/dist/services/definitions/lasuite-meet-agents.d.cts +7 -0
- package/dist/services/definitions/lasuite-meet-agents.d.cts.map +1 -0
- package/dist/services/definitions/lasuite-meet-agents.mjs +1 -1
- package/dist/services/definitions/lasuite-meet-agents.mjs.map +1 -1
- package/dist/services/definitions/lasuite-meet-backend.cjs +136 -0
- package/dist/services/definitions/lasuite-meet-backend.cjs.map +1 -0
- package/dist/services/definitions/lasuite-meet-backend.d.cts +7 -0
- package/dist/services/definitions/lasuite-meet-backend.d.cts.map +1 -0
- package/dist/services/definitions/lasuite-meet-backend.mjs +3 -3
- package/dist/services/definitions/lasuite-meet-backend.mjs.map +1 -1
- package/dist/services/definitions/lasuite-meet-frontend.cjs +50 -0
- package/dist/services/definitions/lasuite-meet-frontend.cjs.map +1 -0
- package/dist/services/definitions/lasuite-meet-frontend.d.cts +7 -0
- package/dist/services/definitions/lasuite-meet-frontend.d.cts.map +1 -0
- package/dist/services/definitions/lasuite-meet-frontend.mjs +3 -3
- package/dist/services/definitions/lasuite-meet-frontend.mjs.map +1 -1
- package/dist/services/definitions/librechat.cjs +52 -0
- package/dist/services/definitions/librechat.cjs.map +1 -0
- package/dist/services/definitions/librechat.d.cts +7 -0
- package/dist/services/definitions/librechat.d.cts.map +1 -0
- package/dist/services/definitions/librechat.mjs +1 -1
- package/dist/services/definitions/librechat.mjs.map +1 -1
- package/dist/services/definitions/lightpanda.cjs +61 -0
- package/dist/services/definitions/lightpanda.cjs.map +1 -0
- package/dist/services/definitions/lightpanda.d.cts +7 -0
- package/dist/services/definitions/lightpanda.d.cts.map +1 -0
- package/dist/services/definitions/litellm.cjs +49 -0
- package/dist/services/definitions/litellm.cjs.map +1 -0
- package/dist/services/definitions/litellm.d.cts +7 -0
- package/dist/services/definitions/litellm.d.cts.map +1 -0
- package/dist/services/definitions/litellm.mjs +1 -1
- package/dist/services/definitions/litellm.mjs.map +1 -1
- package/dist/services/definitions/livekit.cjs +58 -0
- package/dist/services/definitions/livekit.cjs.map +1 -0
- package/dist/services/definitions/livekit.d.cts +7 -0
- package/dist/services/definitions/livekit.d.cts.map +1 -0
- package/dist/services/definitions/loki.cjs +53 -0
- package/dist/services/definitions/loki.cjs.map +1 -0
- package/dist/services/definitions/loki.d.cts +7 -0
- package/dist/services/definitions/loki.d.cts.map +1 -0
- package/dist/services/definitions/loki.mjs +1 -1
- package/dist/services/definitions/loki.mjs.map +1 -1
- package/dist/services/definitions/matomo.cjs +82 -0
- package/dist/services/definitions/matomo.cjs.map +1 -0
- package/dist/services/definitions/matomo.d.cts +7 -0
- package/dist/services/definitions/matomo.d.cts.map +1 -0
- package/dist/services/definitions/matomo.mjs +1 -1
- package/dist/services/definitions/matomo.mjs.map +1 -1
- package/dist/services/definitions/matrix-synapse.cjs +90 -0
- package/dist/services/definitions/matrix-synapse.cjs.map +1 -0
- package/dist/services/definitions/matrix-synapse.d.cts +7 -0
- package/dist/services/definitions/matrix-synapse.d.cts.map +1 -0
- package/dist/services/definitions/matrix-synapse.mjs +1 -1
- package/dist/services/definitions/matrix-synapse.mjs.map +1 -1
- package/dist/services/definitions/mattermost.cjs +63 -0
- package/dist/services/definitions/mattermost.cjs.map +1 -0
- package/dist/services/definitions/mattermost.d.cts +7 -0
- package/dist/services/definitions/mattermost.d.cts.map +1 -0
- package/dist/services/definitions/mattermost.mjs +1 -1
- package/dist/services/definitions/mattermost.mjs.map +1 -1
- package/dist/services/definitions/meilisearch.cjs +80 -0
- package/dist/services/definitions/meilisearch.cjs.map +1 -0
- package/dist/services/definitions/meilisearch.d.cts +7 -0
- package/dist/services/definitions/meilisearch.d.cts.map +1 -0
- package/dist/services/definitions/meilisearch.mjs +4 -4
- package/dist/services/definitions/meilisearch.mjs.map +1 -1
- package/dist/services/definitions/milvus.cjs +77 -0
- package/dist/services/definitions/milvus.cjs.map +1 -0
- package/dist/services/definitions/milvus.d.cts +7 -0
- package/dist/services/definitions/milvus.d.cts.map +1 -0
- package/dist/services/definitions/milvus.mjs +1 -1
- package/dist/services/definitions/milvus.mjs.map +1 -1
- package/dist/services/definitions/minio.cjs +104 -0
- package/dist/services/definitions/minio.cjs.map +1 -0
- package/dist/services/definitions/minio.d.cts +7 -0
- package/dist/services/definitions/minio.d.cts.map +1 -0
- package/dist/services/definitions/minio.mjs +1 -1
- package/dist/services/definitions/minio.mjs.map +1 -1
- package/dist/services/definitions/mission-control.cjs +69 -0
- package/dist/services/definitions/mission-control.cjs.map +1 -0
- package/dist/services/definitions/mission-control.d.cts +7 -0
- package/dist/services/definitions/mission-control.d.cts.map +1 -0
- package/dist/services/definitions/mixpost.cjs +89 -0
- package/dist/services/definitions/mixpost.cjs.map +1 -0
- package/dist/services/definitions/mixpost.d.cts +7 -0
- package/dist/services/definitions/mixpost.d.cts.map +1 -0
- package/dist/services/definitions/mixpost.mjs +1 -1
- package/dist/services/definitions/mixpost.mjs.map +1 -1
- package/dist/services/definitions/motion-canvas.cjs +54 -0
- package/dist/services/definitions/motion-canvas.cjs.map +1 -0
- package/dist/services/definitions/motion-canvas.d.cts +7 -0
- package/dist/services/definitions/motion-canvas.d.cts.map +1 -0
- package/dist/services/definitions/motion-canvas.mjs +1 -1
- package/dist/services/definitions/motion-canvas.mjs.map +1 -1
- package/dist/services/definitions/n8n.cjs +156 -0
- package/dist/services/definitions/n8n.cjs.map +1 -0
- package/dist/services/definitions/n8n.d.cts +7 -0
- package/dist/services/definitions/n8n.d.cts.map +1 -0
- package/dist/services/definitions/n8n.mjs +1 -1
- package/dist/services/definitions/n8n.mjs.map +1 -1
- package/dist/services/definitions/neo4j.cjs +93 -0
- package/dist/services/definitions/neo4j.cjs.map +1 -0
- package/dist/services/definitions/neo4j.d.cts +7 -0
- package/dist/services/definitions/neo4j.d.cts.map +1 -0
- package/dist/services/definitions/neo4j.mjs +1 -1
- package/dist/services/definitions/neo4j.mjs.map +1 -1
- package/dist/services/definitions/nextcloud.cjs +82 -0
- package/dist/services/definitions/nextcloud.cjs.map +1 -0
- package/dist/services/definitions/nextcloud.d.cts +7 -0
- package/dist/services/definitions/nextcloud.d.cts.map +1 -0
- package/dist/services/definitions/nextcloud.mjs +19 -19
- package/dist/services/definitions/nextcloud.mjs.map +1 -1
- package/dist/services/definitions/nocodb.cjs +47 -0
- package/dist/services/definitions/nocodb.cjs.map +1 -0
- package/dist/services/definitions/nocodb.d.cts +7 -0
- package/dist/services/definitions/nocodb.d.cts.map +1 -0
- package/dist/services/definitions/nocodb.mjs +1 -1
- package/dist/services/definitions/nocodb.mjs.map +1 -1
- package/dist/services/definitions/ntfy.cjs +65 -0
- package/dist/services/definitions/ntfy.cjs.map +1 -0
- package/dist/services/definitions/ntfy.d.cts +7 -0
- package/dist/services/definitions/ntfy.d.cts.map +1 -0
- package/dist/services/definitions/ntfy.mjs +2 -2
- package/dist/services/definitions/ntfy.mjs.map +1 -1
- package/dist/services/definitions/ollama.cjs +68 -0
- package/dist/services/definitions/ollama.cjs.map +1 -0
- package/dist/services/definitions/ollama.d.cts +7 -0
- package/dist/services/definitions/ollama.d.cts.map +1 -0
- package/dist/services/definitions/ollama.mjs +1 -1
- package/dist/services/definitions/ollama.mjs.map +1 -1
- package/dist/services/definitions/open-webui.cjs +52 -0
- package/dist/services/definitions/open-webui.cjs.map +1 -0
- package/dist/services/definitions/open-webui.d.cts +7 -0
- package/dist/services/definitions/open-webui.d.cts.map +1 -0
- package/dist/services/definitions/open-webui.mjs +1 -1
- package/dist/services/definitions/open-webui.mjs.map +1 -1
- package/dist/services/definitions/opencode.cjs +52 -0
- package/dist/services/definitions/opencode.cjs.map +1 -0
- package/dist/services/definitions/opencode.d.cts +7 -0
- package/dist/services/definitions/opencode.d.cts.map +1 -0
- package/dist/services/definitions/opencode.mjs +1 -1
- package/dist/services/definitions/opencode.mjs.map +1 -1
- package/dist/services/definitions/openpanel.cjs +48 -0
- package/dist/services/definitions/openpanel.cjs.map +1 -0
- package/dist/services/definitions/openpanel.d.cts +7 -0
- package/dist/services/definitions/openpanel.d.cts.map +1 -0
- package/dist/services/definitions/openpanel.mjs +3 -3
- package/dist/services/definitions/openpanel.mjs.map +1 -1
- package/dist/services/definitions/outline.cjs +76 -0
- package/dist/services/definitions/outline.cjs.map +1 -0
- package/dist/services/definitions/outline.d.cts +7 -0
- package/dist/services/definitions/outline.d.cts.map +1 -0
- package/dist/services/definitions/paperless-ngx.cjs +95 -0
- package/dist/services/definitions/paperless-ngx.cjs.map +1 -0
- package/dist/services/definitions/paperless-ngx.d.cts +7 -0
- package/dist/services/definitions/paperless-ngx.d.cts.map +1 -0
- package/dist/services/definitions/paperless-ngx.mjs +1 -1
- package/dist/services/definitions/paperless-ngx.mjs.map +1 -1
- package/dist/services/definitions/pentagi.cjs +46 -0
- package/dist/services/definitions/pentagi.cjs.map +1 -0
- package/dist/services/definitions/pentagi.d.cts +7 -0
- package/dist/services/definitions/pentagi.d.cts.map +1 -0
- package/dist/services/definitions/pentagi.mjs +2 -2
- package/dist/services/definitions/pentagi.mjs.map +1 -1
- package/dist/services/definitions/pentestagent.cjs +41 -0
- package/dist/services/definitions/pentestagent.cjs.map +1 -0
- package/dist/services/definitions/pentestagent.d.cts +7 -0
- package/dist/services/definitions/pentestagent.d.cts.map +1 -0
- package/dist/services/definitions/playwright-server.cjs +44 -0
- package/dist/services/definitions/playwright-server.cjs.map +1 -0
- package/dist/services/definitions/playwright-server.d.cts +7 -0
- package/dist/services/definitions/playwright-server.d.cts.map +1 -0
- package/dist/services/definitions/playwright-server.mjs +1 -1
- package/dist/services/definitions/playwright-server.mjs.map +1 -1
- package/dist/services/definitions/portainer.cjs +51 -0
- package/dist/services/definitions/portainer.cjs.map +1 -0
- package/dist/services/definitions/portainer.d.cts +7 -0
- package/dist/services/definitions/portainer.d.cts.map +1 -0
- package/dist/services/definitions/portainer.mjs +1 -1
- package/dist/services/definitions/portainer.mjs.map +1 -1
- package/dist/services/definitions/postgresql.cjs +86 -0
- package/dist/services/definitions/postgresql.cjs.map +1 -0
- package/dist/services/definitions/postgresql.d.cts +7 -0
- package/dist/services/definitions/postgresql.d.cts.map +1 -0
- package/dist/services/definitions/postgresql.mjs +1 -1
- package/dist/services/definitions/postgresql.mjs.map +1 -1
- package/dist/services/definitions/postiz.cjs +85 -0
- package/dist/services/definitions/postiz.cjs.map +1 -0
- package/dist/services/definitions/postiz.d.cts +7 -0
- package/dist/services/definitions/postiz.d.cts.map +1 -0
- package/dist/services/definitions/postiz.mjs +1 -1
- package/dist/services/definitions/postiz.mjs.map +1 -1
- package/dist/services/definitions/prometheus.cjs +54 -0
- package/dist/services/definitions/prometheus.cjs.map +1 -0
- package/dist/services/definitions/prometheus.d.cts +7 -0
- package/dist/services/definitions/prometheus.d.cts.map +1 -0
- package/dist/services/definitions/prometheus.mjs +1 -1
- package/dist/services/definitions/prometheus.mjs.map +1 -1
- package/dist/services/definitions/qdrant.cjs +79 -0
- package/dist/services/definitions/qdrant.cjs.map +1 -0
- package/dist/services/definitions/qdrant.d.cts +7 -0
- package/dist/services/definitions/qdrant.d.cts.map +1 -0
- package/dist/services/definitions/qdrant.mjs +1 -1
- package/dist/services/definitions/qdrant.mjs.map +1 -1
- package/dist/services/definitions/redis.cjs +93 -0
- package/dist/services/definitions/redis.cjs.map +1 -0
- package/dist/services/definitions/redis.d.cts +7 -0
- package/dist/services/definitions/redis.d.cts.map +1 -0
- package/dist/services/definitions/redis.mjs +1 -1
- package/dist/services/definitions/redis.mjs.map +1 -1
- package/dist/services/definitions/remotion.cjs +57 -0
- package/dist/services/definitions/remotion.cjs.map +1 -0
- package/dist/services/definitions/remotion.d.cts +7 -0
- package/dist/services/definitions/remotion.d.cts.map +1 -0
- package/dist/services/definitions/remotion.mjs +1 -1
- package/dist/services/definitions/remotion.mjs.map +1 -1
- package/dist/services/definitions/rocketchat.cjs +59 -0
- package/dist/services/definitions/rocketchat.cjs.map +1 -0
- package/dist/services/definitions/rocketchat.d.cts +7 -0
- package/dist/services/definitions/rocketchat.d.cts.map +1 -0
- package/dist/services/definitions/rocketchat.mjs +1 -1
- package/dist/services/definitions/rocketchat.mjs.map +1 -1
- package/dist/services/definitions/scrapling.cjs +46 -0
- package/dist/services/definitions/scrapling.cjs.map +1 -0
- package/dist/services/definitions/scrapling.d.cts +7 -0
- package/dist/services/definitions/scrapling.d.cts.map +1 -0
- package/dist/services/definitions/scrapling.mjs +2 -2
- package/dist/services/definitions/scrapling.mjs.map +1 -1
- package/dist/services/definitions/searxng.cjs +67 -0
- package/dist/services/definitions/searxng.cjs.map +1 -0
- package/dist/services/definitions/searxng.d.cts +7 -0
- package/dist/services/definitions/searxng.d.cts.map +1 -0
- package/dist/services/definitions/searxng.mjs +1 -1
- package/dist/services/definitions/searxng.mjs.map +1 -1
- package/dist/services/definitions/signoz.cjs +56 -0
- package/dist/services/definitions/signoz.cjs.map +1 -0
- package/dist/services/definitions/signoz.d.cts +7 -0
- package/dist/services/definitions/signoz.d.cts.map +1 -0
- package/dist/services/definitions/signoz.mjs +2 -2
- package/dist/services/definitions/signoz.mjs.map +1 -1
- package/dist/services/definitions/solidityguard.cjs +51 -0
- package/dist/services/definitions/solidityguard.cjs.map +1 -0
- package/dist/services/definitions/solidityguard.d.cts +7 -0
- package/dist/services/definitions/solidityguard.d.cts.map +1 -0
- package/dist/services/definitions/solidityguard.mjs +1 -1
- package/dist/services/definitions/solidityguard.mjs.map +1 -1
- package/dist/services/definitions/stable-diffusion.cjs +50 -0
- package/dist/services/definitions/stable-diffusion.cjs.map +1 -0
- package/dist/services/definitions/stable-diffusion.d.cts +7 -0
- package/dist/services/definitions/stable-diffusion.d.cts.map +1 -0
- package/dist/services/definitions/stable-diffusion.mjs +1 -1
- package/dist/services/definitions/stable-diffusion.mjs.map +1 -1
- package/dist/services/definitions/steel-browser.cjs +79 -0
- package/dist/services/definitions/steel-browser.cjs.map +1 -0
- package/dist/services/definitions/steel-browser.d.cts +7 -0
- package/dist/services/definitions/steel-browser.d.cts.map +1 -0
- package/dist/services/definitions/steel-browser.mjs +1 -1
- package/dist/services/definitions/steel-browser.mjs.map +1 -1
- package/dist/services/definitions/stream-gateway.cjs +135 -0
- package/dist/services/definitions/stream-gateway.cjs.map +1 -0
- package/dist/services/definitions/stream-gateway.d.cts +7 -0
- package/dist/services/definitions/stream-gateway.d.cts.map +1 -0
- package/dist/services/definitions/stream-gateway.mjs +2 -2
- package/dist/services/definitions/stream-gateway.mjs.map +1 -1
- package/dist/services/definitions/supabase.cjs +71 -0
- package/dist/services/definitions/supabase.cjs.map +1 -0
- package/dist/services/definitions/supabase.d.cts +7 -0
- package/dist/services/definitions/supabase.d.cts.map +1 -0
- package/dist/services/definitions/supabase.mjs +2 -2
- package/dist/services/definitions/supabase.mjs.map +1 -1
- package/dist/services/definitions/tailscale.cjs +72 -0
- package/dist/services/definitions/tailscale.cjs.map +1 -0
- package/dist/services/definitions/tailscale.d.cts +7 -0
- package/dist/services/definitions/tailscale.d.cts.map +1 -0
- package/dist/services/definitions/tailscale.mjs +1 -1
- package/dist/services/definitions/tailscale.mjs.map +1 -1
- package/dist/services/definitions/temporal.cjs +102 -0
- package/dist/services/definitions/temporal.cjs.map +1 -0
- package/dist/services/definitions/temporal.d.cts +7 -0
- package/dist/services/definitions/temporal.d.cts.map +1 -0
- package/dist/services/definitions/traefik.cjs +59 -0
- package/dist/services/definitions/traefik.cjs.map +1 -0
- package/dist/services/definitions/traefik.d.cts +7 -0
- package/dist/services/definitions/traefik.d.cts.map +1 -0
- package/dist/services/definitions/traefik.mjs +1 -1
- package/dist/services/definitions/traefik.mjs.map +1 -1
- package/dist/services/definitions/umami.cjs +48 -0
- package/dist/services/definitions/umami.cjs.map +1 -0
- package/dist/services/definitions/umami.d.cts +7 -0
- package/dist/services/definitions/umami.d.cts.map +1 -0
- package/dist/services/definitions/uptime-kuma.cjs +53 -0
- package/dist/services/definitions/uptime-kuma.cjs.map +1 -0
- package/dist/services/definitions/uptime-kuma.d.cts +7 -0
- package/dist/services/definitions/uptime-kuma.d.cts.map +1 -0
- package/dist/services/definitions/uptime-kuma.mjs +2 -2
- package/dist/services/definitions/uptime-kuma.mjs.map +1 -1
- package/dist/services/definitions/usesend.cjs +101 -0
- package/dist/services/definitions/usesend.cjs.map +1 -0
- package/dist/services/definitions/usesend.d.cts +7 -0
- package/dist/services/definitions/usesend.d.cts.map +1 -0
- package/dist/services/definitions/usesend.mjs +4 -4
- package/dist/services/definitions/usesend.mjs.map +1 -1
- package/dist/services/definitions/valkey.cjs +63 -0
- package/dist/services/definitions/valkey.cjs.map +1 -0
- package/dist/services/definitions/valkey.d.cts +7 -0
- package/dist/services/definitions/valkey.d.cts.map +1 -0
- package/dist/services/definitions/valkey.mjs +1 -1
- package/dist/services/definitions/valkey.mjs.map +1 -1
- package/dist/services/definitions/vaultwarden.cjs +65 -0
- package/dist/services/definitions/vaultwarden.cjs.map +1 -0
- package/dist/services/definitions/vaultwarden.d.cts +7 -0
- package/dist/services/definitions/vaultwarden.d.cts.map +1 -0
- package/dist/services/definitions/vaultwarden.mjs +2 -2
- package/dist/services/definitions/vaultwarden.mjs.map +1 -1
- package/dist/services/definitions/watchtower.cjs +49 -0
- package/dist/services/definitions/watchtower.cjs.map +1 -0
- package/dist/services/definitions/watchtower.d.cts +7 -0
- package/dist/services/definitions/watchtower.d.cts.map +1 -0
- package/dist/services/definitions/weaviate.cjs +92 -0
- package/dist/services/definitions/weaviate.cjs.map +1 -0
- package/dist/services/definitions/weaviate.d.cts +7 -0
- package/dist/services/definitions/weaviate.d.cts.map +1 -0
- package/dist/services/definitions/weaviate.mjs +2 -2
- package/dist/services/definitions/weaviate.mjs.map +1 -1
- package/dist/services/definitions/whisper.cjs +64 -0
- package/dist/services/definitions/whisper.cjs.map +1 -0
- package/dist/services/definitions/whisper.d.cts +7 -0
- package/dist/services/definitions/whisper.d.cts.map +1 -0
- package/dist/services/definitions/whisper.mjs +2 -2
- package/dist/services/definitions/whisper.mjs.map +1 -1
- package/dist/services/definitions/xyops.cjs +88 -0
- package/dist/services/definitions/xyops.cjs.map +1 -0
- package/dist/services/definitions/xyops.d.cts +7 -0
- package/dist/services/definitions/xyops.d.cts.map +1 -0
- package/dist/services/definitions/xyops.mjs +1 -1
- package/dist/services/definitions/xyops.mjs.map +1 -1
- package/dist/services/registry.cjs +35 -0
- package/dist/services/registry.cjs.map +1 -0
- package/dist/services/registry.d.cts +16 -0
- package/dist/services/registry.d.cts.map +1 -0
- package/dist/services/registry.test.cjs +63 -0
- package/dist/services/registry.test.cjs.map +1 -0
- package/dist/services/registry.test.d.cts +1 -0
- package/dist/services/registry.test.mjs +1 -1
- package/dist/skills/registry.cjs +712 -0
- package/dist/skills/registry.cjs.map +1 -0
- package/dist/skills/registry.d.cts +11 -0
- package/dist/skills/registry.d.cts.map +1 -0
- package/dist/skills/skill-manifest.cjs +23 -0
- package/dist/skills/skill-manifest.cjs.map +1 -0
- package/dist/skills/skill-manifest.d.cts +20 -0
- package/dist/skills/skill-manifest.d.cts.map +1 -0
- package/dist/types.cjs +149 -0
- package/dist/types.cjs.map +1 -0
- package/dist/types.d.cts +82 -0
- package/dist/types.d.cts.map +1 -0
- package/dist/types.d.mts +9 -2
- package/dist/types.d.mts.map +1 -1
- package/dist/types.mjs +40 -0
- package/dist/types.mjs.map +1 -1
- package/dist/validator.cjs +124 -0
- package/dist/validator.cjs.map +1 -0
- package/dist/validator.d.cts +20 -0
- package/dist/validator.d.cts.map +1 -0
- package/dist/validator.test.cjs +64 -0
- package/dist/validator.test.cjs.map +1 -0
- package/dist/validator.test.d.cts +1 -0
- package/dist/validator.test.mjs +1 -1
- package/dist/version-manager.cjs +65 -0
- package/dist/version-manager.cjs.map +1 -0
- package/dist/version-manager.d.cts +14 -0
- package/dist/version-manager.d.cts.map +1 -0
- package/dist/version-manager.test.cjs +101 -0
- package/dist/version-manager.test.cjs.map +1 -0
- package/dist/version-manager.test.d.cts +1 -0
- package/dist/version-manager.test.mjs +1 -1
- package/dist/vi.2VT5v0um-BmRMvymT.cjs +12934 -0
- package/dist/vi.2VT5v0um-BmRMvymT.cjs.map +1 -0
- package/dist/{vi.2VT5v0um-BgmKutxR.mjs → vi.2VT5v0um-CFyDIn0m.mjs} +5 -5
- package/dist/{vi.2VT5v0um-BgmKutxR.mjs.map → vi.2VT5v0um-CFyDIn0m.mjs.map} +1 -1
- package/package.json +24 -3
- package/src/__snapshots__/composer.snapshot.test.ts.snap +20 -20
- package/src/compose-validation.test.ts +102 -0
- package/src/composer.test.ts +1 -1
- package/src/composer.ts +331 -53
- package/src/deployers/coolify.ts +241 -0
- package/src/deployers/dokploy.ts +174 -0
- package/src/deployers/index.ts +36 -0
- package/src/deployers/types.ts +70 -0
- package/src/generate.ts +45 -1
- package/src/generators/cloud-init.ts +137 -0
- package/src/generators/env.ts +112 -6
- package/src/generators/health-check.ts +2 -2
- package/src/generators/openclaw-install-script.ts +77 -0
- package/src/generators/openclaw-json.ts +195 -58
- package/src/generators/postgres-init.ts +11 -0
- package/src/generators/readme.ts +103 -4
- package/src/generators/stack-manifest.ts +127 -0
- package/src/index.ts +30 -0
- package/src/port-scanner.ts +189 -0
- package/src/presets/presets.test.ts +73 -0
- package/src/presets/registry.test.ts +3 -0
- package/src/presets/registry.ts +30 -7
- package/src/resolver.ts +2 -0
- package/src/schema.ts +30 -1
- package/src/services/definitions/anything-llm.ts +2 -2
- package/src/services/definitions/appflowy.ts +1 -1
- package/src/services/definitions/authentik.ts +5 -5
- package/src/services/definitions/beszel.ts +1 -1
- package/src/services/definitions/browserless.ts +1 -1
- package/src/services/definitions/caddy.ts +2 -2
- package/src/services/definitions/cal-com.ts +2 -2
- package/src/services/definitions/chromadb.ts +3 -3
- package/src/services/definitions/claude-code.ts +1 -1
- package/src/services/definitions/code-server.ts +1 -1
- package/src/services/definitions/codex.ts +1 -1
- package/src/services/definitions/comfyui.ts +1 -1
- package/src/services/definitions/coolify.ts +1 -1
- package/src/services/definitions/crowdsec.ts +2 -2
- package/src/services/definitions/desktop-environment.ts +1 -1
- package/src/services/definitions/dify.ts +1 -1
- package/src/services/definitions/docsgpt.ts +1 -1
- package/src/services/definitions/dokploy.ts +2 -2
- package/src/services/definitions/dozzle.ts +1 -1
- package/src/services/definitions/ffmpeg.ts +2 -2
- package/src/services/definitions/flowise.ts +1 -1
- package/src/services/definitions/gemini-cli.ts +8 -1
- package/src/services/definitions/ghost.ts +1 -1
- package/src/services/definitions/gitea.ts +1 -1
- package/src/services/definitions/gotify.ts +3 -3
- package/src/services/definitions/grafana.ts +2 -2
- package/src/services/definitions/headscale.ts +68 -68
- package/src/services/definitions/hexstrike.ts +1 -1
- package/src/services/definitions/homeassistant.ts +1 -1
- package/src/services/definitions/immich.ts +6 -6
- package/src/services/definitions/jellyfin.ts +1 -1
- package/src/services/definitions/jenkins.ts +1 -1
- package/src/services/definitions/kimi.ts +1 -1
- package/src/services/definitions/lasuite-meet-agents.ts +1 -1
- package/src/services/definitions/lasuite-meet-backend.ts +3 -3
- package/src/services/definitions/lasuite-meet-frontend.ts +3 -3
- package/src/services/definitions/librechat.ts +1 -1
- package/src/services/definitions/litellm.ts +1 -1
- package/src/services/definitions/loki.ts +1 -1
- package/src/services/definitions/matomo.ts +1 -1
- package/src/services/definitions/matrix-synapse.ts +1 -1
- package/src/services/definitions/mattermost.ts +1 -1
- package/src/services/definitions/meilisearch.ts +4 -4
- package/src/services/definitions/milvus.ts +1 -1
- package/src/services/definitions/minio.ts +1 -1
- package/src/services/definitions/mixpost.ts +1 -1
- package/src/services/definitions/motion-canvas.ts +1 -1
- package/src/services/definitions/n8n.ts +1 -1
- package/src/services/definitions/neo4j.ts +1 -1
- package/src/services/definitions/nextcloud.ts +19 -19
- package/src/services/definitions/nocodb.ts +1 -1
- package/src/services/definitions/ntfy.ts +2 -2
- package/src/services/definitions/ollama.ts +1 -1
- package/src/services/definitions/open-webui.ts +1 -1
- package/src/services/definitions/opencode.ts +1 -1
- package/src/services/definitions/openpanel.ts +3 -3
- package/src/services/definitions/paperless-ngx.ts +1 -1
- package/src/services/definitions/pentagi.ts +2 -2
- package/src/services/definitions/playwright-server.ts +1 -1
- package/src/services/definitions/portainer.ts +1 -1
- package/src/services/definitions/postgresql.ts +1 -1
- package/src/services/definitions/postiz.ts +1 -1
- package/src/services/definitions/prometheus.ts +1 -1
- package/src/services/definitions/qdrant.ts +1 -1
- package/src/services/definitions/redis.ts +1 -1
- package/src/services/definitions/remotion.ts +1 -1
- package/src/services/definitions/rocketchat.ts +1 -1
- package/src/services/definitions/scrapling.ts +2 -2
- package/src/services/definitions/searxng.ts +1 -1
- package/src/services/definitions/signoz.ts +2 -2
- package/src/services/definitions/solidityguard.ts +1 -1
- package/src/services/definitions/stable-diffusion.ts +1 -1
- package/src/services/definitions/steel-browser.ts +1 -1
- package/src/services/definitions/stream-gateway.ts +2 -2
- package/src/services/definitions/supabase.ts +2 -2
- package/src/services/definitions/tailscale.ts +1 -1
- package/src/services/definitions/traefik.ts +1 -1
- package/src/services/definitions/uptime-kuma.ts +2 -2
- package/src/services/definitions/usesend.ts +4 -4
- package/src/services/definitions/valkey.ts +1 -1
- package/src/services/definitions/vaultwarden.ts +2 -2
- package/src/services/definitions/weaviate.ts +2 -2
- package/src/services/definitions/whisper.ts +2 -2
- package/src/services/definitions/xyops.ts +1 -1
- package/src/types.ts +138 -20
- package/tsdown.config.ts +2 -2
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"health-check.cjs","names":[],"sources":["../../src/generators/health-check.ts"],"sourcesContent":["import type { ResolverOutput } from \"../types.js\";\n\n// ─── Types ──────────────────────────────────────────────────────────────────\n\nexport interface HealthCheckOptions {\n\tprojectName: string;\n\tdeploymentType?: \"docker\" | \"bare-metal\" | \"local\";\n}\n\n// ─── Generator ──────────────────────────────────────────────────────────────\n\n/**\n * Generates stack-specific health check and verification scripts.\n *\n * Each generated script is dynamic — it contains the exact service names,\n * ports, and health check commands from the resolved stack so users get\n * a precise, zero-configuration verifier.\n */\nexport function generateHealthCheck(\n\tresolved: ResolverOutput,\n\toptions: HealthCheckOptions,\n): Record<string, string> {\n\tconst files: Record<string, string> = {};\n\n\tfiles[\"scripts/health-check.sh\"] = generateBashScript(resolved, options);\n\tfiles[\"scripts/health-check.ps1\"] = generatePowerShellScript(resolved, options);\n\n\treturn files;\n}\n\n// ─── Service metadata extraction ────────────────────────────────────────────\n\ninterface ServiceCheck {\n\tid: string;\n\tname: string;\n\ticon: string;\n\tports: Array<{ host: number; container: number; description: string; exposed: boolean }>;\n\thealthCheckCmd: string | null;\n}\n\nfunction extractServiceChecks(resolved: ResolverOutput): ServiceCheck[] {\n\treturn resolved.services.map((svc) => ({\n\t\tid: svc.definition.id,\n\t\tname: svc.definition.name,\n\t\ticon: svc.definition.icon,\n\t\tports: svc.definition.ports.map((p) => ({\n\t\t\thost: p.host,\n\t\t\tcontainer: p.container,\n\t\t\tdescription: p.description,\n\t\t\texposed: p.exposed,\n\t\t})),\n\t\thealthCheckCmd: svc.definition.healthcheck?.test ?? null,\n\t}));\n}\n\nfunction escapeShell(s: string): string {\n\treturn s.replace(/'/g, \"'\\\\''\").replace(/\"/g, '\\\\\"');\n}\n\n// Use L() to build lines safely — avoids template literal issues with shell syntax\nfunction L(...parts: string[]): string {\n\treturn parts.join(\"\");\n}\n\n// ─── Bash Script ────────────────────────────────────────────────────────────\n\nfunction generateBashScript(resolved: ResolverOutput, options: HealthCheckOptions): string {\n\tconst checks = extractServiceChecks(resolved);\n\tconst name = options.projectName;\n\tconst total = checks.length;\n\n\t// Build per-service docker check calls\n\tconst dockerChecks: string[] = [];\n\tfor (const svc of checks) {\n\t\tdockerChecks.push(L(\" # ── \", svc.icon, \" \", svc.name, \" ──\"));\n\t\tdockerChecks.push(L(' check_container \"', svc.id, '\" \"', svc.name, '\" \"', svc.icon, '\"'));\n\t\tfor (const p of svc.ports.filter((pp) => pp.exposed)) {\n\t\t\tdockerChecks.push(\n\t\t\t\tL(' check_port \"', svc.id, '\" ', String(p.host), ' \"', p.description, '\"'),\n\t\t\t);\n\t\t}\n\t\tif (svc.healthCheckCmd) {\n\t\t\tdockerChecks.push(\n\t\t\t\tL(' check_health_cmd \"', svc.id, '\" \"', escapeShell(svc.healthCheckCmd), '\"'),\n\t\t\t);\n\t\t} else {\n\t\t\tdockerChecks.push(\" # No healthcheck defined — skipping exec check\");\n\t\t}\n\t\tdockerChecks.push(\"\");\n\t}\n\n\t// Build per-service bare-metal check calls\n\tconst bmChecks: string[] = [];\n\tfor (const svc of checks) {\n\t\tbmChecks.push(L(\" # ── \", svc.icon, \" \", svc.name, \" (bare-metal) ──\"));\n\t\tbmChecks.push(L(' check_process \"', svc.id, '\" \"', svc.name, '\" \"', svc.icon, '\"'));\n\t\tfor (const p of svc.ports.filter((pp) => pp.exposed)) {\n\t\t\tbmChecks.push(L(' check_port \"', svc.id, '\" ', String(p.host), ' \"', p.description, '\"'));\n\t\t}\n\t\tbmChecks.push(\"\");\n\t}\n\n\t// Build the log-scan service list\n\tconst svcIdList = checks.map((s) => '\"' + s.id + '\"').join(\" \");\n\n\tconst lines: string[] = [\n\t\t\"#!/usr/bin/env bash\",\n\t\t\"set -uo pipefail\",\n\t\t\"\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\tL(\"# 🐾 OpenClaw Stack Health Check — \", name),\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"#\",\n\t\t\"# Auto-generated verification script for your stack.\",\n\t\tL(\"# Checks \", String(total), \" services across 6 phases:\"),\n\t\t\"# 1. Environment — prerequisites, .env, secrets\",\n\t\t\"# 2. Container — running state, health status, restart count\",\n\t\t\"# 3. Port — TCP reachability for exposed ports\",\n\t\t\"# 4. Health Command — execute each service's health check\",\n\t\t\"# 5. Resources — memory/CPU warnings\",\n\t\t\"# 6. Logs — scan for ERROR/FATAL/panic patterns\",\n\t\t\"#\",\n\t\t\"# Usage:\",\n\t\t\"# ./scripts/health-check.sh # Full check\",\n\t\t\"# ./scripts/health-check.sh --quick # Skip log scan\",\n\t\t\"# ./scripts/health-check.sh --json # Output as JSON\",\n\t\t\"#\",\n\t\t\"# Exit codes: 0 = all healthy, 1 = some issues, 2 = critical failure\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"\",\n\t\t'SCRIPT_DIR=\"$(cd \"$(dirname \"${BASH_SOURCE[0]}\")\" && pwd)\"',\n\t\t'PROJECT_DIR=\"$(dirname \"$SCRIPT_DIR\")\"',\n\t\t'cd \"$PROJECT_DIR\"',\n\t\t\"\",\n\t\t\"# ── Arguments ────────────────────────────────────────────────────────\",\n\t\t\"QUICK_MODE=false\",\n\t\t\"JSON_MODE=false\",\n\t\t\"VERBOSE=false\",\n\t\t'for arg in \"$@\"; do',\n\t\t' case \"$arg\" in',\n\t\t\" --quick) QUICK_MODE=true ;;\",\n\t\t\" --json) JSON_MODE=true ;;\",\n\t\t\" --verbose|-v) VERBOSE=true ;;\",\n\t\t\" esac\",\n\t\t\"done\",\n\t\t\"\",\n\t\t\"# ── Colour helpers ───────────────────────────────────────────────────\",\n\t\t'if [ -t 1 ] && [ \"$JSON_MODE\" = false ]; then',\n\t\t\" RED='\\\\033[0;31m'; GREEN='\\\\033[0;32m'; YELLOW='\\\\033[1;33m'\",\n\t\t\" CYAN='\\\\033[0;36m'; DIM='\\\\033[2m'; BOLD='\\\\033[1m'; NC='\\\\033[0m'\",\n\t\t\"else\",\n\t\t\" RED=''; GREEN=''; YELLOW=''; CYAN=''; DIM=''; BOLD=''; NC=''\",\n\t\t\"fi\",\n\t\t\"\",\n\t\t\"# ── Counters ─────────────────────────────────────────────────────────\",\n\t\tL(\"TOTAL=\", String(total)),\n\t\t\"HEALTHY=0\",\n\t\t\"WARNING=0\",\n\t\t\"FAILED=0\",\n\t\t\"ERRORS=()\",\n\t\t\"WARNINGS_LIST=()\",\n\t\t\"SERVICE_RESULTS=()\",\n\t\t\"\",\n\t\t\"# ── Utility functions ────────────────────────────────────────────────\",\n\t\t'pass() { echo -e \" ${GREEN}✅ $*${NC}\"; HEALTHY=$((HEALTHY + 1)); }',\n\t\t'warn_svc() { echo -e \" ${YELLOW}⚠️ $*${NC}\"; WARNING=$((WARNING + 1)); WARNINGS_LIST+=(\"$*\"); }',\n\t\t'fail() { echo -e \" ${RED}❌ $*${NC}\"; FAILED=$((FAILED + 1)); ERRORS+=(\"$*\"); }',\n\t\t'info() { echo -e \" ${CYAN}ℹ $*${NC}\"; }',\n\t\t'dim() { echo -e \" ${DIM}$*${NC}\"; }',\n\t\t\"\",\n\t\t\"pad_name() {\",\n\t\t' local name=\"$1\"',\n\t\t\" local pad=25\",\n\t\t' printf \"%-${pad}s\" \"$name\"',\n\t\t\"}\",\n\t\t\"\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"# Phase 1: Environment\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"phase_environment() {\",\n\t\t' if [ \"$JSON_MODE\" = false ]; then',\n\t\t' echo \"\"',\n\t\t' echo -e \"${BOLD}── Phase 1: Environment ────────────────────────────────────${NC}\"',\n\t\t' echo \"\"',\n\t\t\" fi\",\n\t\t\"\",\n\t\t\" # Docker\",\n\t\t\" if command -v docker &>/dev/null; then\",\n\t\t\" if docker info &>/dev/null 2>&1; then\",\n\t\t' pass \"Docker daemon is running\"',\n\t\t\" DOCKER_AVAILABLE=true\",\n\t\t\" else\",\n\t\t' fail \"Docker daemon is NOT running\"',\n\t\t\" DOCKER_AVAILABLE=false\",\n\t\t\" fi\",\n\t\t\" else\",\n\t\t' warn_svc \"Docker is not installed — running bare-metal checks only\"',\n\t\t\" DOCKER_AVAILABLE=false\",\n\t\t\" fi\",\n\t\t\"\",\n\t\t\" # Docker Compose\",\n\t\t' if [ \"$DOCKER_AVAILABLE\" = true ]; then',\n\t\t\" if docker compose version &>/dev/null 2>&1; then\",\n\t\t' pass \"Docker Compose v2 available\"',\n\t\t\" else\",\n\t\t' warn_svc \"Docker Compose v2 not found\"',\n\t\t\" fi\",\n\t\t\" fi\",\n\t\t\"\",\n\t\t\" # .env\",\n\t\t' if [ -f \".env\" ]; then',\n\t\t' pass \".env file exists\"',\n\t\t\" EMPTY_SECRETS=0\",\n\t\t\" while IFS='=' read -r key value; do\",\n\t\t' [[ \"$key\" =~ ^#.*$ ]] && continue',\n\t\t' [[ -z \"$key\" ]] && continue',\n\t\t' if [[ \"$key\" =~ (PASSWORD|TOKEN|SECRET|SESSION_KEY|COOKIE|API_KEY) ]] && [[ -z \"${value:-}\" ]]; then',\n\t\t' if [ \"$VERBOSE\" = true ]; then',\n\t\t' warn_svc \"Empty secret: $key\"',\n\t\t\" fi\",\n\t\t\" EMPTY_SECRETS=$((EMPTY_SECRETS + 1))\",\n\t\t\" fi\",\n\t\t\" done < .env 2>/dev/null || true\",\n\t\t' if [ \"$EMPTY_SECRETS\" -gt 0 ]; then',\n\t\t' warn_svc \"$EMPTY_SECRETS secret(s) are empty in .env\"',\n\t\t\" else\",\n\t\t' pass \"All secrets are populated\"',\n\t\t\" fi\",\n\t\t\" else\",\n\t\t' warn_svc \".env file not found\"',\n\t\t\" fi\",\n\t\t\"\",\n\t\t\" # Disk space\",\n\t\t\" AVAIL_KB=$(df -k . 2>/dev/null | awk 'NR==2{print $4}' || echo \\\"0\\\")\",\n\t\t\" AVAIL_GB=$((AVAIL_KB / 1024 / 1024))\",\n\t\t' if [ \"$AVAIL_GB\" -lt 2 ]; then',\n\t\t' warn_svc \"Low disk space: ${AVAIL_GB}GB available\"',\n\t\t\" else\",\n\t\t' pass \"Disk space: ${AVAIL_GB}GB available\"',\n\t\t\" fi\",\n\t\t\"}\",\n\t\t\"\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"# Phase 2: Container Status\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"check_container() {\",\n\t\t' local id=\"$1\" name=\"$2\" icon=\"$3\"',\n\t\t\"\",\n\t\t\" local status restarts\",\n\t\t' status=$(docker compose ps \"$id\" --format json 2>/dev/null | head -1)',\n\t\t\"\",\n\t\t' if [ -z \"$status\" ]; then',\n\t\t' fail \"$icon $(pad_name \"$name\") — Container not found\"',\n\t\t\" return\",\n\t\t\" fi\",\n\t\t\"\",\n\t\t\" local state health_status\",\n\t\t' state=$(echo \"$status\" | grep -o \\'\"State\":\"[^\"]*\"\\' | cut -d\\'\"\\' -f4 || echo \"unknown\")',\n\t\t' health_status=$(echo \"$status\" | grep -o \\'\"Health\":\"[^\"]*\"\\' | cut -d\\'\"\\' -f4 || echo \"none\")',\n\t\t\"\",\n\t\t' restarts=$(docker inspect --format=\"{{.RestartCount}}\" \"$(docker compose ps -q \"$id\" 2>/dev/null | head -1)\" 2>/dev/null || echo \"0\")',\n\t\t\"\",\n\t\t' if [ \"$state\" = \"running\" ]; then',\n\t\t' if [ \"$health_status\" = \"healthy\" ]; then',\n\t\t' pass \"$icon $(pad_name \"$name\") Running (healthy) restarts: $restarts\"',\n\t\t' elif [ \"$health_status\" = \"starting\" ]; then',\n\t\t' warn_svc \"$icon $(pad_name \"$name\") Running (starting) restarts: $restarts\"',\n\t\t' elif [ \"$health_status\" = \"unhealthy\" ]; then',\n\t\t' fail \"$icon $(pad_name \"$name\") Running (UNHEALTHY) restarts: $restarts\"',\n\t\t\" else\",\n\t\t' pass \"$icon $(pad_name \"$name\") Running restarts: $restarts\"',\n\t\t\" fi\",\n\t\t' if [ \"$restarts\" -gt 5 ] 2>/dev/null; then',\n\t\t' warn_svc \" └─ $name has restarted $restarts times (possible crash loop)\"',\n\t\t\" fi\",\n\t\t' elif [ \"$state\" = \"exited\" ]; then',\n\t\t\" local exit_code\",\n\t\t' exit_code=$(docker inspect --format=\"{{.State.ExitCode}}\" \"$(docker compose ps -q \"$id\" 2>/dev/null | head -1)\" 2>/dev/null || echo \"?\")',\n\t\t' fail \"$icon $(pad_name \"$name\") Exited (code $exit_code)\"',\n\t\t' if [ \"$exit_code\" = \"137\" ]; then',\n\t\t' ERRORS+=(\" └─ $name: OOM killed (exit 137). Increase memory limits.\")',\n\t\t' elif [ \"$exit_code\" = \"1\" ]; then',\n\t\t' ERRORS+=(\" └─ $name: Application error (exit 1). Check logs: docker compose logs $id\")',\n\t\t\" fi\",\n\t\t\" else\",\n\t\t' fail \"$icon $(pad_name \"$name\") State: $state\"',\n\t\t\" fi\",\n\t\t\"}\",\n\t\t\"\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"# Phase 3: Port Reachability\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"check_port() {\",\n\t\t' local id=\"$1\" port=\"$2\" desc=\"$3\"',\n\t\t\" if command -v nc &>/dev/null; then\",\n\t\t' if nc -z -w 2 localhost \"$port\" 2>/dev/null; then',\n\t\t' dim \" Port $port ($desc) — reachable\"',\n\t\t\" else\",\n\t\t' warn_svc \" └─ $id: Port $port ($desc) — NOT reachable\"',\n\t\t\" fi\",\n\t\t\" elif command -v curl &>/dev/null; then\",\n\t\t' if curl -sf --max-time 2 \"http://localhost:$port/\" &>/dev/null; then',\n\t\t' dim \" Port $port ($desc) — reachable\"',\n\t\t\" else\",\n\t\t' warn_svc \" └─ $id: Port $port ($desc) — NOT reachable\"',\n\t\t\" fi\",\n\t\t\" else\",\n\t\t' dim \" Port $port ($desc) — skipped (no nc or curl)\"',\n\t\t\" fi\",\n\t\t\"}\",\n\t\t\"\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"# Phase 4: Health Check Commands\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"check_health_cmd() {\",\n\t\t' local id=\"$1\" cmd=\"$2\"',\n\t\t' local cid=$(docker compose ps -q \"$id\" 2>/dev/null | head -1)',\n\t\t' [ -z \"$cid\" ] && return',\n\t\t' if docker exec \"$cid\" sh -c \"$cmd\" &>/dev/null; then',\n\t\t' dim \" Health command passed\"',\n\t\t\" else\",\n\t\t' warn_svc \" └─ $id: Health check command failed\"',\n\t\t' if [ \"$VERBOSE\" = true ]; then dim \" Command: $cmd\"; fi',\n\t\t\" fi\",\n\t\t\"}\",\n\t\t\"\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"# Phase 5: Resource Usage\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"phase_resources() {\",\n\t\t' if [ \"$JSON_MODE\" = false ]; then',\n\t\t' echo \"\"',\n\t\t' echo -e \"${BOLD}── Phase 5: Resource Usage ─────────────────────────────────${NC}\"',\n\t\t' echo \"\"',\n\t\t\" fi\",\n\t\t' docker stats --no-stream --format \"table {{.Name}}\\\\t{{.CPUPerc}}\\\\t{{.MemUsage}}\\\\t{{.MemPerc}}\" 2>/dev/null || true',\n\t\t' if [ \"$VERBOSE\" = false ] && [ \"$JSON_MODE\" = false ]; then',\n\t\t' info \"Use --verbose to see warnings for high memory usage\"',\n\t\t\" fi\",\n\t\t\"}\",\n\t\t\"\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"# Phase 6: Log Scan\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"phase_logs() {\",\n\t\t' if [ \"$QUICK_MODE\" = true ]; then',\n\t\t' if [ \"$JSON_MODE\" = false ]; then echo \"\"; info \"Skipping log scan (--quick mode)\"; fi',\n\t\t\" return\",\n\t\t\" fi\",\n\t\t' if [ \"$JSON_MODE\" = false ]; then',\n\t\t' echo \"\"',\n\t\t' echo -e \"${BOLD}── Phase 6: Log Scan ───────────────────────────────────────${NC}\"',\n\t\t' echo \"\"',\n\t\t\" fi\",\n\t\t\" local has_errors=false\",\n\t\tL(\" for svc_id in \", svcIdList, \"; do\"),\n\t\t\" local error_lines\",\n\t\t' error_lines=$(docker compose logs --tail=50 \"$svc_id\" 2>/dev/null | grep -iE \"(error|fatal|panic|exception|segfault|killed|oom)\" | tail -5 || true)',\n\t\t' if [ -n \"$error_lines\" ]; then',\n\t\t\" has_errors=true\",\n\t\t' warn_svc \"$svc_id — found error patterns in logs:\"',\n\t\t' echo \"$error_lines\" | while IFS= read -r eline; do',\n\t\t' dim \" | $eline\"',\n\t\t\" done\",\n\t\t\" fi\",\n\t\t\" done\",\n\t\t' if [ \"$has_errors\" = false ] && [ \"$JSON_MODE\" = false ]; then',\n\t\t' pass \"No error patterns found in recent logs\"',\n\t\t\" fi\",\n\t\t\"}\",\n\t\t\"\",\n\t\t\"# ── Bare-metal process check ────────────────────────────────────────\",\n\t\t\"check_process() {\",\n\t\t' local id=\"$1\" name=\"$2\" icon=\"$3\"',\n\t\t\" if command -v systemctl &>/dev/null; then\",\n\t\t' if systemctl is-active --quiet \"$id\" 2>/dev/null; then',\n\t\t' pass \"$icon $(pad_name \"$name\") Active (systemd)\"',\n\t\t\" return\",\n\t\t\" fi\",\n\t\t\" fi\",\n\t\t' if pgrep -f \"$id\" &>/dev/null; then',\n\t\t' pass \"$icon $(pad_name \"$name\") Running (process)\"',\n\t\t\" else\",\n\t\t' fail \"$icon $(pad_name \"$name\") NOT running\"',\n\t\t\" fi\",\n\t\t\"}\",\n\t\t\"\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"# Main\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"\",\n\t\t\"DOCKER_AVAILABLE=false\",\n\t\t\"\",\n\t\t'if [ \"$JSON_MODE\" = false ]; then',\n\t\t' echo \"\"',\n\t\t' echo \"═══════════════════════════════════════════════════════════════════════\"',\n\t\tL(' echo \" 🐾 OpenClaw Stack Health Report — ', name, '\"'),\n\t\t' echo \"═══════════════════════════════════════════════════════════════════════\"',\n\t\t\"fi\",\n\t\t\"\",\n\t\t\"phase_environment\",\n\t\t\"\",\n\t\t'if [ \"$DOCKER_AVAILABLE\" = true ]; then',\n\t\t' if [ \"$JSON_MODE\" = false ]; then',\n\t\t' echo \"\"',\n\t\t' echo -e \"${BOLD}── Phase 2–4: Service Checks ──────────────────────────────${NC}\"',\n\t\t' echo \"\"',\n\t\t\" fi\",\n\t\t\"\",\n\t\t...dockerChecks,\n\t\t\"\",\n\t\t\" phase_resources\",\n\t\t\" phase_logs\",\n\t\t\"\",\n\t\t\"else\",\n\t\t' if [ \"$JSON_MODE\" = false ]; then',\n\t\t' echo \"\"',\n\t\t' echo -e \"${BOLD}── Bare-Metal Service Checks ──────────────────────────────${NC}\"',\n\t\t' echo \"\"',\n\t\t\" fi\",\n\t\t\"\",\n\t\t...bmChecks,\n\t\t\"fi\",\n\t\t\"\",\n\t\t\"# ── Summary ──────────────────────────────────────────────────────────\",\n\t\t'echo \"\"',\n\t\t'echo \"═══════════════════════════════════════════════════════════════════════\"',\n\t\t'echo \"\"',\n\t\t'echo -e \" ${BOLD}Summary${NC}: $TOTAL services checked\"',\n\t\t'echo -e \" ${GREEN}Healthy: $HEALTHY${NC} | ${YELLOW}Warnings: $WARNING${NC} | ${RED}Failed: $FAILED${NC}\"',\n\t\t\"\",\n\t\t\"if [ ${#ERRORS[@]} -gt 0 ]; then\",\n\t\t' echo \"\"',\n\t\t' echo -e \" ${BOLD}${RED}── Errors ─────────────────────────────────────────────────${NC}\"',\n\t\t' for e in \"${ERRORS[@]}\"; do',\n\t\t' echo -e \" ${RED}$e${NC}\"',\n\t\t\" done\",\n\t\t\"fi\",\n\t\t\"\",\n\t\t'echo \"\"',\n\t\t'echo \"═══════════════════════════════════════════════════════════════════════\"',\n\t\t'echo \"\"',\n\t\t\"\",\n\t\t'if [ \"$FAILED\" -gt 0 ]; then',\n\t\t' echo -e \" ${RED}Some services need attention. Run with --verbose for details.${NC}\"',\n\t\t\" exit 1\",\n\t\t\"else\",\n\t\t' echo -e \" ${GREEN}🎉 All services are running!${NC}\"',\n\t\t\" exit 0\",\n\t\t\"fi\",\n\t];\n\n\treturn lines.join(\"\\n\") + \"\\n\";\n}\n\n// ─── PowerShell Script ──────────────────────────────────────────────────────\n\nfunction generatePowerShellScript(resolved: ResolverOutput, options: HealthCheckOptions): string {\n\tconst checks = extractServiceChecks(resolved);\n\tconst name = options.projectName;\n\tconst total = checks.length;\n\n\t// Build per-service docker check calls\n\tconst dockerChecks: string[] = [];\n\tfor (const svc of checks) {\n\t\tdockerChecks.push(L(\" # \", svc.icon, \" \", svc.name));\n\t\tdockerChecks.push(\n\t\t\tL(\n\t\t\t\t' Test-Container -ServiceId \"',\n\t\t\t\tsvc.id,\n\t\t\t\t'\" -ServiceName \"',\n\t\t\t\tsvc.name,\n\t\t\t\t'\" -Icon \"',\n\t\t\t\tsvc.icon,\n\t\t\t\t'\"',\n\t\t\t),\n\t\t);\n\t\tfor (const p of svc.ports.filter((pp) => pp.exposed)) {\n\t\t\tdockerChecks.push(\n\t\t\t\tL(\n\t\t\t\t\t' Test-Port -ServiceId \"',\n\t\t\t\t\tsvc.id,\n\t\t\t\t\t'\" -Port ',\n\t\t\t\t\tString(p.host),\n\t\t\t\t\t' -Description \"',\n\t\t\t\t\tp.description,\n\t\t\t\t\t'\"',\n\t\t\t\t),\n\t\t\t);\n\t\t}\n\t\tdockerChecks.push(\"\");\n\t}\n\n\t// Build per-service bare-metal check calls\n\tconst bmChecks: string[] = [];\n\tfor (const svc of checks) {\n\t\tbmChecks.push(L(\" # \", svc.icon, \" \", svc.name));\n\t\tbmChecks.push(\n\t\t\tL(\n\t\t\t\t' Test-ProcessRunning -ServiceId \"',\n\t\t\t\tsvc.id,\n\t\t\t\t'\" -ServiceName \"',\n\t\t\t\tsvc.name,\n\t\t\t\t'\" -Icon \"',\n\t\t\t\tsvc.icon,\n\t\t\t\t'\"',\n\t\t\t),\n\t\t);\n\t\tfor (const p of svc.ports.filter((pp) => pp.exposed)) {\n\t\t\tbmChecks.push(\n\t\t\t\tL(\n\t\t\t\t\t' Test-Port -ServiceId \"',\n\t\t\t\t\tsvc.id,\n\t\t\t\t\t'\" -Port ',\n\t\t\t\t\tString(p.host),\n\t\t\t\t\t' -Description \"',\n\t\t\t\t\tp.description,\n\t\t\t\t\t'\"',\n\t\t\t\t),\n\t\t\t);\n\t\t}\n\t\tbmChecks.push(\"\");\n\t}\n\n\tconst svcIdList = checks.map((s) => '\"' + s.id + '\"').join(\", \");\n\n\tconst lines: string[] = [\n\t\t\"#Requires -Version 5.1\",\n\t\t\"<#\",\n\t\t\".SYNOPSIS\",\n\t\tL(\" OpenClaw Stack Health Check — \", name),\n\t\t\"\",\n\t\t\".DESCRIPTION\",\n\t\t\" Auto-generated verification script for your stack.\",\n\t\tL(\n\t\t\t\" Checks \",\n\t\t\tString(total),\n\t\t\t\" services: container status, port reachability, and log errors.\",\n\t\t),\n\t\t\"\",\n\t\t\".PARAMETER Quick\",\n\t\t\" Skip log scanning for faster results.\",\n\t\t\"\",\n\t\t\".PARAMETER Json\",\n\t\t\" Output results as JSON.\",\n\t\t\"\",\n\t\t\".EXAMPLE\",\n\t\t\" .\\\\scripts\\\\health-check.ps1\",\n\t\t\" .\\\\scripts\\\\health-check.ps1 -Quick\",\n\t\t\" .\\\\scripts\\\\health-check.ps1 -Json\",\n\t\t\"#>\",\n\t\t\"param(\",\n\t\t\" [switch]$Quick,\",\n\t\t\" [switch]$Json,\",\n\t\t\" [switch]$Detailed\",\n\t\t\")\",\n\t\t\"\",\n\t\t'$ErrorActionPreference = \"Continue\"',\n\t\t\"\",\n\t\t\"# ── Counters ─────────────────────────────────────────────────────────\",\n\t\tL(\"$script:Total = \", String(total)),\n\t\t\"$script:Healthy = 0\",\n\t\t\"$script:Warnings = 0\",\n\t\t\"$script:Failed = 0\",\n\t\t\"$script:Errors = @()\",\n\t\t\"$script:ServiceResults = @()\",\n\t\t\"$script:DockerAvailable = $false\",\n\t\t\"\",\n\t\t\"# ── Helpers ──────────────────────────────────────────────────────────\",\n\t\t\"\",\n\t\t'function Write-Pass($msg) { Write-Host \" ✅ $msg\" -ForegroundColor Green; $script:Healthy++ }',\n\t\t'function Write-Warn($msg) { Write-Host \" ⚠️ $msg\" -ForegroundColor Yellow; $script:Warnings++ }',\n\t\t'function Write-Fail($msg) { Write-Host \" ❌ $msg\" -ForegroundColor Red; $script:Failed++; $script:Errors += $msg }',\n\t\t'function Write-Info($msg) { Write-Host \" ℹ $msg\" -ForegroundColor Cyan }',\n\t\t'function Write-Dim($msg) { Write-Host \" $msg\" -ForegroundColor DarkGray }',\n\t\t\"\",\n\t\t\"function Pad-Name([string]$n) { return $n.PadRight(25) }\",\n\t\t\"\",\n\t\t\"# ═════════════════════════════════════════════════════════════════════\",\n\t\t\"# Phase 1: Environment\",\n\t\t\"# ═════════════════════════════════════════════════════════════════════\",\n\t\t\"function Test-Environment {\",\n\t\t\" if (-not $Json) {\",\n\t\t' Write-Host \"\"',\n\t\t' Write-Host \"── Phase 1: Environment ────────────────────────────────────\" -ForegroundColor White',\n\t\t' Write-Host \"\"',\n\t\t\" }\",\n\t\t\" try {\",\n\t\t\" $null = docker info 2>&1\",\n\t\t\" if ($LASTEXITCODE -eq 0) {\",\n\t\t' Write-Pass \"Docker daemon is running\"',\n\t\t\" $script:DockerAvailable = $true\",\n\t\t\" } else {\",\n\t\t' Write-Fail \"Docker daemon is NOT running\"',\n\t\t\" }\",\n\t\t\" } catch {\",\n\t\t' Write-Warn \"Docker is not installed — running bare-metal checks only\"',\n\t\t\" }\",\n\t\t\" if ($script:DockerAvailable) {\",\n\t\t\" try {\",\n\t\t\" $null = docker compose version 2>&1\",\n\t\t\" if ($LASTEXITCODE -eq 0) {\",\n\t\t' Write-Pass \"Docker Compose v2 available\"',\n\t\t\" } else {\",\n\t\t' Write-Warn \"Docker Compose v2 not found\"',\n\t\t\" }\",\n\t\t\" } catch {\",\n\t\t' Write-Warn \"Docker Compose not available\"',\n\t\t\" }\",\n\t\t\" }\",\n\t\t' $envPath = Join-Path $PSScriptRoot \"..\\\\.env\"',\n\t\t\" if (Test-Path $envPath) {\",\n\t\t' Write-Pass \".env file exists\"',\n\t\t\" $emptySecrets = 0\",\n\t\t\" Get-Content $envPath | ForEach-Object {\",\n\t\t' if ($_ -match \"(PASSWORD|TOKEN|SECRET|SESSION_KEY|COOKIE|API_KEY).*=\\\\s*$\") { $emptySecrets++ }',\n\t\t\" }\",\n\t\t\" if ($emptySecrets -gt 0) {\",\n\t\t' Write-Warn \"$emptySecrets secret(s) are empty in .env\"',\n\t\t\" } else {\",\n\t\t' Write-Pass \"All secrets are populated\"',\n\t\t\" }\",\n\t\t\" } else {\",\n\t\t' Write-Warn \".env file not found\"',\n\t\t\" }\",\n\t\t\" $drive = (Get-Item $PSScriptRoot).PSDrive\",\n\t\t\" $freeGB = [math]::Round($drive.Free / 1GB, 1)\",\n\t\t\" if ($freeGB -lt 2) {\",\n\t\t' Write-Warn \"Low disk space: ${freeGB}GB available\"',\n\t\t\" } else {\",\n\t\t' Write-Pass \"Disk space: ${freeGB}GB available\"',\n\t\t\" }\",\n\t\t\"}\",\n\t\t\"\",\n\t\t\"# ═════════════════════════════════════════════════════════════════════\",\n\t\t\"# Phase 2: Container Check\",\n\t\t\"# ═════════════════════════════════════════════════════════════════════\",\n\t\t\"function Test-Container {\",\n\t\t\" param([string]$ServiceId, [string]$ServiceName, [string]$Icon)\",\n\t\t\" try {\",\n\t\t\" $status = docker compose ps $ServiceId --format json 2>&1 | ConvertFrom-Json\",\n\t\t\" } catch {\",\n\t\t' Write-Fail \"$Icon $(Pad-Name $ServiceName) Container not found\"',\n\t\t\" return\",\n\t\t\" }\",\n\t\t\" if (-not $status) {\",\n\t\t' Write-Fail \"$Icon $(Pad-Name $ServiceName) Container not found\"',\n\t\t\" return\",\n\t\t\" }\",\n\t\t\" $state = $status.State\",\n\t\t\" $health = $status.Health\",\n\t\t' if ($state -eq \"running\") {',\n\t\t' if ($health -eq \"healthy\") {',\n\t\t' Write-Pass \"$Icon $(Pad-Name $ServiceName) Running (healthy)\"',\n\t\t' } elseif ($health -eq \"starting\") {',\n\t\t' Write-Warn \"$Icon $(Pad-Name $ServiceName) Running (starting)\"',\n\t\t' } elseif ($health -eq \"unhealthy\") {',\n\t\t' Write-Fail \"$Icon $(Pad-Name $ServiceName) Running (UNHEALTHY)\"',\n\t\t\" } else {\",\n\t\t' Write-Pass \"$Icon $(Pad-Name $ServiceName) Running\"',\n\t\t\" }\",\n\t\t' } elseif ($state -eq \"exited\") {',\n\t\t' Write-Fail \"$Icon $(Pad-Name $ServiceName) Exited\"',\n\t\t\" } else {\",\n\t\t' Write-Fail \"$Icon $(Pad-Name $ServiceName) State: $state\"',\n\t\t\" }\",\n\t\t\"}\",\n\t\t\"\",\n\t\t\"# ═════════════════════════════════════════════════════════════════════\",\n\t\t\"# Phase 3: Port Check\",\n\t\t\"# ═════════════════════════════════════════════════════════════════════\",\n\t\t\"function Test-Port {\",\n\t\t\" param([string]$ServiceId, [int]$Port, [string]$Description)\",\n\t\t\" try {\",\n\t\t\" $tcp = New-Object System.Net.Sockets.TcpClient\",\n\t\t' $tcp.ConnectAsync(\"localhost\", $Port).Wait(2000) | Out-Null',\n\t\t\" if ($tcp.Connected) {\",\n\t\t' Write-Dim \" Port $Port ($Description) — reachable\"',\n\t\t\" $tcp.Close()\",\n\t\t\" } else {\",\n\t\t' Write-Warn \" └─ ${ServiceId}: Port $Port ($Description) — NOT reachable\"',\n\t\t\" }\",\n\t\t\" } catch {\",\n\t\t' Write-Warn \" └─ ${ServiceId}: Port $Port ($Description) — NOT reachable\"',\n\t\t\" }\",\n\t\t\"}\",\n\t\t\"\",\n\t\t\"# ── Bare-metal process check ────────────────────────────────────────\",\n\t\t\"function Test-ProcessRunning {\",\n\t\t\" param([string]$ServiceId, [string]$ServiceName, [string]$Icon)\",\n\t\t\" $svc = Get-Service -Name $ServiceId -ErrorAction SilentlyContinue\",\n\t\t\" if ($svc) {\",\n\t\t' if ($svc.Status -eq \"Running\") {',\n\t\t' Write-Pass \"$Icon $(Pad-Name $ServiceName) Active (service)\"',\n\t\t\" } else {\",\n\t\t' Write-Fail \"$Icon $(Pad-Name $ServiceName) $($svc.Status)\"',\n\t\t\" }\",\n\t\t\" return\",\n\t\t\" }\",\n\t\t\" $proc = Get-Process -Name $ServiceId -ErrorAction SilentlyContinue\",\n\t\t\" if ($proc) {\",\n\t\t' Write-Pass \"$Icon $(Pad-Name $ServiceName) Running (process)\"',\n\t\t\" } else {\",\n\t\t' Write-Fail \"$Icon $(Pad-Name $ServiceName) NOT running\"',\n\t\t\" }\",\n\t\t\"}\",\n\t\t\"\",\n\t\t\"# ═════════════════════════════════════════════════════════════════════\",\n\t\t\"# Phase 6: Log Scan\",\n\t\t\"# ═════════════════════════════════════════════════════════════════════\",\n\t\t\"function Test-Logs {\",\n\t\t\" if ($Quick) {\",\n\t\t' if (-not $Json) { Write-Info \"Skipping log scan (-Quick mode)\" }',\n\t\t\" return\",\n\t\t\" }\",\n\t\t\" if (-not $Json) {\",\n\t\t' Write-Host \"\"',\n\t\t' Write-Host \"── Phase 6: Log Scan ───────────────────────────────────────\" -ForegroundColor White',\n\t\t' Write-Host \"\"',\n\t\t\" }\",\n\t\t\" $hasErrors = $false\",\n\t\tL(\" $svcIds = @(\", svcIdList, \")\"),\n\t\t\" foreach ($svcId in $svcIds) {\",\n\t\t\" try {\",\n\t\t\" $logs = docker compose logs --tail=50 $svcId 2>&1 | Out-String\",\n\t\t' $errorLines = $logs -split \"`n\" | Where-Object { $_ -match \"(error|fatal|panic|exception|segfault|killed|oom)\" } | Select-Object -Last 5',\n\t\t\" if ($errorLines) {\",\n\t\t\" $hasErrors = $true\",\n\t\t' Write-Warn \"$svcId — found error patterns in logs:\"',\n\t\t' $errorLines | ForEach-Object { Write-Dim \" | $_\" }',\n\t\t\" }\",\n\t\t\" } catch {}\",\n\t\t\" }\",\n\t\t\" if (-not $hasErrors -and -not $Json) {\",\n\t\t' Write-Pass \"No error patterns found in recent logs\"',\n\t\t\" }\",\n\t\t\"}\",\n\t\t\"\",\n\t\t\"# ═════════════════════════════════════════════════════════════════════\",\n\t\t\"# Main\",\n\t\t\"# ═════════════════════════════════════════════════════════════════════\",\n\t\t\"\",\n\t\t\"if (-not $Json) {\",\n\t\t' Write-Host \"\"',\n\t\t' Write-Host \"═══════════════════════════════════════════════════════════════════════\" ',\n\t\tL(' Write-Host \" 🐾 OpenClaw Stack Health Report — ', name, '\"'),\n\t\t' Write-Host \"═══════════════════════════════════════════════════════════════════════\"',\n\t\t\"}\",\n\t\t\"\",\n\t\t\"Test-Environment\",\n\t\t\"\",\n\t\t\"if ($script:DockerAvailable) {\",\n\t\t\" if (-not $Json) {\",\n\t\t' Write-Host \"\"',\n\t\t' Write-Host \"── Phase 2–4: Service Checks ──────────────────────────────\" -ForegroundColor White',\n\t\t' Write-Host \"\"',\n\t\t\" }\",\n\t\t\"\",\n\t\t...dockerChecks,\n\t\t\" Test-Logs\",\n\t\t\"} else {\",\n\t\t\" if (-not $Json) {\",\n\t\t' Write-Host \"\"',\n\t\t' Write-Host \"── Bare-Metal Service Checks ──────────────────────────────\" -ForegroundColor White',\n\t\t' Write-Host \"\"',\n\t\t\" }\",\n\t\t\"\",\n\t\t...bmChecks,\n\t\t\"}\",\n\t\t\"\",\n\t\t\"# ── Summary ──────────────────────────────────────────────────────────\",\n\t\t\"if ($Json) {\",\n\t\t\" $result = @{\",\n\t\tL(' project = \"', name, '\"'),\n\t\t' timestamp = (Get-Date -Format \"o\")',\n\t\t\" summary = @{\",\n\t\t\" total = $script:Total\",\n\t\t\" healthy = $script:Healthy\",\n\t\t\" warnings = $script:Warnings\",\n\t\t\" failed = $script:Failed\",\n\t\t\" }\",\n\t\t\" errors = $script:Errors\",\n\t\t\" }\",\n\t\t\" $result | ConvertTo-Json -Depth 5\",\n\t\t\"} else {\",\n\t\t' Write-Host \"\"',\n\t\t' Write-Host \"═══════════════════════════════════════════════════════════════════════\"',\n\t\t' Write-Host \"\"',\n\t\t' Write-Host \" Summary: $($script:Total) services checked\"',\n\t\t' Write-Host \" Healthy: $($script:Healthy)\" -ForegroundColor Green -NoNewline',\n\t\t' Write-Host \" | Warnings: $($script:Warnings)\" -ForegroundColor Yellow -NoNewline',\n\t\t' Write-Host \" | Failed: $($script:Failed)\" -ForegroundColor Red',\n\t\t\" if ($script:Errors.Count -gt 0) {\",\n\t\t' Write-Host \"\"',\n\t\t' Write-Host \" ── Errors ──────────────────────────────────────────────────\" -ForegroundColor Red',\n\t\t' $script:Errors | ForEach-Object { Write-Host \" $_\" -ForegroundColor Red }',\n\t\t\" }\",\n\t\t' Write-Host \"\"',\n\t\t' Write-Host \"═══════════════════════════════════════════════════════════════════════\"',\n\t\t' Write-Host \"\"',\n\t\t\" if ($script:Failed -gt 0) {\",\n\t\t' Write-Host \" Some services need attention. Run with -Detailed for more.\" -ForegroundColor Red',\n\t\t\" exit 1\",\n\t\t\" } else {\",\n\t\t' Write-Host \" 🎉 All services are running!\" -ForegroundColor Green',\n\t\t\" exit 0\",\n\t\t\" }\",\n\t\t\"}\",\n\t];\n\n\treturn lines.join(\"\\n\") + \"\\n\";\n}\n"],"mappings":";;;;;;;;;;AAkBA,SAAgB,oBACf,UACA,SACyB;CACzB,MAAM,QAAgC,EAAE;AAExC,OAAM,6BAA6B,mBAAmB,UAAU,QAAQ;AACxE,OAAM,8BAA8B,yBAAyB,UAAU,QAAQ;AAE/E,QAAO;;AAaR,SAAS,qBAAqB,UAA0C;AACvE,QAAO,SAAS,SAAS,KAAK,SAAS;EACtC,IAAI,IAAI,WAAW;EACnB,MAAM,IAAI,WAAW;EACrB,MAAM,IAAI,WAAW;EACrB,OAAO,IAAI,WAAW,MAAM,KAAK,OAAO;GACvC,MAAM,EAAE;GACR,WAAW,EAAE;GACb,aAAa,EAAE;GACf,SAAS,EAAE;GACX,EAAE;EACH,gBAAgB,IAAI,WAAW,aAAa,QAAQ;EACpD,EAAE;;AAGJ,SAAS,YAAY,GAAmB;AACvC,QAAO,EAAE,QAAQ,MAAM,QAAQ,CAAC,QAAQ,MAAM,OAAM;;AAIrD,SAAS,EAAE,GAAG,OAAyB;AACtC,QAAO,MAAM,KAAK,GAAG;;AAKtB,SAAS,mBAAmB,UAA0B,SAAqC;CAC1F,MAAM,SAAS,qBAAqB,SAAS;CAC7C,MAAM,OAAO,QAAQ;CACrB,MAAM,QAAQ,OAAO;CAGrB,MAAM,eAAyB,EAAE;AACjC,MAAK,MAAM,OAAO,QAAQ;AACzB,eAAa,KAAK,EAAE,WAAW,IAAI,MAAM,KAAK,IAAI,MAAM,MAAM,CAAC;AAC/D,eAAa,KAAK,EAAE,wBAAuB,IAAI,IAAI,SAAO,IAAI,MAAM,SAAO,IAAI,MAAM,KAAI,CAAC;AAC1F,OAAK,MAAM,KAAK,IAAI,MAAM,QAAQ,OAAO,GAAG,QAAQ,CACnD,cAAa,KACZ,EAAE,mBAAkB,IAAI,IAAI,OAAM,OAAO,EAAE,KAAK,EAAE,OAAM,EAAE,aAAa,KAAI,CAC3E;AAEF,MAAI,IAAI,eACP,cAAa,KACZ,EAAE,yBAAwB,IAAI,IAAI,SAAO,YAAY,IAAI,eAAe,EAAE,KAAI,CAC9E;MAED,cAAa,KAAK,mDAAmD;AAEtE,eAAa,KAAK,GAAG;;CAItB,MAAM,WAAqB,EAAE;AAC7B,MAAK,MAAM,OAAO,QAAQ;AACzB,WAAS,KAAK,EAAE,WAAW,IAAI,MAAM,KAAK,IAAI,MAAM,mBAAmB,CAAC;AACxE,WAAS,KAAK,EAAE,sBAAqB,IAAI,IAAI,SAAO,IAAI,MAAM,SAAO,IAAI,MAAM,KAAI,CAAC;AACpF,OAAK,MAAM,KAAK,IAAI,MAAM,QAAQ,OAAO,GAAG,QAAQ,CACnD,UAAS,KAAK,EAAE,mBAAkB,IAAI,IAAI,OAAM,OAAO,EAAE,KAAK,EAAE,OAAM,EAAE,aAAa,KAAI,CAAC;AAE3F,WAAS,KAAK,GAAG;;CAIlB,MAAM,YAAY,OAAO,KAAK,MAAM,OAAM,EAAE,KAAK,KAAI,CAAC,KAAK,IAAI;AA6V/D,QA3VwB;EACvB;EACA;EACA;EACA;EACA,EAAE,uCAAuC,KAAK;EAC9C;EACA;EACA;EACA,EAAE,aAAa,OAAO,MAAM,EAAE,6BAA6B;EAC3D;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA,EAAE,UAAU,OAAO,MAAM,CAAC;EAC1B;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA,EAAE,oBAAoB,WAAW,OAAO;EACxC;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA,EAAE,gDAA+C,MAAM,KAAI;EAC3D;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA,GAAG;EACH;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA,GAAG;EACH;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA,CAEY,KAAK,KAAK,GAAG;;AAK3B,SAAS,yBAAyB,UAA0B,SAAqC;CAChG,MAAM,SAAS,qBAAqB,SAAS;CAC7C,MAAM,OAAO,QAAQ;CACrB,MAAM,QAAQ,OAAO;CAGrB,MAAM,eAAyB,EAAE;AACjC,MAAK,MAAM,OAAO,QAAQ;AACzB,eAAa,KAAK,EAAE,UAAU,IAAI,MAAM,KAAK,IAAI,KAAK,CAAC;AACvD,eAAa,KACZ,EACC,oCACA,IAAI,IACJ,sBACA,IAAI,MACJ,eACA,IAAI,MACJ,KACA,CACD;AACD,OAAK,MAAM,KAAK,IAAI,MAAM,QAAQ,OAAO,GAAG,QAAQ,CACnD,cAAa,KACZ,EACC,+BACA,IAAI,IACJ,aACA,OAAO,EAAE,KAAK,EACd,oBACA,EAAE,aACF,KACA,CACD;AAEF,eAAa,KAAK,GAAG;;CAItB,MAAM,WAAqB,EAAE;AAC7B,MAAK,MAAM,OAAO,QAAQ;AACzB,WAAS,KAAK,EAAE,UAAU,IAAI,MAAM,KAAK,IAAI,KAAK,CAAC;AACnD,WAAS,KACR,EACC,yCACA,IAAI,IACJ,sBACA,IAAI,MACJ,eACA,IAAI,MACJ,KACA,CACD;AACD,OAAK,MAAM,KAAK,IAAI,MAAM,QAAQ,OAAO,GAAG,QAAQ,CACnD,UAAS,KACR,EACC,+BACA,IAAI,IACJ,aACA,OAAO,EAAE,KAAK,EACd,oBACA,EAAE,aACF,KACA,CACD;AAEF,WAAS,KAAK,GAAG;;CAGlB,MAAM,YAAY,OAAO,KAAK,MAAM,OAAM,EAAE,KAAK,KAAI,CAAC,KAAK,KAAK;AA8RhE,QA5RwB;EACvB;EACA;EACA;EACA,EAAE,sCAAsC,KAAK;EAC7C;EACA;EACA;EACA,EACC,eACA,OAAO,MAAM,EACb,kEACA;EACD;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA,EAAE,oBAAoB,OAAO,MAAM,CAAC;EACpC;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA,EAAE,oBAAoB,WAAW,IAAI;EACrC;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA,EAAE,wDAAuD,MAAM,KAAI;EACnE;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA,GAAG;EACH;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA,GAAG;EACH;EACA;EACA;EACA;EACA;EACA,EAAE,wBAAuB,MAAM,KAAI;EACnC;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA,CAEY,KAAK,KAAK,GAAG"}
|
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
import { ResolverOutput } from "../types.cjs";
|
|
2
|
+
|
|
3
|
+
//#region src/generators/health-check.d.ts
|
|
4
|
+
interface HealthCheckOptions {
|
|
5
|
+
projectName: string;
|
|
6
|
+
deploymentType?: "docker" | "bare-metal" | "local";
|
|
7
|
+
}
|
|
8
|
+
/**
|
|
9
|
+
* Generates stack-specific health check and verification scripts.
|
|
10
|
+
*
|
|
11
|
+
* Each generated script is dynamic — it contains the exact service names,
|
|
12
|
+
* ports, and health check commands from the resolved stack so users get
|
|
13
|
+
* a precise, zero-configuration verifier.
|
|
14
|
+
*/
|
|
15
|
+
declare function generateHealthCheck(resolved: ResolverOutput, options: HealthCheckOptions): Record<string, string>;
|
|
16
|
+
//#endregion
|
|
17
|
+
export { HealthCheckOptions, generateHealthCheck };
|
|
18
|
+
//# sourceMappingURL=health-check.d.cts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"health-check.d.cts","names":[],"sources":["../../src/generators/health-check.ts"],"mappings":";;;UAIiB,kBAAA;EAChB,WAAA;EACA,cAAA;AAAA;;;;AAYD;;;;iBAAgB,mBAAA,CACf,QAAA,EAAU,cAAA,EACV,OAAA,EAAS,kBAAA,GACP,MAAA"}
|
|
@@ -3,7 +3,7 @@ import { ResolverOutput } from "../types.mjs";
|
|
|
3
3
|
//#region src/generators/health-check.d.ts
|
|
4
4
|
interface HealthCheckOptions {
|
|
5
5
|
projectName: string;
|
|
6
|
-
deploymentType?: "docker" | "bare-metal";
|
|
6
|
+
deploymentType?: "docker" | "bare-metal" | "local";
|
|
7
7
|
}
|
|
8
8
|
/**
|
|
9
9
|
* Generates stack-specific health check and verification scripts.
|
|
@@ -14,5 +14,5 @@ interface HealthCheckOptions {
|
|
|
14
14
|
*/
|
|
15
15
|
declare function generateHealthCheck(resolved: ResolverOutput, options: HealthCheckOptions): Record<string, string>;
|
|
16
16
|
//#endregion
|
|
17
|
-
export { generateHealthCheck };
|
|
17
|
+
export { HealthCheckOptions, generateHealthCheck };
|
|
18
18
|
//# sourceMappingURL=health-check.d.mts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"health-check.d.mts","names":[],"sources":["../../src/generators/health-check.ts"],"mappings":";;;
|
|
1
|
+
{"version":3,"file":"health-check.d.mts","names":[],"sources":["../../src/generators/health-check.ts"],"mappings":";;;UAIiB,kBAAA;EAChB,WAAA;EACA,cAAA;AAAA;;;;AAYD;;;;iBAAgB,mBAAA,CACf,QAAA,EAAU,cAAA,EACV,OAAA,EAAS,kBAAA,GACP,MAAA"}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"health-check.mjs","names":[],"sources":["../../src/generators/health-check.ts"],"sourcesContent":["import type { ResolverOutput } from \"../types.js\";\n\n// ─── Types ──────────────────────────────────────────────────────────────────\n\ninterface HealthCheckOptions {\n\tprojectName: string;\n\tdeploymentType?: \"docker\" | \"bare-metal\";\n}\n\n// ─── Generator ──────────────────────────────────────────────────────────────\n\n/**\n * Generates stack-specific health check and verification scripts.\n *\n * Each generated script is dynamic — it contains the exact service names,\n * ports, and health check commands from the resolved stack so users get\n * a precise, zero-configuration verifier.\n */\nexport function generateHealthCheck(\n\tresolved: ResolverOutput,\n\toptions: HealthCheckOptions,\n): Record<string, string> {\n\tconst files: Record<string, string> = {};\n\n\tfiles[\"scripts/health-check.sh\"] = generateBashScript(resolved, options);\n\tfiles[\"scripts/health-check.ps1\"] = generatePowerShellScript(resolved, options);\n\n\treturn files;\n}\n\n// ─── Service metadata extraction ────────────────────────────────────────────\n\ninterface ServiceCheck {\n\tid: string;\n\tname: string;\n\ticon: string;\n\tports: Array<{ host: number; container: number; description: string; exposed: boolean }>;\n\thealthCheckCmd: string | null;\n}\n\nfunction extractServiceChecks(resolved: ResolverOutput): ServiceCheck[] {\n\treturn resolved.services.map((svc) => ({\n\t\tid: svc.definition.id,\n\t\tname: svc.definition.name,\n\t\ticon: svc.definition.icon,\n\t\tports: svc.definition.ports.map((p) => ({\n\t\t\thost: p.host,\n\t\t\tcontainer: p.container,\n\t\t\tdescription: p.description,\n\t\t\texposed: p.exposed,\n\t\t})),\n\t\thealthCheckCmd: svc.definition.healthcheck?.test ?? null,\n\t}));\n}\n\nfunction escapeShell(s: string): string {\n\treturn s.replace(/'/g, \"'\\\\''\").replace(/\"/g, '\\\\\"');\n}\n\n// Use L() to build lines safely — avoids template literal issues with shell syntax\nfunction L(...parts: string[]): string {\n\treturn parts.join(\"\");\n}\n\n// ─── Bash Script ────────────────────────────────────────────────────────────\n\nfunction generateBashScript(resolved: ResolverOutput, options: HealthCheckOptions): string {\n\tconst checks = extractServiceChecks(resolved);\n\tconst name = options.projectName;\n\tconst total = checks.length;\n\n\t// Build per-service docker check calls\n\tconst dockerChecks: string[] = [];\n\tfor (const svc of checks) {\n\t\tdockerChecks.push(L(\" # ── \", svc.icon, \" \", svc.name, \" ──\"));\n\t\tdockerChecks.push(L(' check_container \"', svc.id, '\" \"', svc.name, '\" \"', svc.icon, '\"'));\n\t\tfor (const p of svc.ports.filter((pp) => pp.exposed)) {\n\t\t\tdockerChecks.push(\n\t\t\t\tL(' check_port \"', svc.id, '\" ', String(p.host), ' \"', p.description, '\"'),\n\t\t\t);\n\t\t}\n\t\tif (svc.healthCheckCmd) {\n\t\t\tdockerChecks.push(\n\t\t\t\tL(' check_health_cmd \"', svc.id, '\" \"', escapeShell(svc.healthCheckCmd), '\"'),\n\t\t\t);\n\t\t} else {\n\t\t\tdockerChecks.push(\" # No healthcheck defined — skipping exec check\");\n\t\t}\n\t\tdockerChecks.push(\"\");\n\t}\n\n\t// Build per-service bare-metal check calls\n\tconst bmChecks: string[] = [];\n\tfor (const svc of checks) {\n\t\tbmChecks.push(L(\" # ── \", svc.icon, \" \", svc.name, \" (bare-metal) ──\"));\n\t\tbmChecks.push(L(' check_process \"', svc.id, '\" \"', svc.name, '\" \"', svc.icon, '\"'));\n\t\tfor (const p of svc.ports.filter((pp) => pp.exposed)) {\n\t\t\tbmChecks.push(L(' check_port \"', svc.id, '\" ', String(p.host), ' \"', p.description, '\"'));\n\t\t}\n\t\tbmChecks.push(\"\");\n\t}\n\n\t// Build the log-scan service list\n\tconst svcIdList = checks.map((s) => '\"' + s.id + '\"').join(\" \");\n\n\tconst lines: string[] = [\n\t\t\"#!/usr/bin/env bash\",\n\t\t\"set -uo pipefail\",\n\t\t\"\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\tL(\"# 🐾 OpenClaw Stack Health Check — \", name),\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"#\",\n\t\t\"# Auto-generated verification script for your stack.\",\n\t\tL(\"# Checks \", String(total), \" services across 6 phases:\"),\n\t\t\"# 1. Environment — prerequisites, .env, secrets\",\n\t\t\"# 2. Container — running state, health status, restart count\",\n\t\t\"# 3. Port — TCP reachability for exposed ports\",\n\t\t\"# 4. Health Command — execute each service's health check\",\n\t\t\"# 5. Resources — memory/CPU warnings\",\n\t\t\"# 6. Logs — scan for ERROR/FATAL/panic patterns\",\n\t\t\"#\",\n\t\t\"# Usage:\",\n\t\t\"# ./scripts/health-check.sh # Full check\",\n\t\t\"# ./scripts/health-check.sh --quick # Skip log scan\",\n\t\t\"# ./scripts/health-check.sh --json # Output as JSON\",\n\t\t\"#\",\n\t\t\"# Exit codes: 0 = all healthy, 1 = some issues, 2 = critical failure\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"\",\n\t\t'SCRIPT_DIR=\"$(cd \"$(dirname \"${BASH_SOURCE[0]}\")\" && pwd)\"',\n\t\t'PROJECT_DIR=\"$(dirname \"$SCRIPT_DIR\")\"',\n\t\t'cd \"$PROJECT_DIR\"',\n\t\t\"\",\n\t\t\"# ── Arguments ────────────────────────────────────────────────────────\",\n\t\t\"QUICK_MODE=false\",\n\t\t\"JSON_MODE=false\",\n\t\t\"VERBOSE=false\",\n\t\t'for arg in \"$@\"; do',\n\t\t' case \"$arg\" in',\n\t\t\" --quick) QUICK_MODE=true ;;\",\n\t\t\" --json) JSON_MODE=true ;;\",\n\t\t\" --verbose|-v) VERBOSE=true ;;\",\n\t\t\" esac\",\n\t\t\"done\",\n\t\t\"\",\n\t\t\"# ── Colour helpers ───────────────────────────────────────────────────\",\n\t\t'if [ -t 1 ] && [ \"$JSON_MODE\" = false ]; then',\n\t\t\" RED='\\\\033[0;31m'; GREEN='\\\\033[0;32m'; YELLOW='\\\\033[1;33m'\",\n\t\t\" CYAN='\\\\033[0;36m'; DIM='\\\\033[2m'; BOLD='\\\\033[1m'; NC='\\\\033[0m'\",\n\t\t\"else\",\n\t\t\" RED=''; GREEN=''; YELLOW=''; CYAN=''; DIM=''; BOLD=''; NC=''\",\n\t\t\"fi\",\n\t\t\"\",\n\t\t\"# ── Counters ─────────────────────────────────────────────────────────\",\n\t\tL(\"TOTAL=\", String(total)),\n\t\t\"HEALTHY=0\",\n\t\t\"WARNING=0\",\n\t\t\"FAILED=0\",\n\t\t\"ERRORS=()\",\n\t\t\"WARNINGS_LIST=()\",\n\t\t\"SERVICE_RESULTS=()\",\n\t\t\"\",\n\t\t\"# ── Utility functions ────────────────────────────────────────────────\",\n\t\t'pass() { echo -e \" ${GREEN}✅ $*${NC}\"; HEALTHY=$((HEALTHY + 1)); }',\n\t\t'warn_svc() { echo -e \" ${YELLOW}⚠️ $*${NC}\"; WARNING=$((WARNING + 1)); WARNINGS_LIST+=(\"$*\"); }',\n\t\t'fail() { echo -e \" ${RED}❌ $*${NC}\"; FAILED=$((FAILED + 1)); ERRORS+=(\"$*\"); }',\n\t\t'info() { echo -e \" ${CYAN}ℹ $*${NC}\"; }',\n\t\t'dim() { echo -e \" ${DIM}$*${NC}\"; }',\n\t\t\"\",\n\t\t\"pad_name() {\",\n\t\t' local name=\"$1\"',\n\t\t\" local pad=25\",\n\t\t' printf \"%-${pad}s\" \"$name\"',\n\t\t\"}\",\n\t\t\"\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"# Phase 1: Environment\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"phase_environment() {\",\n\t\t' if [ \"$JSON_MODE\" = false ]; then',\n\t\t' echo \"\"',\n\t\t' echo -e \"${BOLD}── Phase 1: Environment ────────────────────────────────────${NC}\"',\n\t\t' echo \"\"',\n\t\t\" fi\",\n\t\t\"\",\n\t\t\" # Docker\",\n\t\t\" if command -v docker &>/dev/null; then\",\n\t\t\" if docker info &>/dev/null 2>&1; then\",\n\t\t' pass \"Docker daemon is running\"',\n\t\t\" DOCKER_AVAILABLE=true\",\n\t\t\" else\",\n\t\t' fail \"Docker daemon is NOT running\"',\n\t\t\" DOCKER_AVAILABLE=false\",\n\t\t\" fi\",\n\t\t\" else\",\n\t\t' warn_svc \"Docker is not installed — running bare-metal checks only\"',\n\t\t\" DOCKER_AVAILABLE=false\",\n\t\t\" fi\",\n\t\t\"\",\n\t\t\" # Docker Compose\",\n\t\t' if [ \"$DOCKER_AVAILABLE\" = true ]; then',\n\t\t\" if docker compose version &>/dev/null 2>&1; then\",\n\t\t' pass \"Docker Compose v2 available\"',\n\t\t\" else\",\n\t\t' warn_svc \"Docker Compose v2 not found\"',\n\t\t\" fi\",\n\t\t\" fi\",\n\t\t\"\",\n\t\t\" # .env\",\n\t\t' if [ -f \".env\" ]; then',\n\t\t' pass \".env file exists\"',\n\t\t\" EMPTY_SECRETS=0\",\n\t\t\" while IFS='=' read -r key value; do\",\n\t\t' [[ \"$key\" =~ ^#.*$ ]] && continue',\n\t\t' [[ -z \"$key\" ]] && continue',\n\t\t' if [[ \"$key\" =~ (PASSWORD|TOKEN|SECRET|SESSION_KEY|COOKIE|API_KEY) ]] && [[ -z \"${value:-}\" ]]; then',\n\t\t' if [ \"$VERBOSE\" = true ]; then',\n\t\t' warn_svc \"Empty secret: $key\"',\n\t\t\" fi\",\n\t\t\" EMPTY_SECRETS=$((EMPTY_SECRETS + 1))\",\n\t\t\" fi\",\n\t\t\" done < .env 2>/dev/null || true\",\n\t\t' if [ \"$EMPTY_SECRETS\" -gt 0 ]; then',\n\t\t' warn_svc \"$EMPTY_SECRETS secret(s) are empty in .env\"',\n\t\t\" else\",\n\t\t' pass \"All secrets are populated\"',\n\t\t\" fi\",\n\t\t\" else\",\n\t\t' warn_svc \".env file not found\"',\n\t\t\" fi\",\n\t\t\"\",\n\t\t\" # Disk space\",\n\t\t\" AVAIL_KB=$(df -k . 2>/dev/null | awk 'NR==2{print $4}' || echo \\\"0\\\")\",\n\t\t\" AVAIL_GB=$((AVAIL_KB / 1024 / 1024))\",\n\t\t' if [ \"$AVAIL_GB\" -lt 2 ]; then',\n\t\t' warn_svc \"Low disk space: ${AVAIL_GB}GB available\"',\n\t\t\" else\",\n\t\t' pass \"Disk space: ${AVAIL_GB}GB available\"',\n\t\t\" fi\",\n\t\t\"}\",\n\t\t\"\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"# Phase 2: Container Status\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"check_container() {\",\n\t\t' local id=\"$1\" name=\"$2\" icon=\"$3\"',\n\t\t\"\",\n\t\t\" local status restarts\",\n\t\t' status=$(docker compose ps \"$id\" --format json 2>/dev/null | head -1)',\n\t\t\"\",\n\t\t' if [ -z \"$status\" ]; then',\n\t\t' fail \"$icon $(pad_name \"$name\") — Container not found\"',\n\t\t\" return\",\n\t\t\" fi\",\n\t\t\"\",\n\t\t\" local state health_status\",\n\t\t' state=$(echo \"$status\" | grep -o \\'\"State\":\"[^\"]*\"\\' | cut -d\\'\"\\' -f4 || echo \"unknown\")',\n\t\t' health_status=$(echo \"$status\" | grep -o \\'\"Health\":\"[^\"]*\"\\' | cut -d\\'\"\\' -f4 || echo \"none\")',\n\t\t\"\",\n\t\t' restarts=$(docker inspect --format=\"{{.RestartCount}}\" \"$(docker compose ps -q \"$id\" 2>/dev/null | head -1)\" 2>/dev/null || echo \"0\")',\n\t\t\"\",\n\t\t' if [ \"$state\" = \"running\" ]; then',\n\t\t' if [ \"$health_status\" = \"healthy\" ]; then',\n\t\t' pass \"$icon $(pad_name \"$name\") Running (healthy) restarts: $restarts\"',\n\t\t' elif [ \"$health_status\" = \"starting\" ]; then',\n\t\t' warn_svc \"$icon $(pad_name \"$name\") Running (starting) restarts: $restarts\"',\n\t\t' elif [ \"$health_status\" = \"unhealthy\" ]; then',\n\t\t' fail \"$icon $(pad_name \"$name\") Running (UNHEALTHY) restarts: $restarts\"',\n\t\t\" else\",\n\t\t' pass \"$icon $(pad_name \"$name\") Running restarts: $restarts\"',\n\t\t\" fi\",\n\t\t' if [ \"$restarts\" -gt 5 ] 2>/dev/null; then',\n\t\t' warn_svc \" └─ $name has restarted $restarts times (possible crash loop)\"',\n\t\t\" fi\",\n\t\t' elif [ \"$state\" = \"exited\" ]; then',\n\t\t\" local exit_code\",\n\t\t' exit_code=$(docker inspect --format=\"{{.State.ExitCode}}\" \"$(docker compose ps -q \"$id\" 2>/dev/null | head -1)\" 2>/dev/null || echo \"?\")',\n\t\t' fail \"$icon $(pad_name \"$name\") Exited (code $exit_code)\"',\n\t\t' if [ \"$exit_code\" = \"137\" ]; then',\n\t\t' ERRORS+=(\" └─ $name: OOM killed (exit 137). Increase memory limits.\")',\n\t\t' elif [ \"$exit_code\" = \"1\" ]; then',\n\t\t' ERRORS+=(\" └─ $name: Application error (exit 1). Check logs: docker compose logs $id\")',\n\t\t\" fi\",\n\t\t\" else\",\n\t\t' fail \"$icon $(pad_name \"$name\") State: $state\"',\n\t\t\" fi\",\n\t\t\"}\",\n\t\t\"\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"# Phase 3: Port Reachability\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"check_port() {\",\n\t\t' local id=\"$1\" port=\"$2\" desc=\"$3\"',\n\t\t\" if command -v nc &>/dev/null; then\",\n\t\t' if nc -z -w 2 localhost \"$port\" 2>/dev/null; then',\n\t\t' dim \" Port $port ($desc) — reachable\"',\n\t\t\" else\",\n\t\t' warn_svc \" └─ $id: Port $port ($desc) — NOT reachable\"',\n\t\t\" fi\",\n\t\t\" elif command -v curl &>/dev/null; then\",\n\t\t' if curl -sf --max-time 2 \"http://localhost:$port/\" &>/dev/null; then',\n\t\t' dim \" Port $port ($desc) — reachable\"',\n\t\t\" else\",\n\t\t' warn_svc \" └─ $id: Port $port ($desc) — NOT reachable\"',\n\t\t\" fi\",\n\t\t\" else\",\n\t\t' dim \" Port $port ($desc) — skipped (no nc or curl)\"',\n\t\t\" fi\",\n\t\t\"}\",\n\t\t\"\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"# Phase 4: Health Check Commands\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"check_health_cmd() {\",\n\t\t' local id=\"$1\" cmd=\"$2\"',\n\t\t' local cid=$(docker compose ps -q \"$id\" 2>/dev/null | head -1)',\n\t\t' [ -z \"$cid\" ] && return',\n\t\t' if docker exec \"$cid\" sh -c \"$cmd\" &>/dev/null; then',\n\t\t' dim \" Health command passed\"',\n\t\t\" else\",\n\t\t' warn_svc \" └─ $id: Health check command failed\"',\n\t\t' if [ \"$VERBOSE\" = true ]; then dim \" Command: $cmd\"; fi',\n\t\t\" fi\",\n\t\t\"}\",\n\t\t\"\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"# Phase 5: Resource Usage\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"phase_resources() {\",\n\t\t' if [ \"$JSON_MODE\" = false ]; then',\n\t\t' echo \"\"',\n\t\t' echo -e \"${BOLD}── Phase 5: Resource Usage ─────────────────────────────────${NC}\"',\n\t\t' echo \"\"',\n\t\t\" fi\",\n\t\t' docker stats --no-stream --format \"table {{.Name}}\\\\t{{.CPUPerc}}\\\\t{{.MemUsage}}\\\\t{{.MemPerc}}\" 2>/dev/null || true',\n\t\t' if [ \"$VERBOSE\" = false ] && [ \"$JSON_MODE\" = false ]; then',\n\t\t' info \"Use --verbose to see warnings for high memory usage\"',\n\t\t\" fi\",\n\t\t\"}\",\n\t\t\"\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"# Phase 6: Log Scan\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"phase_logs() {\",\n\t\t' if [ \"$QUICK_MODE\" = true ]; then',\n\t\t' if [ \"$JSON_MODE\" = false ]; then echo \"\"; info \"Skipping log scan (--quick mode)\"; fi',\n\t\t\" return\",\n\t\t\" fi\",\n\t\t' if [ \"$JSON_MODE\" = false ]; then',\n\t\t' echo \"\"',\n\t\t' echo -e \"${BOLD}── Phase 6: Log Scan ───────────────────────────────────────${NC}\"',\n\t\t' echo \"\"',\n\t\t\" fi\",\n\t\t\" local has_errors=false\",\n\t\tL(\" for svc_id in \", svcIdList, \"; do\"),\n\t\t\" local error_lines\",\n\t\t' error_lines=$(docker compose logs --tail=50 \"$svc_id\" 2>/dev/null | grep -iE \"(error|fatal|panic|exception|segfault|killed|oom)\" | tail -5 || true)',\n\t\t' if [ -n \"$error_lines\" ]; then',\n\t\t\" has_errors=true\",\n\t\t' warn_svc \"$svc_id — found error patterns in logs:\"',\n\t\t' echo \"$error_lines\" | while IFS= read -r eline; do',\n\t\t' dim \" | $eline\"',\n\t\t\" done\",\n\t\t\" fi\",\n\t\t\" done\",\n\t\t' if [ \"$has_errors\" = false ] && [ \"$JSON_MODE\" = false ]; then',\n\t\t' pass \"No error patterns found in recent logs\"',\n\t\t\" fi\",\n\t\t\"}\",\n\t\t\"\",\n\t\t\"# ── Bare-metal process check ────────────────────────────────────────\",\n\t\t\"check_process() {\",\n\t\t' local id=\"$1\" name=\"$2\" icon=\"$3\"',\n\t\t\" if command -v systemctl &>/dev/null; then\",\n\t\t' if systemctl is-active --quiet \"$id\" 2>/dev/null; then',\n\t\t' pass \"$icon $(pad_name \"$name\") Active (systemd)\"',\n\t\t\" return\",\n\t\t\" fi\",\n\t\t\" fi\",\n\t\t' if pgrep -f \"$id\" &>/dev/null; then',\n\t\t' pass \"$icon $(pad_name \"$name\") Running (process)\"',\n\t\t\" else\",\n\t\t' fail \"$icon $(pad_name \"$name\") NOT running\"',\n\t\t\" fi\",\n\t\t\"}\",\n\t\t\"\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"# Main\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"\",\n\t\t\"DOCKER_AVAILABLE=false\",\n\t\t\"\",\n\t\t'if [ \"$JSON_MODE\" = false ]; then',\n\t\t' echo \"\"',\n\t\t' echo \"═══════════════════════════════════════════════════════════════════════\"',\n\t\tL(' echo \" 🐾 OpenClaw Stack Health Report — ', name, '\"'),\n\t\t' echo \"═══════════════════════════════════════════════════════════════════════\"',\n\t\t\"fi\",\n\t\t\"\",\n\t\t\"phase_environment\",\n\t\t\"\",\n\t\t'if [ \"$DOCKER_AVAILABLE\" = true ]; then',\n\t\t' if [ \"$JSON_MODE\" = false ]; then',\n\t\t' echo \"\"',\n\t\t' echo -e \"${BOLD}── Phase 2–4: Service Checks ──────────────────────────────${NC}\"',\n\t\t' echo \"\"',\n\t\t\" fi\",\n\t\t\"\",\n\t\t...dockerChecks,\n\t\t\"\",\n\t\t\" phase_resources\",\n\t\t\" phase_logs\",\n\t\t\"\",\n\t\t\"else\",\n\t\t' if [ \"$JSON_MODE\" = false ]; then',\n\t\t' echo \"\"',\n\t\t' echo -e \"${BOLD}── Bare-Metal Service Checks ──────────────────────────────${NC}\"',\n\t\t' echo \"\"',\n\t\t\" fi\",\n\t\t\"\",\n\t\t...bmChecks,\n\t\t\"fi\",\n\t\t\"\",\n\t\t\"# ── Summary ──────────────────────────────────────────────────────────\",\n\t\t'echo \"\"',\n\t\t'echo \"═══════════════════════════════════════════════════════════════════════\"',\n\t\t'echo \"\"',\n\t\t'echo -e \" ${BOLD}Summary${NC}: $TOTAL services checked\"',\n\t\t'echo -e \" ${GREEN}Healthy: $HEALTHY${NC} | ${YELLOW}Warnings: $WARNING${NC} | ${RED}Failed: $FAILED${NC}\"',\n\t\t\"\",\n\t\t\"if [ ${#ERRORS[@]} -gt 0 ]; then\",\n\t\t' echo \"\"',\n\t\t' echo -e \" ${BOLD}${RED}── Errors ─────────────────────────────────────────────────${NC}\"',\n\t\t' for e in \"${ERRORS[@]}\"; do',\n\t\t' echo -e \" ${RED}$e${NC}\"',\n\t\t\" done\",\n\t\t\"fi\",\n\t\t\"\",\n\t\t'echo \"\"',\n\t\t'echo \"═══════════════════════════════════════════════════════════════════════\"',\n\t\t'echo \"\"',\n\t\t\"\",\n\t\t'if [ \"$FAILED\" -gt 0 ]; then',\n\t\t' echo -e \" ${RED}Some services need attention. Run with --verbose for details.${NC}\"',\n\t\t\" exit 1\",\n\t\t\"else\",\n\t\t' echo -e \" ${GREEN}🎉 All services are running!${NC}\"',\n\t\t\" exit 0\",\n\t\t\"fi\",\n\t];\n\n\treturn lines.join(\"\\n\") + \"\\n\";\n}\n\n// ─── PowerShell Script ──────────────────────────────────────────────────────\n\nfunction generatePowerShellScript(resolved: ResolverOutput, options: HealthCheckOptions): string {\n\tconst checks = extractServiceChecks(resolved);\n\tconst name = options.projectName;\n\tconst total = checks.length;\n\n\t// Build per-service docker check calls\n\tconst dockerChecks: string[] = [];\n\tfor (const svc of checks) {\n\t\tdockerChecks.push(L(\" # \", svc.icon, \" \", svc.name));\n\t\tdockerChecks.push(\n\t\t\tL(\n\t\t\t\t' Test-Container -ServiceId \"',\n\t\t\t\tsvc.id,\n\t\t\t\t'\" -ServiceName \"',\n\t\t\t\tsvc.name,\n\t\t\t\t'\" -Icon \"',\n\t\t\t\tsvc.icon,\n\t\t\t\t'\"',\n\t\t\t),\n\t\t);\n\t\tfor (const p of svc.ports.filter((pp) => pp.exposed)) {\n\t\t\tdockerChecks.push(\n\t\t\t\tL(\n\t\t\t\t\t' Test-Port -ServiceId \"',\n\t\t\t\t\tsvc.id,\n\t\t\t\t\t'\" -Port ',\n\t\t\t\t\tString(p.host),\n\t\t\t\t\t' -Description \"',\n\t\t\t\t\tp.description,\n\t\t\t\t\t'\"',\n\t\t\t\t),\n\t\t\t);\n\t\t}\n\t\tdockerChecks.push(\"\");\n\t}\n\n\t// Build per-service bare-metal check calls\n\tconst bmChecks: string[] = [];\n\tfor (const svc of checks) {\n\t\tbmChecks.push(L(\" # \", svc.icon, \" \", svc.name));\n\t\tbmChecks.push(\n\t\t\tL(\n\t\t\t\t' Test-ProcessRunning -ServiceId \"',\n\t\t\t\tsvc.id,\n\t\t\t\t'\" -ServiceName \"',\n\t\t\t\tsvc.name,\n\t\t\t\t'\" -Icon \"',\n\t\t\t\tsvc.icon,\n\t\t\t\t'\"',\n\t\t\t),\n\t\t);\n\t\tfor (const p of svc.ports.filter((pp) => pp.exposed)) {\n\t\t\tbmChecks.push(\n\t\t\t\tL(\n\t\t\t\t\t' Test-Port -ServiceId \"',\n\t\t\t\t\tsvc.id,\n\t\t\t\t\t'\" -Port ',\n\t\t\t\t\tString(p.host),\n\t\t\t\t\t' -Description \"',\n\t\t\t\t\tp.description,\n\t\t\t\t\t'\"',\n\t\t\t\t),\n\t\t\t);\n\t\t}\n\t\tbmChecks.push(\"\");\n\t}\n\n\tconst svcIdList = checks.map((s) => '\"' + s.id + '\"').join(\", \");\n\n\tconst lines: string[] = [\n\t\t\"#Requires -Version 5.1\",\n\t\t\"<#\",\n\t\t\".SYNOPSIS\",\n\t\tL(\" OpenClaw Stack Health Check — \", name),\n\t\t\"\",\n\t\t\".DESCRIPTION\",\n\t\t\" Auto-generated verification script for your stack.\",\n\t\tL(\n\t\t\t\" Checks \",\n\t\t\tString(total),\n\t\t\t\" services: container status, port reachability, and log errors.\",\n\t\t),\n\t\t\"\",\n\t\t\".PARAMETER Quick\",\n\t\t\" Skip log scanning for faster results.\",\n\t\t\"\",\n\t\t\".PARAMETER Json\",\n\t\t\" Output results as JSON.\",\n\t\t\"\",\n\t\t\".EXAMPLE\",\n\t\t\" .\\\\scripts\\\\health-check.ps1\",\n\t\t\" .\\\\scripts\\\\health-check.ps1 -Quick\",\n\t\t\" .\\\\scripts\\\\health-check.ps1 -Json\",\n\t\t\"#>\",\n\t\t\"param(\",\n\t\t\" [switch]$Quick,\",\n\t\t\" [switch]$Json,\",\n\t\t\" [switch]$Detailed\",\n\t\t\")\",\n\t\t\"\",\n\t\t'$ErrorActionPreference = \"Continue\"',\n\t\t\"\",\n\t\t\"# ── Counters ─────────────────────────────────────────────────────────\",\n\t\tL(\"$script:Total = \", String(total)),\n\t\t\"$script:Healthy = 0\",\n\t\t\"$script:Warnings = 0\",\n\t\t\"$script:Failed = 0\",\n\t\t\"$script:Errors = @()\",\n\t\t\"$script:ServiceResults = @()\",\n\t\t\"$script:DockerAvailable = $false\",\n\t\t\"\",\n\t\t\"# ── Helpers ──────────────────────────────────────────────────────────\",\n\t\t\"\",\n\t\t'function Write-Pass($msg) { Write-Host \" ✅ $msg\" -ForegroundColor Green; $script:Healthy++ }',\n\t\t'function Write-Warn($msg) { Write-Host \" ⚠️ $msg\" -ForegroundColor Yellow; $script:Warnings++ }',\n\t\t'function Write-Fail($msg) { Write-Host \" ❌ $msg\" -ForegroundColor Red; $script:Failed++; $script:Errors += $msg }',\n\t\t'function Write-Info($msg) { Write-Host \" ℹ $msg\" -ForegroundColor Cyan }',\n\t\t'function Write-Dim($msg) { Write-Host \" $msg\" -ForegroundColor DarkGray }',\n\t\t\"\",\n\t\t\"function Pad-Name([string]$n) { return $n.PadRight(25) }\",\n\t\t\"\",\n\t\t\"# ═════════════════════════════════════════════════════════════════════\",\n\t\t\"# Phase 1: Environment\",\n\t\t\"# ═════════════════════════════════════════════════════════════════════\",\n\t\t\"function Test-Environment {\",\n\t\t\" if (-not $Json) {\",\n\t\t' Write-Host \"\"',\n\t\t' Write-Host \"── Phase 1: Environment ────────────────────────────────────\" -ForegroundColor White',\n\t\t' Write-Host \"\"',\n\t\t\" }\",\n\t\t\" try {\",\n\t\t\" $null = docker info 2>&1\",\n\t\t\" if ($LASTEXITCODE -eq 0) {\",\n\t\t' Write-Pass \"Docker daemon is running\"',\n\t\t\" $script:DockerAvailable = $true\",\n\t\t\" } else {\",\n\t\t' Write-Fail \"Docker daemon is NOT running\"',\n\t\t\" }\",\n\t\t\" } catch {\",\n\t\t' Write-Warn \"Docker is not installed — running bare-metal checks only\"',\n\t\t\" }\",\n\t\t\" if ($script:DockerAvailable) {\",\n\t\t\" try {\",\n\t\t\" $null = docker compose version 2>&1\",\n\t\t\" if ($LASTEXITCODE -eq 0) {\",\n\t\t' Write-Pass \"Docker Compose v2 available\"',\n\t\t\" } else {\",\n\t\t' Write-Warn \"Docker Compose v2 not found\"',\n\t\t\" }\",\n\t\t\" } catch {\",\n\t\t' Write-Warn \"Docker Compose not available\"',\n\t\t\" }\",\n\t\t\" }\",\n\t\t' $envPath = Join-Path $PSScriptRoot \"..\\\\.env\"',\n\t\t\" if (Test-Path $envPath) {\",\n\t\t' Write-Pass \".env file exists\"',\n\t\t\" $emptySecrets = 0\",\n\t\t\" Get-Content $envPath | ForEach-Object {\",\n\t\t' if ($_ -match \"(PASSWORD|TOKEN|SECRET|SESSION_KEY|COOKIE|API_KEY).*=\\\\s*$\") { $emptySecrets++ }',\n\t\t\" }\",\n\t\t\" if ($emptySecrets -gt 0) {\",\n\t\t' Write-Warn \"$emptySecrets secret(s) are empty in .env\"',\n\t\t\" } else {\",\n\t\t' Write-Pass \"All secrets are populated\"',\n\t\t\" }\",\n\t\t\" } else {\",\n\t\t' Write-Warn \".env file not found\"',\n\t\t\" }\",\n\t\t\" $drive = (Get-Item $PSScriptRoot).PSDrive\",\n\t\t\" $freeGB = [math]::Round($drive.Free / 1GB, 1)\",\n\t\t\" if ($freeGB -lt 2) {\",\n\t\t' Write-Warn \"Low disk space: ${freeGB}GB available\"',\n\t\t\" } else {\",\n\t\t' Write-Pass \"Disk space: ${freeGB}GB available\"',\n\t\t\" }\",\n\t\t\"}\",\n\t\t\"\",\n\t\t\"# ═════════════════════════════════════════════════════════════════════\",\n\t\t\"# Phase 2: Container Check\",\n\t\t\"# ═════════════════════════════════════════════════════════════════════\",\n\t\t\"function Test-Container {\",\n\t\t\" param([string]$ServiceId, [string]$ServiceName, [string]$Icon)\",\n\t\t\" try {\",\n\t\t\" $status = docker compose ps $ServiceId --format json 2>&1 | ConvertFrom-Json\",\n\t\t\" } catch {\",\n\t\t' Write-Fail \"$Icon $(Pad-Name $ServiceName) Container not found\"',\n\t\t\" return\",\n\t\t\" }\",\n\t\t\" if (-not $status) {\",\n\t\t' Write-Fail \"$Icon $(Pad-Name $ServiceName) Container not found\"',\n\t\t\" return\",\n\t\t\" }\",\n\t\t\" $state = $status.State\",\n\t\t\" $health = $status.Health\",\n\t\t' if ($state -eq \"running\") {',\n\t\t' if ($health -eq \"healthy\") {',\n\t\t' Write-Pass \"$Icon $(Pad-Name $ServiceName) Running (healthy)\"',\n\t\t' } elseif ($health -eq \"starting\") {',\n\t\t' Write-Warn \"$Icon $(Pad-Name $ServiceName) Running (starting)\"',\n\t\t' } elseif ($health -eq \"unhealthy\") {',\n\t\t' Write-Fail \"$Icon $(Pad-Name $ServiceName) Running (UNHEALTHY)\"',\n\t\t\" } else {\",\n\t\t' Write-Pass \"$Icon $(Pad-Name $ServiceName) Running\"',\n\t\t\" }\",\n\t\t' } elseif ($state -eq \"exited\") {',\n\t\t' Write-Fail \"$Icon $(Pad-Name $ServiceName) Exited\"',\n\t\t\" } else {\",\n\t\t' Write-Fail \"$Icon $(Pad-Name $ServiceName) State: $state\"',\n\t\t\" }\",\n\t\t\"}\",\n\t\t\"\",\n\t\t\"# ═════════════════════════════════════════════════════════════════════\",\n\t\t\"# Phase 3: Port Check\",\n\t\t\"# ═════════════════════════════════════════════════════════════════════\",\n\t\t\"function Test-Port {\",\n\t\t\" param([string]$ServiceId, [int]$Port, [string]$Description)\",\n\t\t\" try {\",\n\t\t\" $tcp = New-Object System.Net.Sockets.TcpClient\",\n\t\t' $tcp.ConnectAsync(\"localhost\", $Port).Wait(2000) | Out-Null',\n\t\t\" if ($tcp.Connected) {\",\n\t\t' Write-Dim \" Port $Port ($Description) — reachable\"',\n\t\t\" $tcp.Close()\",\n\t\t\" } else {\",\n\t\t' Write-Warn \" └─ ${ServiceId}: Port $Port ($Description) — NOT reachable\"',\n\t\t\" }\",\n\t\t\" } catch {\",\n\t\t' Write-Warn \" └─ ${ServiceId}: Port $Port ($Description) — NOT reachable\"',\n\t\t\" }\",\n\t\t\"}\",\n\t\t\"\",\n\t\t\"# ── Bare-metal process check ────────────────────────────────────────\",\n\t\t\"function Test-ProcessRunning {\",\n\t\t\" param([string]$ServiceId, [string]$ServiceName, [string]$Icon)\",\n\t\t\" $svc = Get-Service -Name $ServiceId -ErrorAction SilentlyContinue\",\n\t\t\" if ($svc) {\",\n\t\t' if ($svc.Status -eq \"Running\") {',\n\t\t' Write-Pass \"$Icon $(Pad-Name $ServiceName) Active (service)\"',\n\t\t\" } else {\",\n\t\t' Write-Fail \"$Icon $(Pad-Name $ServiceName) $($svc.Status)\"',\n\t\t\" }\",\n\t\t\" return\",\n\t\t\" }\",\n\t\t\" $proc = Get-Process -Name $ServiceId -ErrorAction SilentlyContinue\",\n\t\t\" if ($proc) {\",\n\t\t' Write-Pass \"$Icon $(Pad-Name $ServiceName) Running (process)\"',\n\t\t\" } else {\",\n\t\t' Write-Fail \"$Icon $(Pad-Name $ServiceName) NOT running\"',\n\t\t\" }\",\n\t\t\"}\",\n\t\t\"\",\n\t\t\"# ═════════════════════════════════════════════════════════════════════\",\n\t\t\"# Phase 6: Log Scan\",\n\t\t\"# ═════════════════════════════════════════════════════════════════════\",\n\t\t\"function Test-Logs {\",\n\t\t\" if ($Quick) {\",\n\t\t' if (-not $Json) { Write-Info \"Skipping log scan (-Quick mode)\" }',\n\t\t\" return\",\n\t\t\" }\",\n\t\t\" if (-not $Json) {\",\n\t\t' Write-Host \"\"',\n\t\t' Write-Host \"── Phase 6: Log Scan ───────────────────────────────────────\" -ForegroundColor White',\n\t\t' Write-Host \"\"',\n\t\t\" }\",\n\t\t\" $hasErrors = $false\",\n\t\tL(\" $svcIds = @(\", svcIdList, \")\"),\n\t\t\" foreach ($svcId in $svcIds) {\",\n\t\t\" try {\",\n\t\t\" $logs = docker compose logs --tail=50 $svcId 2>&1 | Out-String\",\n\t\t' $errorLines = $logs -split \"`n\" | Where-Object { $_ -match \"(error|fatal|panic|exception|segfault|killed|oom)\" } | Select-Object -Last 5',\n\t\t\" if ($errorLines) {\",\n\t\t\" $hasErrors = $true\",\n\t\t' Write-Warn \"$svcId — found error patterns in logs:\"',\n\t\t' $errorLines | ForEach-Object { Write-Dim \" | $_\" }',\n\t\t\" }\",\n\t\t\" } catch {}\",\n\t\t\" }\",\n\t\t\" if (-not $hasErrors -and -not $Json) {\",\n\t\t' Write-Pass \"No error patterns found in recent logs\"',\n\t\t\" }\",\n\t\t\"}\",\n\t\t\"\",\n\t\t\"# ═════════════════════════════════════════════════════════════════════\",\n\t\t\"# Main\",\n\t\t\"# ═════════════════════════════════════════════════════════════════════\",\n\t\t\"\",\n\t\t\"if (-not $Json) {\",\n\t\t' Write-Host \"\"',\n\t\t' Write-Host \"═══════════════════════════════════════════════════════════════════════\" ',\n\t\tL(' Write-Host \" 🐾 OpenClaw Stack Health Report — ', name, '\"'),\n\t\t' Write-Host \"═══════════════════════════════════════════════════════════════════════\"',\n\t\t\"}\",\n\t\t\"\",\n\t\t\"Test-Environment\",\n\t\t\"\",\n\t\t\"if ($script:DockerAvailable) {\",\n\t\t\" if (-not $Json) {\",\n\t\t' Write-Host \"\"',\n\t\t' Write-Host \"── Phase 2–4: Service Checks ──────────────────────────────\" -ForegroundColor White',\n\t\t' Write-Host \"\"',\n\t\t\" }\",\n\t\t\"\",\n\t\t...dockerChecks,\n\t\t\" Test-Logs\",\n\t\t\"} else {\",\n\t\t\" if (-not $Json) {\",\n\t\t' Write-Host \"\"',\n\t\t' Write-Host \"── Bare-Metal Service Checks ──────────────────────────────\" -ForegroundColor White',\n\t\t' Write-Host \"\"',\n\t\t\" }\",\n\t\t\"\",\n\t\t...bmChecks,\n\t\t\"}\",\n\t\t\"\",\n\t\t\"# ── Summary ──────────────────────────────────────────────────────────\",\n\t\t\"if ($Json) {\",\n\t\t\" $result = @{\",\n\t\tL(' project = \"', name, '\"'),\n\t\t' timestamp = (Get-Date -Format \"o\")',\n\t\t\" summary = @{\",\n\t\t\" total = $script:Total\",\n\t\t\" healthy = $script:Healthy\",\n\t\t\" warnings = $script:Warnings\",\n\t\t\" failed = $script:Failed\",\n\t\t\" }\",\n\t\t\" errors = $script:Errors\",\n\t\t\" }\",\n\t\t\" $result | ConvertTo-Json -Depth 5\",\n\t\t\"} else {\",\n\t\t' Write-Host \"\"',\n\t\t' Write-Host \"═══════════════════════════════════════════════════════════════════════\"',\n\t\t' Write-Host \"\"',\n\t\t' Write-Host \" Summary: $($script:Total) services checked\"',\n\t\t' Write-Host \" Healthy: $($script:Healthy)\" -ForegroundColor Green -NoNewline',\n\t\t' Write-Host \" | Warnings: $($script:Warnings)\" -ForegroundColor Yellow -NoNewline',\n\t\t' Write-Host \" | Failed: $($script:Failed)\" -ForegroundColor Red',\n\t\t\" if ($script:Errors.Count -gt 0) {\",\n\t\t' Write-Host \"\"',\n\t\t' Write-Host \" ── Errors ──────────────────────────────────────────────────\" -ForegroundColor Red',\n\t\t' $script:Errors | ForEach-Object { Write-Host \" $_\" -ForegroundColor Red }',\n\t\t\" }\",\n\t\t' Write-Host \"\"',\n\t\t' Write-Host \"═══════════════════════════════════════════════════════════════════════\"',\n\t\t' Write-Host \"\"',\n\t\t\" if ($script:Failed -gt 0) {\",\n\t\t' Write-Host \" Some services need attention. Run with -Detailed for more.\" -ForegroundColor Red',\n\t\t\" exit 1\",\n\t\t\" } else {\",\n\t\t' Write-Host \" 🎉 All services are running!\" -ForegroundColor Green',\n\t\t\" exit 0\",\n\t\t\" }\",\n\t\t\"}\",\n\t];\n\n\treturn lines.join(\"\\n\") + \"\\n\";\n}\n"],"mappings":";;;;;;;;AAkBA,SAAgB,oBACf,UACA,SACyB;CACzB,MAAM,QAAgC,EAAE;AAExC,OAAM,6BAA6B,mBAAmB,UAAU,QAAQ;AACxE,OAAM,8BAA8B,yBAAyB,UAAU,QAAQ;AAE/E,QAAO;;AAaR,SAAS,qBAAqB,UAA0C;AACvE,QAAO,SAAS,SAAS,KAAK,SAAS;EACtC,IAAI,IAAI,WAAW;EACnB,MAAM,IAAI,WAAW;EACrB,MAAM,IAAI,WAAW;EACrB,OAAO,IAAI,WAAW,MAAM,KAAK,OAAO;GACvC,MAAM,EAAE;GACR,WAAW,EAAE;GACb,aAAa,EAAE;GACf,SAAS,EAAE;GACX,EAAE;EACH,gBAAgB,IAAI,WAAW,aAAa,QAAQ;EACpD,EAAE;;AAGJ,SAAS,YAAY,GAAmB;AACvC,QAAO,EAAE,QAAQ,MAAM,QAAQ,CAAC,QAAQ,MAAM,OAAM;;AAIrD,SAAS,EAAE,GAAG,OAAyB;AACtC,QAAO,MAAM,KAAK,GAAG;;AAKtB,SAAS,mBAAmB,UAA0B,SAAqC;CAC1F,MAAM,SAAS,qBAAqB,SAAS;CAC7C,MAAM,OAAO,QAAQ;CACrB,MAAM,QAAQ,OAAO;CAGrB,MAAM,eAAyB,EAAE;AACjC,MAAK,MAAM,OAAO,QAAQ;AACzB,eAAa,KAAK,EAAE,WAAW,IAAI,MAAM,KAAK,IAAI,MAAM,MAAM,CAAC;AAC/D,eAAa,KAAK,EAAE,wBAAuB,IAAI,IAAI,SAAO,IAAI,MAAM,SAAO,IAAI,MAAM,KAAI,CAAC;AAC1F,OAAK,MAAM,KAAK,IAAI,MAAM,QAAQ,OAAO,GAAG,QAAQ,CACnD,cAAa,KACZ,EAAE,mBAAkB,IAAI,IAAI,OAAM,OAAO,EAAE,KAAK,EAAE,OAAM,EAAE,aAAa,KAAI,CAC3E;AAEF,MAAI,IAAI,eACP,cAAa,KACZ,EAAE,yBAAwB,IAAI,IAAI,SAAO,YAAY,IAAI,eAAe,EAAE,KAAI,CAC9E;MAED,cAAa,KAAK,mDAAmD;AAEtE,eAAa,KAAK,GAAG;;CAItB,MAAM,WAAqB,EAAE;AAC7B,MAAK,MAAM,OAAO,QAAQ;AACzB,WAAS,KAAK,EAAE,WAAW,IAAI,MAAM,KAAK,IAAI,MAAM,mBAAmB,CAAC;AACxE,WAAS,KAAK,EAAE,sBAAqB,IAAI,IAAI,SAAO,IAAI,MAAM,SAAO,IAAI,MAAM,KAAI,CAAC;AACpF,OAAK,MAAM,KAAK,IAAI,MAAM,QAAQ,OAAO,GAAG,QAAQ,CACnD,UAAS,KAAK,EAAE,mBAAkB,IAAI,IAAI,OAAM,OAAO,EAAE,KAAK,EAAE,OAAM,EAAE,aAAa,KAAI,CAAC;AAE3F,WAAS,KAAK,GAAG;;CAIlB,MAAM,YAAY,OAAO,KAAK,MAAM,OAAM,EAAE,KAAK,KAAI,CAAC,KAAK,IAAI;AA6V/D,QA3VwB;EACvB;EACA;EACA;EACA;EACA,EAAE,uCAAuC,KAAK;EAC9C;EACA;EACA;EACA,EAAE,aAAa,OAAO,MAAM,EAAE,6BAA6B;EAC3D;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA,EAAE,UAAU,OAAO,MAAM,CAAC;EAC1B;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA,EAAE,oBAAoB,WAAW,OAAO;EACxC;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA,EAAE,gDAA+C,MAAM,KAAI;EAC3D;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA,GAAG;EACH;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA,GAAG;EACH;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA,CAEY,KAAK,KAAK,GAAG;;AAK3B,SAAS,yBAAyB,UAA0B,SAAqC;CAChG,MAAM,SAAS,qBAAqB,SAAS;CAC7C,MAAM,OAAO,QAAQ;CACrB,MAAM,QAAQ,OAAO;CAGrB,MAAM,eAAyB,EAAE;AACjC,MAAK,MAAM,OAAO,QAAQ;AACzB,eAAa,KAAK,EAAE,UAAU,IAAI,MAAM,KAAK,IAAI,KAAK,CAAC;AACvD,eAAa,KACZ,EACC,oCACA,IAAI,IACJ,sBACA,IAAI,MACJ,eACA,IAAI,MACJ,KACA,CACD;AACD,OAAK,MAAM,KAAK,IAAI,MAAM,QAAQ,OAAO,GAAG,QAAQ,CACnD,cAAa,KACZ,EACC,+BACA,IAAI,IACJ,aACA,OAAO,EAAE,KAAK,EACd,oBACA,EAAE,aACF,KACA,CACD;AAEF,eAAa,KAAK,GAAG;;CAItB,MAAM,WAAqB,EAAE;AAC7B,MAAK,MAAM,OAAO,QAAQ;AACzB,WAAS,KAAK,EAAE,UAAU,IAAI,MAAM,KAAK,IAAI,KAAK,CAAC;AACnD,WAAS,KACR,EACC,yCACA,IAAI,IACJ,sBACA,IAAI,MACJ,eACA,IAAI,MACJ,KACA,CACD;AACD,OAAK,MAAM,KAAK,IAAI,MAAM,QAAQ,OAAO,GAAG,QAAQ,CACnD,UAAS,KACR,EACC,+BACA,IAAI,IACJ,aACA,OAAO,EAAE,KAAK,EACd,oBACA,EAAE,aACF,KACA,CACD;AAEF,WAAS,KAAK,GAAG;;CAGlB,MAAM,YAAY,OAAO,KAAK,MAAM,OAAM,EAAE,KAAK,KAAI,CAAC,KAAK,KAAK;AA8RhE,QA5RwB;EACvB;EACA;EACA;EACA,EAAE,sCAAsC,KAAK;EAC7C;EACA;EACA;EACA,EACC,eACA,OAAO,MAAM,EACb,kEACA;EACD;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA,EAAE,oBAAoB,OAAO,MAAM,CAAC;EACpC;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA,EAAE,oBAAoB,WAAW,IAAI;EACrC;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA,EAAE,wDAAuD,MAAM,KAAI;EACnE;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA,GAAG;EACH;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA,GAAG;EACH;EACA;EACA;EACA;EACA;EACA,EAAE,wBAAuB,MAAM,KAAI;EACnC;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA,CAEY,KAAK,KAAK,GAAG"}
|
|
1
|
+
{"version":3,"file":"health-check.mjs","names":[],"sources":["../../src/generators/health-check.ts"],"sourcesContent":["import type { ResolverOutput } from \"../types.js\";\n\n// ─── Types ──────────────────────────────────────────────────────────────────\n\nexport interface HealthCheckOptions {\n\tprojectName: string;\n\tdeploymentType?: \"docker\" | \"bare-metal\" | \"local\";\n}\n\n// ─── Generator ──────────────────────────────────────────────────────────────\n\n/**\n * Generates stack-specific health check and verification scripts.\n *\n * Each generated script is dynamic — it contains the exact service names,\n * ports, and health check commands from the resolved stack so users get\n * a precise, zero-configuration verifier.\n */\nexport function generateHealthCheck(\n\tresolved: ResolverOutput,\n\toptions: HealthCheckOptions,\n): Record<string, string> {\n\tconst files: Record<string, string> = {};\n\n\tfiles[\"scripts/health-check.sh\"] = generateBashScript(resolved, options);\n\tfiles[\"scripts/health-check.ps1\"] = generatePowerShellScript(resolved, options);\n\n\treturn files;\n}\n\n// ─── Service metadata extraction ────────────────────────────────────────────\n\ninterface ServiceCheck {\n\tid: string;\n\tname: string;\n\ticon: string;\n\tports: Array<{ host: number; container: number; description: string; exposed: boolean }>;\n\thealthCheckCmd: string | null;\n}\n\nfunction extractServiceChecks(resolved: ResolverOutput): ServiceCheck[] {\n\treturn resolved.services.map((svc) => ({\n\t\tid: svc.definition.id,\n\t\tname: svc.definition.name,\n\t\ticon: svc.definition.icon,\n\t\tports: svc.definition.ports.map((p) => ({\n\t\t\thost: p.host,\n\t\t\tcontainer: p.container,\n\t\t\tdescription: p.description,\n\t\t\texposed: p.exposed,\n\t\t})),\n\t\thealthCheckCmd: svc.definition.healthcheck?.test ?? null,\n\t}));\n}\n\nfunction escapeShell(s: string): string {\n\treturn s.replace(/'/g, \"'\\\\''\").replace(/\"/g, '\\\\\"');\n}\n\n// Use L() to build lines safely — avoids template literal issues with shell syntax\nfunction L(...parts: string[]): string {\n\treturn parts.join(\"\");\n}\n\n// ─── Bash Script ────────────────────────────────────────────────────────────\n\nfunction generateBashScript(resolved: ResolverOutput, options: HealthCheckOptions): string {\n\tconst checks = extractServiceChecks(resolved);\n\tconst name = options.projectName;\n\tconst total = checks.length;\n\n\t// Build per-service docker check calls\n\tconst dockerChecks: string[] = [];\n\tfor (const svc of checks) {\n\t\tdockerChecks.push(L(\" # ── \", svc.icon, \" \", svc.name, \" ──\"));\n\t\tdockerChecks.push(L(' check_container \"', svc.id, '\" \"', svc.name, '\" \"', svc.icon, '\"'));\n\t\tfor (const p of svc.ports.filter((pp) => pp.exposed)) {\n\t\t\tdockerChecks.push(\n\t\t\t\tL(' check_port \"', svc.id, '\" ', String(p.host), ' \"', p.description, '\"'),\n\t\t\t);\n\t\t}\n\t\tif (svc.healthCheckCmd) {\n\t\t\tdockerChecks.push(\n\t\t\t\tL(' check_health_cmd \"', svc.id, '\" \"', escapeShell(svc.healthCheckCmd), '\"'),\n\t\t\t);\n\t\t} else {\n\t\t\tdockerChecks.push(\" # No healthcheck defined — skipping exec check\");\n\t\t}\n\t\tdockerChecks.push(\"\");\n\t}\n\n\t// Build per-service bare-metal check calls\n\tconst bmChecks: string[] = [];\n\tfor (const svc of checks) {\n\t\tbmChecks.push(L(\" # ── \", svc.icon, \" \", svc.name, \" (bare-metal) ──\"));\n\t\tbmChecks.push(L(' check_process \"', svc.id, '\" \"', svc.name, '\" \"', svc.icon, '\"'));\n\t\tfor (const p of svc.ports.filter((pp) => pp.exposed)) {\n\t\t\tbmChecks.push(L(' check_port \"', svc.id, '\" ', String(p.host), ' \"', p.description, '\"'));\n\t\t}\n\t\tbmChecks.push(\"\");\n\t}\n\n\t// Build the log-scan service list\n\tconst svcIdList = checks.map((s) => '\"' + s.id + '\"').join(\" \");\n\n\tconst lines: string[] = [\n\t\t\"#!/usr/bin/env bash\",\n\t\t\"set -uo pipefail\",\n\t\t\"\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\tL(\"# 🐾 OpenClaw Stack Health Check — \", name),\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"#\",\n\t\t\"# Auto-generated verification script for your stack.\",\n\t\tL(\"# Checks \", String(total), \" services across 6 phases:\"),\n\t\t\"# 1. Environment — prerequisites, .env, secrets\",\n\t\t\"# 2. Container — running state, health status, restart count\",\n\t\t\"# 3. Port — TCP reachability for exposed ports\",\n\t\t\"# 4. Health Command — execute each service's health check\",\n\t\t\"# 5. Resources — memory/CPU warnings\",\n\t\t\"# 6. Logs — scan for ERROR/FATAL/panic patterns\",\n\t\t\"#\",\n\t\t\"# Usage:\",\n\t\t\"# ./scripts/health-check.sh # Full check\",\n\t\t\"# ./scripts/health-check.sh --quick # Skip log scan\",\n\t\t\"# ./scripts/health-check.sh --json # Output as JSON\",\n\t\t\"#\",\n\t\t\"# Exit codes: 0 = all healthy, 1 = some issues, 2 = critical failure\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"\",\n\t\t'SCRIPT_DIR=\"$(cd \"$(dirname \"${BASH_SOURCE[0]}\")\" && pwd)\"',\n\t\t'PROJECT_DIR=\"$(dirname \"$SCRIPT_DIR\")\"',\n\t\t'cd \"$PROJECT_DIR\"',\n\t\t\"\",\n\t\t\"# ── Arguments ────────────────────────────────────────────────────────\",\n\t\t\"QUICK_MODE=false\",\n\t\t\"JSON_MODE=false\",\n\t\t\"VERBOSE=false\",\n\t\t'for arg in \"$@\"; do',\n\t\t' case \"$arg\" in',\n\t\t\" --quick) QUICK_MODE=true ;;\",\n\t\t\" --json) JSON_MODE=true ;;\",\n\t\t\" --verbose|-v) VERBOSE=true ;;\",\n\t\t\" esac\",\n\t\t\"done\",\n\t\t\"\",\n\t\t\"# ── Colour helpers ───────────────────────────────────────────────────\",\n\t\t'if [ -t 1 ] && [ \"$JSON_MODE\" = false ]; then',\n\t\t\" RED='\\\\033[0;31m'; GREEN='\\\\033[0;32m'; YELLOW='\\\\033[1;33m'\",\n\t\t\" CYAN='\\\\033[0;36m'; DIM='\\\\033[2m'; BOLD='\\\\033[1m'; NC='\\\\033[0m'\",\n\t\t\"else\",\n\t\t\" RED=''; GREEN=''; YELLOW=''; CYAN=''; DIM=''; BOLD=''; NC=''\",\n\t\t\"fi\",\n\t\t\"\",\n\t\t\"# ── Counters ─────────────────────────────────────────────────────────\",\n\t\tL(\"TOTAL=\", String(total)),\n\t\t\"HEALTHY=0\",\n\t\t\"WARNING=0\",\n\t\t\"FAILED=0\",\n\t\t\"ERRORS=()\",\n\t\t\"WARNINGS_LIST=()\",\n\t\t\"SERVICE_RESULTS=()\",\n\t\t\"\",\n\t\t\"# ── Utility functions ────────────────────────────────────────────────\",\n\t\t'pass() { echo -e \" ${GREEN}✅ $*${NC}\"; HEALTHY=$((HEALTHY + 1)); }',\n\t\t'warn_svc() { echo -e \" ${YELLOW}⚠️ $*${NC}\"; WARNING=$((WARNING + 1)); WARNINGS_LIST+=(\"$*\"); }',\n\t\t'fail() { echo -e \" ${RED}❌ $*${NC}\"; FAILED=$((FAILED + 1)); ERRORS+=(\"$*\"); }',\n\t\t'info() { echo -e \" ${CYAN}ℹ $*${NC}\"; }',\n\t\t'dim() { echo -e \" ${DIM}$*${NC}\"; }',\n\t\t\"\",\n\t\t\"pad_name() {\",\n\t\t' local name=\"$1\"',\n\t\t\" local pad=25\",\n\t\t' printf \"%-${pad}s\" \"$name\"',\n\t\t\"}\",\n\t\t\"\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"# Phase 1: Environment\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"phase_environment() {\",\n\t\t' if [ \"$JSON_MODE\" = false ]; then',\n\t\t' echo \"\"',\n\t\t' echo -e \"${BOLD}── Phase 1: Environment ────────────────────────────────────${NC}\"',\n\t\t' echo \"\"',\n\t\t\" fi\",\n\t\t\"\",\n\t\t\" # Docker\",\n\t\t\" if command -v docker &>/dev/null; then\",\n\t\t\" if docker info &>/dev/null 2>&1; then\",\n\t\t' pass \"Docker daemon is running\"',\n\t\t\" DOCKER_AVAILABLE=true\",\n\t\t\" else\",\n\t\t' fail \"Docker daemon is NOT running\"',\n\t\t\" DOCKER_AVAILABLE=false\",\n\t\t\" fi\",\n\t\t\" else\",\n\t\t' warn_svc \"Docker is not installed — running bare-metal checks only\"',\n\t\t\" DOCKER_AVAILABLE=false\",\n\t\t\" fi\",\n\t\t\"\",\n\t\t\" # Docker Compose\",\n\t\t' if [ \"$DOCKER_AVAILABLE\" = true ]; then',\n\t\t\" if docker compose version &>/dev/null 2>&1; then\",\n\t\t' pass \"Docker Compose v2 available\"',\n\t\t\" else\",\n\t\t' warn_svc \"Docker Compose v2 not found\"',\n\t\t\" fi\",\n\t\t\" fi\",\n\t\t\"\",\n\t\t\" # .env\",\n\t\t' if [ -f \".env\" ]; then',\n\t\t' pass \".env file exists\"',\n\t\t\" EMPTY_SECRETS=0\",\n\t\t\" while IFS='=' read -r key value; do\",\n\t\t' [[ \"$key\" =~ ^#.*$ ]] && continue',\n\t\t' [[ -z \"$key\" ]] && continue',\n\t\t' if [[ \"$key\" =~ (PASSWORD|TOKEN|SECRET|SESSION_KEY|COOKIE|API_KEY) ]] && [[ -z \"${value:-}\" ]]; then',\n\t\t' if [ \"$VERBOSE\" = true ]; then',\n\t\t' warn_svc \"Empty secret: $key\"',\n\t\t\" fi\",\n\t\t\" EMPTY_SECRETS=$((EMPTY_SECRETS + 1))\",\n\t\t\" fi\",\n\t\t\" done < .env 2>/dev/null || true\",\n\t\t' if [ \"$EMPTY_SECRETS\" -gt 0 ]; then',\n\t\t' warn_svc \"$EMPTY_SECRETS secret(s) are empty in .env\"',\n\t\t\" else\",\n\t\t' pass \"All secrets are populated\"',\n\t\t\" fi\",\n\t\t\" else\",\n\t\t' warn_svc \".env file not found\"',\n\t\t\" fi\",\n\t\t\"\",\n\t\t\" # Disk space\",\n\t\t\" AVAIL_KB=$(df -k . 2>/dev/null | awk 'NR==2{print $4}' || echo \\\"0\\\")\",\n\t\t\" AVAIL_GB=$((AVAIL_KB / 1024 / 1024))\",\n\t\t' if [ \"$AVAIL_GB\" -lt 2 ]; then',\n\t\t' warn_svc \"Low disk space: ${AVAIL_GB}GB available\"',\n\t\t\" else\",\n\t\t' pass \"Disk space: ${AVAIL_GB}GB available\"',\n\t\t\" fi\",\n\t\t\"}\",\n\t\t\"\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"# Phase 2: Container Status\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"check_container() {\",\n\t\t' local id=\"$1\" name=\"$2\" icon=\"$3\"',\n\t\t\"\",\n\t\t\" local status restarts\",\n\t\t' status=$(docker compose ps \"$id\" --format json 2>/dev/null | head -1)',\n\t\t\"\",\n\t\t' if [ -z \"$status\" ]; then',\n\t\t' fail \"$icon $(pad_name \"$name\") — Container not found\"',\n\t\t\" return\",\n\t\t\" fi\",\n\t\t\"\",\n\t\t\" local state health_status\",\n\t\t' state=$(echo \"$status\" | grep -o \\'\"State\":\"[^\"]*\"\\' | cut -d\\'\"\\' -f4 || echo \"unknown\")',\n\t\t' health_status=$(echo \"$status\" | grep -o \\'\"Health\":\"[^\"]*\"\\' | cut -d\\'\"\\' -f4 || echo \"none\")',\n\t\t\"\",\n\t\t' restarts=$(docker inspect --format=\"{{.RestartCount}}\" \"$(docker compose ps -q \"$id\" 2>/dev/null | head -1)\" 2>/dev/null || echo \"0\")',\n\t\t\"\",\n\t\t' if [ \"$state\" = \"running\" ]; then',\n\t\t' if [ \"$health_status\" = \"healthy\" ]; then',\n\t\t' pass \"$icon $(pad_name \"$name\") Running (healthy) restarts: $restarts\"',\n\t\t' elif [ \"$health_status\" = \"starting\" ]; then',\n\t\t' warn_svc \"$icon $(pad_name \"$name\") Running (starting) restarts: $restarts\"',\n\t\t' elif [ \"$health_status\" = \"unhealthy\" ]; then',\n\t\t' fail \"$icon $(pad_name \"$name\") Running (UNHEALTHY) restarts: $restarts\"',\n\t\t\" else\",\n\t\t' pass \"$icon $(pad_name \"$name\") Running restarts: $restarts\"',\n\t\t\" fi\",\n\t\t' if [ \"$restarts\" -gt 5 ] 2>/dev/null; then',\n\t\t' warn_svc \" └─ $name has restarted $restarts times (possible crash loop)\"',\n\t\t\" fi\",\n\t\t' elif [ \"$state\" = \"exited\" ]; then',\n\t\t\" local exit_code\",\n\t\t' exit_code=$(docker inspect --format=\"{{.State.ExitCode}}\" \"$(docker compose ps -q \"$id\" 2>/dev/null | head -1)\" 2>/dev/null || echo \"?\")',\n\t\t' fail \"$icon $(pad_name \"$name\") Exited (code $exit_code)\"',\n\t\t' if [ \"$exit_code\" = \"137\" ]; then',\n\t\t' ERRORS+=(\" └─ $name: OOM killed (exit 137). Increase memory limits.\")',\n\t\t' elif [ \"$exit_code\" = \"1\" ]; then',\n\t\t' ERRORS+=(\" └─ $name: Application error (exit 1). Check logs: docker compose logs $id\")',\n\t\t\" fi\",\n\t\t\" else\",\n\t\t' fail \"$icon $(pad_name \"$name\") State: $state\"',\n\t\t\" fi\",\n\t\t\"}\",\n\t\t\"\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"# Phase 3: Port Reachability\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"check_port() {\",\n\t\t' local id=\"$1\" port=\"$2\" desc=\"$3\"',\n\t\t\" if command -v nc &>/dev/null; then\",\n\t\t' if nc -z -w 2 localhost \"$port\" 2>/dev/null; then',\n\t\t' dim \" Port $port ($desc) — reachable\"',\n\t\t\" else\",\n\t\t' warn_svc \" └─ $id: Port $port ($desc) — NOT reachable\"',\n\t\t\" fi\",\n\t\t\" elif command -v curl &>/dev/null; then\",\n\t\t' if curl -sf --max-time 2 \"http://localhost:$port/\" &>/dev/null; then',\n\t\t' dim \" Port $port ($desc) — reachable\"',\n\t\t\" else\",\n\t\t' warn_svc \" └─ $id: Port $port ($desc) — NOT reachable\"',\n\t\t\" fi\",\n\t\t\" else\",\n\t\t' dim \" Port $port ($desc) — skipped (no nc or curl)\"',\n\t\t\" fi\",\n\t\t\"}\",\n\t\t\"\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"# Phase 4: Health Check Commands\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"check_health_cmd() {\",\n\t\t' local id=\"$1\" cmd=\"$2\"',\n\t\t' local cid=$(docker compose ps -q \"$id\" 2>/dev/null | head -1)',\n\t\t' [ -z \"$cid\" ] && return',\n\t\t' if docker exec \"$cid\" sh -c \"$cmd\" &>/dev/null; then',\n\t\t' dim \" Health command passed\"',\n\t\t\" else\",\n\t\t' warn_svc \" └─ $id: Health check command failed\"',\n\t\t' if [ \"$VERBOSE\" = true ]; then dim \" Command: $cmd\"; fi',\n\t\t\" fi\",\n\t\t\"}\",\n\t\t\"\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"# Phase 5: Resource Usage\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"phase_resources() {\",\n\t\t' if [ \"$JSON_MODE\" = false ]; then',\n\t\t' echo \"\"',\n\t\t' echo -e \"${BOLD}── Phase 5: Resource Usage ─────────────────────────────────${NC}\"',\n\t\t' echo \"\"',\n\t\t\" fi\",\n\t\t' docker stats --no-stream --format \"table {{.Name}}\\\\t{{.CPUPerc}}\\\\t{{.MemUsage}}\\\\t{{.MemPerc}}\" 2>/dev/null || true',\n\t\t' if [ \"$VERBOSE\" = false ] && [ \"$JSON_MODE\" = false ]; then',\n\t\t' info \"Use --verbose to see warnings for high memory usage\"',\n\t\t\" fi\",\n\t\t\"}\",\n\t\t\"\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"# Phase 6: Log Scan\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"phase_logs() {\",\n\t\t' if [ \"$QUICK_MODE\" = true ]; then',\n\t\t' if [ \"$JSON_MODE\" = false ]; then echo \"\"; info \"Skipping log scan (--quick mode)\"; fi',\n\t\t\" return\",\n\t\t\" fi\",\n\t\t' if [ \"$JSON_MODE\" = false ]; then',\n\t\t' echo \"\"',\n\t\t' echo -e \"${BOLD}── Phase 6: Log Scan ───────────────────────────────────────${NC}\"',\n\t\t' echo \"\"',\n\t\t\" fi\",\n\t\t\" local has_errors=false\",\n\t\tL(\" for svc_id in \", svcIdList, \"; do\"),\n\t\t\" local error_lines\",\n\t\t' error_lines=$(docker compose logs --tail=50 \"$svc_id\" 2>/dev/null | grep -iE \"(error|fatal|panic|exception|segfault|killed|oom)\" | tail -5 || true)',\n\t\t' if [ -n \"$error_lines\" ]; then',\n\t\t\" has_errors=true\",\n\t\t' warn_svc \"$svc_id — found error patterns in logs:\"',\n\t\t' echo \"$error_lines\" | while IFS= read -r eline; do',\n\t\t' dim \" | $eline\"',\n\t\t\" done\",\n\t\t\" fi\",\n\t\t\" done\",\n\t\t' if [ \"$has_errors\" = false ] && [ \"$JSON_MODE\" = false ]; then',\n\t\t' pass \"No error patterns found in recent logs\"',\n\t\t\" fi\",\n\t\t\"}\",\n\t\t\"\",\n\t\t\"# ── Bare-metal process check ────────────────────────────────────────\",\n\t\t\"check_process() {\",\n\t\t' local id=\"$1\" name=\"$2\" icon=\"$3\"',\n\t\t\" if command -v systemctl &>/dev/null; then\",\n\t\t' if systemctl is-active --quiet \"$id\" 2>/dev/null; then',\n\t\t' pass \"$icon $(pad_name \"$name\") Active (systemd)\"',\n\t\t\" return\",\n\t\t\" fi\",\n\t\t\" fi\",\n\t\t' if pgrep -f \"$id\" &>/dev/null; then',\n\t\t' pass \"$icon $(pad_name \"$name\") Running (process)\"',\n\t\t\" else\",\n\t\t' fail \"$icon $(pad_name \"$name\") NOT running\"',\n\t\t\" fi\",\n\t\t\"}\",\n\t\t\"\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"# Main\",\n\t\t\"# ═══════════════════════════════════════════════════════════════════════\",\n\t\t\"\",\n\t\t\"DOCKER_AVAILABLE=false\",\n\t\t\"\",\n\t\t'if [ \"$JSON_MODE\" = false ]; then',\n\t\t' echo \"\"',\n\t\t' echo \"═══════════════════════════════════════════════════════════════════════\"',\n\t\tL(' echo \" 🐾 OpenClaw Stack Health Report — ', name, '\"'),\n\t\t' echo \"═══════════════════════════════════════════════════════════════════════\"',\n\t\t\"fi\",\n\t\t\"\",\n\t\t\"phase_environment\",\n\t\t\"\",\n\t\t'if [ \"$DOCKER_AVAILABLE\" = true ]; then',\n\t\t' if [ \"$JSON_MODE\" = false ]; then',\n\t\t' echo \"\"',\n\t\t' echo -e \"${BOLD}── Phase 2–4: Service Checks ──────────────────────────────${NC}\"',\n\t\t' echo \"\"',\n\t\t\" fi\",\n\t\t\"\",\n\t\t...dockerChecks,\n\t\t\"\",\n\t\t\" phase_resources\",\n\t\t\" phase_logs\",\n\t\t\"\",\n\t\t\"else\",\n\t\t' if [ \"$JSON_MODE\" = false ]; then',\n\t\t' echo \"\"',\n\t\t' echo -e \"${BOLD}── Bare-Metal Service Checks ──────────────────────────────${NC}\"',\n\t\t' echo \"\"',\n\t\t\" fi\",\n\t\t\"\",\n\t\t...bmChecks,\n\t\t\"fi\",\n\t\t\"\",\n\t\t\"# ── Summary ──────────────────────────────────────────────────────────\",\n\t\t'echo \"\"',\n\t\t'echo \"═══════════════════════════════════════════════════════════════════════\"',\n\t\t'echo \"\"',\n\t\t'echo -e \" ${BOLD}Summary${NC}: $TOTAL services checked\"',\n\t\t'echo -e \" ${GREEN}Healthy: $HEALTHY${NC} | ${YELLOW}Warnings: $WARNING${NC} | ${RED}Failed: $FAILED${NC}\"',\n\t\t\"\",\n\t\t\"if [ ${#ERRORS[@]} -gt 0 ]; then\",\n\t\t' echo \"\"',\n\t\t' echo -e \" ${BOLD}${RED}── Errors ─────────────────────────────────────────────────${NC}\"',\n\t\t' for e in \"${ERRORS[@]}\"; do',\n\t\t' echo -e \" ${RED}$e${NC}\"',\n\t\t\" done\",\n\t\t\"fi\",\n\t\t\"\",\n\t\t'echo \"\"',\n\t\t'echo \"═══════════════════════════════════════════════════════════════════════\"',\n\t\t'echo \"\"',\n\t\t\"\",\n\t\t'if [ \"$FAILED\" -gt 0 ]; then',\n\t\t' echo -e \" ${RED}Some services need attention. Run with --verbose for details.${NC}\"',\n\t\t\" exit 1\",\n\t\t\"else\",\n\t\t' echo -e \" ${GREEN}🎉 All services are running!${NC}\"',\n\t\t\" exit 0\",\n\t\t\"fi\",\n\t];\n\n\treturn lines.join(\"\\n\") + \"\\n\";\n}\n\n// ─── PowerShell Script ──────────────────────────────────────────────────────\n\nfunction generatePowerShellScript(resolved: ResolverOutput, options: HealthCheckOptions): string {\n\tconst checks = extractServiceChecks(resolved);\n\tconst name = options.projectName;\n\tconst total = checks.length;\n\n\t// Build per-service docker check calls\n\tconst dockerChecks: string[] = [];\n\tfor (const svc of checks) {\n\t\tdockerChecks.push(L(\" # \", svc.icon, \" \", svc.name));\n\t\tdockerChecks.push(\n\t\t\tL(\n\t\t\t\t' Test-Container -ServiceId \"',\n\t\t\t\tsvc.id,\n\t\t\t\t'\" -ServiceName \"',\n\t\t\t\tsvc.name,\n\t\t\t\t'\" -Icon \"',\n\t\t\t\tsvc.icon,\n\t\t\t\t'\"',\n\t\t\t),\n\t\t);\n\t\tfor (const p of svc.ports.filter((pp) => pp.exposed)) {\n\t\t\tdockerChecks.push(\n\t\t\t\tL(\n\t\t\t\t\t' Test-Port -ServiceId \"',\n\t\t\t\t\tsvc.id,\n\t\t\t\t\t'\" -Port ',\n\t\t\t\t\tString(p.host),\n\t\t\t\t\t' -Description \"',\n\t\t\t\t\tp.description,\n\t\t\t\t\t'\"',\n\t\t\t\t),\n\t\t\t);\n\t\t}\n\t\tdockerChecks.push(\"\");\n\t}\n\n\t// Build per-service bare-metal check calls\n\tconst bmChecks: string[] = [];\n\tfor (const svc of checks) {\n\t\tbmChecks.push(L(\" # \", svc.icon, \" \", svc.name));\n\t\tbmChecks.push(\n\t\t\tL(\n\t\t\t\t' Test-ProcessRunning -ServiceId \"',\n\t\t\t\tsvc.id,\n\t\t\t\t'\" -ServiceName \"',\n\t\t\t\tsvc.name,\n\t\t\t\t'\" -Icon \"',\n\t\t\t\tsvc.icon,\n\t\t\t\t'\"',\n\t\t\t),\n\t\t);\n\t\tfor (const p of svc.ports.filter((pp) => pp.exposed)) {\n\t\t\tbmChecks.push(\n\t\t\t\tL(\n\t\t\t\t\t' Test-Port -ServiceId \"',\n\t\t\t\t\tsvc.id,\n\t\t\t\t\t'\" -Port ',\n\t\t\t\t\tString(p.host),\n\t\t\t\t\t' -Description \"',\n\t\t\t\t\tp.description,\n\t\t\t\t\t'\"',\n\t\t\t\t),\n\t\t\t);\n\t\t}\n\t\tbmChecks.push(\"\");\n\t}\n\n\tconst svcIdList = checks.map((s) => '\"' + s.id + '\"').join(\", \");\n\n\tconst lines: string[] = [\n\t\t\"#Requires -Version 5.1\",\n\t\t\"<#\",\n\t\t\".SYNOPSIS\",\n\t\tL(\" OpenClaw Stack Health Check — \", name),\n\t\t\"\",\n\t\t\".DESCRIPTION\",\n\t\t\" Auto-generated verification script for your stack.\",\n\t\tL(\n\t\t\t\" Checks \",\n\t\t\tString(total),\n\t\t\t\" services: container status, port reachability, and log errors.\",\n\t\t),\n\t\t\"\",\n\t\t\".PARAMETER Quick\",\n\t\t\" Skip log scanning for faster results.\",\n\t\t\"\",\n\t\t\".PARAMETER Json\",\n\t\t\" Output results as JSON.\",\n\t\t\"\",\n\t\t\".EXAMPLE\",\n\t\t\" .\\\\scripts\\\\health-check.ps1\",\n\t\t\" .\\\\scripts\\\\health-check.ps1 -Quick\",\n\t\t\" .\\\\scripts\\\\health-check.ps1 -Json\",\n\t\t\"#>\",\n\t\t\"param(\",\n\t\t\" [switch]$Quick,\",\n\t\t\" [switch]$Json,\",\n\t\t\" [switch]$Detailed\",\n\t\t\")\",\n\t\t\"\",\n\t\t'$ErrorActionPreference = \"Continue\"',\n\t\t\"\",\n\t\t\"# ── Counters ─────────────────────────────────────────────────────────\",\n\t\tL(\"$script:Total = \", String(total)),\n\t\t\"$script:Healthy = 0\",\n\t\t\"$script:Warnings = 0\",\n\t\t\"$script:Failed = 0\",\n\t\t\"$script:Errors = @()\",\n\t\t\"$script:ServiceResults = @()\",\n\t\t\"$script:DockerAvailable = $false\",\n\t\t\"\",\n\t\t\"# ── Helpers ──────────────────────────────────────────────────────────\",\n\t\t\"\",\n\t\t'function Write-Pass($msg) { Write-Host \" ✅ $msg\" -ForegroundColor Green; $script:Healthy++ }',\n\t\t'function Write-Warn($msg) { Write-Host \" ⚠️ $msg\" -ForegroundColor Yellow; $script:Warnings++ }',\n\t\t'function Write-Fail($msg) { Write-Host \" ❌ $msg\" -ForegroundColor Red; $script:Failed++; $script:Errors += $msg }',\n\t\t'function Write-Info($msg) { Write-Host \" ℹ $msg\" -ForegroundColor Cyan }',\n\t\t'function Write-Dim($msg) { Write-Host \" $msg\" -ForegroundColor DarkGray }',\n\t\t\"\",\n\t\t\"function Pad-Name([string]$n) { return $n.PadRight(25) }\",\n\t\t\"\",\n\t\t\"# ═════════════════════════════════════════════════════════════════════\",\n\t\t\"# Phase 1: Environment\",\n\t\t\"# ═════════════════════════════════════════════════════════════════════\",\n\t\t\"function Test-Environment {\",\n\t\t\" if (-not $Json) {\",\n\t\t' Write-Host \"\"',\n\t\t' Write-Host \"── Phase 1: Environment ────────────────────────────────────\" -ForegroundColor White',\n\t\t' Write-Host \"\"',\n\t\t\" }\",\n\t\t\" try {\",\n\t\t\" $null = docker info 2>&1\",\n\t\t\" if ($LASTEXITCODE -eq 0) {\",\n\t\t' Write-Pass \"Docker daemon is running\"',\n\t\t\" $script:DockerAvailable = $true\",\n\t\t\" } else {\",\n\t\t' Write-Fail \"Docker daemon is NOT running\"',\n\t\t\" }\",\n\t\t\" } catch {\",\n\t\t' Write-Warn \"Docker is not installed — running bare-metal checks only\"',\n\t\t\" }\",\n\t\t\" if ($script:DockerAvailable) {\",\n\t\t\" try {\",\n\t\t\" $null = docker compose version 2>&1\",\n\t\t\" if ($LASTEXITCODE -eq 0) {\",\n\t\t' Write-Pass \"Docker Compose v2 available\"',\n\t\t\" } else {\",\n\t\t' Write-Warn \"Docker Compose v2 not found\"',\n\t\t\" }\",\n\t\t\" } catch {\",\n\t\t' Write-Warn \"Docker Compose not available\"',\n\t\t\" }\",\n\t\t\" }\",\n\t\t' $envPath = Join-Path $PSScriptRoot \"..\\\\.env\"',\n\t\t\" if (Test-Path $envPath) {\",\n\t\t' Write-Pass \".env file exists\"',\n\t\t\" $emptySecrets = 0\",\n\t\t\" Get-Content $envPath | ForEach-Object {\",\n\t\t' if ($_ -match \"(PASSWORD|TOKEN|SECRET|SESSION_KEY|COOKIE|API_KEY).*=\\\\s*$\") { $emptySecrets++ }',\n\t\t\" }\",\n\t\t\" if ($emptySecrets -gt 0) {\",\n\t\t' Write-Warn \"$emptySecrets secret(s) are empty in .env\"',\n\t\t\" } else {\",\n\t\t' Write-Pass \"All secrets are populated\"',\n\t\t\" }\",\n\t\t\" } else {\",\n\t\t' Write-Warn \".env file not found\"',\n\t\t\" }\",\n\t\t\" $drive = (Get-Item $PSScriptRoot).PSDrive\",\n\t\t\" $freeGB = [math]::Round($drive.Free / 1GB, 1)\",\n\t\t\" if ($freeGB -lt 2) {\",\n\t\t' Write-Warn \"Low disk space: ${freeGB}GB available\"',\n\t\t\" } else {\",\n\t\t' Write-Pass \"Disk space: ${freeGB}GB available\"',\n\t\t\" }\",\n\t\t\"}\",\n\t\t\"\",\n\t\t\"# ═════════════════════════════════════════════════════════════════════\",\n\t\t\"# Phase 2: Container Check\",\n\t\t\"# ═════════════════════════════════════════════════════════════════════\",\n\t\t\"function Test-Container {\",\n\t\t\" param([string]$ServiceId, [string]$ServiceName, [string]$Icon)\",\n\t\t\" try {\",\n\t\t\" $status = docker compose ps $ServiceId --format json 2>&1 | ConvertFrom-Json\",\n\t\t\" } catch {\",\n\t\t' Write-Fail \"$Icon $(Pad-Name $ServiceName) Container not found\"',\n\t\t\" return\",\n\t\t\" }\",\n\t\t\" if (-not $status) {\",\n\t\t' Write-Fail \"$Icon $(Pad-Name $ServiceName) Container not found\"',\n\t\t\" return\",\n\t\t\" }\",\n\t\t\" $state = $status.State\",\n\t\t\" $health = $status.Health\",\n\t\t' if ($state -eq \"running\") {',\n\t\t' if ($health -eq \"healthy\") {',\n\t\t' Write-Pass \"$Icon $(Pad-Name $ServiceName) Running (healthy)\"',\n\t\t' } elseif ($health -eq \"starting\") {',\n\t\t' Write-Warn \"$Icon $(Pad-Name $ServiceName) Running (starting)\"',\n\t\t' } elseif ($health -eq \"unhealthy\") {',\n\t\t' Write-Fail \"$Icon $(Pad-Name $ServiceName) Running (UNHEALTHY)\"',\n\t\t\" } else {\",\n\t\t' Write-Pass \"$Icon $(Pad-Name $ServiceName) Running\"',\n\t\t\" }\",\n\t\t' } elseif ($state -eq \"exited\") {',\n\t\t' Write-Fail \"$Icon $(Pad-Name $ServiceName) Exited\"',\n\t\t\" } else {\",\n\t\t' Write-Fail \"$Icon $(Pad-Name $ServiceName) State: $state\"',\n\t\t\" }\",\n\t\t\"}\",\n\t\t\"\",\n\t\t\"# ═════════════════════════════════════════════════════════════════════\",\n\t\t\"# Phase 3: Port Check\",\n\t\t\"# ═════════════════════════════════════════════════════════════════════\",\n\t\t\"function Test-Port {\",\n\t\t\" param([string]$ServiceId, [int]$Port, [string]$Description)\",\n\t\t\" try {\",\n\t\t\" $tcp = New-Object System.Net.Sockets.TcpClient\",\n\t\t' $tcp.ConnectAsync(\"localhost\", $Port).Wait(2000) | Out-Null',\n\t\t\" if ($tcp.Connected) {\",\n\t\t' Write-Dim \" Port $Port ($Description) — reachable\"',\n\t\t\" $tcp.Close()\",\n\t\t\" } else {\",\n\t\t' Write-Warn \" └─ ${ServiceId}: Port $Port ($Description) — NOT reachable\"',\n\t\t\" }\",\n\t\t\" } catch {\",\n\t\t' Write-Warn \" └─ ${ServiceId}: Port $Port ($Description) — NOT reachable\"',\n\t\t\" }\",\n\t\t\"}\",\n\t\t\"\",\n\t\t\"# ── Bare-metal process check ────────────────────────────────────────\",\n\t\t\"function Test-ProcessRunning {\",\n\t\t\" param([string]$ServiceId, [string]$ServiceName, [string]$Icon)\",\n\t\t\" $svc = Get-Service -Name $ServiceId -ErrorAction SilentlyContinue\",\n\t\t\" if ($svc) {\",\n\t\t' if ($svc.Status -eq \"Running\") {',\n\t\t' Write-Pass \"$Icon $(Pad-Name $ServiceName) Active (service)\"',\n\t\t\" } else {\",\n\t\t' Write-Fail \"$Icon $(Pad-Name $ServiceName) $($svc.Status)\"',\n\t\t\" }\",\n\t\t\" return\",\n\t\t\" }\",\n\t\t\" $proc = Get-Process -Name $ServiceId -ErrorAction SilentlyContinue\",\n\t\t\" if ($proc) {\",\n\t\t' Write-Pass \"$Icon $(Pad-Name $ServiceName) Running (process)\"',\n\t\t\" } else {\",\n\t\t' Write-Fail \"$Icon $(Pad-Name $ServiceName) NOT running\"',\n\t\t\" }\",\n\t\t\"}\",\n\t\t\"\",\n\t\t\"# ═════════════════════════════════════════════════════════════════════\",\n\t\t\"# Phase 6: Log Scan\",\n\t\t\"# ═════════════════════════════════════════════════════════════════════\",\n\t\t\"function Test-Logs {\",\n\t\t\" if ($Quick) {\",\n\t\t' if (-not $Json) { Write-Info \"Skipping log scan (-Quick mode)\" }',\n\t\t\" return\",\n\t\t\" }\",\n\t\t\" if (-not $Json) {\",\n\t\t' Write-Host \"\"',\n\t\t' Write-Host \"── Phase 6: Log Scan ───────────────────────────────────────\" -ForegroundColor White',\n\t\t' Write-Host \"\"',\n\t\t\" }\",\n\t\t\" $hasErrors = $false\",\n\t\tL(\" $svcIds = @(\", svcIdList, \")\"),\n\t\t\" foreach ($svcId in $svcIds) {\",\n\t\t\" try {\",\n\t\t\" $logs = docker compose logs --tail=50 $svcId 2>&1 | Out-String\",\n\t\t' $errorLines = $logs -split \"`n\" | Where-Object { $_ -match \"(error|fatal|panic|exception|segfault|killed|oom)\" } | Select-Object -Last 5',\n\t\t\" if ($errorLines) {\",\n\t\t\" $hasErrors = $true\",\n\t\t' Write-Warn \"$svcId — found error patterns in logs:\"',\n\t\t' $errorLines | ForEach-Object { Write-Dim \" | $_\" }',\n\t\t\" }\",\n\t\t\" } catch {}\",\n\t\t\" }\",\n\t\t\" if (-not $hasErrors -and -not $Json) {\",\n\t\t' Write-Pass \"No error patterns found in recent logs\"',\n\t\t\" }\",\n\t\t\"}\",\n\t\t\"\",\n\t\t\"# ═════════════════════════════════════════════════════════════════════\",\n\t\t\"# Main\",\n\t\t\"# ═════════════════════════════════════════════════════════════════════\",\n\t\t\"\",\n\t\t\"if (-not $Json) {\",\n\t\t' Write-Host \"\"',\n\t\t' Write-Host \"═══════════════════════════════════════════════════════════════════════\" ',\n\t\tL(' Write-Host \" 🐾 OpenClaw Stack Health Report — ', name, '\"'),\n\t\t' Write-Host \"═══════════════════════════════════════════════════════════════════════\"',\n\t\t\"}\",\n\t\t\"\",\n\t\t\"Test-Environment\",\n\t\t\"\",\n\t\t\"if ($script:DockerAvailable) {\",\n\t\t\" if (-not $Json) {\",\n\t\t' Write-Host \"\"',\n\t\t' Write-Host \"── Phase 2–4: Service Checks ──────────────────────────────\" -ForegroundColor White',\n\t\t' Write-Host \"\"',\n\t\t\" }\",\n\t\t\"\",\n\t\t...dockerChecks,\n\t\t\" Test-Logs\",\n\t\t\"} else {\",\n\t\t\" if (-not $Json) {\",\n\t\t' Write-Host \"\"',\n\t\t' Write-Host \"── Bare-Metal Service Checks ──────────────────────────────\" -ForegroundColor White',\n\t\t' Write-Host \"\"',\n\t\t\" }\",\n\t\t\"\",\n\t\t...bmChecks,\n\t\t\"}\",\n\t\t\"\",\n\t\t\"# ── Summary ──────────────────────────────────────────────────────────\",\n\t\t\"if ($Json) {\",\n\t\t\" $result = @{\",\n\t\tL(' project = \"', name, '\"'),\n\t\t' timestamp = (Get-Date -Format \"o\")',\n\t\t\" summary = @{\",\n\t\t\" total = $script:Total\",\n\t\t\" healthy = $script:Healthy\",\n\t\t\" warnings = $script:Warnings\",\n\t\t\" failed = $script:Failed\",\n\t\t\" }\",\n\t\t\" errors = $script:Errors\",\n\t\t\" }\",\n\t\t\" $result | ConvertTo-Json -Depth 5\",\n\t\t\"} else {\",\n\t\t' Write-Host \"\"',\n\t\t' Write-Host \"═══════════════════════════════════════════════════════════════════════\"',\n\t\t' Write-Host \"\"',\n\t\t' Write-Host \" Summary: $($script:Total) services checked\"',\n\t\t' Write-Host \" Healthy: $($script:Healthy)\" -ForegroundColor Green -NoNewline',\n\t\t' Write-Host \" | Warnings: $($script:Warnings)\" -ForegroundColor Yellow -NoNewline',\n\t\t' Write-Host \" | Failed: $($script:Failed)\" -ForegroundColor Red',\n\t\t\" if ($script:Errors.Count -gt 0) {\",\n\t\t' Write-Host \"\"',\n\t\t' Write-Host \" ── Errors ──────────────────────────────────────────────────\" -ForegroundColor Red',\n\t\t' $script:Errors | ForEach-Object { Write-Host \" $_\" -ForegroundColor Red }',\n\t\t\" }\",\n\t\t' Write-Host \"\"',\n\t\t' Write-Host \"═══════════════════════════════════════════════════════════════════════\"',\n\t\t' Write-Host \"\"',\n\t\t\" if ($script:Failed -gt 0) {\",\n\t\t' Write-Host \" Some services need attention. Run with -Detailed for more.\" -ForegroundColor Red',\n\t\t\" exit 1\",\n\t\t\" } else {\",\n\t\t' Write-Host \" 🎉 All services are running!\" -ForegroundColor Green',\n\t\t\" exit 0\",\n\t\t\" }\",\n\t\t\"}\",\n\t];\n\n\treturn lines.join(\"\\n\") + \"\\n\";\n}\n"],"mappings":";;;;;;;;AAkBA,SAAgB,oBACf,UACA,SACyB;CACzB,MAAM,QAAgC,EAAE;AAExC,OAAM,6BAA6B,mBAAmB,UAAU,QAAQ;AACxE,OAAM,8BAA8B,yBAAyB,UAAU,QAAQ;AAE/E,QAAO;;AAaR,SAAS,qBAAqB,UAA0C;AACvE,QAAO,SAAS,SAAS,KAAK,SAAS;EACtC,IAAI,IAAI,WAAW;EACnB,MAAM,IAAI,WAAW;EACrB,MAAM,IAAI,WAAW;EACrB,OAAO,IAAI,WAAW,MAAM,KAAK,OAAO;GACvC,MAAM,EAAE;GACR,WAAW,EAAE;GACb,aAAa,EAAE;GACf,SAAS,EAAE;GACX,EAAE;EACH,gBAAgB,IAAI,WAAW,aAAa,QAAQ;EACpD,EAAE;;AAGJ,SAAS,YAAY,GAAmB;AACvC,QAAO,EAAE,QAAQ,MAAM,QAAQ,CAAC,QAAQ,MAAM,OAAM;;AAIrD,SAAS,EAAE,GAAG,OAAyB;AACtC,QAAO,MAAM,KAAK,GAAG;;AAKtB,SAAS,mBAAmB,UAA0B,SAAqC;CAC1F,MAAM,SAAS,qBAAqB,SAAS;CAC7C,MAAM,OAAO,QAAQ;CACrB,MAAM,QAAQ,OAAO;CAGrB,MAAM,eAAyB,EAAE;AACjC,MAAK,MAAM,OAAO,QAAQ;AACzB,eAAa,KAAK,EAAE,WAAW,IAAI,MAAM,KAAK,IAAI,MAAM,MAAM,CAAC;AAC/D,eAAa,KAAK,EAAE,wBAAuB,IAAI,IAAI,SAAO,IAAI,MAAM,SAAO,IAAI,MAAM,KAAI,CAAC;AAC1F,OAAK,MAAM,KAAK,IAAI,MAAM,QAAQ,OAAO,GAAG,QAAQ,CACnD,cAAa,KACZ,EAAE,mBAAkB,IAAI,IAAI,OAAM,OAAO,EAAE,KAAK,EAAE,OAAM,EAAE,aAAa,KAAI,CAC3E;AAEF,MAAI,IAAI,eACP,cAAa,KACZ,EAAE,yBAAwB,IAAI,IAAI,SAAO,YAAY,IAAI,eAAe,EAAE,KAAI,CAC9E;MAED,cAAa,KAAK,mDAAmD;AAEtE,eAAa,KAAK,GAAG;;CAItB,MAAM,WAAqB,EAAE;AAC7B,MAAK,MAAM,OAAO,QAAQ;AACzB,WAAS,KAAK,EAAE,WAAW,IAAI,MAAM,KAAK,IAAI,MAAM,mBAAmB,CAAC;AACxE,WAAS,KAAK,EAAE,sBAAqB,IAAI,IAAI,SAAO,IAAI,MAAM,SAAO,IAAI,MAAM,KAAI,CAAC;AACpF,OAAK,MAAM,KAAK,IAAI,MAAM,QAAQ,OAAO,GAAG,QAAQ,CACnD,UAAS,KAAK,EAAE,mBAAkB,IAAI,IAAI,OAAM,OAAO,EAAE,KAAK,EAAE,OAAM,EAAE,aAAa,KAAI,CAAC;AAE3F,WAAS,KAAK,GAAG;;CAIlB,MAAM,YAAY,OAAO,KAAK,MAAM,OAAM,EAAE,KAAK,KAAI,CAAC,KAAK,IAAI;AA6V/D,QA3VwB;EACvB;EACA;EACA;EACA;EACA,EAAE,uCAAuC,KAAK;EAC9C;EACA;EACA;EACA,EAAE,aAAa,OAAO,MAAM,EAAE,6BAA6B;EAC3D;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA,EAAE,UAAU,OAAO,MAAM,CAAC;EAC1B;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA,EAAE,oBAAoB,WAAW,OAAO;EACxC;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA,EAAE,gDAA+C,MAAM,KAAI;EAC3D;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA,GAAG;EACH;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA,GAAG;EACH;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA,CAEY,KAAK,KAAK,GAAG;;AAK3B,SAAS,yBAAyB,UAA0B,SAAqC;CAChG,MAAM,SAAS,qBAAqB,SAAS;CAC7C,MAAM,OAAO,QAAQ;CACrB,MAAM,QAAQ,OAAO;CAGrB,MAAM,eAAyB,EAAE;AACjC,MAAK,MAAM,OAAO,QAAQ;AACzB,eAAa,KAAK,EAAE,UAAU,IAAI,MAAM,KAAK,IAAI,KAAK,CAAC;AACvD,eAAa,KACZ,EACC,oCACA,IAAI,IACJ,sBACA,IAAI,MACJ,eACA,IAAI,MACJ,KACA,CACD;AACD,OAAK,MAAM,KAAK,IAAI,MAAM,QAAQ,OAAO,GAAG,QAAQ,CACnD,cAAa,KACZ,EACC,+BACA,IAAI,IACJ,aACA,OAAO,EAAE,KAAK,EACd,oBACA,EAAE,aACF,KACA,CACD;AAEF,eAAa,KAAK,GAAG;;CAItB,MAAM,WAAqB,EAAE;AAC7B,MAAK,MAAM,OAAO,QAAQ;AACzB,WAAS,KAAK,EAAE,UAAU,IAAI,MAAM,KAAK,IAAI,KAAK,CAAC;AACnD,WAAS,KACR,EACC,yCACA,IAAI,IACJ,sBACA,IAAI,MACJ,eACA,IAAI,MACJ,KACA,CACD;AACD,OAAK,MAAM,KAAK,IAAI,MAAM,QAAQ,OAAO,GAAG,QAAQ,CACnD,UAAS,KACR,EACC,+BACA,IAAI,IACJ,aACA,OAAO,EAAE,KAAK,EACd,oBACA,EAAE,aACF,KACA,CACD;AAEF,WAAS,KAAK,GAAG;;CAGlB,MAAM,YAAY,OAAO,KAAK,MAAM,OAAM,EAAE,KAAK,KAAI,CAAC,KAAK,KAAK;AA8RhE,QA5RwB;EACvB;EACA;EACA;EACA,EAAE,sCAAsC,KAAK;EAC7C;EACA;EACA;EACA,EACC,eACA,OAAO,MAAM,EACb,kEACA;EACD;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA,EAAE,oBAAoB,OAAO,MAAM,CAAC;EACpC;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA,EAAE,oBAAoB,WAAW,IAAI;EACrC;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA,EAAE,wDAAuD,MAAM,KAAI;EACnE;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA,GAAG;EACH;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA,GAAG;EACH;EACA;EACA;EACA;EACA;EACA,EAAE,wBAAuB,MAAM,KAAI;EACnC;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA,CAEY,KAAK,KAAK,GAAG"}
|
|
@@ -0,0 +1,84 @@
|
|
|
1
|
+
const require_vi_2VT5v0um = require('../vi.2VT5v0um-BmRMvymT.cjs');
|
|
2
|
+
const require_generate = require('../generate.cjs');
|
|
3
|
+
|
|
4
|
+
//#region src/generators/health-check.test.ts
|
|
5
|
+
require_vi_2VT5v0um.describe("generateHealthCheck (via generate)", () => {
|
|
6
|
+
const baseInput = {
|
|
7
|
+
projectName: "health-test",
|
|
8
|
+
services: ["redis", "postgresql"],
|
|
9
|
+
skillPacks: [],
|
|
10
|
+
proxy: "none",
|
|
11
|
+
gpu: false,
|
|
12
|
+
platform: "linux/amd64",
|
|
13
|
+
deployment: "local",
|
|
14
|
+
generateSecrets: true,
|
|
15
|
+
openclawVersion: "latest"
|
|
16
|
+
};
|
|
17
|
+
require_vi_2VT5v0um.it("generates health-check.sh and health-check.ps1", () => {
|
|
18
|
+
const result = require_generate.generate(baseInput);
|
|
19
|
+
require_vi_2VT5v0um.globalExpect(result.files).toHaveProperty("scripts/health-check.sh");
|
|
20
|
+
require_vi_2VT5v0um.globalExpect(result.files).toHaveProperty("scripts/health-check.ps1");
|
|
21
|
+
});
|
|
22
|
+
require_vi_2VT5v0um.it("health-check.sh contains project name", () => {
|
|
23
|
+
const sh = require_generate.generate(baseInput).files["scripts/health-check.sh"];
|
|
24
|
+
require_vi_2VT5v0um.globalExpect(sh).toContain("health-test");
|
|
25
|
+
});
|
|
26
|
+
require_vi_2VT5v0um.it("health-check.sh contains service-specific checks for each resolved service", () => {
|
|
27
|
+
const sh = require_generate.generate(baseInput).files["scripts/health-check.sh"];
|
|
28
|
+
require_vi_2VT5v0um.globalExpect(sh).toContain("redis");
|
|
29
|
+
require_vi_2VT5v0um.globalExpect(sh).toContain("postgresql");
|
|
30
|
+
});
|
|
31
|
+
require_vi_2VT5v0um.it("health-check.sh includes port checks for exposed ports", () => {
|
|
32
|
+
const sh = require_generate.generate(baseInput).files["scripts/health-check.sh"];
|
|
33
|
+
require_vi_2VT5v0um.globalExpect(sh).toContain("6379");
|
|
34
|
+
require_vi_2VT5v0um.globalExpect(sh).toContain("5432");
|
|
35
|
+
});
|
|
36
|
+
require_vi_2VT5v0um.it("health-check.sh has usage/help docs", () => {
|
|
37
|
+
const sh = require_generate.generate(baseInput).files["scripts/health-check.sh"];
|
|
38
|
+
require_vi_2VT5v0um.globalExpect(sh).toContain("--quick");
|
|
39
|
+
require_vi_2VT5v0um.globalExpect(sh).toContain("--json");
|
|
40
|
+
require_vi_2VT5v0um.globalExpect(sh).toContain("--verbose");
|
|
41
|
+
});
|
|
42
|
+
require_vi_2VT5v0um.it("health-check.sh starts with shebang", () => {
|
|
43
|
+
const sh = require_generate.generate(baseInput).files["scripts/health-check.sh"];
|
|
44
|
+
require_vi_2VT5v0um.globalExpect(sh.startsWith("#!/usr/bin/env bash")).toBe(true);
|
|
45
|
+
});
|
|
46
|
+
require_vi_2VT5v0um.it("health-check.ps1 contains project name", () => {
|
|
47
|
+
const ps1 = require_generate.generate(baseInput).files["scripts/health-check.ps1"];
|
|
48
|
+
require_vi_2VT5v0um.globalExpect(ps1).toContain("health-test");
|
|
49
|
+
});
|
|
50
|
+
require_vi_2VT5v0um.it("health-check.ps1 contains service checks", () => {
|
|
51
|
+
const ps1 = require_generate.generate(baseInput).files["scripts/health-check.ps1"];
|
|
52
|
+
require_vi_2VT5v0um.globalExpect(ps1).toContain("redis");
|
|
53
|
+
require_vi_2VT5v0um.globalExpect(ps1).toContain("postgresql");
|
|
54
|
+
});
|
|
55
|
+
require_vi_2VT5v0um.it("health-check.ps1 includes PowerShell-style parameters", () => {
|
|
56
|
+
const ps1 = require_generate.generate(baseInput).files["scripts/health-check.ps1"];
|
|
57
|
+
require_vi_2VT5v0um.globalExpect(ps1).toContain("[switch]$Quick");
|
|
58
|
+
require_vi_2VT5v0um.globalExpect(ps1).toContain("[switch]$Json");
|
|
59
|
+
});
|
|
60
|
+
require_vi_2VT5v0um.it("health check scripts include all resolved services including dependencies", () => {
|
|
61
|
+
const sh = require_generate.generate({
|
|
62
|
+
...baseInput,
|
|
63
|
+
services: ["postiz"]
|
|
64
|
+
}).files["scripts/health-check.sh"];
|
|
65
|
+
require_vi_2VT5v0um.globalExpect(sh).toContain("postiz");
|
|
66
|
+
require_vi_2VT5v0um.globalExpect(sh).toContain("redis");
|
|
67
|
+
require_vi_2VT5v0um.globalExpect(sh).toContain("postgresql");
|
|
68
|
+
});
|
|
69
|
+
require_vi_2VT5v0um.it("health check script includes health check commands when service defines one", () => {
|
|
70
|
+
const sh = require_generate.generate(baseInput).files["scripts/health-check.sh"];
|
|
71
|
+
require_vi_2VT5v0um.globalExpect(sh).toContain("check_health_cmd");
|
|
72
|
+
});
|
|
73
|
+
require_vi_2VT5v0um.it("health check scripts contain the phase structure", () => {
|
|
74
|
+
const sh = require_generate.generate(baseInput).files["scripts/health-check.sh"];
|
|
75
|
+
require_vi_2VT5v0um.globalExpect(sh).toContain("Phase 1");
|
|
76
|
+
require_vi_2VT5v0um.globalExpect(sh).toContain("phase_environment");
|
|
77
|
+
require_vi_2VT5v0um.globalExpect(sh).toContain("check_container");
|
|
78
|
+
require_vi_2VT5v0um.globalExpect(sh).toContain("check_port");
|
|
79
|
+
require_vi_2VT5v0um.globalExpect(sh).toContain("phase_logs");
|
|
80
|
+
});
|
|
81
|
+
});
|
|
82
|
+
|
|
83
|
+
//#endregion
|
|
84
|
+
//# sourceMappingURL=health-check.test.cjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"health-check.test.cjs","names":["describe","generate"],"sources":["../../src/generators/health-check.test.ts"],"sourcesContent":["import { describe, expect, it } from \"vitest\";\nimport { generate } from \"../generate.js\";\n\ndescribe(\"generateHealthCheck (via generate)\", () => {\n\tconst baseInput = {\n\t\tprojectName: \"health-test\",\n\t\tservices: [\"redis\", \"postgresql\"],\n\t\tskillPacks: [] as string[],\n\t\tproxy: \"none\" as const,\n\t\tgpu: false,\n\t\tplatform: \"linux/amd64\" as const,\n\t\tdeployment: \"local\" as const,\n\t\tgenerateSecrets: true,\n\t\topenclawVersion: \"latest\",\n\t};\n\n\tit(\"generates health-check.sh and health-check.ps1\", () => {\n\t\tconst result = generate(baseInput);\n\n\t\texpect(result.files).toHaveProperty(\"scripts/health-check.sh\");\n\t\texpect(result.files).toHaveProperty(\"scripts/health-check.ps1\");\n\t});\n\n\tit(\"health-check.sh contains project name\", () => {\n\t\tconst result = generate(baseInput);\n\t\tconst sh = result.files[\"scripts/health-check.sh\"]!;\n\n\t\texpect(sh).toContain(\"health-test\");\n\t});\n\n\tit(\"health-check.sh contains service-specific checks for each resolved service\", () => {\n\t\tconst result = generate(baseInput);\n\t\tconst sh = result.files[\"scripts/health-check.sh\"]!;\n\n\t\texpect(sh).toContain(\"redis\");\n\t\texpect(sh).toContain(\"postgresql\");\n\t});\n\n\tit(\"health-check.sh includes port checks for exposed ports\", () => {\n\t\tconst result = generate(baseInput);\n\t\tconst sh = result.files[\"scripts/health-check.sh\"]!;\n\n\t\t// Redis exposes port 6379, PostgreSQL exposes 5432\n\t\texpect(sh).toContain(\"6379\");\n\t\texpect(sh).toContain(\"5432\");\n\t});\n\n\tit(\"health-check.sh has usage/help docs\", () => {\n\t\tconst result = generate(baseInput);\n\t\tconst sh = result.files[\"scripts/health-check.sh\"]!;\n\n\t\texpect(sh).toContain(\"--quick\");\n\t\texpect(sh).toContain(\"--json\");\n\t\texpect(sh).toContain(\"--verbose\");\n\t});\n\n\tit(\"health-check.sh starts with shebang\", () => {\n\t\tconst result = generate(baseInput);\n\t\tconst sh = result.files[\"scripts/health-check.sh\"]!;\n\n\t\texpect(sh.startsWith(\"#!/usr/bin/env bash\")).toBe(true);\n\t});\n\n\tit(\"health-check.ps1 contains project name\", () => {\n\t\tconst result = generate(baseInput);\n\t\tconst ps1 = result.files[\"scripts/health-check.ps1\"]!;\n\n\t\texpect(ps1).toContain(\"health-test\");\n\t});\n\n\tit(\"health-check.ps1 contains service checks\", () => {\n\t\tconst result = generate(baseInput);\n\t\tconst ps1 = result.files[\"scripts/health-check.ps1\"]!;\n\n\t\texpect(ps1).toContain(\"redis\");\n\t\texpect(ps1).toContain(\"postgresql\");\n\t});\n\n\tit(\"health-check.ps1 includes PowerShell-style parameters\", () => {\n\t\tconst result = generate(baseInput);\n\t\tconst ps1 = result.files[\"scripts/health-check.ps1\"]!;\n\n\t\texpect(ps1).toContain(\"[switch]$Quick\");\n\t\texpect(ps1).toContain(\"[switch]$Json\");\n\t});\n\n\tit(\"health check scripts include all resolved services including dependencies\", () => {\n\t\tconst result = generate({\n\t\t\t...baseInput,\n\t\t\tservices: [\"postiz\"], // postiz depends on redis and postgresql\n\t\t});\n\t\tconst sh = result.files[\"scripts/health-check.sh\"]!;\n\n\t\texpect(sh).toContain(\"postiz\");\n\t\t// Dependencies should be auto-resolved and included\n\t\texpect(sh).toContain(\"redis\");\n\t\texpect(sh).toContain(\"postgresql\");\n\t});\n\n\tit(\"health check script includes health check commands when service defines one\", () => {\n\t\tconst result = generate(baseInput);\n\t\tconst sh = result.files[\"scripts/health-check.sh\"]!;\n\n\t\t// Redis has a healthcheck command (redis-cli ping)\n\t\texpect(sh).toContain(\"check_health_cmd\");\n\t});\n\n\tit(\"health check scripts contain the phase structure\", () => {\n\t\tconst result = generate(baseInput);\n\t\tconst sh = result.files[\"scripts/health-check.sh\"]!;\n\n\t\texpect(sh).toContain(\"Phase 1\");\n\t\texpect(sh).toContain(\"phase_environment\");\n\t\texpect(sh).toContain(\"check_container\");\n\t\texpect(sh).toContain(\"check_port\");\n\t\texpect(sh).toContain(\"phase_logs\");\n\t});\n});\n"],"mappings":";;;;AAGAA,6BAAS,4CAA4C;CACpD,MAAM,YAAY;EACjB,aAAa;EACb,UAAU,CAAC,SAAS,aAAa;EACjC,YAAY,EAAE;EACd,OAAO;EACP,KAAK;EACL,UAAU;EACV,YAAY;EACZ,iBAAiB;EACjB,iBAAiB;EACjB;AAED,wBAAG,wDAAwD;EAC1D,MAAM,SAASC,0BAAS,UAAU;AAElC,mCAAO,OAAO,MAAM,CAAC,eAAe,0BAA0B;AAC9D,mCAAO,OAAO,MAAM,CAAC,eAAe,2BAA2B;GAC9D;AAEF,wBAAG,+CAA+C;EAEjD,MAAM,KADSA,0BAAS,UAAU,CAChB,MAAM;AAExB,mCAAO,GAAG,CAAC,UAAU,cAAc;GAClC;AAEF,wBAAG,oFAAoF;EAEtF,MAAM,KADSA,0BAAS,UAAU,CAChB,MAAM;AAExB,mCAAO,GAAG,CAAC,UAAU,QAAQ;AAC7B,mCAAO,GAAG,CAAC,UAAU,aAAa;GACjC;AAEF,wBAAG,gEAAgE;EAElE,MAAM,KADSA,0BAAS,UAAU,CAChB,MAAM;AAGxB,mCAAO,GAAG,CAAC,UAAU,OAAO;AAC5B,mCAAO,GAAG,CAAC,UAAU,OAAO;GAC3B;AAEF,wBAAG,6CAA6C;EAE/C,MAAM,KADSA,0BAAS,UAAU,CAChB,MAAM;AAExB,mCAAO,GAAG,CAAC,UAAU,UAAU;AAC/B,mCAAO,GAAG,CAAC,UAAU,SAAS;AAC9B,mCAAO,GAAG,CAAC,UAAU,YAAY;GAChC;AAEF,wBAAG,6CAA6C;EAE/C,MAAM,KADSA,0BAAS,UAAU,CAChB,MAAM;AAExB,mCAAO,GAAG,WAAW,sBAAsB,CAAC,CAAC,KAAK,KAAK;GACtD;AAEF,wBAAG,gDAAgD;EAElD,MAAM,MADSA,0BAAS,UAAU,CACf,MAAM;AAEzB,mCAAO,IAAI,CAAC,UAAU,cAAc;GACnC;AAEF,wBAAG,kDAAkD;EAEpD,MAAM,MADSA,0BAAS,UAAU,CACf,MAAM;AAEzB,mCAAO,IAAI,CAAC,UAAU,QAAQ;AAC9B,mCAAO,IAAI,CAAC,UAAU,aAAa;GAClC;AAEF,wBAAG,+DAA+D;EAEjE,MAAM,MADSA,0BAAS,UAAU,CACf,MAAM;AAEzB,mCAAO,IAAI,CAAC,UAAU,iBAAiB;AACvC,mCAAO,IAAI,CAAC,UAAU,gBAAgB;GACrC;AAEF,wBAAG,mFAAmF;EAKrF,MAAM,KAJSA,0BAAS;GACvB,GAAG;GACH,UAAU,CAAC,SAAS;GACpB,CAAC,CACgB,MAAM;AAExB,mCAAO,GAAG,CAAC,UAAU,SAAS;AAE9B,mCAAO,GAAG,CAAC,UAAU,QAAQ;AAC7B,mCAAO,GAAG,CAAC,UAAU,aAAa;GACjC;AAEF,wBAAG,qFAAqF;EAEvF,MAAM,KADSA,0BAAS,UAAU,CAChB,MAAM;AAGxB,mCAAO,GAAG,CAAC,UAAU,mBAAmB;GACvC;AAEF,wBAAG,0DAA0D;EAE5D,MAAM,KADSA,0BAAS,UAAU,CAChB,MAAM;AAExB,mCAAO,GAAG,CAAC,UAAU,UAAU;AAC/B,mCAAO,GAAG,CAAC,UAAU,oBAAoB;AACzC,mCAAO,GAAG,CAAC,UAAU,kBAAkB;AACvC,mCAAO,GAAG,CAAC,UAAU,aAAa;AAClC,mCAAO,GAAG,CAAC,UAAU,aAAa;GACjC;EACD"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export { };
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { n as describe, r as it, t as globalExpect } from "../vi.2VT5v0um-
|
|
1
|
+
import { n as describe, r as it, t as globalExpect } from "../vi.2VT5v0um-CFyDIn0m.mjs";
|
|
2
2
|
import { generate } from "../generate.mjs";
|
|
3
3
|
|
|
4
4
|
//#region src/generators/health-check.test.ts
|