@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
|
@@ -0,0 +1,490 @@
|
|
|
1
|
+
import { parse } from "yaml";
|
|
2
|
+
import { describe, expect, it } from "vitest";
|
|
3
|
+
import { generateAddonStack, updateAddonStack } from "./addon-stack.js";
|
|
4
|
+
import { AddonStackInputSchema } from "./schema.js";
|
|
5
|
+
|
|
6
|
+
describe("generateAddonStack", () => {
|
|
7
|
+
it("generates valid compose YAML with a single service (qdrant)", () => {
|
|
8
|
+
const result = generateAddonStack({
|
|
9
|
+
instanceId: "test-instance",
|
|
10
|
+
services: ["qdrant"],
|
|
11
|
+
});
|
|
12
|
+
|
|
13
|
+
// Resolver may generate warnings about recommended services; that's fine
|
|
14
|
+
expect(result.metadata.serviceCount).toBeGreaterThan(0);
|
|
15
|
+
expect(result.metadata.resolvedServices).toContain("qdrant");
|
|
16
|
+
|
|
17
|
+
// Parse YAML to verify it's valid
|
|
18
|
+
const composed = parse(result.composeOverride);
|
|
19
|
+
expect(composed.services).toHaveProperty("qdrant");
|
|
20
|
+
|
|
21
|
+
// Should NOT contain infrastructure services
|
|
22
|
+
expect(composed.services).not.toHaveProperty("openclaw-gateway");
|
|
23
|
+
expect(composed.services).not.toHaveProperty("openclaw-cli");
|
|
24
|
+
expect(composed.services).not.toHaveProperty("redis");
|
|
25
|
+
expect(composed.services).not.toHaveProperty("postgresql");
|
|
26
|
+
expect(composed.services).not.toHaveProperty("open-webui");
|
|
27
|
+
expect(composed.services).not.toHaveProperty("caddy");
|
|
28
|
+
|
|
29
|
+
// Should have openclaw-network as external
|
|
30
|
+
expect(composed.networks).toHaveProperty("openclaw-network");
|
|
31
|
+
expect(composed.networks["openclaw-network"].external).toBe(true);
|
|
32
|
+
});
|
|
33
|
+
|
|
34
|
+
it("does not include profiles on any service", () => {
|
|
35
|
+
const result = generateAddonStack({
|
|
36
|
+
instanceId: "test-instance",
|
|
37
|
+
services: ["qdrant", "meilisearch"],
|
|
38
|
+
});
|
|
39
|
+
|
|
40
|
+
const composed = parse(result.composeOverride);
|
|
41
|
+
for (const [, svc] of Object.entries(composed.services)) {
|
|
42
|
+
expect(svc).not.toHaveProperty("profiles");
|
|
43
|
+
}
|
|
44
|
+
});
|
|
45
|
+
|
|
46
|
+
it("does not apply cap_drop or security_opt by default", () => {
|
|
47
|
+
const result = generateAddonStack({
|
|
48
|
+
instanceId: "test-instance",
|
|
49
|
+
services: ["qdrant"],
|
|
50
|
+
});
|
|
51
|
+
|
|
52
|
+
const composed = parse(result.composeOverride);
|
|
53
|
+
const qdrant = composed.services.qdrant;
|
|
54
|
+
expect(qdrant).not.toHaveProperty("cap_drop");
|
|
55
|
+
expect(qdrant).not.toHaveProperty("security_opt");
|
|
56
|
+
});
|
|
57
|
+
|
|
58
|
+
it("includes postgres-setup when a DB-dependent service is requested (n8n)", () => {
|
|
59
|
+
const result = generateAddonStack({
|
|
60
|
+
instanceId: "test-instance",
|
|
61
|
+
services: ["n8n"],
|
|
62
|
+
});
|
|
63
|
+
|
|
64
|
+
const composed = parse(result.composeOverride);
|
|
65
|
+
expect(composed.services).toHaveProperty("n8n");
|
|
66
|
+
expect(composed.services).toHaveProperty("postgres-setup");
|
|
67
|
+
|
|
68
|
+
// postgres-setup should depend on existing postgresql
|
|
69
|
+
expect(composed.services["postgres-setup"].depends_on).toHaveProperty("postgresql");
|
|
70
|
+
|
|
71
|
+
// n8n should depend on postgres-setup
|
|
72
|
+
expect(composed.services.n8n.depends_on).toHaveProperty("postgres-setup");
|
|
73
|
+
});
|
|
74
|
+
|
|
75
|
+
it("generates DB passwords in env file for DB-dependent services", () => {
|
|
76
|
+
const result = generateAddonStack({
|
|
77
|
+
instanceId: "test-instance",
|
|
78
|
+
services: ["n8n"],
|
|
79
|
+
generateSecrets: true,
|
|
80
|
+
});
|
|
81
|
+
|
|
82
|
+
expect(result.envFile).toContain("N8N_DB_PASSWORD=");
|
|
83
|
+
// Password should be non-empty (48 hex chars = 24 bytes)
|
|
84
|
+
const match = result.envFile.match(/N8N_DB_PASSWORD=([a-f0-9]+)/);
|
|
85
|
+
expect(match).not.toBeNull();
|
|
86
|
+
expect(match![1].length).toBe(48);
|
|
87
|
+
});
|
|
88
|
+
|
|
89
|
+
it("gracefully handles unknown service IDs without throwing", () => {
|
|
90
|
+
const result = generateAddonStack({
|
|
91
|
+
instanceId: "test-instance",
|
|
92
|
+
services: ["nonexistent-service", "qdrant"],
|
|
93
|
+
});
|
|
94
|
+
|
|
95
|
+
// Should NOT throw
|
|
96
|
+
expect(result.metadata.skippedServices).toEqual(
|
|
97
|
+
expect.arrayContaining([
|
|
98
|
+
expect.objectContaining({
|
|
99
|
+
serviceId: "nonexistent-service",
|
|
100
|
+
reason: "unknown_service",
|
|
101
|
+
}),
|
|
102
|
+
]),
|
|
103
|
+
);
|
|
104
|
+
|
|
105
|
+
// Valid service should still be included
|
|
106
|
+
expect(result.metadata.resolvedServices).toContain("qdrant");
|
|
107
|
+
});
|
|
108
|
+
|
|
109
|
+
it("excludes infrastructure services from the request", () => {
|
|
110
|
+
const result = generateAddonStack({
|
|
111
|
+
instanceId: "test-instance",
|
|
112
|
+
services: ["redis", "postgresql", "qdrant"],
|
|
113
|
+
});
|
|
114
|
+
|
|
115
|
+
expect(result.warnings).toEqual(
|
|
116
|
+
expect.arrayContaining([
|
|
117
|
+
expect.stringContaining("redis"),
|
|
118
|
+
expect.stringContaining("postgresql"),
|
|
119
|
+
]),
|
|
120
|
+
);
|
|
121
|
+
expect(result.metadata.resolvedServices).not.toContain("redis");
|
|
122
|
+
expect(result.metadata.resolvedServices).not.toContain("postgresql");
|
|
123
|
+
expect(result.metadata.resolvedServices).toContain("qdrant");
|
|
124
|
+
});
|
|
125
|
+
|
|
126
|
+
it("generates proxy routes from proxyPath", () => {
|
|
127
|
+
const result = generateAddonStack({
|
|
128
|
+
instanceId: "test-instance",
|
|
129
|
+
services: ["n8n"],
|
|
130
|
+
});
|
|
131
|
+
|
|
132
|
+
const n8nRoute = result.proxyRoutes.find((r) => r.serviceId === "n8n");
|
|
133
|
+
expect(n8nRoute).toBeDefined();
|
|
134
|
+
expect(n8nRoute!.path).toBe("/n8n");
|
|
135
|
+
expect(n8nRoute!.port).toBe(5678);
|
|
136
|
+
});
|
|
137
|
+
|
|
138
|
+
it("generates skill files and openclaw config patch", () => {
|
|
139
|
+
const result = generateAddonStack({
|
|
140
|
+
instanceId: "test-instance",
|
|
141
|
+
services: ["n8n"],
|
|
142
|
+
});
|
|
143
|
+
|
|
144
|
+
// n8n has skill binding: n8n-trigger
|
|
145
|
+
expect(result.openclawConfigPatch.skills.entries).toHaveProperty("n8n-trigger");
|
|
146
|
+
expect(result.openclawConfigPatch.skills.entries["n8n-trigger"].enabled).toBe(true);
|
|
147
|
+
});
|
|
148
|
+
|
|
149
|
+
it("resolves port conflicts with reserved ports", () => {
|
|
150
|
+
const result = generateAddonStack({
|
|
151
|
+
instanceId: "test-instance",
|
|
152
|
+
services: ["n8n"],
|
|
153
|
+
reservedPorts: [5678], // n8n's default port
|
|
154
|
+
});
|
|
155
|
+
|
|
156
|
+
// Port should be reassigned
|
|
157
|
+
const portAssignments = result.metadata.portAssignments;
|
|
158
|
+
const n8nAssignment = Object.entries(portAssignments).find(([key]) =>
|
|
159
|
+
key.startsWith("n8n:"),
|
|
160
|
+
);
|
|
161
|
+
expect(n8nAssignment).toBeDefined();
|
|
162
|
+
expect(n8nAssignment![1]).not.toBe(5678);
|
|
163
|
+
});
|
|
164
|
+
|
|
165
|
+
it("sanitizes project name from instanceId", () => {
|
|
166
|
+
const result = generateAddonStack({
|
|
167
|
+
instanceId: "My_Instance_123",
|
|
168
|
+
services: ["qdrant"],
|
|
169
|
+
});
|
|
170
|
+
|
|
171
|
+
// Should succeed without error
|
|
172
|
+
expect(result.metadata.serviceCount).toBeGreaterThan(0);
|
|
173
|
+
});
|
|
174
|
+
|
|
175
|
+
it("returns env vars grouped by service", () => {
|
|
176
|
+
const result = generateAddonStack({
|
|
177
|
+
instanceId: "test-instance",
|
|
178
|
+
services: ["qdrant"],
|
|
179
|
+
});
|
|
180
|
+
|
|
181
|
+
expect(result.envVars.length).toBeGreaterThanOrEqual(0);
|
|
182
|
+
});
|
|
183
|
+
|
|
184
|
+
it("returns empty result for no services", () => {
|
|
185
|
+
const result = generateAddonStack({
|
|
186
|
+
instanceId: "test-instance",
|
|
187
|
+
services: [],
|
|
188
|
+
});
|
|
189
|
+
|
|
190
|
+
expect(result.composeOverride).toContain("services: {}");
|
|
191
|
+
expect(result.warnings.length).toBeGreaterThan(0);
|
|
192
|
+
});
|
|
193
|
+
|
|
194
|
+
it("returns empty result when all services are infrastructure", () => {
|
|
195
|
+
const result = generateAddonStack({
|
|
196
|
+
instanceId: "test-instance",
|
|
197
|
+
services: ["redis", "postgresql", "caddy"],
|
|
198
|
+
});
|
|
199
|
+
|
|
200
|
+
expect(result.metadata.serviceCount).toBe(0);
|
|
201
|
+
expect(result.warnings.length).toBeGreaterThan(0);
|
|
202
|
+
});
|
|
203
|
+
|
|
204
|
+
it("applies env quirks to envFile output (meilisearch MEILI_MASTER_KEY min_length)", () => {
|
|
205
|
+
const result = generateAddonStack({
|
|
206
|
+
instanceId: "test-instance",
|
|
207
|
+
services: ["meilisearch"],
|
|
208
|
+
generateSecrets: true,
|
|
209
|
+
});
|
|
210
|
+
|
|
211
|
+
// MEILI_MASTER_KEY should be a base64url string of at least 32 chars (24 bytes)
|
|
212
|
+
const match = result.envFile.match(/MEILI_MASTER_KEY=([^\n]+)/);
|
|
213
|
+
expect(match).not.toBeNull();
|
|
214
|
+
expect(match![1].length).toBeGreaterThanOrEqual(32);
|
|
215
|
+
// Should not be empty
|
|
216
|
+
expect(match![1]).not.toBe("");
|
|
217
|
+
});
|
|
218
|
+
|
|
219
|
+
it("applies env quirks to envFile output (grafana GF_SECURITY_ADMIN_PASSWORD)", () => {
|
|
220
|
+
const result = generateAddonStack({
|
|
221
|
+
instanceId: "test-instance",
|
|
222
|
+
services: ["grafana"],
|
|
223
|
+
generateSecrets: true,
|
|
224
|
+
});
|
|
225
|
+
|
|
226
|
+
const match = result.envFile.match(/GF_SECURITY_ADMIN_PASSWORD=([^\n]+)/);
|
|
227
|
+
expect(match).not.toBeNull();
|
|
228
|
+
// Should be at least 22 chars (16 bytes base64url)
|
|
229
|
+
expect(match![1].length).toBeGreaterThanOrEqual(22);
|
|
230
|
+
});
|
|
231
|
+
|
|
232
|
+
it("syncs DB_POSTGRESDB_PASSWORD with N8N_DB_PASSWORD via must_sync quirk", () => {
|
|
233
|
+
const result = generateAddonStack({
|
|
234
|
+
instanceId: "test-instance",
|
|
235
|
+
services: ["n8n"],
|
|
236
|
+
generateSecrets: true,
|
|
237
|
+
});
|
|
238
|
+
|
|
239
|
+
const n8nDbPw = result.envFile.match(/N8N_DB_PASSWORD=([^\n]+)/);
|
|
240
|
+
const dbPostgresPw = result.envFile.match(/DB_POSTGRESDB_PASSWORD=([^\n]+)/);
|
|
241
|
+
expect(n8nDbPw).not.toBeNull();
|
|
242
|
+
expect(dbPostgresPw).not.toBeNull();
|
|
243
|
+
// Both passwords must match due to must_sync quirk
|
|
244
|
+
expect(n8nDbPw![1]).toBe(dbPostgresPw![1]);
|
|
245
|
+
});
|
|
246
|
+
|
|
247
|
+
it("syncs user-provided credential with DB password reference", () => {
|
|
248
|
+
const customPassword = "my_custom_secure_password";
|
|
249
|
+
const result = generateAddonStack({
|
|
250
|
+
instanceId: "test-instance",
|
|
251
|
+
services: ["n8n"],
|
|
252
|
+
generateSecrets: true,
|
|
253
|
+
credentials: {
|
|
254
|
+
n8n: { DB_POSTGRESDB_PASSWORD: customPassword },
|
|
255
|
+
},
|
|
256
|
+
});
|
|
257
|
+
|
|
258
|
+
// User value should be used for both the service env var and the ref key
|
|
259
|
+
expect(result.envFile).toContain(`DB_POSTGRESDB_PASSWORD=${customPassword}`);
|
|
260
|
+
expect(result.envFile).toContain(`N8N_DB_PASSWORD=${customPassword}`);
|
|
261
|
+
});
|
|
262
|
+
|
|
263
|
+
it("uses existingServices ports for conflict detection", () => {
|
|
264
|
+
const result = generateAddonStack({
|
|
265
|
+
instanceId: "test-instance",
|
|
266
|
+
services: ["n8n"],
|
|
267
|
+
// n8n uses port 5678, qdrant uses 6333 — qdrant as existing should not conflict
|
|
268
|
+
existingServices: ["qdrant"],
|
|
269
|
+
});
|
|
270
|
+
|
|
271
|
+
// n8n should still get its default port (no conflict with qdrant)
|
|
272
|
+
const portAssignments = result.metadata.portAssignments;
|
|
273
|
+
const n8nAssignment = Object.entries(portAssignments).find(([key]) =>
|
|
274
|
+
key.startsWith("n8n:"),
|
|
275
|
+
);
|
|
276
|
+
expect(n8nAssignment).toBeDefined();
|
|
277
|
+
expect(n8nAssignment![1]).toBe(5678);
|
|
278
|
+
});
|
|
279
|
+
|
|
280
|
+
it("does not dual-list GPU services in both skipped and resolved", () => {
|
|
281
|
+
const result = generateAddonStack({
|
|
282
|
+
instanceId: "test-instance",
|
|
283
|
+
services: ["ollama"],
|
|
284
|
+
gpu: false,
|
|
285
|
+
});
|
|
286
|
+
|
|
287
|
+
// ollama requires GPU — without gpu: true, it should be skipped only
|
|
288
|
+
const isSkipped = result.metadata.skippedServices.some(
|
|
289
|
+
(s) => s.serviceId === "ollama" && s.reason === "gpu_required",
|
|
290
|
+
);
|
|
291
|
+
const isResolved = result.metadata.resolvedServices.includes("ollama");
|
|
292
|
+
|
|
293
|
+
// Must be in exactly one list, not both
|
|
294
|
+
if (isSkipped) {
|
|
295
|
+
expect(isResolved).toBe(false);
|
|
296
|
+
}
|
|
297
|
+
});
|
|
298
|
+
|
|
299
|
+
it("includes GPU services when gpu: true is set", () => {
|
|
300
|
+
const result = generateAddonStack({
|
|
301
|
+
instanceId: "test-instance",
|
|
302
|
+
services: ["ollama"],
|
|
303
|
+
gpu: true,
|
|
304
|
+
});
|
|
305
|
+
|
|
306
|
+
// Should NOT be skipped when GPU support is available
|
|
307
|
+
const isSkipped = result.metadata.skippedServices.some(
|
|
308
|
+
(s) => s.serviceId === "ollama" && s.reason === "gpu_required",
|
|
309
|
+
);
|
|
310
|
+
expect(isSkipped).toBe(false);
|
|
311
|
+
expect(result.metadata.resolvedServices).toContain("ollama");
|
|
312
|
+
});
|
|
313
|
+
});
|
|
314
|
+
|
|
315
|
+
describe("updateAddonStack", () => {
|
|
316
|
+
it("adds a service to an existing stack", () => {
|
|
317
|
+
// First generate a base stack
|
|
318
|
+
const base = generateAddonStack({
|
|
319
|
+
instanceId: "test-instance",
|
|
320
|
+
services: ["qdrant"],
|
|
321
|
+
});
|
|
322
|
+
|
|
323
|
+
// Then add meilisearch
|
|
324
|
+
const result = updateAddonStack({
|
|
325
|
+
instanceId: "test-instance",
|
|
326
|
+
currentCompose: base.composeOverride,
|
|
327
|
+
currentEnv: base.envFile,
|
|
328
|
+
addServices: ["meilisearch"],
|
|
329
|
+
});
|
|
330
|
+
|
|
331
|
+
expect(result.metadata.added).toContain("meilisearch");
|
|
332
|
+
expect(result.metadata.unchanged).toContain("qdrant");
|
|
333
|
+
|
|
334
|
+
const composed = parse(result.composeOverride);
|
|
335
|
+
expect(composed.services).toHaveProperty("qdrant");
|
|
336
|
+
expect(composed.services).toHaveProperty("meilisearch");
|
|
337
|
+
});
|
|
338
|
+
|
|
339
|
+
it("removes a service from an existing stack", () => {
|
|
340
|
+
const base = generateAddonStack({
|
|
341
|
+
instanceId: "test-instance",
|
|
342
|
+
services: ["qdrant", "meilisearch"],
|
|
343
|
+
});
|
|
344
|
+
|
|
345
|
+
const result = updateAddonStack({
|
|
346
|
+
instanceId: "test-instance",
|
|
347
|
+
currentCompose: base.composeOverride,
|
|
348
|
+
currentEnv: base.envFile,
|
|
349
|
+
removeServices: ["meilisearch"],
|
|
350
|
+
});
|
|
351
|
+
|
|
352
|
+
expect(result.metadata.removed).toContain("meilisearch");
|
|
353
|
+
expect(result.metadata.unchanged).toContain("qdrant");
|
|
354
|
+
|
|
355
|
+
const composed = parse(result.composeOverride);
|
|
356
|
+
expect(composed.services).toHaveProperty("qdrant");
|
|
357
|
+
expect(composed.services).not.toHaveProperty("meilisearch");
|
|
358
|
+
});
|
|
359
|
+
|
|
360
|
+
it("preserves existing env values when adding a service", () => {
|
|
361
|
+
const base = generateAddonStack({
|
|
362
|
+
instanceId: "test-instance",
|
|
363
|
+
services: ["qdrant"],
|
|
364
|
+
generateSecrets: true,
|
|
365
|
+
});
|
|
366
|
+
|
|
367
|
+
// Simulate user editing an env value
|
|
368
|
+
const customEnv = base.envFile.replace(
|
|
369
|
+
/QDRANT_API_KEY=[a-f0-9]+/,
|
|
370
|
+
"QDRANT_API_KEY=my_custom_key",
|
|
371
|
+
);
|
|
372
|
+
|
|
373
|
+
const result = updateAddonStack({
|
|
374
|
+
instanceId: "test-instance",
|
|
375
|
+
currentCompose: base.composeOverride,
|
|
376
|
+
currentEnv: customEnv,
|
|
377
|
+
addServices: ["meilisearch"],
|
|
378
|
+
generateSecrets: true,
|
|
379
|
+
});
|
|
380
|
+
|
|
381
|
+
// If QDRANT_API_KEY was in the original env, it should be preserved
|
|
382
|
+
if (customEnv.includes("QDRANT_API_KEY=my_custom_key")) {
|
|
383
|
+
expect(result.envFile).toContain("QDRANT_API_KEY=my_custom_key");
|
|
384
|
+
}
|
|
385
|
+
});
|
|
386
|
+
|
|
387
|
+
it("returns images to pull for added services", () => {
|
|
388
|
+
const base = generateAddonStack({
|
|
389
|
+
instanceId: "test-instance",
|
|
390
|
+
services: ["qdrant"],
|
|
391
|
+
});
|
|
392
|
+
|
|
393
|
+
const result = updateAddonStack({
|
|
394
|
+
instanceId: "test-instance",
|
|
395
|
+
currentCompose: base.composeOverride,
|
|
396
|
+
currentEnv: base.envFile,
|
|
397
|
+
addServices: ["meilisearch"],
|
|
398
|
+
});
|
|
399
|
+
|
|
400
|
+
expect(result.imagesToPull.length).toBeGreaterThan(0);
|
|
401
|
+
expect(result.imagesToPull.some((img) => img.includes("meilisearch"))).toBe(true);
|
|
402
|
+
});
|
|
403
|
+
|
|
404
|
+
it("handles empty update gracefully", () => {
|
|
405
|
+
const base = generateAddonStack({
|
|
406
|
+
instanceId: "test-instance",
|
|
407
|
+
services: ["qdrant"],
|
|
408
|
+
});
|
|
409
|
+
|
|
410
|
+
const result = updateAddonStack({
|
|
411
|
+
instanceId: "test-instance",
|
|
412
|
+
currentCompose: base.composeOverride,
|
|
413
|
+
currentEnv: base.envFile,
|
|
414
|
+
});
|
|
415
|
+
|
|
416
|
+
expect(result.metadata.added).toEqual([]);
|
|
417
|
+
expect(result.metadata.removed).toEqual([]);
|
|
418
|
+
expect(result.metadata.unchanged).toContain("qdrant");
|
|
419
|
+
});
|
|
420
|
+
|
|
421
|
+
it("respects generateSecrets: false during update", () => {
|
|
422
|
+
const base = generateAddonStack({
|
|
423
|
+
instanceId: "test-instance",
|
|
424
|
+
services: ["qdrant"],
|
|
425
|
+
generateSecrets: true,
|
|
426
|
+
});
|
|
427
|
+
|
|
428
|
+
// With generateSecrets: false, services requiring secrets (like meilisearch)
|
|
429
|
+
// may be skipped as missing_credentials. Provide credentials explicitly.
|
|
430
|
+
const result = updateAddonStack({
|
|
431
|
+
instanceId: "test-instance",
|
|
432
|
+
currentCompose: base.composeOverride,
|
|
433
|
+
currentEnv: base.envFile,
|
|
434
|
+
addServices: ["meilisearch"],
|
|
435
|
+
generateSecrets: false,
|
|
436
|
+
credentials: {
|
|
437
|
+
meilisearch: { MEILI_MASTER_KEY: "test-master-key-1234567890" },
|
|
438
|
+
},
|
|
439
|
+
});
|
|
440
|
+
|
|
441
|
+
// Meilisearch should be added since we provided the required credential
|
|
442
|
+
expect(result.metadata.added).toContain("meilisearch");
|
|
443
|
+
});
|
|
444
|
+
|
|
445
|
+
it("forwards portOverrides during update", () => {
|
|
446
|
+
const base = generateAddonStack({
|
|
447
|
+
instanceId: "test-instance",
|
|
448
|
+
services: ["qdrant"],
|
|
449
|
+
});
|
|
450
|
+
|
|
451
|
+
const result = updateAddonStack({
|
|
452
|
+
instanceId: "test-instance",
|
|
453
|
+
currentCompose: base.composeOverride,
|
|
454
|
+
currentEnv: base.envFile,
|
|
455
|
+
addServices: ["n8n"],
|
|
456
|
+
portOverrides: { n8n: { "5678": 9999 } },
|
|
457
|
+
});
|
|
458
|
+
|
|
459
|
+
// n8n should be present with the port override
|
|
460
|
+
expect(result.metadata.added).toContain("n8n");
|
|
461
|
+
});
|
|
462
|
+
});
|
|
463
|
+
|
|
464
|
+
describe("AddonStackInputSchema", () => {
|
|
465
|
+
it("accepts valid input", () => {
|
|
466
|
+
const result = AddonStackInputSchema.safeParse({
|
|
467
|
+
instanceId: "test-instance",
|
|
468
|
+
services: ["qdrant", "n8n"],
|
|
469
|
+
});
|
|
470
|
+
expect(result.success).toBe(true);
|
|
471
|
+
});
|
|
472
|
+
|
|
473
|
+
it("requires instanceId", () => {
|
|
474
|
+
const result = AddonStackInputSchema.safeParse({
|
|
475
|
+
services: ["qdrant"],
|
|
476
|
+
});
|
|
477
|
+
expect(result.success).toBe(false);
|
|
478
|
+
});
|
|
479
|
+
|
|
480
|
+
it("defaults empty arrays and booleans", () => {
|
|
481
|
+
const result = AddonStackInputSchema.parse({
|
|
482
|
+
instanceId: "test",
|
|
483
|
+
services: ["qdrant"],
|
|
484
|
+
});
|
|
485
|
+
expect(result.skillPacks).toEqual([]);
|
|
486
|
+
expect(result.reservedPorts).toEqual([]);
|
|
487
|
+
expect(result.generateSecrets).toBe(true);
|
|
488
|
+
expect(result.platform).toBe("linux/amd64");
|
|
489
|
+
});
|
|
490
|
+
});
|