@better-openclaw/core 1.0.23 → 1.0.25
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/dist/addon-stack.cjs +673 -0
- package/dist/addon-stack.cjs.map +1 -0
- package/dist/addon-stack.d.cts +23 -0
- package/dist/addon-stack.d.cts.map +1 -0
- package/dist/addon-stack.d.mts +23 -0
- package/dist/addon-stack.d.mts.map +1 -0
- package/dist/addon-stack.mjs +671 -0
- package/dist/addon-stack.mjs.map +1 -0
- package/dist/addon-stack.test.cjs +349 -0
- package/dist/addon-stack.test.cjs.map +1 -0
- package/dist/addon-stack.test.d.cts +1 -0
- package/dist/addon-stack.test.d.mts +1 -0
- package/dist/addon-stack.test.mjs +349 -0
- package/dist/addon-stack.test.mjs.map +1 -0
- package/dist/bare-metal-partition.test.cjs +20 -21
- package/dist/bare-metal-partition.test.cjs.map +1 -1
- package/dist/bare-metal-partition.test.mjs +4 -5
- package/dist/bare-metal-partition.test.mjs.map +1 -1
- package/dist/composer.cjs +17 -1
- package/dist/composer.cjs.map +1 -1
- package/dist/composer.d.cts +24 -1
- package/dist/composer.d.cts.map +1 -1
- package/dist/composer.d.mts +24 -1
- package/dist/composer.d.mts.map +1 -1
- package/dist/composer.mjs +14 -2
- package/dist/composer.mjs.map +1 -1
- package/dist/composer.snapshot.test.cjs +20 -20
- package/dist/composer.snapshot.test.cjs.map +1 -1
- package/dist/composer.snapshot.test.mjs +2 -2
- package/dist/composer.test.cjs +53 -52
- package/dist/composer.test.cjs.map +1 -1
- package/dist/composer.test.mjs +4 -3
- package/dist/composer.test.mjs.map +1 -1
- package/dist/deployers/strip-host-ports.test.cjs +26 -26
- package/dist/deployers/strip-host-ports.test.cjs.map +1 -1
- package/dist/deployers/strip-host-ports.test.mjs +1 -1
- package/dist/generate.cjs +8 -4
- package/dist/generate.cjs.map +1 -1
- package/dist/generate.d.cts.map +1 -1
- package/dist/generate.d.mts.map +1 -1
- package/dist/generate.mjs +9 -5
- package/dist/generate.mjs.map +1 -1
- package/dist/generate.test.cjs +55 -55
- package/dist/generate.test.cjs.map +1 -1
- package/dist/generate.test.mjs +2 -2
- package/dist/generate.test.mjs.map +1 -1
- package/dist/generators/bare-metal-install.test.cjs +18 -18
- package/dist/generators/bare-metal-install.test.cjs.map +1 -1
- package/dist/generators/bare-metal-install.test.mjs +1 -1
- package/dist/generators/caddy.test.cjs +13 -13
- package/dist/generators/caddy.test.cjs.map +1 -1
- package/dist/generators/caddy.test.mjs +1 -1
- package/dist/generators/clone-repos.cjs +140 -0
- package/dist/generators/clone-repos.cjs.map +1 -0
- package/dist/generators/clone-repos.d.cts +11 -0
- package/dist/generators/clone-repos.d.cts.map +1 -0
- package/dist/generators/clone-repos.d.mts +11 -0
- package/dist/generators/clone-repos.d.mts.map +1 -0
- package/dist/generators/clone-repos.mjs +139 -0
- package/dist/generators/clone-repos.mjs.map +1 -0
- package/dist/generators/clone-repos.test.cjs +140 -0
- package/dist/generators/clone-repos.test.cjs.map +1 -0
- package/dist/generators/clone-repos.test.d.cts +1 -0
- package/dist/generators/clone-repos.test.d.mts +1 -0
- package/dist/generators/clone-repos.test.mjs +141 -0
- package/dist/generators/clone-repos.test.mjs.map +1 -0
- package/dist/generators/env.test.cjs +17 -17
- package/dist/generators/env.test.cjs.map +1 -1
- package/dist/generators/env.test.mjs +1 -1
- package/dist/generators/health-check.test.cjs +39 -39
- package/dist/generators/health-check.test.cjs.map +1 -1
- package/dist/generators/health-check.test.mjs +1 -1
- package/dist/generators/postgres-init.cjs +20 -0
- package/dist/generators/postgres-init.cjs.map +1 -1
- package/dist/generators/postgres-init.d.cts.map +1 -1
- package/dist/generators/postgres-init.d.mts.map +1 -1
- package/dist/generators/postgres-init.mjs +20 -0
- package/dist/generators/postgres-init.mjs.map +1 -1
- package/dist/generators/scripts.cjs +332 -3
- package/dist/generators/scripts.cjs.map +1 -1
- package/dist/generators/scripts.d.cts +3 -1
- package/dist/generators/scripts.d.cts.map +1 -1
- package/dist/generators/scripts.d.mts +3 -1
- package/dist/generators/scripts.d.mts.map +1 -1
- package/dist/generators/scripts.mjs +332 -3
- package/dist/generators/scripts.mjs.map +1 -1
- package/dist/generators/scripts.test.cjs +57 -23
- package/dist/generators/scripts.test.cjs.map +1 -1
- package/dist/generators/scripts.test.mjs +39 -5
- package/dist/generators/scripts.test.mjs.map +1 -1
- package/dist/generators/stack-manifest.cjs +1 -0
- package/dist/generators/stack-manifest.cjs.map +1 -1
- package/dist/generators/stack-manifest.d.cts +3 -2
- package/dist/generators/stack-manifest.d.cts.map +1 -1
- package/dist/generators/stack-manifest.d.mts +3 -2
- package/dist/generators/stack-manifest.d.mts.map +1 -1
- package/dist/generators/stack-manifest.mjs +1 -0
- package/dist/generators/stack-manifest.mjs.map +1 -1
- package/dist/generators/traefik.test.cjs +32 -32
- package/dist/generators/traefik.test.cjs.map +1 -1
- package/dist/generators/traefik.test.mjs +1 -1
- package/dist/index.cjs +28 -5
- package/dist/index.d.cts +7 -4
- package/dist/index.d.mts +7 -4
- package/dist/index.mjs +10 -7
- package/dist/migrations.test.cjs +16 -16
- package/dist/migrations.test.cjs.map +1 -1
- package/dist/migrations.test.mjs +1 -1
- package/dist/presets/registry.cjs.map +1 -1
- package/dist/presets/registry.d.cts.map +1 -1
- package/dist/presets/registry.d.mts.map +1 -1
- package/dist/presets/registry.mjs.map +1 -1
- package/dist/presets/registry.test.cjs +14 -14
- package/dist/presets/registry.test.cjs.map +1 -1
- package/dist/presets/registry.test.mjs +1 -1
- package/dist/resolver.cjs +8 -0
- package/dist/resolver.cjs.map +1 -1
- package/dist/resolver.mjs +9 -1
- package/dist/resolver.mjs.map +1 -1
- package/dist/resolver.test.cjs +125 -90
- package/dist/resolver.test.cjs.map +1 -1
- package/dist/resolver.test.mjs +47 -12
- package/dist/resolver.test.mjs.map +1 -1
- package/dist/{schema-B4c64P8N.d.cts → schema-CKBRu-Rt.d.cts} +355 -8
- package/dist/schema-CKBRu-Rt.d.cts.map +1 -0
- package/dist/{schema-CXNhYci1.d.mts → schema-Dn-_Jpb6.d.mts} +355 -8
- package/dist/schema-Dn-_Jpb6.d.mts.map +1 -0
- package/dist/schema.cjs +160 -5
- package/dist/schema.cjs.map +1 -1
- package/dist/schema.d.cts +2 -2
- package/dist/schema.d.mts +2 -2
- package/dist/schema.mjs +150 -6
- package/dist/schema.mjs.map +1 -1
- package/dist/schema.test.cjs +86 -86
- package/dist/schema.test.cjs.map +1 -1
- package/dist/schema.test.mjs +1 -1
- package/dist/services/definitions/apptension-saas.cjs +87 -0
- package/dist/services/definitions/apptension-saas.cjs.map +1 -0
- package/dist/services/definitions/apptension-saas.d.cts +7 -0
- package/dist/services/definitions/apptension-saas.d.cts.map +1 -0
- package/dist/services/definitions/apptension-saas.d.mts +7 -0
- package/dist/services/definitions/apptension-saas.d.mts.map +1 -0
- package/dist/services/definitions/apptension-saas.mjs +86 -0
- package/dist/services/definitions/apptension-saas.mjs.map +1 -0
- package/dist/services/definitions/boxyhq-saas.cjs +88 -0
- package/dist/services/definitions/boxyhq-saas.cjs.map +1 -0
- package/dist/services/definitions/boxyhq-saas.d.cts +7 -0
- package/dist/services/definitions/boxyhq-saas.d.cts.map +1 -0
- package/dist/services/definitions/boxyhq-saas.d.mts +7 -0
- package/dist/services/definitions/boxyhq-saas.d.mts.map +1 -0
- package/dist/services/definitions/boxyhq-saas.mjs +87 -0
- package/dist/services/definitions/boxyhq-saas.mjs.map +1 -0
- package/dist/services/definitions/browserless.cjs +4 -1
- package/dist/services/definitions/browserless.cjs.map +1 -1
- package/dist/services/definitions/browserless.mjs +4 -1
- package/dist/services/definitions/browserless.mjs.map +1 -1
- package/dist/services/definitions/cmsaas-starter.cjs +86 -0
- package/dist/services/definitions/cmsaas-starter.cjs.map +1 -0
- package/dist/services/definitions/cmsaas-starter.d.cts +7 -0
- package/dist/services/definitions/cmsaas-starter.d.cts.map +1 -0
- package/dist/services/definitions/cmsaas-starter.d.mts +7 -0
- package/dist/services/definitions/cmsaas-starter.d.mts.map +1 -0
- package/dist/services/definitions/cmsaas-starter.mjs +85 -0
- package/dist/services/definitions/cmsaas-starter.mjs.map +1 -0
- package/dist/services/definitions/convex.cjs +43 -1
- package/dist/services/definitions/convex.cjs.map +1 -1
- package/dist/services/definitions/convex.mjs +43 -1
- package/dist/services/definitions/convex.mjs.map +1 -1
- package/dist/services/definitions/grafana.cjs +11 -1
- package/dist/services/definitions/grafana.cjs.map +1 -1
- package/dist/services/definitions/grafana.mjs +11 -1
- package/dist/services/definitions/grafana.mjs.map +1 -1
- package/dist/services/definitions/index.cjs +51 -36
- package/dist/services/definitions/index.cjs.map +1 -1
- package/dist/services/definitions/index.d.cts +30 -25
- package/dist/services/definitions/index.d.cts.map +1 -1
- package/dist/services/definitions/index.d.mts +30 -25
- package/dist/services/definitions/index.d.mts.map +1 -1
- package/dist/services/definitions/index.mjs +47 -37
- package/dist/services/definitions/index.mjs.map +1 -1
- package/dist/services/definitions/ixartz-saas.cjs +88 -0
- package/dist/services/definitions/ixartz-saas.cjs.map +1 -0
- package/dist/services/definitions/ixartz-saas.d.cts +7 -0
- package/dist/services/definitions/ixartz-saas.d.cts.map +1 -0
- package/dist/services/definitions/ixartz-saas.d.mts +7 -0
- package/dist/services/definitions/ixartz-saas.d.mts.map +1 -0
- package/dist/services/definitions/ixartz-saas.mjs +87 -0
- package/dist/services/definitions/ixartz-saas.mjs.map +1 -0
- package/dist/services/definitions/meilisearch.cjs +11 -1
- package/dist/services/definitions/meilisearch.cjs.map +1 -1
- package/dist/services/definitions/meilisearch.mjs +11 -1
- package/dist/services/definitions/meilisearch.mjs.map +1 -1
- package/dist/services/definitions/minio.cjs +3 -1
- package/dist/services/definitions/minio.cjs.map +1 -1
- package/dist/services/definitions/minio.mjs +3 -1
- package/dist/services/definitions/minio.mjs.map +1 -1
- package/dist/services/definitions/mission-control.cjs +16 -2
- package/dist/services/definitions/mission-control.cjs.map +1 -1
- package/dist/services/definitions/mission-control.mjs +16 -2
- package/dist/services/definitions/mission-control.mjs.map +1 -1
- package/dist/services/definitions/n8n.cjs +11 -1
- package/dist/services/definitions/n8n.cjs.map +1 -1
- package/dist/services/definitions/n8n.mjs +11 -1
- package/dist/services/definitions/n8n.mjs.map +1 -1
- package/dist/services/definitions/ollama.cjs +3 -1
- package/dist/services/definitions/ollama.cjs.map +1 -1
- package/dist/services/definitions/ollama.mjs +3 -1
- package/dist/services/definitions/ollama.mjs.map +1 -1
- package/dist/services/definitions/open-saas.cjs +81 -0
- package/dist/services/definitions/open-saas.cjs.map +1 -0
- package/dist/services/definitions/open-saas.d.cts +7 -0
- package/dist/services/definitions/open-saas.d.cts.map +1 -0
- package/dist/services/definitions/open-saas.d.mts +7 -0
- package/dist/services/definitions/open-saas.d.mts.map +1 -0
- package/dist/services/definitions/open-saas.mjs +80 -0
- package/dist/services/definitions/open-saas.mjs.map +1 -0
- package/dist/services/definitions/qdrant.cjs +3 -1
- package/dist/services/definitions/qdrant.cjs.map +1 -1
- package/dist/services/definitions/qdrant.mjs +3 -1
- package/dist/services/definitions/qdrant.mjs.map +1 -1
- package/dist/services/definitions/searxng.cjs +8 -1
- package/dist/services/definitions/searxng.cjs.map +1 -1
- package/dist/services/definitions/searxng.mjs +8 -1
- package/dist/services/definitions/searxng.mjs.map +1 -1
- package/dist/services/definitions/uptime-kuma.cjs +8 -1
- package/dist/services/definitions/uptime-kuma.cjs.map +1 -1
- package/dist/services/definitions/uptime-kuma.mjs +8 -1
- package/dist/services/definitions/uptime-kuma.mjs.map +1 -1
- package/dist/services/registry.cjs +3 -0
- package/dist/services/registry.cjs.map +1 -1
- package/dist/services/registry.d.cts.map +1 -1
- package/dist/services/registry.d.mts.map +1 -1
- package/dist/services/registry.mjs +3 -0
- package/dist/services/registry.mjs.map +1 -1
- package/dist/services/registry.test.cjs +40 -33
- package/dist/services/registry.test.cjs.map +1 -1
- package/dist/services/registry.test.mjs +8 -1
- package/dist/services/registry.test.mjs.map +1 -1
- package/dist/{skill-manifest-BVUXU0__.mjs → skill-manifest-6XhrhWsG.mjs} +49 -1
- package/dist/{skill-manifest--IgY9REK.cjs.map → skill-manifest-6XhrhWsG.mjs.map} +1 -1
- package/dist/{skill-manifest--IgY9REK.cjs → skill-manifest-B8znSsym.cjs} +49 -1
- package/dist/{skill-manifest-BVUXU0__.mjs.map → skill-manifest-B8znSsym.cjs.map} +1 -1
- package/dist/skills/registry.cjs +3 -3
- package/dist/skills/registry.cjs.map +1 -1
- package/dist/skills/registry.mjs +3 -3
- package/dist/skills/registry.mjs.map +1 -1
- package/dist/skills/skill-manifest.cjs +1 -1
- package/dist/skills/skill-manifest.mjs +1 -1
- package/dist/{vi.2VT5v0um-DvC3SVNc.mjs → test.CTcmp4Su-ClCHJ3FA.mjs} +6793 -6403
- package/dist/test.CTcmp4Su-ClCHJ3FA.mjs.map +1 -0
- package/dist/{vi.2VT5v0um-CRqXre87.cjs → test.CTcmp4Su-DlzTarwH.cjs} +6793 -6403
- package/dist/test.CTcmp4Su-DlzTarwH.cjs.map +1 -0
- package/dist/track-analytics.cjs +50 -0
- package/dist/track-analytics.cjs.map +1 -0
- package/dist/track-analytics.d.cts +34 -0
- package/dist/track-analytics.d.cts.map +1 -0
- package/dist/track-analytics.d.mts +34 -0
- package/dist/track-analytics.d.mts.map +1 -0
- package/dist/track-analytics.mjs +48 -0
- package/dist/track-analytics.mjs.map +1 -0
- package/dist/track-analytics.test.cjs +91 -0
- package/dist/track-analytics.test.cjs.map +1 -0
- package/dist/track-analytics.test.d.cts +1 -0
- package/dist/track-analytics.test.d.mts +1 -0
- package/dist/track-analytics.test.mjs +92 -0
- package/dist/track-analytics.test.mjs.map +1 -0
- package/dist/types.cjs +7 -0
- package/dist/types.cjs.map +1 -1
- package/dist/types.d.cts +12 -2
- package/dist/types.d.cts.map +1 -1
- package/dist/types.d.mts +12 -2
- package/dist/types.d.mts.map +1 -1
- package/dist/types.mjs +7 -0
- package/dist/types.mjs.map +1 -1
- package/dist/validator.test.cjs +15 -15
- package/dist/validator.test.cjs.map +1 -1
- package/dist/validator.test.mjs +2 -2
- package/dist/version-manager.cjs +1 -1
- package/dist/version-manager.cjs.map +1 -1
- package/dist/version-manager.mjs +1 -1
- package/dist/version-manager.mjs.map +1 -1
- package/dist/version-manager.test.cjs +40 -38
- package/dist/version-manager.test.cjs.map +1 -1
- package/dist/version-manager.test.mjs +7 -5
- package/dist/version-manager.test.mjs.map +1 -1
- package/package.json +4 -4
- package/src/__snapshots__/composer.snapshot.test.ts.snap +160 -0
- package/src/addon-stack.test.ts +490 -0
- package/src/addon-stack.ts +998 -0
- package/src/bare-metal-partition.test.ts +4 -3
- package/src/composer.test.ts +4 -2
- package/src/composer.ts +24 -5
- package/src/generate.test.ts +2 -1
- package/src/generate.ts +10 -1
- package/src/generators/clone-repos.test.ts +154 -0
- package/src/generators/clone-repos.ts +159 -0
- package/src/generators/postgres-init.ts +17 -0
- package/src/generators/scripts.test.ts +52 -4
- package/src/generators/scripts.ts +351 -3
- package/src/generators/stack-manifest.ts +4 -2
- package/src/index.ts +28 -2
- package/src/presets/registry.ts +241 -329
- package/src/resolver.test.ts +53 -15
- package/src/resolver.ts +13 -1
- package/src/schema.ts +216 -4
- package/src/services/definitions/apptension-saas.ts +84 -0
- package/src/services/definitions/boxyhq-saas.ts +84 -0
- package/src/services/definitions/browserless.ts +3 -0
- package/src/services/definitions/cmsaas-starter.ts +84 -0
- package/src/services/definitions/convex.ts +31 -0
- package/src/services/definitions/grafana.ts +9 -0
- package/src/services/definitions/index.ts +90 -70
- package/src/services/definitions/ixartz-saas.ts +84 -0
- package/src/services/definitions/meilisearch.ts +9 -0
- package/src/services/definitions/minio.ts +2 -0
- package/src/services/definitions/mission-control.ts +19 -2
- package/src/services/definitions/n8n.ts +9 -0
- package/src/services/definitions/ollama.ts +2 -0
- package/src/services/definitions/open-saas.ts +79 -0
- package/src/services/definitions/qdrant.ts +2 -0
- package/src/services/definitions/searxng.ts +3 -0
- package/src/services/definitions/uptime-kuma.ts +3 -0
- package/src/services/registry.test.ts +8 -0
- package/src/services/registry.ts +7 -0
- package/src/skills/manifest.json +64 -0
- package/src/skills/registry.ts +3 -3
- package/src/track-analytics.test.ts +82 -0
- package/src/track-analytics.ts +76 -0
- package/src/types.ts +29 -0
- package/src/version-manager.test.ts +10 -5
- package/src/version-manager.ts +1 -1
- package/dist/schema-B4c64P8N.d.cts.map +0 -1
- package/dist/schema-CXNhYci1.d.mts.map +0 -1
- package/dist/vi.2VT5v0um-CRqXre87.cjs.map +0 -1
- package/dist/vi.2VT5v0um-DvC3SVNc.mjs.map +0 -1
|
@@ -4,7 +4,9 @@
|
|
|
4
4
|
*
|
|
5
5
|
* Returns a map of file paths (relative to project root) to file contents.
|
|
6
6
|
*/
|
|
7
|
-
declare function generateScripts(
|
|
7
|
+
declare function generateScripts(options?: {
|
|
8
|
+
hasGitServices?: boolean;
|
|
9
|
+
}): Record<string, string>;
|
|
8
10
|
//#endregion
|
|
9
11
|
export { generateScripts };
|
|
10
12
|
//# sourceMappingURL=scripts.d.cts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"scripts.d.cts","names":[],"sources":["../../src/generators/scripts.ts"],"mappings":";;AAKA;;;;iBAAgB,eAAA,
|
|
1
|
+
{"version":3,"file":"scripts.d.cts","names":[],"sources":["../../src/generators/scripts.ts"],"mappings":";;AAKA;;;;iBAAgB,eAAA,CAAgB,OAAA;EAAY,cAAA;AAAA,IAA6B,MAAA"}
|
|
@@ -4,7 +4,9 @@
|
|
|
4
4
|
*
|
|
5
5
|
* Returns a map of file paths (relative to project root) to file contents.
|
|
6
6
|
*/
|
|
7
|
-
declare function generateScripts(
|
|
7
|
+
declare function generateScripts(options?: {
|
|
8
|
+
hasGitServices?: boolean;
|
|
9
|
+
}): Record<string, string>;
|
|
8
10
|
//#endregion
|
|
9
11
|
export { generateScripts };
|
|
10
12
|
//# sourceMappingURL=scripts.d.mts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"scripts.d.mts","names":[],"sources":["../../src/generators/scripts.ts"],"mappings":";;AAKA;;;;iBAAgB,eAAA,
|
|
1
|
+
{"version":3,"file":"scripts.d.mts","names":[],"sources":["../../src/generators/scripts.ts"],"mappings":";;AAKA;;;;iBAAgB,eAAA,CAAgB,OAAA;EAAY,cAAA;AAAA,IAA6B,MAAA"}
|
|
@@ -4,7 +4,8 @@
|
|
|
4
4
|
*
|
|
5
5
|
* Returns a map of file paths (relative to project root) to file contents.
|
|
6
6
|
*/
|
|
7
|
-
function generateScripts() {
|
|
7
|
+
function generateScripts(options) {
|
|
8
|
+
const hasGit = options?.hasGitServices ?? false;
|
|
8
9
|
const files = {};
|
|
9
10
|
files["scripts/start.sh"] = `#!/usr/bin/env bash
|
|
10
11
|
set -euo pipefail
|
|
@@ -134,6 +135,14 @@ if [ "$EMPTY_SECRETS" -gt 0 ]; then
|
|
|
134
135
|
echo ""
|
|
135
136
|
fi
|
|
136
137
|
|
|
138
|
+
# ── Clone git-based repositories (if any) ────────────────────────────────────
|
|
139
|
+
|
|
140
|
+
if [ -f "$SCRIPT_DIR/clone-repos.sh" ]; then
|
|
141
|
+
info "Cloning/updating SaaS boilerplate repositories..."
|
|
142
|
+
bash "$SCRIPT_DIR/clone-repos.sh"
|
|
143
|
+
ok "Repositories ready."
|
|
144
|
+
fi
|
|
145
|
+
|
|
137
146
|
# ── Pull and start ───────────────────────────────────────────────────────────
|
|
138
147
|
|
|
139
148
|
echo ""
|
|
@@ -142,7 +151,7 @@ docker compose pull --quiet 2>/dev/null || docker compose pull
|
|
|
142
151
|
|
|
143
152
|
echo ""
|
|
144
153
|
info "Starting services..."
|
|
145
|
-
docker compose up -d --remove-orphans
|
|
154
|
+
docker compose up -d --remove-orphans${hasGit ? " --build" : ""}
|
|
146
155
|
|
|
147
156
|
# ── Health-check loop ────────────────────────────────────────────────────────
|
|
148
157
|
|
|
@@ -235,13 +244,19 @@ cd "$PROJECT_DIR"
|
|
|
235
244
|
echo "🐾 OpenClaw — Updating services..."
|
|
236
245
|
echo ""
|
|
237
246
|
|
|
247
|
+
# Update git-based repositories (if any)
|
|
248
|
+
if [ -f "$SCRIPT_DIR/clone-repos.sh" ]; then
|
|
249
|
+
echo "📂 Updating SaaS boilerplate repositories..."
|
|
250
|
+
bash "$SCRIPT_DIR/clone-repos.sh"
|
|
251
|
+
fi
|
|
252
|
+
|
|
238
253
|
# Pull latest images
|
|
239
254
|
echo "📦 Pulling latest images..."
|
|
240
255
|
docker compose pull
|
|
241
256
|
|
|
242
257
|
echo ""
|
|
243
258
|
echo "🔄 Restarting services with new images..."
|
|
244
|
-
docker compose up -d --remove-orphans
|
|
259
|
+
docker compose up -d --remove-orphans${hasGit ? " --build" : ""}
|
|
245
260
|
|
|
246
261
|
echo ""
|
|
247
262
|
echo "⏳ Waiting for services to stabilize..."
|
|
@@ -342,6 +357,320 @@ echo "── Network ───────────────────
|
|
|
342
357
|
echo ""
|
|
343
358
|
|
|
344
359
|
docker network ls --filter "name=openclaw" --format "table {{.Name}}\\t{{.Driver}}\\t{{.Scope}}" 2>/dev/null || true
|
|
360
|
+
`;
|
|
361
|
+
files["scripts/start.ps1"] = `#Requires -Version 5.1
|
|
362
|
+
<#
|
|
363
|
+
.SYNOPSIS
|
|
364
|
+
OpenClaw Start Script — validates prerequisites, auto-generates secrets,
|
|
365
|
+
creates required directories, and starts all services via Docker Compose.
|
|
366
|
+
#>
|
|
367
|
+
|
|
368
|
+
$ErrorActionPreference = 'Stop'
|
|
369
|
+
$ScriptDir = Split-Path -Parent $MyInvocation.MyCommand.Path
|
|
370
|
+
$ProjectDir = Split-Path -Parent $ScriptDir
|
|
371
|
+
Set-Location $ProjectDir
|
|
372
|
+
|
|
373
|
+
function Info { param($Msg) Write-Host " i $Msg" -ForegroundColor Cyan }
|
|
374
|
+
function Ok { param($Msg) Write-Host " + $Msg" -ForegroundColor Green }
|
|
375
|
+
function Warn { param($Msg) Write-Host " ! $Msg" -ForegroundColor Yellow }
|
|
376
|
+
function Err { param($Msg) Write-Host " x $Msg" -ForegroundColor Red }
|
|
377
|
+
|
|
378
|
+
Write-Host ""
|
|
379
|
+
Write-Host "OpenClaw - Starting services..." -ForegroundColor White
|
|
380
|
+
Write-Host ""
|
|
381
|
+
|
|
382
|
+
# ── Prerequisite checks ─────────────────────────────────────────────────
|
|
383
|
+
if (-not (Get-Command docker -ErrorAction SilentlyContinue)) {
|
|
384
|
+
Err "Docker is not installed. Please install Docker Desktop."
|
|
385
|
+
Write-Host " https://docs.docker.com/desktop/install/windows-install/"
|
|
386
|
+
exit 1
|
|
387
|
+
}
|
|
388
|
+
|
|
389
|
+
$composeCheck = docker compose version 2>&1
|
|
390
|
+
if ($LASTEXITCODE -ne 0) {
|
|
391
|
+
Err "Docker Compose (v2) is not available."
|
|
392
|
+
Write-Host " https://docs.docker.com/compose/install/"
|
|
393
|
+
exit 1
|
|
394
|
+
}
|
|
395
|
+
|
|
396
|
+
$dockerInfo = docker info 2>&1
|
|
397
|
+
if ($LASTEXITCODE -ne 0) {
|
|
398
|
+
Err "Docker daemon is not running. Please start Docker Desktop."
|
|
399
|
+
exit 1
|
|
400
|
+
}
|
|
401
|
+
|
|
402
|
+
# ── Source .env if it exists ─────────────────────────────────────────────
|
|
403
|
+
if (Test-Path ".env") {
|
|
404
|
+
Info "Loading existing .env file..."
|
|
405
|
+
Get-Content ".env" | ForEach-Object {
|
|
406
|
+
if ($_ -match '^([^#=]+)=(.*)$') {
|
|
407
|
+
[Environment]::SetEnvironmentVariable($Matches[1].Trim(), $Matches[2].Trim(), 'Process')
|
|
408
|
+
}
|
|
409
|
+
}
|
|
410
|
+
} else {
|
|
411
|
+
Warn ".env file not found - will create one from .env.example if available."
|
|
412
|
+
if (Test-Path ".env.example") {
|
|
413
|
+
Copy-Item ".env.example" ".env"
|
|
414
|
+
Info "Created .env from .env.example"
|
|
415
|
+
Get-Content ".env" | ForEach-Object {
|
|
416
|
+
if ($_ -match '^([^#=]+)=(.*)$') {
|
|
417
|
+
[Environment]::SetEnvironmentVariable($Matches[1].Trim(), $Matches[2].Trim(), 'Process')
|
|
418
|
+
}
|
|
419
|
+
}
|
|
420
|
+
}
|
|
421
|
+
}
|
|
422
|
+
|
|
423
|
+
# ── Auto-generate OPENCLAW_GATEWAY_TOKEN if missing ──────────────────────
|
|
424
|
+
if (-not $env:OPENCLAW_GATEWAY_TOKEN) {
|
|
425
|
+
Info "Generating OPENCLAW_GATEWAY_TOKEN..."
|
|
426
|
+
$bytes = New-Object byte[] 32
|
|
427
|
+
[System.Security.Cryptography.RandomNumberGenerator]::Create().GetBytes($bytes)
|
|
428
|
+
$env:OPENCLAW_GATEWAY_TOKEN = ($bytes | ForEach-Object { $_.ToString("x2") }) -join ''
|
|
429
|
+
|
|
430
|
+
if (Test-Path ".env") {
|
|
431
|
+
$envContent = Get-Content ".env" -Raw
|
|
432
|
+
if ($envContent -match '(?m)^OPENCLAW_GATEWAY_TOKEN=') {
|
|
433
|
+
$envContent = $envContent -replace '(?m)^OPENCLAW_GATEWAY_TOKEN=.*', "OPENCLAW_GATEWAY_TOKEN=$($env:OPENCLAW_GATEWAY_TOKEN)"
|
|
434
|
+
Set-Content ".env" $envContent -NoNewline
|
|
435
|
+
} else {
|
|
436
|
+
Add-Content ".env" "OPENCLAW_GATEWAY_TOKEN=$($env:OPENCLAW_GATEWAY_TOKEN)"
|
|
437
|
+
}
|
|
438
|
+
}
|
|
439
|
+
Ok "Gateway token generated and saved."
|
|
440
|
+
}
|
|
441
|
+
|
|
442
|
+
# ── Apply defaults ───────────────────────────────────────────────────────
|
|
443
|
+
if (-not $env:OPENCLAW_VERSION) { $env:OPENCLAW_VERSION = "latest" }
|
|
444
|
+
if (-not $env:OPENCLAW_GATEWAY_PORT) { $env:OPENCLAW_GATEWAY_PORT = "18789" }
|
|
445
|
+
if (-not $env:OPENCLAW_BRIDGE_PORT) { $env:OPENCLAW_BRIDGE_PORT = "18790" }
|
|
446
|
+
if (-not $env:OPENCLAW_GATEWAY_BIND) { $env:OPENCLAW_GATEWAY_BIND = "lan" }
|
|
447
|
+
if (-not $env:OPENCLAW_CONFIG_DIR) { $env:OPENCLAW_CONFIG_DIR = "./openclaw/config" }
|
|
448
|
+
if (-not $env:OPENCLAW_WORKSPACE_DIR) { $env:OPENCLAW_WORKSPACE_DIR = "./openclaw/workspace" }
|
|
449
|
+
|
|
450
|
+
# ── Create required host directories ────────────────────────────────────
|
|
451
|
+
Info "Ensuring host directories exist..."
|
|
452
|
+
New-Item -ItemType Directory -Force -Path $env:OPENCLAW_CONFIG_DIR | Out-Null
|
|
453
|
+
New-Item -ItemType Directory -Force -Path $env:OPENCLAW_WORKSPACE_DIR | Out-Null
|
|
454
|
+
Ok "Directories ready: $($env:OPENCLAW_CONFIG_DIR), $($env:OPENCLAW_WORKSPACE_DIR)"
|
|
455
|
+
|
|
456
|
+
# ── Clone git-based repositories (if any) ─────────────────────────────
|
|
457
|
+
$cloneScript = Join-Path $ScriptDir "clone-repos.ps1"
|
|
458
|
+
if (Test-Path $cloneScript) {
|
|
459
|
+
Info "Cloning/updating SaaS boilerplate repositories..."
|
|
460
|
+
& $cloneScript
|
|
461
|
+
Ok "Repositories ready."
|
|
462
|
+
}
|
|
463
|
+
|
|
464
|
+
# ── Pull and start ───────────────────────────────────────────────────────
|
|
465
|
+
Write-Host ""
|
|
466
|
+
Info "Pulling latest images..."
|
|
467
|
+
docker compose pull 2>$null
|
|
468
|
+
if ($LASTEXITCODE -ne 0) { docker compose pull }
|
|
469
|
+
|
|
470
|
+
Write-Host ""
|
|
471
|
+
Info "Starting services..."
|
|
472
|
+
docker compose up -d --remove-orphans${hasGit ? " --build" : ""}
|
|
473
|
+
|
|
474
|
+
# ── Health-check loop ────────────────────────────────────────────────────
|
|
475
|
+
Write-Host ""
|
|
476
|
+
Info "Waiting for services to become healthy..."
|
|
477
|
+
Start-Sleep -Seconds 5
|
|
478
|
+
|
|
479
|
+
$retries = 0
|
|
480
|
+
$maxRetries = 30
|
|
481
|
+
while ($retries -lt $maxRetries) {
|
|
482
|
+
$psOutput = docker compose ps --format json 2>$null
|
|
483
|
+
$unhealthy = ($psOutput | Select-String -Pattern '"unhealthy"' -SimpleMatch).Count
|
|
484
|
+
$starting = ($psOutput | Select-String -Pattern '"starting"' -SimpleMatch).Count
|
|
485
|
+
if ($unhealthy -eq 0 -and $starting -eq 0) { break }
|
|
486
|
+
$retries++
|
|
487
|
+
Start-Sleep -Seconds 2
|
|
488
|
+
}
|
|
489
|
+
|
|
490
|
+
Write-Host ""
|
|
491
|
+
docker compose ps
|
|
492
|
+
Write-Host ""
|
|
493
|
+
|
|
494
|
+
if ($retries -ge $maxRetries) {
|
|
495
|
+
Warn "Some services may still be starting. Check: docker compose ps"
|
|
496
|
+
} else {
|
|
497
|
+
Ok "All services are running!"
|
|
498
|
+
}
|
|
499
|
+
|
|
500
|
+
# ── Print service URLs & token ───────────────────────────────────────────
|
|
501
|
+
$gatewayHost = if ($env:OPENCLAW_GATEWAY_BIND -eq "lan") { "0.0.0.0" } else { "localhost" }
|
|
502
|
+
|
|
503
|
+
Write-Host ""
|
|
504
|
+
Write-Host "==============================================================================="
|
|
505
|
+
Write-Host " OpenClaw is ready!"
|
|
506
|
+
Write-Host "==============================================================================="
|
|
507
|
+
Write-Host ""
|
|
508
|
+
Write-Host " Gateway URL: http://\${gatewayHost}:$($env:OPENCLAW_GATEWAY_PORT)"
|
|
509
|
+
Write-Host " Bridge (WebSocket): ws://\${gatewayHost}:$($env:OPENCLAW_BRIDGE_PORT)"
|
|
510
|
+
Write-Host " Config directory: $($env:OPENCLAW_CONFIG_DIR)"
|
|
511
|
+
Write-Host " Workspace directory: $($env:OPENCLAW_WORKSPACE_DIR)"
|
|
512
|
+
Write-Host ""
|
|
513
|
+
Write-Host " Gateway Token: $($env:OPENCLAW_GATEWAY_TOKEN)"
|
|
514
|
+
Write-Host ""
|
|
515
|
+
Write-Host " Manage:"
|
|
516
|
+
Write-Host " Stop: .\\scripts\\stop.ps1"
|
|
517
|
+
Write-Host " Status: .\\scripts\\status.ps1"
|
|
518
|
+
Write-Host " Update: .\\scripts\\update.ps1"
|
|
519
|
+
Write-Host " Logs: docker compose logs -f"
|
|
520
|
+
Write-Host ""
|
|
521
|
+
Write-Host "==============================================================================="
|
|
522
|
+
`;
|
|
523
|
+
files["scripts/stop.ps1"] = `#Requires -Version 5.1
|
|
524
|
+
<#
|
|
525
|
+
.SYNOPSIS
|
|
526
|
+
OpenClaw Stop Script — gracefully stops all services.
|
|
527
|
+
#>
|
|
528
|
+
|
|
529
|
+
$ErrorActionPreference = 'Stop'
|
|
530
|
+
$ScriptDir = Split-Path -Parent $MyInvocation.MyCommand.Path
|
|
531
|
+
$ProjectDir = Split-Path -Parent $ScriptDir
|
|
532
|
+
Set-Location $ProjectDir
|
|
533
|
+
|
|
534
|
+
Write-Host ""
|
|
535
|
+
Write-Host "OpenClaw - Stopping services..." -ForegroundColor White
|
|
536
|
+
Write-Host ""
|
|
537
|
+
|
|
538
|
+
docker compose down --timeout 30
|
|
539
|
+
|
|
540
|
+
Write-Host ""
|
|
541
|
+
Write-Host " + All services stopped." -ForegroundColor Green
|
|
542
|
+
`;
|
|
543
|
+
files["scripts/update.ps1"] = `#Requires -Version 5.1
|
|
544
|
+
<#
|
|
545
|
+
.SYNOPSIS
|
|
546
|
+
OpenClaw Update Script — pulls latest images and restarts services.
|
|
547
|
+
#>
|
|
548
|
+
|
|
549
|
+
$ErrorActionPreference = 'Stop'
|
|
550
|
+
$ScriptDir = Split-Path -Parent $MyInvocation.MyCommand.Path
|
|
551
|
+
$ProjectDir = Split-Path -Parent $ScriptDir
|
|
552
|
+
Set-Location $ProjectDir
|
|
553
|
+
|
|
554
|
+
Write-Host ""
|
|
555
|
+
Write-Host "OpenClaw - Updating services..." -ForegroundColor White
|
|
556
|
+
Write-Host ""
|
|
557
|
+
|
|
558
|
+
# Update git-based repositories (if any)
|
|
559
|
+
$cloneScript = Join-Path $ScriptDir "clone-repos.ps1"
|
|
560
|
+
if (Test-Path $cloneScript) {
|
|
561
|
+
Write-Host " Updating SaaS boilerplate repositories..." -ForegroundColor Cyan
|
|
562
|
+
& $cloneScript
|
|
563
|
+
}
|
|
564
|
+
|
|
565
|
+
Write-Host " Pulling latest images..." -ForegroundColor Cyan
|
|
566
|
+
docker compose pull
|
|
567
|
+
|
|
568
|
+
Write-Host ""
|
|
569
|
+
Write-Host " Restarting services with new images..." -ForegroundColor Cyan
|
|
570
|
+
docker compose up -d --remove-orphans${hasGit ? " --build" : ""}
|
|
571
|
+
|
|
572
|
+
Write-Host ""
|
|
573
|
+
Write-Host " Waiting for services to stabilize..." -ForegroundColor Cyan
|
|
574
|
+
Start-Sleep -Seconds 10
|
|
575
|
+
|
|
576
|
+
docker compose ps
|
|
577
|
+
|
|
578
|
+
Write-Host ""
|
|
579
|
+
Write-Host " + Update complete!" -ForegroundColor Green
|
|
580
|
+
`;
|
|
581
|
+
files["scripts/backup.ps1"] = `#Requires -Version 5.1
|
|
582
|
+
<#
|
|
583
|
+
.SYNOPSIS
|
|
584
|
+
OpenClaw Backup Script — backs up all named Docker volumes to a timestamped directory.
|
|
585
|
+
#>
|
|
586
|
+
|
|
587
|
+
$ErrorActionPreference = 'Stop'
|
|
588
|
+
$ScriptDir = Split-Path -Parent $MyInvocation.MyCommand.Path
|
|
589
|
+
$ProjectDir = Split-Path -Parent $ScriptDir
|
|
590
|
+
Set-Location $ProjectDir
|
|
591
|
+
|
|
592
|
+
$timestamp = Get-Date -Format "yyyyMMdd_HHmmss"
|
|
593
|
+
$backupDir = Join-Path $ProjectDir "backups\\$timestamp"
|
|
594
|
+
New-Item -ItemType Directory -Force -Path $backupDir | Out-Null
|
|
595
|
+
|
|
596
|
+
Write-Host ""
|
|
597
|
+
Write-Host "OpenClaw - Backing up volumes..." -ForegroundColor White
|
|
598
|
+
Write-Host " Backup directory: $backupDir"
|
|
599
|
+
Write-Host ""
|
|
600
|
+
|
|
601
|
+
# Get project name from docker compose
|
|
602
|
+
$projectName = "openclaw"
|
|
603
|
+
try {
|
|
604
|
+
$configJson = docker compose config --format json 2>$null | ConvertFrom-Json
|
|
605
|
+
if ($configJson.name) { $projectName = $configJson.name }
|
|
606
|
+
} catch {}
|
|
607
|
+
|
|
608
|
+
# List all volumes for this project
|
|
609
|
+
$volumes = docker volume ls --filter "name=$projectName" --format "{{.Name}}" 2>$null
|
|
610
|
+
if (-not $volumes) {
|
|
611
|
+
Write-Host " ! No volumes found for project: $projectName" -ForegroundColor Yellow
|
|
612
|
+
Write-Host " Trying to list all openclaw volumes..."
|
|
613
|
+
$volumes = docker volume ls --filter "name=openclaw" --format "{{.Name}}" 2>$null
|
|
614
|
+
}
|
|
615
|
+
|
|
616
|
+
if (-not $volumes) {
|
|
617
|
+
Write-Host " x No volumes found to back up." -ForegroundColor Red
|
|
618
|
+
exit 1
|
|
619
|
+
}
|
|
620
|
+
|
|
621
|
+
$backedUp = 0
|
|
622
|
+
foreach ($volume in $volumes) {
|
|
623
|
+
$vol = $volume.Trim()
|
|
624
|
+
if (-not $vol) { continue }
|
|
625
|
+
Write-Host " Backing up: $vol" -ForegroundColor Cyan
|
|
626
|
+
docker run --rm -v "\${vol}:/source:ro" -v "\${backupDir}:/backup" alpine tar czf "/backup/$vol.tar.gz" -C /source .
|
|
627
|
+
$size = (Get-Item (Join-Path $backupDir "$vol.tar.gz")).Length / 1MB
|
|
628
|
+
Write-Host " + $vol ($([math]::Round($size, 1))MB)" -ForegroundColor Green
|
|
629
|
+
$backedUp++
|
|
630
|
+
}
|
|
631
|
+
|
|
632
|
+
$totalSize = ((Get-ChildItem $backupDir -Recurse | Measure-Object -Property Length -Sum).Sum) / 1MB
|
|
633
|
+
Write-Host ""
|
|
634
|
+
Write-Host " + Backed up $backedUp volume(s) ($([math]::Round($totalSize, 1))MB total)" -ForegroundColor Green
|
|
635
|
+
Write-Host " Location: $backupDir"
|
|
636
|
+
`;
|
|
637
|
+
files["scripts/status.ps1"] = `#Requires -Version 5.1
|
|
638
|
+
<#
|
|
639
|
+
.SYNOPSIS
|
|
640
|
+
OpenClaw Status Script — shows the current status of all services.
|
|
641
|
+
#>
|
|
642
|
+
|
|
643
|
+
$ErrorActionPreference = 'Stop'
|
|
644
|
+
$ScriptDir = Split-Path -Parent $MyInvocation.MyCommand.Path
|
|
645
|
+
$ProjectDir = Split-Path -Parent $ScriptDir
|
|
646
|
+
Set-Location $ProjectDir
|
|
647
|
+
|
|
648
|
+
Write-Host ""
|
|
649
|
+
Write-Host "OpenClaw - Service Status" -ForegroundColor White
|
|
650
|
+
Write-Host ""
|
|
651
|
+
|
|
652
|
+
# Show compose status
|
|
653
|
+
docker compose ps
|
|
654
|
+
|
|
655
|
+
Write-Host ""
|
|
656
|
+
Write-Host "-- Resource Usage ----------------------------------------------------------"
|
|
657
|
+
Write-Host ""
|
|
658
|
+
|
|
659
|
+
try { docker compose top 2>$null } catch {}
|
|
660
|
+
|
|
661
|
+
Write-Host ""
|
|
662
|
+
Write-Host "-- Disk Usage --------------------------------------------------------------"
|
|
663
|
+
Write-Host ""
|
|
664
|
+
|
|
665
|
+
try { docker system df 2>$null } catch {}
|
|
666
|
+
|
|
667
|
+
Write-Host ""
|
|
668
|
+
Write-Host "-- Network -----------------------------------------------------------------"
|
|
669
|
+
Write-Host ""
|
|
670
|
+
|
|
671
|
+
try {
|
|
672
|
+
docker network ls --filter "name=openclaw" --format "table {{.Name}}\\t{{.Driver}}\\t{{.Scope}}" 2>$null
|
|
673
|
+
} catch {}
|
|
345
674
|
`;
|
|
346
675
|
return files;
|
|
347
676
|
}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"scripts.mjs","names":[],"sources":["../../src/generators/scripts.ts"],"sourcesContent":["/**\n * Generates helper shell scripts for managing the OpenClaw Docker Compose stack.\n *\n * Returns a map of file paths (relative to project root) to file contents.\n */\nexport function generateScripts(): Record<string, string> {\n\tconst files: Record<string, string> = {};\n\n\t// ── scripts/start.sh ────────────────────────────────────────────────────\n\n\tfiles[\"scripts/start.sh\"] = `#!/usr/bin/env bash\nset -euo pipefail\n\n# ─── OpenClaw Start Script ──────────────────────────────────────────────────\n# Production-quality bootstrap: validates prerequisites, auto-generates secrets,\n# creates required directories, and starts all services via Docker Compose.\n# Modelled after the official docker-setup.sh patterns.\n\nSCRIPT_DIR=\"$(cd \"$(dirname \"\\${BASH_SOURCE[0]}\")\" && pwd)\"\nPROJECT_DIR=\"$(dirname \"$SCRIPT_DIR\")\"\n\ncd \"$PROJECT_DIR\"\n\necho \"🐾 OpenClaw — Starting services...\"\necho \"\"\n\n# ── Colour helpers (no-op when not a TTY) ────────────────────────────────────\nif [ -t 1 ]; then\n RED='\\\\033[0;31m'; GREEN='\\\\033[0;32m'; YELLOW='\\\\033[1;33m'; CYAN='\\\\033[0;36m'; NC='\\\\033[0m'\nelse\n RED=''; GREEN=''; YELLOW=''; CYAN=''; NC=''\nfi\n\ninfo() { echo -e \"\\${CYAN}ℹ $*\\${NC}\"; }\nok() { echo -e \"\\${GREEN}✅ $*\\${NC}\"; }\nwarn() { echo -e \"\\${YELLOW}⚠️ $*\\${NC}\"; }\nerr() { echo -e \"\\${RED}❌ $*\\${NC}\" >&2; }\n\n# ── Prerequisite checks ─────────────────────────────────────────────────────\n\nif ! command -v docker &> /dev/null; then\n err \"Docker is not installed. Please install Docker first.\"\n echo \" https://docs.docker.com/get-docker/\"\n exit 1\nfi\n\nif ! docker compose version &> /dev/null; then\n err \"Docker Compose (v2) is not available.\"\n echo \" https://docs.docker.com/compose/install/\"\n exit 1\nfi\n\nif ! docker info &> /dev/null 2>&1; then\n err \"Docker daemon is not running. Please start Docker.\"\n exit 1\nfi\n\n# ── Source .env if it exists ─────────────────────────────────────────────────\n\nif [ -f \".env\" ]; then\n info \"Loading existing .env file...\"\n set -a\n # shellcheck disable=SC1091\n source .env\n set +a\nelse\n warn \".env file not found — will create one from .env.example if available.\"\n if [ -f \".env.example\" ]; then\n cp .env.example .env\n info \"Created .env from .env.example\"\n set -a\n # shellcheck disable=SC1091\n source .env\n set +a\n fi\nfi\n\n# ── Auto-generate OPENCLAW_GATEWAY_TOKEN if missing ──────────────────────────\n\nif [ -z \"\\${OPENCLAW_GATEWAY_TOKEN:-}\" ]; then\n info \"Generating OPENCLAW_GATEWAY_TOKEN...\"\n if command -v openssl &> /dev/null; then\n OPENCLAW_GATEWAY_TOKEN=\"$(openssl rand -hex 32)\"\n elif command -v python3 &> /dev/null; then\n OPENCLAW_GATEWAY_TOKEN=\"$(python3 -c 'import secrets; print(secrets.token_hex(32))')\"\n elif command -v python &> /dev/null; then\n OPENCLAW_GATEWAY_TOKEN=\"$(python -c 'import secrets; print(secrets.token_hex(32))')\"\n else\n err \"Cannot generate token: neither openssl nor python3 found.\"\n exit 1\n fi\n export OPENCLAW_GATEWAY_TOKEN\n\n # Persist into .env\n if [ -f \".env\" ]; then\n if grep -q \"^OPENCLAW_GATEWAY_TOKEN=\" .env 2>/dev/null; then\n sed -i.bak \"s|^OPENCLAW_GATEWAY_TOKEN=.*|OPENCLAW_GATEWAY_TOKEN=\\${OPENCLAW_GATEWAY_TOKEN}|\" .env && rm -f .env.bak\n else\n echo \"OPENCLAW_GATEWAY_TOKEN=\\${OPENCLAW_GATEWAY_TOKEN}\" >> .env\n fi\n fi\n ok \"Gateway token generated and saved.\"\nfi\n\n# ── Apply defaults for optional variables ────────────────────────────────────\n\nexport OPENCLAW_VERSION=\"\\${OPENCLAW_VERSION:-latest}\"\nexport OPENCLAW_GATEWAY_PORT=\"\\${OPENCLAW_GATEWAY_PORT:-18789}\"\nexport OPENCLAW_BRIDGE_PORT=\"\\${OPENCLAW_BRIDGE_PORT:-18790}\"\nexport OPENCLAW_GATEWAY_BIND=\"\\${OPENCLAW_GATEWAY_BIND:-lan}\"\nexport OPENCLAW_CONFIG_DIR=\"\\${OPENCLAW_CONFIG_DIR:-./openclaw/config}\"\nexport OPENCLAW_WORKSPACE_DIR=\"\\${OPENCLAW_WORKSPACE_DIR:-./openclaw/workspace}\"\n\n# ── Create required host directories ────────────────────────────────────────\n\ninfo \"Ensuring host directories exist...\"\nmkdir -p \"\\${OPENCLAW_CONFIG_DIR}\"\nmkdir -p \"\\${OPENCLAW_WORKSPACE_DIR}\"\nok \"Directories ready: \\${OPENCLAW_CONFIG_DIR}, \\${OPENCLAW_WORKSPACE_DIR}\"\n\n# ── Check for empty secret values ────────────────────────────────────────────\n\nEMPTY_SECRETS=0\nwhile IFS='=' read -r key value; do\n [[ \"$key\" =~ ^#.*$ ]] && continue\n [[ -z \"$key\" ]] && continue\n if [[ \"$key\" =~ (PASSWORD|TOKEN|SECRET|SESSION_KEY|COOKIE) ]] && [[ -z \"\\${value:-}\" ]]; then\n warn \"Warning: $key is empty in .env\"\n EMPTY_SECRETS=$((EMPTY_SECRETS + 1))\n fi\ndone < .env 2>/dev/null || true\n\nif [ \"$EMPTY_SECRETS\" -gt 0 ]; then\n echo \"\"\n warn \"$EMPTY_SECRETS secret(s) are empty. Some services may not function until they are set.\"\n echo \"\"\nfi\n\n# ── Pull and start ───────────────────────────────────────────────────────────\n\necho \"\"\ninfo \"Pulling latest images...\"\ndocker compose pull --quiet 2>/dev/null || docker compose pull\n\necho \"\"\ninfo \"Starting services...\"\ndocker compose up -d --remove-orphans\n\n# ── Health-check loop ────────────────────────────────────────────────────────\n\necho \"\"\ninfo \"Waiting for services to become healthy...\"\nsleep 5\n\nRETRIES=0\nMAX_RETRIES=30\nwhile [ $RETRIES -lt $MAX_RETRIES ]; do\n UNHEALTHY=$(docker compose ps --format json 2>/dev/null | grep -c '\"unhealthy\"' || true)\n STARTING=$(docker compose ps --format json 2>/dev/null | grep -c '\"starting\"' || true)\n\n if [ \"$UNHEALTHY\" -eq 0 ] && [ \"$STARTING\" -eq 0 ]; then\n break\n fi\n\n RETRIES=$((RETRIES + 1))\n sleep 2\ndone\n\necho \"\"\ndocker compose ps\necho \"\"\n\nif [ $RETRIES -ge $MAX_RETRIES ]; then\n warn \"Some services may still be starting. Check: docker compose ps\"\nelse\n ok \"All services are running!\"\nfi\n\n# ── Print service URLs & token ───────────────────────────────────────────────\n\nGATEWAY_HOST=\"localhost\"\nif [ \"\\${OPENCLAW_GATEWAY_BIND}\" = \"lan\" ]; then\n GATEWAY_HOST=\"0.0.0.0\"\nfi\n\necho \"\"\necho \"═══════════════════════════════════════════════════════════════════════════════\"\necho \" 🐾 OpenClaw is ready!\"\necho \"═══════════════════════════════════════════════════════════════════════════════\"\necho \"\"\necho \" Gateway URL: http://\\${GATEWAY_HOST}:\\${OPENCLAW_GATEWAY_PORT}\"\necho \" Bridge (WebSocket): ws://\\${GATEWAY_HOST}:\\${OPENCLAW_BRIDGE_PORT}\"\necho \" Config directory: \\${OPENCLAW_CONFIG_DIR}\"\necho \" Workspace directory:\\${OPENCLAW_WORKSPACE_DIR}\"\necho \"\"\necho \" Gateway Token: \\${OPENCLAW_GATEWAY_TOKEN}\"\necho \"\"\necho \" Manage:\"\necho \" Stop: ./scripts/stop.sh\"\necho \" Status: ./scripts/status.sh\"\necho \" Update: ./scripts/update.sh\"\necho \" Logs: docker compose logs -f\"\necho \"\"\necho \"═══════════════════════════════════════════════════════════════════════════════\"\n`;\n\n\t// ── scripts/stop.sh ─────────────────────────────────────────────────────\n\n\tfiles[\"scripts/stop.sh\"] = `#!/usr/bin/env bash\nset -euo pipefail\n\n# ─── OpenClaw Stop Script ───────────────────────────────────────────────────\n# Gracefully stops all services.\n\nSCRIPT_DIR=\"$(cd \"$(dirname \"\\${BASH_SOURCE[0]}\")\" && pwd)\"\nPROJECT_DIR=\"$(dirname \"$SCRIPT_DIR\")\"\n\ncd \"$PROJECT_DIR\"\n\necho \"🐾 OpenClaw — Stopping services...\"\necho \"\"\n\n# Graceful shutdown with timeout\ndocker compose down --timeout 30\n\necho \"\"\necho \"✅ All services stopped.\"\n`;\n\n\t// ── scripts/update.sh ───────────────────────────────────────────────────\n\n\tfiles[\"scripts/update.sh\"] = `#!/usr/bin/env bash\nset -euo pipefail\n\n# ─── OpenClaw Update Script ─────────────────────────────────────────────────\n# Pulls latest images and restarts services.\n\nSCRIPT_DIR=\"$(cd \"$(dirname \"\\${BASH_SOURCE[0]}\")\" && pwd)\"\nPROJECT_DIR=\"$(dirname \"$SCRIPT_DIR\")\"\n\ncd \"$PROJECT_DIR\"\n\necho \"🐾 OpenClaw — Updating services...\"\necho \"\"\n\n# Pull latest images\necho \"📦 Pulling latest images...\"\ndocker compose pull\n\necho \"\"\necho \"🔄 Restarting services with new images...\"\ndocker compose up -d --remove-orphans\n\necho \"\"\necho \"⏳ Waiting for services to stabilize...\"\nsleep 10\n\ndocker compose ps\n\necho \"\"\necho \"✅ Update complete!\"\n`;\n\n\t// ── scripts/backup.sh ───────────────────────────────────────────────────\n\n\tfiles[\"scripts/backup.sh\"] = `#!/usr/bin/env bash\nset -euo pipefail\n\n# ─── OpenClaw Backup Script ─────────────────────────────────────────────────\n# Backs up all named Docker volumes to a timestamped directory.\n\nSCRIPT_DIR=\"$(cd \"$(dirname \"\\${BASH_SOURCE[0]}\")\" && pwd)\"\nPROJECT_DIR=\"$(dirname \"$SCRIPT_DIR\")\"\n\ncd \"$PROJECT_DIR\"\n\nTIMESTAMP=$(date +%Y%m%d_%H%M%S)\nBACKUP_DIR=\"$PROJECT_DIR/backups/$TIMESTAMP\"\nmkdir -p \"$BACKUP_DIR\"\n\necho \"🐾 OpenClaw — Backing up volumes...\"\necho \" Backup directory: $BACKUP_DIR\"\necho \"\"\n\n# Get project name from docker compose\nPROJECT_NAME=$(docker compose config --format json 2>/dev/null | grep -o '\"name\":\"[^\"]*\"' | head -1 | cut -d'\"' -f4 || echo \"openclaw\")\n\n# List all volumes for this project\nVOLUMES=$(docker volume ls --filter \"name=\\${PROJECT_NAME}\" --format \"{{.Name}}\" 2>/dev/null || true)\n\nif [ -z \"$VOLUMES\" ]; then\n echo \"⚠️ No volumes found for project: $PROJECT_NAME\"\n echo \" Trying to list all openclaw volumes...\"\n VOLUMES=$(docker volume ls --filter \"name=openclaw\" --format \"{{.Name}}\" 2>/dev/null || true)\nfi\n\nif [ -z \"$VOLUMES\" ]; then\n echo \"❌ No volumes found to back up.\"\n exit 1\nfi\n\nBACKED_UP=0\nfor VOLUME in $VOLUMES; do\n echo \"📦 Backing up: $VOLUME\"\n docker run --rm \\\\\n -v \"\\${VOLUME}:/source:ro\" \\\\\n -v \"$BACKUP_DIR:/backup\" \\\\\n alpine tar czf \"/backup/\\${VOLUME}.tar.gz\" -C /source .\n\n SIZE=$(du -sh \"$BACKUP_DIR/\\${VOLUME}.tar.gz\" | cut -f1)\n echo \" ✓ $VOLUME ($SIZE)\"\n BACKED_UP=$((BACKED_UP + 1))\ndone\n\nTOTAL_SIZE=$(du -sh \"$BACKUP_DIR\" | cut -f1)\necho \"\"\necho \"✅ Backed up $BACKED_UP volume(s) ($TOTAL_SIZE total)\"\necho \" Location: $BACKUP_DIR\"\n`;\n\n\t// ── scripts/status.sh ───────────────────────────────────────────────────\n\n\tfiles[\"scripts/status.sh\"] = `#!/usr/bin/env bash\nset -euo pipefail\n\n# ─── OpenClaw Status Script ─────────────────────────────────────────────────\n# Shows the current status of all services.\n\nSCRIPT_DIR=\"$(cd \"$(dirname \"\\${BASH_SOURCE[0]}\")\" && pwd)\"\nPROJECT_DIR=\"$(dirname \"$SCRIPT_DIR\")\"\n\ncd \"$PROJECT_DIR\"\n\necho \"🐾 OpenClaw — Service Status\"\necho \"\"\n\n# Show compose status\ndocker compose ps\n\necho \"\"\necho \"── Resource Usage ──────────────────────────────────────────────────────\"\necho \"\"\n\n# Show resource usage\ndocker compose top 2>/dev/null || true\n\necho \"\"\necho \"── Disk Usage ─────────────────────────────────────────────────────────\"\necho \"\"\n\n# Show volume sizes\ndocker system df -v 2>/dev/null | head -30 || docker system df 2>/dev/null || true\n\necho \"\"\necho \"── Network ────────────────────────────────────────────────────────────\"\necho \"\"\n\ndocker network ls --filter \"name=openclaw\" --format \"table {{.Name}}\\\\t{{.Driver}}\\\\t{{.Scope}}\" 2>/dev/null || true\n`;\n\n\treturn files;\n}\n"],"mappings":";;;;;;AAKA,SAAgB,kBAA0C;CACzD,MAAM,QAAgC,EAAE;AAIxC,OAAM,sBAAsB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAsM5B,OAAM,qBAAqB;;;;;;;;;;;;;;;;;;;;AAuB3B,OAAM,uBAAuB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAkC7B,OAAM,uBAAuB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAyD7B,OAAM,uBAAuB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAsC7B,QAAO"}
|
|
1
|
+
{"version":3,"file":"scripts.mjs","names":[],"sources":["../../src/generators/scripts.ts"],"sourcesContent":["/**\n * Generates helper shell scripts for managing the OpenClaw Docker Compose stack.\n *\n * Returns a map of file paths (relative to project root) to file contents.\n */\nexport function generateScripts(options?: { hasGitServices?: boolean }): Record<string, string> {\n\tconst hasGit = options?.hasGitServices ?? false;\n\tconst files: Record<string, string> = {};\n\n\t// ── scripts/start.sh ────────────────────────────────────────────────────\n\n\tfiles[\"scripts/start.sh\"] = `#!/usr/bin/env bash\nset -euo pipefail\n\n# ─── OpenClaw Start Script ──────────────────────────────────────────────────\n# Production-quality bootstrap: validates prerequisites, auto-generates secrets,\n# creates required directories, and starts all services via Docker Compose.\n# Modelled after the official docker-setup.sh patterns.\n\nSCRIPT_DIR=\"$(cd \"$(dirname \"\\${BASH_SOURCE[0]}\")\" && pwd)\"\nPROJECT_DIR=\"$(dirname \"$SCRIPT_DIR\")\"\n\ncd \"$PROJECT_DIR\"\n\necho \"🐾 OpenClaw — Starting services...\"\necho \"\"\n\n# ── Colour helpers (no-op when not a TTY) ────────────────────────────────────\nif [ -t 1 ]; then\n RED='\\\\033[0;31m'; GREEN='\\\\033[0;32m'; YELLOW='\\\\033[1;33m'; CYAN='\\\\033[0;36m'; NC='\\\\033[0m'\nelse\n RED=''; GREEN=''; YELLOW=''; CYAN=''; NC=''\nfi\n\ninfo() { echo -e \"\\${CYAN}ℹ $*\\${NC}\"; }\nok() { echo -e \"\\${GREEN}✅ $*\\${NC}\"; }\nwarn() { echo -e \"\\${YELLOW}⚠️ $*\\${NC}\"; }\nerr() { echo -e \"\\${RED}❌ $*\\${NC}\" >&2; }\n\n# ── Prerequisite checks ─────────────────────────────────────────────────────\n\nif ! command -v docker &> /dev/null; then\n err \"Docker is not installed. Please install Docker first.\"\n echo \" https://docs.docker.com/get-docker/\"\n exit 1\nfi\n\nif ! docker compose version &> /dev/null; then\n err \"Docker Compose (v2) is not available.\"\n echo \" https://docs.docker.com/compose/install/\"\n exit 1\nfi\n\nif ! docker info &> /dev/null 2>&1; then\n err \"Docker daemon is not running. Please start Docker.\"\n exit 1\nfi\n\n# ── Source .env if it exists ─────────────────────────────────────────────────\n\nif [ -f \".env\" ]; then\n info \"Loading existing .env file...\"\n set -a\n # shellcheck disable=SC1091\n source .env\n set +a\nelse\n warn \".env file not found — will create one from .env.example if available.\"\n if [ -f \".env.example\" ]; then\n cp .env.example .env\n info \"Created .env from .env.example\"\n set -a\n # shellcheck disable=SC1091\n source .env\n set +a\n fi\nfi\n\n# ── Auto-generate OPENCLAW_GATEWAY_TOKEN if missing ──────────────────────────\n\nif [ -z \"\\${OPENCLAW_GATEWAY_TOKEN:-}\" ]; then\n info \"Generating OPENCLAW_GATEWAY_TOKEN...\"\n if command -v openssl &> /dev/null; then\n OPENCLAW_GATEWAY_TOKEN=\"$(openssl rand -hex 32)\"\n elif command -v python3 &> /dev/null; then\n OPENCLAW_GATEWAY_TOKEN=\"$(python3 -c 'import secrets; print(secrets.token_hex(32))')\"\n elif command -v python &> /dev/null; then\n OPENCLAW_GATEWAY_TOKEN=\"$(python -c 'import secrets; print(secrets.token_hex(32))')\"\n else\n err \"Cannot generate token: neither openssl nor python3 found.\"\n exit 1\n fi\n export OPENCLAW_GATEWAY_TOKEN\n\n # Persist into .env\n if [ -f \".env\" ]; then\n if grep -q \"^OPENCLAW_GATEWAY_TOKEN=\" .env 2>/dev/null; then\n sed -i.bak \"s|^OPENCLAW_GATEWAY_TOKEN=.*|OPENCLAW_GATEWAY_TOKEN=\\${OPENCLAW_GATEWAY_TOKEN}|\" .env && rm -f .env.bak\n else\n echo \"OPENCLAW_GATEWAY_TOKEN=\\${OPENCLAW_GATEWAY_TOKEN}\" >> .env\n fi\n fi\n ok \"Gateway token generated and saved.\"\nfi\n\n# ── Apply defaults for optional variables ────────────────────────────────────\n\nexport OPENCLAW_VERSION=\"\\${OPENCLAW_VERSION:-latest}\"\nexport OPENCLAW_GATEWAY_PORT=\"\\${OPENCLAW_GATEWAY_PORT:-18789}\"\nexport OPENCLAW_BRIDGE_PORT=\"\\${OPENCLAW_BRIDGE_PORT:-18790}\"\nexport OPENCLAW_GATEWAY_BIND=\"\\${OPENCLAW_GATEWAY_BIND:-lan}\"\nexport OPENCLAW_CONFIG_DIR=\"\\${OPENCLAW_CONFIG_DIR:-./openclaw/config}\"\nexport OPENCLAW_WORKSPACE_DIR=\"\\${OPENCLAW_WORKSPACE_DIR:-./openclaw/workspace}\"\n\n# ── Create required host directories ────────────────────────────────────────\n\ninfo \"Ensuring host directories exist...\"\nmkdir -p \"\\${OPENCLAW_CONFIG_DIR}\"\nmkdir -p \"\\${OPENCLAW_WORKSPACE_DIR}\"\nok \"Directories ready: \\${OPENCLAW_CONFIG_DIR}, \\${OPENCLAW_WORKSPACE_DIR}\"\n\n# ── Check for empty secret values ────────────────────────────────────────────\n\nEMPTY_SECRETS=0\nwhile IFS='=' read -r key value; do\n [[ \"$key\" =~ ^#.*$ ]] && continue\n [[ -z \"$key\" ]] && continue\n if [[ \"$key\" =~ (PASSWORD|TOKEN|SECRET|SESSION_KEY|COOKIE) ]] && [[ -z \"\\${value:-}\" ]]; then\n warn \"Warning: $key is empty in .env\"\n EMPTY_SECRETS=$((EMPTY_SECRETS + 1))\n fi\ndone < .env 2>/dev/null || true\n\nif [ \"$EMPTY_SECRETS\" -gt 0 ]; then\n echo \"\"\n warn \"$EMPTY_SECRETS secret(s) are empty. Some services may not function until they are set.\"\n echo \"\"\nfi\n\n# ── Clone git-based repositories (if any) ────────────────────────────────────\n\nif [ -f \"$SCRIPT_DIR/clone-repos.sh\" ]; then\n info \"Cloning/updating SaaS boilerplate repositories...\"\n bash \"$SCRIPT_DIR/clone-repos.sh\"\n ok \"Repositories ready.\"\nfi\n\n# ── Pull and start ───────────────────────────────────────────────────────────\n\necho \"\"\ninfo \"Pulling latest images...\"\ndocker compose pull --quiet 2>/dev/null || docker compose pull\n\necho \"\"\ninfo \"Starting services...\"\ndocker compose up -d --remove-orphans${hasGit ? \" --build\" : \"\"}\n\n# ── Health-check loop ────────────────────────────────────────────────────────\n\necho \"\"\ninfo \"Waiting for services to become healthy...\"\nsleep 5\n\nRETRIES=0\nMAX_RETRIES=30\nwhile [ $RETRIES -lt $MAX_RETRIES ]; do\n UNHEALTHY=$(docker compose ps --format json 2>/dev/null | grep -c '\"unhealthy\"' || true)\n STARTING=$(docker compose ps --format json 2>/dev/null | grep -c '\"starting\"' || true)\n\n if [ \"$UNHEALTHY\" -eq 0 ] && [ \"$STARTING\" -eq 0 ]; then\n break\n fi\n\n RETRIES=$((RETRIES + 1))\n sleep 2\ndone\n\necho \"\"\ndocker compose ps\necho \"\"\n\nif [ $RETRIES -ge $MAX_RETRIES ]; then\n warn \"Some services may still be starting. Check: docker compose ps\"\nelse\n ok \"All services are running!\"\nfi\n\n# ── Print service URLs & token ───────────────────────────────────────────────\n\nGATEWAY_HOST=\"localhost\"\nif [ \"\\${OPENCLAW_GATEWAY_BIND}\" = \"lan\" ]; then\n GATEWAY_HOST=\"0.0.0.0\"\nfi\n\necho \"\"\necho \"═══════════════════════════════════════════════════════════════════════════════\"\necho \" 🐾 OpenClaw is ready!\"\necho \"═══════════════════════════════════════════════════════════════════════════════\"\necho \"\"\necho \" Gateway URL: http://\\${GATEWAY_HOST}:\\${OPENCLAW_GATEWAY_PORT}\"\necho \" Bridge (WebSocket): ws://\\${GATEWAY_HOST}:\\${OPENCLAW_BRIDGE_PORT}\"\necho \" Config directory: \\${OPENCLAW_CONFIG_DIR}\"\necho \" Workspace directory:\\${OPENCLAW_WORKSPACE_DIR}\"\necho \"\"\necho \" Gateway Token: \\${OPENCLAW_GATEWAY_TOKEN}\"\necho \"\"\necho \" Manage:\"\necho \" Stop: ./scripts/stop.sh\"\necho \" Status: ./scripts/status.sh\"\necho \" Update: ./scripts/update.sh\"\necho \" Logs: docker compose logs -f\"\necho \"\"\necho \"═══════════════════════════════════════════════════════════════════════════════\"\n`;\n\n\t// ── scripts/stop.sh ─────────────────────────────────────────────────────\n\n\tfiles[\"scripts/stop.sh\"] = `#!/usr/bin/env bash\nset -euo pipefail\n\n# ─── OpenClaw Stop Script ───────────────────────────────────────────────────\n# Gracefully stops all services.\n\nSCRIPT_DIR=\"$(cd \"$(dirname \"\\${BASH_SOURCE[0]}\")\" && pwd)\"\nPROJECT_DIR=\"$(dirname \"$SCRIPT_DIR\")\"\n\ncd \"$PROJECT_DIR\"\n\necho \"🐾 OpenClaw — Stopping services...\"\necho \"\"\n\n# Graceful shutdown with timeout\ndocker compose down --timeout 30\n\necho \"\"\necho \"✅ All services stopped.\"\n`;\n\n\t// ── scripts/update.sh ───────────────────────────────────────────────────\n\n\tfiles[\"scripts/update.sh\"] = `#!/usr/bin/env bash\nset -euo pipefail\n\n# ─── OpenClaw Update Script ─────────────────────────────────────────────────\n# Pulls latest images and restarts services.\n\nSCRIPT_DIR=\"$(cd \"$(dirname \"\\${BASH_SOURCE[0]}\")\" && pwd)\"\nPROJECT_DIR=\"$(dirname \"$SCRIPT_DIR\")\"\n\ncd \"$PROJECT_DIR\"\n\necho \"🐾 OpenClaw — Updating services...\"\necho \"\"\n\n# Update git-based repositories (if any)\nif [ -f \"$SCRIPT_DIR/clone-repos.sh\" ]; then\n echo \"📂 Updating SaaS boilerplate repositories...\"\n bash \"$SCRIPT_DIR/clone-repos.sh\"\nfi\n\n# Pull latest images\necho \"📦 Pulling latest images...\"\ndocker compose pull\n\necho \"\"\necho \"🔄 Restarting services with new images...\"\ndocker compose up -d --remove-orphans${hasGit ? \" --build\" : \"\"}\n\necho \"\"\necho \"⏳ Waiting for services to stabilize...\"\nsleep 10\n\ndocker compose ps\n\necho \"\"\necho \"✅ Update complete!\"\n`;\n\n\t// ── scripts/backup.sh ───────────────────────────────────────────────────\n\n\tfiles[\"scripts/backup.sh\"] = `#!/usr/bin/env bash\nset -euo pipefail\n\n# ─── OpenClaw Backup Script ─────────────────────────────────────────────────\n# Backs up all named Docker volumes to a timestamped directory.\n\nSCRIPT_DIR=\"$(cd \"$(dirname \"\\${BASH_SOURCE[0]}\")\" && pwd)\"\nPROJECT_DIR=\"$(dirname \"$SCRIPT_DIR\")\"\n\ncd \"$PROJECT_DIR\"\n\nTIMESTAMP=$(date +%Y%m%d_%H%M%S)\nBACKUP_DIR=\"$PROJECT_DIR/backups/$TIMESTAMP\"\nmkdir -p \"$BACKUP_DIR\"\n\necho \"🐾 OpenClaw — Backing up volumes...\"\necho \" Backup directory: $BACKUP_DIR\"\necho \"\"\n\n# Get project name from docker compose\nPROJECT_NAME=$(docker compose config --format json 2>/dev/null | grep -o '\"name\":\"[^\"]*\"' | head -1 | cut -d'\"' -f4 || echo \"openclaw\")\n\n# List all volumes for this project\nVOLUMES=$(docker volume ls --filter \"name=\\${PROJECT_NAME}\" --format \"{{.Name}}\" 2>/dev/null || true)\n\nif [ -z \"$VOLUMES\" ]; then\n echo \"⚠️ No volumes found for project: $PROJECT_NAME\"\n echo \" Trying to list all openclaw volumes...\"\n VOLUMES=$(docker volume ls --filter \"name=openclaw\" --format \"{{.Name}}\" 2>/dev/null || true)\nfi\n\nif [ -z \"$VOLUMES\" ]; then\n echo \"❌ No volumes found to back up.\"\n exit 1\nfi\n\nBACKED_UP=0\nfor VOLUME in $VOLUMES; do\n echo \"📦 Backing up: $VOLUME\"\n docker run --rm \\\\\n -v \"\\${VOLUME}:/source:ro\" \\\\\n -v \"$BACKUP_DIR:/backup\" \\\\\n alpine tar czf \"/backup/\\${VOLUME}.tar.gz\" -C /source .\n\n SIZE=$(du -sh \"$BACKUP_DIR/\\${VOLUME}.tar.gz\" | cut -f1)\n echo \" ✓ $VOLUME ($SIZE)\"\n BACKED_UP=$((BACKED_UP + 1))\ndone\n\nTOTAL_SIZE=$(du -sh \"$BACKUP_DIR\" | cut -f1)\necho \"\"\necho \"✅ Backed up $BACKED_UP volume(s) ($TOTAL_SIZE total)\"\necho \" Location: $BACKUP_DIR\"\n`;\n\n\t// ── scripts/status.sh ───────────────────────────────────────────────────\n\n\tfiles[\"scripts/status.sh\"] = `#!/usr/bin/env bash\nset -euo pipefail\n\n# ─── OpenClaw Status Script ─────────────────────────────────────────────────\n# Shows the current status of all services.\n\nSCRIPT_DIR=\"$(cd \"$(dirname \"\\${BASH_SOURCE[0]}\")\" && pwd)\"\nPROJECT_DIR=\"$(dirname \"$SCRIPT_DIR\")\"\n\ncd \"$PROJECT_DIR\"\n\necho \"🐾 OpenClaw — Service Status\"\necho \"\"\n\n# Show compose status\ndocker compose ps\n\necho \"\"\necho \"── Resource Usage ──────────────────────────────────────────────────────\"\necho \"\"\n\n# Show resource usage\ndocker compose top 2>/dev/null || true\n\necho \"\"\necho \"── Disk Usage ─────────────────────────────────────────────────────────\"\necho \"\"\n\n# Show volume sizes\ndocker system df -v 2>/dev/null | head -30 || docker system df 2>/dev/null || true\n\necho \"\"\necho \"── Network ────────────────────────────────────────────────────────────\"\necho \"\"\n\ndocker network ls --filter \"name=openclaw\" --format \"table {{.Name}}\\\\t{{.Driver}}\\\\t{{.Scope}}\" 2>/dev/null || true\n`;\n\n\t// ═══════════════════════════════════════════════════════════════════════\n\t// PowerShell equivalents\n\t// ═══════════════════════════════════════════════════════════════════════\n\n\t// ── scripts/start.ps1 ───────────────────────────────────────────────────\n\n\tfiles[\"scripts/start.ps1\"] = `#Requires -Version 5.1\n<#\n.SYNOPSIS\n OpenClaw Start Script — validates prerequisites, auto-generates secrets,\n creates required directories, and starts all services via Docker Compose.\n#>\n\n$ErrorActionPreference = 'Stop'\n$ScriptDir = Split-Path -Parent $MyInvocation.MyCommand.Path\n$ProjectDir = Split-Path -Parent $ScriptDir\nSet-Location $ProjectDir\n\nfunction Info { param($Msg) Write-Host \" i $Msg\" -ForegroundColor Cyan }\nfunction Ok { param($Msg) Write-Host \" + $Msg\" -ForegroundColor Green }\nfunction Warn { param($Msg) Write-Host \" ! $Msg\" -ForegroundColor Yellow }\nfunction Err { param($Msg) Write-Host \" x $Msg\" -ForegroundColor Red }\n\nWrite-Host \"\"\nWrite-Host \"OpenClaw - Starting services...\" -ForegroundColor White\nWrite-Host \"\"\n\n# ── Prerequisite checks ─────────────────────────────────────────────────\nif (-not (Get-Command docker -ErrorAction SilentlyContinue)) {\n Err \"Docker is not installed. Please install Docker Desktop.\"\n Write-Host \" https://docs.docker.com/desktop/install/windows-install/\"\n exit 1\n}\n\n$composeCheck = docker compose version 2>&1\nif ($LASTEXITCODE -ne 0) {\n Err \"Docker Compose (v2) is not available.\"\n Write-Host \" https://docs.docker.com/compose/install/\"\n exit 1\n}\n\n$dockerInfo = docker info 2>&1\nif ($LASTEXITCODE -ne 0) {\n Err \"Docker daemon is not running. Please start Docker Desktop.\"\n exit 1\n}\n\n# ── Source .env if it exists ─────────────────────────────────────────────\nif (Test-Path \".env\") {\n Info \"Loading existing .env file...\"\n Get-Content \".env\" | ForEach-Object {\n if ($_ -match '^([^#=]+)=(.*)$') {\n [Environment]::SetEnvironmentVariable($Matches[1].Trim(), $Matches[2].Trim(), 'Process')\n }\n }\n} else {\n Warn \".env file not found - will create one from .env.example if available.\"\n if (Test-Path \".env.example\") {\n Copy-Item \".env.example\" \".env\"\n Info \"Created .env from .env.example\"\n Get-Content \".env\" | ForEach-Object {\n if ($_ -match '^([^#=]+)=(.*)$') {\n [Environment]::SetEnvironmentVariable($Matches[1].Trim(), $Matches[2].Trim(), 'Process')\n }\n }\n }\n}\n\n# ── Auto-generate OPENCLAW_GATEWAY_TOKEN if missing ──────────────────────\nif (-not $env:OPENCLAW_GATEWAY_TOKEN) {\n Info \"Generating OPENCLAW_GATEWAY_TOKEN...\"\n $bytes = New-Object byte[] 32\n [System.Security.Cryptography.RandomNumberGenerator]::Create().GetBytes($bytes)\n $env:OPENCLAW_GATEWAY_TOKEN = ($bytes | ForEach-Object { $_.ToString(\"x2\") }) -join ''\n\n if (Test-Path \".env\") {\n $envContent = Get-Content \".env\" -Raw\n if ($envContent -match '(?m)^OPENCLAW_GATEWAY_TOKEN=') {\n $envContent = $envContent -replace '(?m)^OPENCLAW_GATEWAY_TOKEN=.*', \"OPENCLAW_GATEWAY_TOKEN=$($env:OPENCLAW_GATEWAY_TOKEN)\"\n Set-Content \".env\" $envContent -NoNewline\n } else {\n Add-Content \".env\" \"OPENCLAW_GATEWAY_TOKEN=$($env:OPENCLAW_GATEWAY_TOKEN)\"\n }\n }\n Ok \"Gateway token generated and saved.\"\n}\n\n# ── Apply defaults ───────────────────────────────────────────────────────\nif (-not $env:OPENCLAW_VERSION) { $env:OPENCLAW_VERSION = \"latest\" }\nif (-not $env:OPENCLAW_GATEWAY_PORT) { $env:OPENCLAW_GATEWAY_PORT = \"18789\" }\nif (-not $env:OPENCLAW_BRIDGE_PORT) { $env:OPENCLAW_BRIDGE_PORT = \"18790\" }\nif (-not $env:OPENCLAW_GATEWAY_BIND) { $env:OPENCLAW_GATEWAY_BIND = \"lan\" }\nif (-not $env:OPENCLAW_CONFIG_DIR) { $env:OPENCLAW_CONFIG_DIR = \"./openclaw/config\" }\nif (-not $env:OPENCLAW_WORKSPACE_DIR) { $env:OPENCLAW_WORKSPACE_DIR = \"./openclaw/workspace\" }\n\n# ── Create required host directories ────────────────────────────────────\nInfo \"Ensuring host directories exist...\"\nNew-Item -ItemType Directory -Force -Path $env:OPENCLAW_CONFIG_DIR | Out-Null\nNew-Item -ItemType Directory -Force -Path $env:OPENCLAW_WORKSPACE_DIR | Out-Null\nOk \"Directories ready: $($env:OPENCLAW_CONFIG_DIR), $($env:OPENCLAW_WORKSPACE_DIR)\"\n\n# ── Clone git-based repositories (if any) ─────────────────────────────\n$cloneScript = Join-Path $ScriptDir \"clone-repos.ps1\"\nif (Test-Path $cloneScript) {\n Info \"Cloning/updating SaaS boilerplate repositories...\"\n & $cloneScript\n Ok \"Repositories ready.\"\n}\n\n# ── Pull and start ───────────────────────────────────────────────────────\nWrite-Host \"\"\nInfo \"Pulling latest images...\"\ndocker compose pull 2>$null\nif ($LASTEXITCODE -ne 0) { docker compose pull }\n\nWrite-Host \"\"\nInfo \"Starting services...\"\ndocker compose up -d --remove-orphans${hasGit ? \" --build\" : \"\"}\n\n# ── Health-check loop ────────────────────────────────────────────────────\nWrite-Host \"\"\nInfo \"Waiting for services to become healthy...\"\nStart-Sleep -Seconds 5\n\n$retries = 0\n$maxRetries = 30\nwhile ($retries -lt $maxRetries) {\n $psOutput = docker compose ps --format json 2>$null\n $unhealthy = ($psOutput | Select-String -Pattern '\"unhealthy\"' -SimpleMatch).Count\n $starting = ($psOutput | Select-String -Pattern '\"starting\"' -SimpleMatch).Count\n if ($unhealthy -eq 0 -and $starting -eq 0) { break }\n $retries++\n Start-Sleep -Seconds 2\n}\n\nWrite-Host \"\"\ndocker compose ps\nWrite-Host \"\"\n\nif ($retries -ge $maxRetries) {\n Warn \"Some services may still be starting. Check: docker compose ps\"\n} else {\n Ok \"All services are running!\"\n}\n\n# ── Print service URLs & token ───────────────────────────────────────────\n$gatewayHost = if ($env:OPENCLAW_GATEWAY_BIND -eq \"lan\") { \"0.0.0.0\" } else { \"localhost\" }\n\nWrite-Host \"\"\nWrite-Host \"===============================================================================\"\nWrite-Host \" OpenClaw is ready!\"\nWrite-Host \"===============================================================================\"\nWrite-Host \"\"\nWrite-Host \" Gateway URL: http://\\${gatewayHost}:$($env:OPENCLAW_GATEWAY_PORT)\"\nWrite-Host \" Bridge (WebSocket): ws://\\${gatewayHost}:$($env:OPENCLAW_BRIDGE_PORT)\"\nWrite-Host \" Config directory: $($env:OPENCLAW_CONFIG_DIR)\"\nWrite-Host \" Workspace directory: $($env:OPENCLAW_WORKSPACE_DIR)\"\nWrite-Host \"\"\nWrite-Host \" Gateway Token: $($env:OPENCLAW_GATEWAY_TOKEN)\"\nWrite-Host \"\"\nWrite-Host \" Manage:\"\nWrite-Host \" Stop: .\\\\scripts\\\\stop.ps1\"\nWrite-Host \" Status: .\\\\scripts\\\\status.ps1\"\nWrite-Host \" Update: .\\\\scripts\\\\update.ps1\"\nWrite-Host \" Logs: docker compose logs -f\"\nWrite-Host \"\"\nWrite-Host \"===============================================================================\"\n`;\n\n\t// ── scripts/stop.ps1 ────────────────────────────────────────────────────\n\n\tfiles[\"scripts/stop.ps1\"] = `#Requires -Version 5.1\n<#\n.SYNOPSIS\n OpenClaw Stop Script — gracefully stops all services.\n#>\n\n$ErrorActionPreference = 'Stop'\n$ScriptDir = Split-Path -Parent $MyInvocation.MyCommand.Path\n$ProjectDir = Split-Path -Parent $ScriptDir\nSet-Location $ProjectDir\n\nWrite-Host \"\"\nWrite-Host \"OpenClaw - Stopping services...\" -ForegroundColor White\nWrite-Host \"\"\n\ndocker compose down --timeout 30\n\nWrite-Host \"\"\nWrite-Host \" + All services stopped.\" -ForegroundColor Green\n`;\n\n\t// ── scripts/update.ps1 ──────────────────────────────────────────────────\n\n\tfiles[\"scripts/update.ps1\"] = `#Requires -Version 5.1\n<#\n.SYNOPSIS\n OpenClaw Update Script — pulls latest images and restarts services.\n#>\n\n$ErrorActionPreference = 'Stop'\n$ScriptDir = Split-Path -Parent $MyInvocation.MyCommand.Path\n$ProjectDir = Split-Path -Parent $ScriptDir\nSet-Location $ProjectDir\n\nWrite-Host \"\"\nWrite-Host \"OpenClaw - Updating services...\" -ForegroundColor White\nWrite-Host \"\"\n\n# Update git-based repositories (if any)\n$cloneScript = Join-Path $ScriptDir \"clone-repos.ps1\"\nif (Test-Path $cloneScript) {\n Write-Host \" Updating SaaS boilerplate repositories...\" -ForegroundColor Cyan\n & $cloneScript\n}\n\nWrite-Host \" Pulling latest images...\" -ForegroundColor Cyan\ndocker compose pull\n\nWrite-Host \"\"\nWrite-Host \" Restarting services with new images...\" -ForegroundColor Cyan\ndocker compose up -d --remove-orphans${hasGit ? \" --build\" : \"\"}\n\nWrite-Host \"\"\nWrite-Host \" Waiting for services to stabilize...\" -ForegroundColor Cyan\nStart-Sleep -Seconds 10\n\ndocker compose ps\n\nWrite-Host \"\"\nWrite-Host \" + Update complete!\" -ForegroundColor Green\n`;\n\n\t// ── scripts/backup.ps1 ──────────────────────────────────────────────────\n\n\tfiles[\"scripts/backup.ps1\"] = `#Requires -Version 5.1\n<#\n.SYNOPSIS\n OpenClaw Backup Script — backs up all named Docker volumes to a timestamped directory.\n#>\n\n$ErrorActionPreference = 'Stop'\n$ScriptDir = Split-Path -Parent $MyInvocation.MyCommand.Path\n$ProjectDir = Split-Path -Parent $ScriptDir\nSet-Location $ProjectDir\n\n$timestamp = Get-Date -Format \"yyyyMMdd_HHmmss\"\n$backupDir = Join-Path $ProjectDir \"backups\\\\$timestamp\"\nNew-Item -ItemType Directory -Force -Path $backupDir | Out-Null\n\nWrite-Host \"\"\nWrite-Host \"OpenClaw - Backing up volumes...\" -ForegroundColor White\nWrite-Host \" Backup directory: $backupDir\"\nWrite-Host \"\"\n\n# Get project name from docker compose\n$projectName = \"openclaw\"\ntry {\n $configJson = docker compose config --format json 2>$null | ConvertFrom-Json\n if ($configJson.name) { $projectName = $configJson.name }\n} catch {}\n\n# List all volumes for this project\n$volumes = docker volume ls --filter \"name=$projectName\" --format \"{{.Name}}\" 2>$null\nif (-not $volumes) {\n Write-Host \" ! No volumes found for project: $projectName\" -ForegroundColor Yellow\n Write-Host \" Trying to list all openclaw volumes...\"\n $volumes = docker volume ls --filter \"name=openclaw\" --format \"{{.Name}}\" 2>$null\n}\n\nif (-not $volumes) {\n Write-Host \" x No volumes found to back up.\" -ForegroundColor Red\n exit 1\n}\n\n$backedUp = 0\nforeach ($volume in $volumes) {\n $vol = $volume.Trim()\n if (-not $vol) { continue }\n Write-Host \" Backing up: $vol\" -ForegroundColor Cyan\n docker run --rm -v \"\\${vol}:/source:ro\" -v \"\\${backupDir}:/backup\" alpine tar czf \"/backup/$vol.tar.gz\" -C /source .\n $size = (Get-Item (Join-Path $backupDir \"$vol.tar.gz\")).Length / 1MB\n Write-Host \" + $vol ($([math]::Round($size, 1))MB)\" -ForegroundColor Green\n $backedUp++\n}\n\n$totalSize = ((Get-ChildItem $backupDir -Recurse | Measure-Object -Property Length -Sum).Sum) / 1MB\nWrite-Host \"\"\nWrite-Host \" + Backed up $backedUp volume(s) ($([math]::Round($totalSize, 1))MB total)\" -ForegroundColor Green\nWrite-Host \" Location: $backupDir\"\n`;\n\n\t// ── scripts/status.ps1 ──────────────────────────────────────────────────\n\n\tfiles[\"scripts/status.ps1\"] = `#Requires -Version 5.1\n<#\n.SYNOPSIS\n OpenClaw Status Script — shows the current status of all services.\n#>\n\n$ErrorActionPreference = 'Stop'\n$ScriptDir = Split-Path -Parent $MyInvocation.MyCommand.Path\n$ProjectDir = Split-Path -Parent $ScriptDir\nSet-Location $ProjectDir\n\nWrite-Host \"\"\nWrite-Host \"OpenClaw - Service Status\" -ForegroundColor White\nWrite-Host \"\"\n\n# Show compose status\ndocker compose ps\n\nWrite-Host \"\"\nWrite-Host \"-- Resource Usage ----------------------------------------------------------\"\nWrite-Host \"\"\n\ntry { docker compose top 2>$null } catch {}\n\nWrite-Host \"\"\nWrite-Host \"-- Disk Usage --------------------------------------------------------------\"\nWrite-Host \"\"\n\ntry { docker system df 2>$null } catch {}\n\nWrite-Host \"\"\nWrite-Host \"-- Network -----------------------------------------------------------------\"\nWrite-Host \"\"\n\ntry {\n docker network ls --filter \"name=openclaw\" --format \"table {{.Name}}\\\\t{{.Driver}}\\\\t{{.Scope}}\" 2>$null\n} catch {}\n`;\n\n\treturn files;\n}\n"],"mappings":";;;;;;AAKA,SAAgB,gBAAgB,SAAgE;CAC/F,MAAM,SAAS,SAAS,kBAAkB;CAC1C,MAAM,QAAgC,EAAE;AAIxC,OAAM,sBAAsB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;uCAgJU,SAAS,aAAa,GAAG;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AA8D/D,OAAM,qBAAqB;;;;;;;;;;;;;;;;;;;;AAuB3B,OAAM,uBAAuB;;;;;;;;;;;;;;;;;;;;;;;;;;uCA0BS,SAAS,aAAa,GAAG;;;;;;;;;;;AAc/D,OAAM,uBAAuB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAyD7B,OAAM,uBAAuB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AA4C7B,OAAM,uBAAuB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;uCA+GS,SAAS,aAAa,GAAG;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAsD/D,OAAM,sBAAsB;;;;;;;;;;;;;;;;;;;;AAuB5B,OAAM,wBAAwB;;;;;;;;;;;;;;;;;;;;;;;;;;;uCA2BQ,SAAS,aAAa,GAAG;;;;;;;;;;;AAc/D,OAAM,wBAAwB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AA2D9B,OAAM,wBAAwB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAuC9B,QAAO"}
|
|
@@ -1,48 +1,82 @@
|
|
|
1
|
-
const
|
|
1
|
+
const require_test_CTcmp4Su = require("../test.CTcmp4Su-DlzTarwH.cjs");
|
|
2
2
|
const require_generators_scripts = require("./scripts.cjs");
|
|
3
3
|
//#region src/generators/scripts.test.ts
|
|
4
|
-
|
|
5
|
-
|
|
4
|
+
require_test_CTcmp4Su.describe("generateScripts", () => {
|
|
5
|
+
require_test_CTcmp4Su.it("generates all 10 expected scripts (5 bash + 5 PowerShell)", () => {
|
|
6
6
|
const result = require_generators_scripts.generateScripts();
|
|
7
7
|
for (const script of [
|
|
8
8
|
"scripts/start.sh",
|
|
9
9
|
"scripts/stop.sh",
|
|
10
10
|
"scripts/update.sh",
|
|
11
11
|
"scripts/backup.sh",
|
|
12
|
-
"scripts/status.sh"
|
|
12
|
+
"scripts/status.sh",
|
|
13
|
+
"scripts/start.ps1",
|
|
14
|
+
"scripts/stop.ps1",
|
|
15
|
+
"scripts/update.ps1",
|
|
16
|
+
"scripts/backup.ps1",
|
|
17
|
+
"scripts/status.ps1"
|
|
13
18
|
]) {
|
|
14
|
-
|
|
15
|
-
|
|
19
|
+
require_test_CTcmp4Su.globalExpect(result).toHaveProperty(script);
|
|
20
|
+
require_test_CTcmp4Su.globalExpect(result[script].length).toBeGreaterThan(0);
|
|
16
21
|
}
|
|
17
22
|
});
|
|
18
|
-
|
|
23
|
+
require_test_CTcmp4Su.it("start.sh calls docker compose up", () => {
|
|
19
24
|
const result = require_generators_scripts.generateScripts();
|
|
20
|
-
|
|
21
|
-
|
|
25
|
+
require_test_CTcmp4Su.globalExpect(result["scripts/start.sh"]).toContain("docker compose");
|
|
26
|
+
require_test_CTcmp4Su.globalExpect(result["scripts/start.sh"]).toContain("up");
|
|
22
27
|
});
|
|
23
|
-
|
|
28
|
+
require_test_CTcmp4Su.it("stop.sh calls docker compose down", () => {
|
|
24
29
|
const result = require_generators_scripts.generateScripts();
|
|
25
|
-
|
|
26
|
-
|
|
30
|
+
require_test_CTcmp4Su.globalExpect(result["scripts/stop.sh"]).toContain("docker compose");
|
|
31
|
+
require_test_CTcmp4Su.globalExpect(result["scripts/stop.sh"]).toContain("down");
|
|
27
32
|
});
|
|
28
|
-
|
|
33
|
+
require_test_CTcmp4Su.it("update.sh calls docker compose pull", () => {
|
|
29
34
|
const result = require_generators_scripts.generateScripts();
|
|
30
|
-
|
|
31
|
-
|
|
35
|
+
require_test_CTcmp4Su.globalExpect(result["scripts/update.sh"]).toContain("docker compose");
|
|
36
|
+
require_test_CTcmp4Su.globalExpect(result["scripts/update.sh"]).toContain("pull");
|
|
32
37
|
});
|
|
33
|
-
|
|
38
|
+
require_test_CTcmp4Su.it("backup.sh references volumes or backup", () => {
|
|
34
39
|
const backup = require_generators_scripts.generateScripts()["scripts/backup.sh"];
|
|
35
|
-
|
|
36
|
-
|
|
40
|
+
require_test_CTcmp4Su.globalExpect(backup).toBeDefined();
|
|
41
|
+
require_test_CTcmp4Su.globalExpect(backup.length).toBeGreaterThan(50);
|
|
37
42
|
});
|
|
38
|
-
|
|
43
|
+
require_test_CTcmp4Su.it("status.sh calls docker compose ps", () => {
|
|
39
44
|
const result = require_generators_scripts.generateScripts();
|
|
40
|
-
|
|
41
|
-
|
|
45
|
+
require_test_CTcmp4Su.globalExpect(result["scripts/status.sh"]).toContain("docker compose");
|
|
46
|
+
require_test_CTcmp4Su.globalExpect(result["scripts/status.sh"]).toContain("ps");
|
|
42
47
|
});
|
|
43
|
-
|
|
48
|
+
require_test_CTcmp4Su.it("all bash scripts start with shebang", () => {
|
|
44
49
|
const result = require_generators_scripts.generateScripts();
|
|
45
|
-
for (const [, content] of Object.entries(result))
|
|
50
|
+
for (const [path, content] of Object.entries(result)) if (path.endsWith(".sh")) require_test_CTcmp4Su.globalExpect(content.startsWith("#!/")).toBe(true);
|
|
51
|
+
});
|
|
52
|
+
require_test_CTcmp4Su.it("all PowerShell scripts start with #Requires", () => {
|
|
53
|
+
const result = require_generators_scripts.generateScripts();
|
|
54
|
+
for (const [path, content] of Object.entries(result)) if (path.endsWith(".ps1")) require_test_CTcmp4Su.globalExpect(content.startsWith("#Requires")).toBe(true);
|
|
55
|
+
});
|
|
56
|
+
require_test_CTcmp4Su.it("start.ps1 calls docker compose up", () => {
|
|
57
|
+
const result = require_generators_scripts.generateScripts();
|
|
58
|
+
require_test_CTcmp4Su.globalExpect(result["scripts/start.ps1"]).toContain("docker compose");
|
|
59
|
+
require_test_CTcmp4Su.globalExpect(result["scripts/start.ps1"]).toContain("up");
|
|
60
|
+
});
|
|
61
|
+
require_test_CTcmp4Su.it("stop.ps1 calls docker compose down", () => {
|
|
62
|
+
const result = require_generators_scripts.generateScripts();
|
|
63
|
+
require_test_CTcmp4Su.globalExpect(result["scripts/stop.ps1"]).toContain("docker compose");
|
|
64
|
+
require_test_CTcmp4Su.globalExpect(result["scripts/stop.ps1"]).toContain("down");
|
|
65
|
+
});
|
|
66
|
+
require_test_CTcmp4Su.it("update.ps1 calls docker compose pull", () => {
|
|
67
|
+
const result = require_generators_scripts.generateScripts();
|
|
68
|
+
require_test_CTcmp4Su.globalExpect(result["scripts/update.ps1"]).toContain("docker compose");
|
|
69
|
+
require_test_CTcmp4Su.globalExpect(result["scripts/update.ps1"]).toContain("pull");
|
|
70
|
+
});
|
|
71
|
+
require_test_CTcmp4Su.it("backup.ps1 references volumes or backup", () => {
|
|
72
|
+
const backup = require_generators_scripts.generateScripts()["scripts/backup.ps1"];
|
|
73
|
+
require_test_CTcmp4Su.globalExpect(backup).toBeDefined();
|
|
74
|
+
require_test_CTcmp4Su.globalExpect(backup.length).toBeGreaterThan(50);
|
|
75
|
+
});
|
|
76
|
+
require_test_CTcmp4Su.it("status.ps1 calls docker compose ps", () => {
|
|
77
|
+
const result = require_generators_scripts.generateScripts();
|
|
78
|
+
require_test_CTcmp4Su.globalExpect(result["scripts/status.ps1"]).toContain("docker compose");
|
|
79
|
+
require_test_CTcmp4Su.globalExpect(result["scripts/status.ps1"]).toContain("ps");
|
|
46
80
|
});
|
|
47
81
|
});
|
|
48
82
|
//#endregion
|