@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,349 @@
|
|
|
1
|
+
import { n as describe, r as it, t as globalExpect } from "./test.CTcmp4Su-ClCHJ3FA.mjs";
|
|
2
|
+
import { AddonStackInputSchema } from "./schema.mjs";
|
|
3
|
+
import { generateAddonStack, updateAddonStack } from "./addon-stack.mjs";
|
|
4
|
+
import { parse } from "yaml";
|
|
5
|
+
//#region src/addon-stack.test.ts
|
|
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
|
+
globalExpect(result.metadata.serviceCount).toBeGreaterThan(0);
|
|
13
|
+
globalExpect(result.metadata.resolvedServices).toContain("qdrant");
|
|
14
|
+
const composed = parse(result.composeOverride);
|
|
15
|
+
globalExpect(composed.services).toHaveProperty("qdrant");
|
|
16
|
+
globalExpect(composed.services).not.toHaveProperty("openclaw-gateway");
|
|
17
|
+
globalExpect(composed.services).not.toHaveProperty("openclaw-cli");
|
|
18
|
+
globalExpect(composed.services).not.toHaveProperty("redis");
|
|
19
|
+
globalExpect(composed.services).not.toHaveProperty("postgresql");
|
|
20
|
+
globalExpect(composed.services).not.toHaveProperty("open-webui");
|
|
21
|
+
globalExpect(composed.services).not.toHaveProperty("caddy");
|
|
22
|
+
globalExpect(composed.networks).toHaveProperty("openclaw-network");
|
|
23
|
+
globalExpect(composed.networks["openclaw-network"].external).toBe(true);
|
|
24
|
+
});
|
|
25
|
+
it("does not include profiles on any service", () => {
|
|
26
|
+
const composed = parse(generateAddonStack({
|
|
27
|
+
instanceId: "test-instance",
|
|
28
|
+
services: ["qdrant", "meilisearch"]
|
|
29
|
+
}).composeOverride);
|
|
30
|
+
for (const [, svc] of Object.entries(composed.services)) globalExpect(svc).not.toHaveProperty("profiles");
|
|
31
|
+
});
|
|
32
|
+
it("does not apply cap_drop or security_opt by default", () => {
|
|
33
|
+
const qdrant = parse(generateAddonStack({
|
|
34
|
+
instanceId: "test-instance",
|
|
35
|
+
services: ["qdrant"]
|
|
36
|
+
}).composeOverride).services.qdrant;
|
|
37
|
+
globalExpect(qdrant).not.toHaveProperty("cap_drop");
|
|
38
|
+
globalExpect(qdrant).not.toHaveProperty("security_opt");
|
|
39
|
+
});
|
|
40
|
+
it("includes postgres-setup when a DB-dependent service is requested (n8n)", () => {
|
|
41
|
+
const composed = parse(generateAddonStack({
|
|
42
|
+
instanceId: "test-instance",
|
|
43
|
+
services: ["n8n"]
|
|
44
|
+
}).composeOverride);
|
|
45
|
+
globalExpect(composed.services).toHaveProperty("n8n");
|
|
46
|
+
globalExpect(composed.services).toHaveProperty("postgres-setup");
|
|
47
|
+
globalExpect(composed.services["postgres-setup"].depends_on).toHaveProperty("postgresql");
|
|
48
|
+
globalExpect(composed.services.n8n.depends_on).toHaveProperty("postgres-setup");
|
|
49
|
+
});
|
|
50
|
+
it("generates DB passwords in env file for DB-dependent services", () => {
|
|
51
|
+
const result = generateAddonStack({
|
|
52
|
+
instanceId: "test-instance",
|
|
53
|
+
services: ["n8n"],
|
|
54
|
+
generateSecrets: true
|
|
55
|
+
});
|
|
56
|
+
globalExpect(result.envFile).toContain("N8N_DB_PASSWORD=");
|
|
57
|
+
const match = result.envFile.match(/N8N_DB_PASSWORD=([a-f0-9]+)/);
|
|
58
|
+
globalExpect(match).not.toBeNull();
|
|
59
|
+
globalExpect(match[1].length).toBe(48);
|
|
60
|
+
});
|
|
61
|
+
it("gracefully handles unknown service IDs without throwing", () => {
|
|
62
|
+
const result = generateAddonStack({
|
|
63
|
+
instanceId: "test-instance",
|
|
64
|
+
services: ["nonexistent-service", "qdrant"]
|
|
65
|
+
});
|
|
66
|
+
globalExpect(result.metadata.skippedServices).toEqual(globalExpect.arrayContaining([globalExpect.objectContaining({
|
|
67
|
+
serviceId: "nonexistent-service",
|
|
68
|
+
reason: "unknown_service"
|
|
69
|
+
})]));
|
|
70
|
+
globalExpect(result.metadata.resolvedServices).toContain("qdrant");
|
|
71
|
+
});
|
|
72
|
+
it("excludes infrastructure services from the request", () => {
|
|
73
|
+
const result = generateAddonStack({
|
|
74
|
+
instanceId: "test-instance",
|
|
75
|
+
services: [
|
|
76
|
+
"redis",
|
|
77
|
+
"postgresql",
|
|
78
|
+
"qdrant"
|
|
79
|
+
]
|
|
80
|
+
});
|
|
81
|
+
globalExpect(result.warnings).toEqual(globalExpect.arrayContaining([globalExpect.stringContaining("redis"), globalExpect.stringContaining("postgresql")]));
|
|
82
|
+
globalExpect(result.metadata.resolvedServices).not.toContain("redis");
|
|
83
|
+
globalExpect(result.metadata.resolvedServices).not.toContain("postgresql");
|
|
84
|
+
globalExpect(result.metadata.resolvedServices).toContain("qdrant");
|
|
85
|
+
});
|
|
86
|
+
it("generates proxy routes from proxyPath", () => {
|
|
87
|
+
const n8nRoute = generateAddonStack({
|
|
88
|
+
instanceId: "test-instance",
|
|
89
|
+
services: ["n8n"]
|
|
90
|
+
}).proxyRoutes.find((r) => r.serviceId === "n8n");
|
|
91
|
+
globalExpect(n8nRoute).toBeDefined();
|
|
92
|
+
globalExpect(n8nRoute.path).toBe("/n8n");
|
|
93
|
+
globalExpect(n8nRoute.port).toBe(5678);
|
|
94
|
+
});
|
|
95
|
+
it("generates skill files and openclaw config patch", () => {
|
|
96
|
+
const result = generateAddonStack({
|
|
97
|
+
instanceId: "test-instance",
|
|
98
|
+
services: ["n8n"]
|
|
99
|
+
});
|
|
100
|
+
globalExpect(result.openclawConfigPatch.skills.entries).toHaveProperty("n8n-trigger");
|
|
101
|
+
globalExpect(result.openclawConfigPatch.skills.entries["n8n-trigger"].enabled).toBe(true);
|
|
102
|
+
});
|
|
103
|
+
it("resolves port conflicts with reserved ports", () => {
|
|
104
|
+
const portAssignments = generateAddonStack({
|
|
105
|
+
instanceId: "test-instance",
|
|
106
|
+
services: ["n8n"],
|
|
107
|
+
reservedPorts: [5678]
|
|
108
|
+
}).metadata.portAssignments;
|
|
109
|
+
const n8nAssignment = Object.entries(portAssignments).find(([key]) => key.startsWith("n8n:"));
|
|
110
|
+
globalExpect(n8nAssignment).toBeDefined();
|
|
111
|
+
globalExpect(n8nAssignment[1]).not.toBe(5678);
|
|
112
|
+
});
|
|
113
|
+
it("sanitizes project name from instanceId", () => {
|
|
114
|
+
globalExpect(generateAddonStack({
|
|
115
|
+
instanceId: "My_Instance_123",
|
|
116
|
+
services: ["qdrant"]
|
|
117
|
+
}).metadata.serviceCount).toBeGreaterThan(0);
|
|
118
|
+
});
|
|
119
|
+
it("returns env vars grouped by service", () => {
|
|
120
|
+
globalExpect(generateAddonStack({
|
|
121
|
+
instanceId: "test-instance",
|
|
122
|
+
services: ["qdrant"]
|
|
123
|
+
}).envVars.length).toBeGreaterThanOrEqual(0);
|
|
124
|
+
});
|
|
125
|
+
it("returns empty result for no services", () => {
|
|
126
|
+
const result = generateAddonStack({
|
|
127
|
+
instanceId: "test-instance",
|
|
128
|
+
services: []
|
|
129
|
+
});
|
|
130
|
+
globalExpect(result.composeOverride).toContain("services: {}");
|
|
131
|
+
globalExpect(result.warnings.length).toBeGreaterThan(0);
|
|
132
|
+
});
|
|
133
|
+
it("returns empty result when all services are infrastructure", () => {
|
|
134
|
+
const result = generateAddonStack({
|
|
135
|
+
instanceId: "test-instance",
|
|
136
|
+
services: [
|
|
137
|
+
"redis",
|
|
138
|
+
"postgresql",
|
|
139
|
+
"caddy"
|
|
140
|
+
]
|
|
141
|
+
});
|
|
142
|
+
globalExpect(result.metadata.serviceCount).toBe(0);
|
|
143
|
+
globalExpect(result.warnings.length).toBeGreaterThan(0);
|
|
144
|
+
});
|
|
145
|
+
it("applies env quirks to envFile output (meilisearch MEILI_MASTER_KEY min_length)", () => {
|
|
146
|
+
const match = generateAddonStack({
|
|
147
|
+
instanceId: "test-instance",
|
|
148
|
+
services: ["meilisearch"],
|
|
149
|
+
generateSecrets: true
|
|
150
|
+
}).envFile.match(/MEILI_MASTER_KEY=([^\n]+)/);
|
|
151
|
+
globalExpect(match).not.toBeNull();
|
|
152
|
+
globalExpect(match[1].length).toBeGreaterThanOrEqual(32);
|
|
153
|
+
globalExpect(match[1]).not.toBe("");
|
|
154
|
+
});
|
|
155
|
+
it("applies env quirks to envFile output (grafana GF_SECURITY_ADMIN_PASSWORD)", () => {
|
|
156
|
+
const match = generateAddonStack({
|
|
157
|
+
instanceId: "test-instance",
|
|
158
|
+
services: ["grafana"],
|
|
159
|
+
generateSecrets: true
|
|
160
|
+
}).envFile.match(/GF_SECURITY_ADMIN_PASSWORD=([^\n]+)/);
|
|
161
|
+
globalExpect(match).not.toBeNull();
|
|
162
|
+
globalExpect(match[1].length).toBeGreaterThanOrEqual(22);
|
|
163
|
+
});
|
|
164
|
+
it("syncs DB_POSTGRESDB_PASSWORD with N8N_DB_PASSWORD via must_sync quirk", () => {
|
|
165
|
+
const result = generateAddonStack({
|
|
166
|
+
instanceId: "test-instance",
|
|
167
|
+
services: ["n8n"],
|
|
168
|
+
generateSecrets: true
|
|
169
|
+
});
|
|
170
|
+
const n8nDbPw = result.envFile.match(/N8N_DB_PASSWORD=([^\n]+)/);
|
|
171
|
+
const dbPostgresPw = result.envFile.match(/DB_POSTGRESDB_PASSWORD=([^\n]+)/);
|
|
172
|
+
globalExpect(n8nDbPw).not.toBeNull();
|
|
173
|
+
globalExpect(dbPostgresPw).not.toBeNull();
|
|
174
|
+
globalExpect(n8nDbPw[1]).toBe(dbPostgresPw[1]);
|
|
175
|
+
});
|
|
176
|
+
it("syncs user-provided credential with DB password reference", () => {
|
|
177
|
+
const customPassword = "my_custom_secure_password";
|
|
178
|
+
const result = generateAddonStack({
|
|
179
|
+
instanceId: "test-instance",
|
|
180
|
+
services: ["n8n"],
|
|
181
|
+
generateSecrets: true,
|
|
182
|
+
credentials: { n8n: { DB_POSTGRESDB_PASSWORD: customPassword } }
|
|
183
|
+
});
|
|
184
|
+
globalExpect(result.envFile).toContain(`DB_POSTGRESDB_PASSWORD=${customPassword}`);
|
|
185
|
+
globalExpect(result.envFile).toContain(`N8N_DB_PASSWORD=${customPassword}`);
|
|
186
|
+
});
|
|
187
|
+
it("uses existingServices ports for conflict detection", () => {
|
|
188
|
+
const portAssignments = generateAddonStack({
|
|
189
|
+
instanceId: "test-instance",
|
|
190
|
+
services: ["n8n"],
|
|
191
|
+
existingServices: ["qdrant"]
|
|
192
|
+
}).metadata.portAssignments;
|
|
193
|
+
const n8nAssignment = Object.entries(portAssignments).find(([key]) => key.startsWith("n8n:"));
|
|
194
|
+
globalExpect(n8nAssignment).toBeDefined();
|
|
195
|
+
globalExpect(n8nAssignment[1]).toBe(5678);
|
|
196
|
+
});
|
|
197
|
+
it("does not dual-list GPU services in both skipped and resolved", () => {
|
|
198
|
+
const result = generateAddonStack({
|
|
199
|
+
instanceId: "test-instance",
|
|
200
|
+
services: ["ollama"],
|
|
201
|
+
gpu: false
|
|
202
|
+
});
|
|
203
|
+
const isSkipped = result.metadata.skippedServices.some((s) => s.serviceId === "ollama" && s.reason === "gpu_required");
|
|
204
|
+
const isResolved = result.metadata.resolvedServices.includes("ollama");
|
|
205
|
+
if (isSkipped) globalExpect(isResolved).toBe(false);
|
|
206
|
+
});
|
|
207
|
+
it("includes GPU services when gpu: true is set", () => {
|
|
208
|
+
const result = generateAddonStack({
|
|
209
|
+
instanceId: "test-instance",
|
|
210
|
+
services: ["ollama"],
|
|
211
|
+
gpu: true
|
|
212
|
+
});
|
|
213
|
+
globalExpect(result.metadata.skippedServices.some((s) => s.serviceId === "ollama" && s.reason === "gpu_required")).toBe(false);
|
|
214
|
+
globalExpect(result.metadata.resolvedServices).toContain("ollama");
|
|
215
|
+
});
|
|
216
|
+
});
|
|
217
|
+
describe("updateAddonStack", () => {
|
|
218
|
+
it("adds a service to an existing stack", () => {
|
|
219
|
+
const base = generateAddonStack({
|
|
220
|
+
instanceId: "test-instance",
|
|
221
|
+
services: ["qdrant"]
|
|
222
|
+
});
|
|
223
|
+
const result = updateAddonStack({
|
|
224
|
+
instanceId: "test-instance",
|
|
225
|
+
currentCompose: base.composeOverride,
|
|
226
|
+
currentEnv: base.envFile,
|
|
227
|
+
addServices: ["meilisearch"]
|
|
228
|
+
});
|
|
229
|
+
globalExpect(result.metadata.added).toContain("meilisearch");
|
|
230
|
+
globalExpect(result.metadata.unchanged).toContain("qdrant");
|
|
231
|
+
const composed = parse(result.composeOverride);
|
|
232
|
+
globalExpect(composed.services).toHaveProperty("qdrant");
|
|
233
|
+
globalExpect(composed.services).toHaveProperty("meilisearch");
|
|
234
|
+
});
|
|
235
|
+
it("removes a service from an existing stack", () => {
|
|
236
|
+
const base = generateAddonStack({
|
|
237
|
+
instanceId: "test-instance",
|
|
238
|
+
services: ["qdrant", "meilisearch"]
|
|
239
|
+
});
|
|
240
|
+
const result = updateAddonStack({
|
|
241
|
+
instanceId: "test-instance",
|
|
242
|
+
currentCompose: base.composeOverride,
|
|
243
|
+
currentEnv: base.envFile,
|
|
244
|
+
removeServices: ["meilisearch"]
|
|
245
|
+
});
|
|
246
|
+
globalExpect(result.metadata.removed).toContain("meilisearch");
|
|
247
|
+
globalExpect(result.metadata.unchanged).toContain("qdrant");
|
|
248
|
+
const composed = parse(result.composeOverride);
|
|
249
|
+
globalExpect(composed.services).toHaveProperty("qdrant");
|
|
250
|
+
globalExpect(composed.services).not.toHaveProperty("meilisearch");
|
|
251
|
+
});
|
|
252
|
+
it("preserves existing env values when adding a service", () => {
|
|
253
|
+
const base = generateAddonStack({
|
|
254
|
+
instanceId: "test-instance",
|
|
255
|
+
services: ["qdrant"],
|
|
256
|
+
generateSecrets: true
|
|
257
|
+
});
|
|
258
|
+
const customEnv = base.envFile.replace(/QDRANT_API_KEY=[a-f0-9]+/, "QDRANT_API_KEY=my_custom_key");
|
|
259
|
+
const result = updateAddonStack({
|
|
260
|
+
instanceId: "test-instance",
|
|
261
|
+
currentCompose: base.composeOverride,
|
|
262
|
+
currentEnv: customEnv,
|
|
263
|
+
addServices: ["meilisearch"],
|
|
264
|
+
generateSecrets: true
|
|
265
|
+
});
|
|
266
|
+
if (customEnv.includes("QDRANT_API_KEY=my_custom_key")) globalExpect(result.envFile).toContain("QDRANT_API_KEY=my_custom_key");
|
|
267
|
+
});
|
|
268
|
+
it("returns images to pull for added services", () => {
|
|
269
|
+
const base = generateAddonStack({
|
|
270
|
+
instanceId: "test-instance",
|
|
271
|
+
services: ["qdrant"]
|
|
272
|
+
});
|
|
273
|
+
const result = updateAddonStack({
|
|
274
|
+
instanceId: "test-instance",
|
|
275
|
+
currentCompose: base.composeOverride,
|
|
276
|
+
currentEnv: base.envFile,
|
|
277
|
+
addServices: ["meilisearch"]
|
|
278
|
+
});
|
|
279
|
+
globalExpect(result.imagesToPull.length).toBeGreaterThan(0);
|
|
280
|
+
globalExpect(result.imagesToPull.some((img) => img.includes("meilisearch"))).toBe(true);
|
|
281
|
+
});
|
|
282
|
+
it("handles empty update gracefully", () => {
|
|
283
|
+
const base = generateAddonStack({
|
|
284
|
+
instanceId: "test-instance",
|
|
285
|
+
services: ["qdrant"]
|
|
286
|
+
});
|
|
287
|
+
const result = updateAddonStack({
|
|
288
|
+
instanceId: "test-instance",
|
|
289
|
+
currentCompose: base.composeOverride,
|
|
290
|
+
currentEnv: base.envFile
|
|
291
|
+
});
|
|
292
|
+
globalExpect(result.metadata.added).toEqual([]);
|
|
293
|
+
globalExpect(result.metadata.removed).toEqual([]);
|
|
294
|
+
globalExpect(result.metadata.unchanged).toContain("qdrant");
|
|
295
|
+
});
|
|
296
|
+
it("respects generateSecrets: false during update", () => {
|
|
297
|
+
const base = generateAddonStack({
|
|
298
|
+
instanceId: "test-instance",
|
|
299
|
+
services: ["qdrant"],
|
|
300
|
+
generateSecrets: true
|
|
301
|
+
});
|
|
302
|
+
globalExpect(updateAddonStack({
|
|
303
|
+
instanceId: "test-instance",
|
|
304
|
+
currentCompose: base.composeOverride,
|
|
305
|
+
currentEnv: base.envFile,
|
|
306
|
+
addServices: ["meilisearch"],
|
|
307
|
+
generateSecrets: false,
|
|
308
|
+
credentials: { meilisearch: { MEILI_MASTER_KEY: "test-master-key-1234567890" } }
|
|
309
|
+
}).metadata.added).toContain("meilisearch");
|
|
310
|
+
});
|
|
311
|
+
it("forwards portOverrides during update", () => {
|
|
312
|
+
const base = generateAddonStack({
|
|
313
|
+
instanceId: "test-instance",
|
|
314
|
+
services: ["qdrant"]
|
|
315
|
+
});
|
|
316
|
+
globalExpect(updateAddonStack({
|
|
317
|
+
instanceId: "test-instance",
|
|
318
|
+
currentCompose: base.composeOverride,
|
|
319
|
+
currentEnv: base.envFile,
|
|
320
|
+
addServices: ["n8n"],
|
|
321
|
+
portOverrides: { n8n: { "5678": 9999 } }
|
|
322
|
+
}).metadata.added).toContain("n8n");
|
|
323
|
+
});
|
|
324
|
+
});
|
|
325
|
+
describe("AddonStackInputSchema", () => {
|
|
326
|
+
it("accepts valid input", () => {
|
|
327
|
+
globalExpect(AddonStackInputSchema.safeParse({
|
|
328
|
+
instanceId: "test-instance",
|
|
329
|
+
services: ["qdrant", "n8n"]
|
|
330
|
+
}).success).toBe(true);
|
|
331
|
+
});
|
|
332
|
+
it("requires instanceId", () => {
|
|
333
|
+
globalExpect(AddonStackInputSchema.safeParse({ services: ["qdrant"] }).success).toBe(false);
|
|
334
|
+
});
|
|
335
|
+
it("defaults empty arrays and booleans", () => {
|
|
336
|
+
const result = AddonStackInputSchema.parse({
|
|
337
|
+
instanceId: "test",
|
|
338
|
+
services: ["qdrant"]
|
|
339
|
+
});
|
|
340
|
+
globalExpect(result.skillPacks).toEqual([]);
|
|
341
|
+
globalExpect(result.reservedPorts).toEqual([]);
|
|
342
|
+
globalExpect(result.generateSecrets).toBe(true);
|
|
343
|
+
globalExpect(result.platform).toBe("linux/amd64");
|
|
344
|
+
});
|
|
345
|
+
});
|
|
346
|
+
//#endregion
|
|
347
|
+
export {};
|
|
348
|
+
|
|
349
|
+
//# sourceMappingURL=addon-stack.test.mjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"addon-stack.test.mjs","names":["expect"],"sources":["../src/addon-stack.test.ts"],"sourcesContent":["import { parse } from \"yaml\";\nimport { describe, expect, it } from \"vitest\";\nimport { generateAddonStack, updateAddonStack } from \"./addon-stack.js\";\nimport { AddonStackInputSchema } from \"./schema.js\";\n\ndescribe(\"generateAddonStack\", () => {\n\tit(\"generates valid compose YAML with a single service (qdrant)\", () => {\n\t\tconst result = generateAddonStack({\n\t\t\tinstanceId: \"test-instance\",\n\t\t\tservices: [\"qdrant\"],\n\t\t});\n\n\t\t// Resolver may generate warnings about recommended services; that's fine\n\t\texpect(result.metadata.serviceCount).toBeGreaterThan(0);\n\t\texpect(result.metadata.resolvedServices).toContain(\"qdrant\");\n\n\t\t// Parse YAML to verify it's valid\n\t\tconst composed = parse(result.composeOverride);\n\t\texpect(composed.services).toHaveProperty(\"qdrant\");\n\n\t\t// Should NOT contain infrastructure services\n\t\texpect(composed.services).not.toHaveProperty(\"openclaw-gateway\");\n\t\texpect(composed.services).not.toHaveProperty(\"openclaw-cli\");\n\t\texpect(composed.services).not.toHaveProperty(\"redis\");\n\t\texpect(composed.services).not.toHaveProperty(\"postgresql\");\n\t\texpect(composed.services).not.toHaveProperty(\"open-webui\");\n\t\texpect(composed.services).not.toHaveProperty(\"caddy\");\n\n\t\t// Should have openclaw-network as external\n\t\texpect(composed.networks).toHaveProperty(\"openclaw-network\");\n\t\texpect(composed.networks[\"openclaw-network\"].external).toBe(true);\n\t});\n\n\tit(\"does not include profiles on any service\", () => {\n\t\tconst result = generateAddonStack({\n\t\t\tinstanceId: \"test-instance\",\n\t\t\tservices: [\"qdrant\", \"meilisearch\"],\n\t\t});\n\n\t\tconst composed = parse(result.composeOverride);\n\t\tfor (const [, svc] of Object.entries(composed.services)) {\n\t\t\texpect(svc).not.toHaveProperty(\"profiles\");\n\t\t}\n\t});\n\n\tit(\"does not apply cap_drop or security_opt by default\", () => {\n\t\tconst result = generateAddonStack({\n\t\t\tinstanceId: \"test-instance\",\n\t\t\tservices: [\"qdrant\"],\n\t\t});\n\n\t\tconst composed = parse(result.composeOverride);\n\t\tconst qdrant = composed.services.qdrant;\n\t\texpect(qdrant).not.toHaveProperty(\"cap_drop\");\n\t\texpect(qdrant).not.toHaveProperty(\"security_opt\");\n\t});\n\n\tit(\"includes postgres-setup when a DB-dependent service is requested (n8n)\", () => {\n\t\tconst result = generateAddonStack({\n\t\t\tinstanceId: \"test-instance\",\n\t\t\tservices: [\"n8n\"],\n\t\t});\n\n\t\tconst composed = parse(result.composeOverride);\n\t\texpect(composed.services).toHaveProperty(\"n8n\");\n\t\texpect(composed.services).toHaveProperty(\"postgres-setup\");\n\n\t\t// postgres-setup should depend on existing postgresql\n\t\texpect(composed.services[\"postgres-setup\"].depends_on).toHaveProperty(\"postgresql\");\n\n\t\t// n8n should depend on postgres-setup\n\t\texpect(composed.services.n8n.depends_on).toHaveProperty(\"postgres-setup\");\n\t});\n\n\tit(\"generates DB passwords in env file for DB-dependent services\", () => {\n\t\tconst result = generateAddonStack({\n\t\t\tinstanceId: \"test-instance\",\n\t\t\tservices: [\"n8n\"],\n\t\t\tgenerateSecrets: true,\n\t\t});\n\n\t\texpect(result.envFile).toContain(\"N8N_DB_PASSWORD=\");\n\t\t// Password should be non-empty (48 hex chars = 24 bytes)\n\t\tconst match = result.envFile.match(/N8N_DB_PASSWORD=([a-f0-9]+)/);\n\t\texpect(match).not.toBeNull();\n\t\texpect(match![1].length).toBe(48);\n\t});\n\n\tit(\"gracefully handles unknown service IDs without throwing\", () => {\n\t\tconst result = generateAddonStack({\n\t\t\tinstanceId: \"test-instance\",\n\t\t\tservices: [\"nonexistent-service\", \"qdrant\"],\n\t\t});\n\n\t\t// Should NOT throw\n\t\texpect(result.metadata.skippedServices).toEqual(\n\t\t\texpect.arrayContaining([\n\t\t\t\texpect.objectContaining({\n\t\t\t\t\tserviceId: \"nonexistent-service\",\n\t\t\t\t\treason: \"unknown_service\",\n\t\t\t\t}),\n\t\t\t]),\n\t\t);\n\n\t\t// Valid service should still be included\n\t\texpect(result.metadata.resolvedServices).toContain(\"qdrant\");\n\t});\n\n\tit(\"excludes infrastructure services from the request\", () => {\n\t\tconst result = generateAddonStack({\n\t\t\tinstanceId: \"test-instance\",\n\t\t\tservices: [\"redis\", \"postgresql\", \"qdrant\"],\n\t\t});\n\n\t\texpect(result.warnings).toEqual(\n\t\t\texpect.arrayContaining([\n\t\t\t\texpect.stringContaining(\"redis\"),\n\t\t\t\texpect.stringContaining(\"postgresql\"),\n\t\t\t]),\n\t\t);\n\t\texpect(result.metadata.resolvedServices).not.toContain(\"redis\");\n\t\texpect(result.metadata.resolvedServices).not.toContain(\"postgresql\");\n\t\texpect(result.metadata.resolvedServices).toContain(\"qdrant\");\n\t});\n\n\tit(\"generates proxy routes from proxyPath\", () => {\n\t\tconst result = generateAddonStack({\n\t\t\tinstanceId: \"test-instance\",\n\t\t\tservices: [\"n8n\"],\n\t\t});\n\n\t\tconst n8nRoute = result.proxyRoutes.find((r) => r.serviceId === \"n8n\");\n\t\texpect(n8nRoute).toBeDefined();\n\t\texpect(n8nRoute!.path).toBe(\"/n8n\");\n\t\texpect(n8nRoute!.port).toBe(5678);\n\t});\n\n\tit(\"generates skill files and openclaw config patch\", () => {\n\t\tconst result = generateAddonStack({\n\t\t\tinstanceId: \"test-instance\",\n\t\t\tservices: [\"n8n\"],\n\t\t});\n\n\t\t// n8n has skill binding: n8n-trigger\n\t\texpect(result.openclawConfigPatch.skills.entries).toHaveProperty(\"n8n-trigger\");\n\t\texpect(result.openclawConfigPatch.skills.entries[\"n8n-trigger\"].enabled).toBe(true);\n\t});\n\n\tit(\"resolves port conflicts with reserved ports\", () => {\n\t\tconst result = generateAddonStack({\n\t\t\tinstanceId: \"test-instance\",\n\t\t\tservices: [\"n8n\"],\n\t\t\treservedPorts: [5678], // n8n's default port\n\t\t});\n\n\t\t// Port should be reassigned\n\t\tconst portAssignments = result.metadata.portAssignments;\n\t\tconst n8nAssignment = Object.entries(portAssignments).find(([key]) =>\n\t\t\tkey.startsWith(\"n8n:\"),\n\t\t);\n\t\texpect(n8nAssignment).toBeDefined();\n\t\texpect(n8nAssignment![1]).not.toBe(5678);\n\t});\n\n\tit(\"sanitizes project name from instanceId\", () => {\n\t\tconst result = generateAddonStack({\n\t\t\tinstanceId: \"My_Instance_123\",\n\t\t\tservices: [\"qdrant\"],\n\t\t});\n\n\t\t// Should succeed without error\n\t\texpect(result.metadata.serviceCount).toBeGreaterThan(0);\n\t});\n\n\tit(\"returns env vars grouped by service\", () => {\n\t\tconst result = generateAddonStack({\n\t\t\tinstanceId: \"test-instance\",\n\t\t\tservices: [\"qdrant\"],\n\t\t});\n\n\t\texpect(result.envVars.length).toBeGreaterThanOrEqual(0);\n\t});\n\n\tit(\"returns empty result for no services\", () => {\n\t\tconst result = generateAddonStack({\n\t\t\tinstanceId: \"test-instance\",\n\t\t\tservices: [],\n\t\t});\n\n\t\texpect(result.composeOverride).toContain(\"services: {}\");\n\t\texpect(result.warnings.length).toBeGreaterThan(0);\n\t});\n\n\tit(\"returns empty result when all services are infrastructure\", () => {\n\t\tconst result = generateAddonStack({\n\t\t\tinstanceId: \"test-instance\",\n\t\t\tservices: [\"redis\", \"postgresql\", \"caddy\"],\n\t\t});\n\n\t\texpect(result.metadata.serviceCount).toBe(0);\n\t\texpect(result.warnings.length).toBeGreaterThan(0);\n\t});\n\n\tit(\"applies env quirks to envFile output (meilisearch MEILI_MASTER_KEY min_length)\", () => {\n\t\tconst result = generateAddonStack({\n\t\t\tinstanceId: \"test-instance\",\n\t\t\tservices: [\"meilisearch\"],\n\t\t\tgenerateSecrets: true,\n\t\t});\n\n\t\t// MEILI_MASTER_KEY should be a base64url string of at least 32 chars (24 bytes)\n\t\tconst match = result.envFile.match(/MEILI_MASTER_KEY=([^\\n]+)/);\n\t\texpect(match).not.toBeNull();\n\t\texpect(match![1].length).toBeGreaterThanOrEqual(32);\n\t\t// Should not be empty\n\t\texpect(match![1]).not.toBe(\"\");\n\t});\n\n\tit(\"applies env quirks to envFile output (grafana GF_SECURITY_ADMIN_PASSWORD)\", () => {\n\t\tconst result = generateAddonStack({\n\t\t\tinstanceId: \"test-instance\",\n\t\t\tservices: [\"grafana\"],\n\t\t\tgenerateSecrets: true,\n\t\t});\n\n\t\tconst match = result.envFile.match(/GF_SECURITY_ADMIN_PASSWORD=([^\\n]+)/);\n\t\texpect(match).not.toBeNull();\n\t\t// Should be at least 22 chars (16 bytes base64url)\n\t\texpect(match![1].length).toBeGreaterThanOrEqual(22);\n\t});\n\n\tit(\"syncs DB_POSTGRESDB_PASSWORD with N8N_DB_PASSWORD via must_sync quirk\", () => {\n\t\tconst result = generateAddonStack({\n\t\t\tinstanceId: \"test-instance\",\n\t\t\tservices: [\"n8n\"],\n\t\t\tgenerateSecrets: true,\n\t\t});\n\n\t\tconst n8nDbPw = result.envFile.match(/N8N_DB_PASSWORD=([^\\n]+)/);\n\t\tconst dbPostgresPw = result.envFile.match(/DB_POSTGRESDB_PASSWORD=([^\\n]+)/);\n\t\texpect(n8nDbPw).not.toBeNull();\n\t\texpect(dbPostgresPw).not.toBeNull();\n\t\t// Both passwords must match due to must_sync quirk\n\t\texpect(n8nDbPw![1]).toBe(dbPostgresPw![1]);\n\t});\n\n\tit(\"syncs user-provided credential with DB password reference\", () => {\n\t\tconst customPassword = \"my_custom_secure_password\";\n\t\tconst result = generateAddonStack({\n\t\t\tinstanceId: \"test-instance\",\n\t\t\tservices: [\"n8n\"],\n\t\t\tgenerateSecrets: true,\n\t\t\tcredentials: {\n\t\t\t\tn8n: { DB_POSTGRESDB_PASSWORD: customPassword },\n\t\t\t},\n\t\t});\n\n\t\t// User value should be used for both the service env var and the ref key\n\t\texpect(result.envFile).toContain(`DB_POSTGRESDB_PASSWORD=${customPassword}`);\n\t\texpect(result.envFile).toContain(`N8N_DB_PASSWORD=${customPassword}`);\n\t});\n\n\tit(\"uses existingServices ports for conflict detection\", () => {\n\t\tconst result = generateAddonStack({\n\t\t\tinstanceId: \"test-instance\",\n\t\t\tservices: [\"n8n\"],\n\t\t\t// n8n uses port 5678, qdrant uses 6333 — qdrant as existing should not conflict\n\t\t\texistingServices: [\"qdrant\"],\n\t\t});\n\n\t\t// n8n should still get its default port (no conflict with qdrant)\n\t\tconst portAssignments = result.metadata.portAssignments;\n\t\tconst n8nAssignment = Object.entries(portAssignments).find(([key]) =>\n\t\t\tkey.startsWith(\"n8n:\"),\n\t\t);\n\t\texpect(n8nAssignment).toBeDefined();\n\t\texpect(n8nAssignment![1]).toBe(5678);\n\t});\n\n\tit(\"does not dual-list GPU services in both skipped and resolved\", () => {\n\t\tconst result = generateAddonStack({\n\t\t\tinstanceId: \"test-instance\",\n\t\t\tservices: [\"ollama\"],\n\t\t\tgpu: false,\n\t\t});\n\n\t\t// ollama requires GPU — without gpu: true, it should be skipped only\n\t\tconst isSkipped = result.metadata.skippedServices.some(\n\t\t\t(s) => s.serviceId === \"ollama\" && s.reason === \"gpu_required\",\n\t\t);\n\t\tconst isResolved = result.metadata.resolvedServices.includes(\"ollama\");\n\n\t\t// Must be in exactly one list, not both\n\t\tif (isSkipped) {\n\t\t\texpect(isResolved).toBe(false);\n\t\t}\n\t});\n\n\tit(\"includes GPU services when gpu: true is set\", () => {\n\t\tconst result = generateAddonStack({\n\t\t\tinstanceId: \"test-instance\",\n\t\t\tservices: [\"ollama\"],\n\t\t\tgpu: true,\n\t\t});\n\n\t\t// Should NOT be skipped when GPU support is available\n\t\tconst isSkipped = result.metadata.skippedServices.some(\n\t\t\t(s) => s.serviceId === \"ollama\" && s.reason === \"gpu_required\",\n\t\t);\n\t\texpect(isSkipped).toBe(false);\n\t\texpect(result.metadata.resolvedServices).toContain(\"ollama\");\n\t});\n});\n\ndescribe(\"updateAddonStack\", () => {\n\tit(\"adds a service to an existing stack\", () => {\n\t\t// First generate a base stack\n\t\tconst base = generateAddonStack({\n\t\t\tinstanceId: \"test-instance\",\n\t\t\tservices: [\"qdrant\"],\n\t\t});\n\n\t\t// Then add meilisearch\n\t\tconst result = updateAddonStack({\n\t\t\tinstanceId: \"test-instance\",\n\t\t\tcurrentCompose: base.composeOverride,\n\t\t\tcurrentEnv: base.envFile,\n\t\t\taddServices: [\"meilisearch\"],\n\t\t});\n\n\t\texpect(result.metadata.added).toContain(\"meilisearch\");\n\t\texpect(result.metadata.unchanged).toContain(\"qdrant\");\n\n\t\tconst composed = parse(result.composeOverride);\n\t\texpect(composed.services).toHaveProperty(\"qdrant\");\n\t\texpect(composed.services).toHaveProperty(\"meilisearch\");\n\t});\n\n\tit(\"removes a service from an existing stack\", () => {\n\t\tconst base = generateAddonStack({\n\t\t\tinstanceId: \"test-instance\",\n\t\t\tservices: [\"qdrant\", \"meilisearch\"],\n\t\t});\n\n\t\tconst result = updateAddonStack({\n\t\t\tinstanceId: \"test-instance\",\n\t\t\tcurrentCompose: base.composeOverride,\n\t\t\tcurrentEnv: base.envFile,\n\t\t\tremoveServices: [\"meilisearch\"],\n\t\t});\n\n\t\texpect(result.metadata.removed).toContain(\"meilisearch\");\n\t\texpect(result.metadata.unchanged).toContain(\"qdrant\");\n\n\t\tconst composed = parse(result.composeOverride);\n\t\texpect(composed.services).toHaveProperty(\"qdrant\");\n\t\texpect(composed.services).not.toHaveProperty(\"meilisearch\");\n\t});\n\n\tit(\"preserves existing env values when adding a service\", () => {\n\t\tconst base = generateAddonStack({\n\t\t\tinstanceId: \"test-instance\",\n\t\t\tservices: [\"qdrant\"],\n\t\t\tgenerateSecrets: true,\n\t\t});\n\n\t\t// Simulate user editing an env value\n\t\tconst customEnv = base.envFile.replace(\n\t\t\t/QDRANT_API_KEY=[a-f0-9]+/,\n\t\t\t\"QDRANT_API_KEY=my_custom_key\",\n\t\t);\n\n\t\tconst result = updateAddonStack({\n\t\t\tinstanceId: \"test-instance\",\n\t\t\tcurrentCompose: base.composeOverride,\n\t\t\tcurrentEnv: customEnv,\n\t\t\taddServices: [\"meilisearch\"],\n\t\t\tgenerateSecrets: true,\n\t\t});\n\n\t\t// If QDRANT_API_KEY was in the original env, it should be preserved\n\t\tif (customEnv.includes(\"QDRANT_API_KEY=my_custom_key\")) {\n\t\t\texpect(result.envFile).toContain(\"QDRANT_API_KEY=my_custom_key\");\n\t\t}\n\t});\n\n\tit(\"returns images to pull for added services\", () => {\n\t\tconst base = generateAddonStack({\n\t\t\tinstanceId: \"test-instance\",\n\t\t\tservices: [\"qdrant\"],\n\t\t});\n\n\t\tconst result = updateAddonStack({\n\t\t\tinstanceId: \"test-instance\",\n\t\t\tcurrentCompose: base.composeOverride,\n\t\t\tcurrentEnv: base.envFile,\n\t\t\taddServices: [\"meilisearch\"],\n\t\t});\n\n\t\texpect(result.imagesToPull.length).toBeGreaterThan(0);\n\t\texpect(result.imagesToPull.some((img) => img.includes(\"meilisearch\"))).toBe(true);\n\t});\n\n\tit(\"handles empty update gracefully\", () => {\n\t\tconst base = generateAddonStack({\n\t\t\tinstanceId: \"test-instance\",\n\t\t\tservices: [\"qdrant\"],\n\t\t});\n\n\t\tconst result = updateAddonStack({\n\t\t\tinstanceId: \"test-instance\",\n\t\t\tcurrentCompose: base.composeOverride,\n\t\t\tcurrentEnv: base.envFile,\n\t\t});\n\n\t\texpect(result.metadata.added).toEqual([]);\n\t\texpect(result.metadata.removed).toEqual([]);\n\t\texpect(result.metadata.unchanged).toContain(\"qdrant\");\n\t});\n\n\tit(\"respects generateSecrets: false during update\", () => {\n\t\tconst base = generateAddonStack({\n\t\t\tinstanceId: \"test-instance\",\n\t\t\tservices: [\"qdrant\"],\n\t\t\tgenerateSecrets: true,\n\t\t});\n\n\t\t// With generateSecrets: false, services requiring secrets (like meilisearch)\n\t\t// may be skipped as missing_credentials. Provide credentials explicitly.\n\t\tconst result = updateAddonStack({\n\t\t\tinstanceId: \"test-instance\",\n\t\t\tcurrentCompose: base.composeOverride,\n\t\t\tcurrentEnv: base.envFile,\n\t\t\taddServices: [\"meilisearch\"],\n\t\t\tgenerateSecrets: false,\n\t\t\tcredentials: {\n\t\t\t\tmeilisearch: { MEILI_MASTER_KEY: \"test-master-key-1234567890\" },\n\t\t\t},\n\t\t});\n\n\t\t// Meilisearch should be added since we provided the required credential\n\t\texpect(result.metadata.added).toContain(\"meilisearch\");\n\t});\n\n\tit(\"forwards portOverrides during update\", () => {\n\t\tconst base = generateAddonStack({\n\t\t\tinstanceId: \"test-instance\",\n\t\t\tservices: [\"qdrant\"],\n\t\t});\n\n\t\tconst result = updateAddonStack({\n\t\t\tinstanceId: \"test-instance\",\n\t\t\tcurrentCompose: base.composeOverride,\n\t\t\tcurrentEnv: base.envFile,\n\t\t\taddServices: [\"n8n\"],\n\t\t\tportOverrides: { n8n: { \"5678\": 9999 } },\n\t\t});\n\n\t\t// n8n should be present with the port override\n\t\texpect(result.metadata.added).toContain(\"n8n\");\n\t});\n});\n\ndescribe(\"AddonStackInputSchema\", () => {\n\tit(\"accepts valid input\", () => {\n\t\tconst result = AddonStackInputSchema.safeParse({\n\t\t\tinstanceId: \"test-instance\",\n\t\t\tservices: [\"qdrant\", \"n8n\"],\n\t\t});\n\t\texpect(result.success).toBe(true);\n\t});\n\n\tit(\"requires instanceId\", () => {\n\t\tconst result = AddonStackInputSchema.safeParse({\n\t\t\tservices: [\"qdrant\"],\n\t\t});\n\t\texpect(result.success).toBe(false);\n\t});\n\n\tit(\"defaults empty arrays and booleans\", () => {\n\t\tconst result = AddonStackInputSchema.parse({\n\t\t\tinstanceId: \"test\",\n\t\t\tservices: [\"qdrant\"],\n\t\t});\n\t\texpect(result.skillPacks).toEqual([]);\n\t\texpect(result.reservedPorts).toEqual([]);\n\t\texpect(result.generateSecrets).toBe(true);\n\t\texpect(result.platform).toBe(\"linux/amd64\");\n\t});\n});\n"],"mappings":"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"}
|
|
@@ -1,15 +1,15 @@
|
|
|
1
|
-
const
|
|
2
|
-
const require_bare_metal_partition = require("./bare-metal-partition.cjs");
|
|
1
|
+
const require_test_CTcmp4Su = require("./test.CTcmp4Su-DlzTarwH.cjs");
|
|
3
2
|
const require_resolver = require("./resolver.cjs");
|
|
3
|
+
const require_bare_metal_partition = require("./bare-metal-partition.cjs");
|
|
4
4
|
//#region src/bare-metal-partition.test.ts
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
5
|
+
require_test_CTcmp4Su.describe("bare-metal partition", () => {
|
|
6
|
+
require_test_CTcmp4Su.it("platformToNativePlatform maps platform to native", () => {
|
|
7
|
+
require_test_CTcmp4Su.globalExpect(require_bare_metal_partition.platformToNativePlatform("linux/amd64")).toBe("linux");
|
|
8
|
+
require_test_CTcmp4Su.globalExpect(require_bare_metal_partition.platformToNativePlatform("linux/arm64")).toBe("linux");
|
|
9
|
+
require_test_CTcmp4Su.globalExpect(require_bare_metal_partition.platformToNativePlatform("windows/amd64")).toBe("windows");
|
|
10
|
+
require_test_CTcmp4Su.globalExpect(require_bare_metal_partition.platformToNativePlatform("macos/arm64")).toBe("macos");
|
|
11
11
|
});
|
|
12
|
-
|
|
12
|
+
require_test_CTcmp4Su.it("partitions redis as native on Linux when redis has native recipe", () => {
|
|
13
13
|
const result = require_bare_metal_partition.partitionBareMetal(require_resolver.resolve({
|
|
14
14
|
services: ["redis"],
|
|
15
15
|
skillPacks: [],
|
|
@@ -18,12 +18,11 @@ require_vi_2VT5v0um.describe("bare-metal partition", () => {
|
|
|
18
18
|
platform: "linux/amd64",
|
|
19
19
|
monitoring: false
|
|
20
20
|
}), "linux/amd64");
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
require_vi_2VT5v0um.globalExpect(result.dockerOnlyServices.length).toBe(0);
|
|
21
|
+
require_test_CTcmp4Su.globalExpect(result.nativeIds.has("redis")).toBe(true);
|
|
22
|
+
require_test_CTcmp4Su.globalExpect(result.nativeServices.find((s) => s.definition.id === "redis")).toBeDefined();
|
|
23
|
+
require_test_CTcmp4Su.globalExpect(result.dockerOnlyServices.length).toBeGreaterThanOrEqual(0);
|
|
25
24
|
});
|
|
26
|
-
|
|
25
|
+
require_test_CTcmp4Su.it("partitions n8n as Docker-only (no native recipe)", () => {
|
|
27
26
|
const result = require_bare_metal_partition.partitionBareMetal(require_resolver.resolve({
|
|
28
27
|
services: ["n8n"],
|
|
29
28
|
skillPacks: [],
|
|
@@ -32,12 +31,12 @@ require_vi_2VT5v0um.describe("bare-metal partition", () => {
|
|
|
32
31
|
platform: "linux/amd64",
|
|
33
32
|
monitoring: false
|
|
34
33
|
}), "linux/amd64");
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
34
|
+
require_test_CTcmp4Su.globalExpect(result.nativeIds.has("n8n")).toBe(false);
|
|
35
|
+
require_test_CTcmp4Su.globalExpect(result.nativeServices.length).toBe(0);
|
|
36
|
+
require_test_CTcmp4Su.globalExpect(result.dockerOnlyServices.length).toBeGreaterThanOrEqual(1);
|
|
37
|
+
require_test_CTcmp4Su.globalExpect(result.dockerOnlyServices.some((s) => s.definition.id === "n8n")).toBe(true);
|
|
39
38
|
});
|
|
40
|
-
|
|
39
|
+
require_test_CTcmp4Su.it("resolvedWithOnlyServices filters to given service list", () => {
|
|
41
40
|
const resolved = require_resolver.resolve({
|
|
42
41
|
services: ["redis", "n8n"],
|
|
43
42
|
skillPacks: [],
|
|
@@ -47,8 +46,8 @@ require_vi_2VT5v0um.describe("bare-metal partition", () => {
|
|
|
47
46
|
monitoring: false
|
|
48
47
|
});
|
|
49
48
|
const filtered = require_bare_metal_partition.resolvedWithOnlyServices(resolved, require_bare_metal_partition.partitionBareMetal(resolved, "linux/amd64").dockerOnlyServices);
|
|
50
|
-
|
|
51
|
-
|
|
49
|
+
require_test_CTcmp4Su.globalExpect(filtered.services.every((s) => s.definition.id !== "redis")).toBe(true);
|
|
50
|
+
require_test_CTcmp4Su.globalExpect(filtered.services.some((s) => s.definition.id === "n8n")).toBe(true);
|
|
52
51
|
});
|
|
53
52
|
});
|
|
54
53
|
//#endregion
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"bare-metal-partition.test.cjs","names":["describe","platformToNativePlatform","partitionBareMetal","resolve","resolvedWithOnlyServices"],"sources":["../src/bare-metal-partition.test.ts"],"sourcesContent":["import { describe, expect, it } from \"vitest\";\nimport {\n\tpartitionBareMetal,\n\tplatformToNativePlatform,\n\tresolvedWithOnlyServices,\n} from \"./bare-metal-partition.js\";\nimport { resolve } from \"./resolver.js\";\n\ndescribe(\"bare-metal partition\", () => {\n\tit(\"platformToNativePlatform maps platform to native\", () => {\n\t\texpect(platformToNativePlatform(\"linux/amd64\")).toBe(\"linux\");\n\t\texpect(platformToNativePlatform(\"linux/arm64\")).toBe(\"linux\");\n\t\texpect(platformToNativePlatform(\"windows/amd64\")).toBe(\"windows\");\n\t\texpect(platformToNativePlatform(\"macos/arm64\")).toBe(\"macos\");\n\t});\n\n\tit(\"partitions redis as native on Linux when redis has native recipe\", () => {\n\t\tconst resolved = resolve({\n\t\t\tservices: [\"redis\"],\n\t\t\tskillPacks: [],\n\t\t\tproxy: \"none\",\n\t\t\tgpu: false,\n\t\t\tplatform: \"linux/amd64\",\n\t\t\tmonitoring: false,\n\t\t});\n\t\tconst result = partitionBareMetal(resolved, \"linux/amd64\");\n\t\texpect(result.nativeIds.has(\"redis\")).toBe(true);\n\t\
|
|
1
|
+
{"version":3,"file":"bare-metal-partition.test.cjs","names":["describe","platformToNativePlatform","partitionBareMetal","resolve","resolvedWithOnlyServices"],"sources":["../src/bare-metal-partition.test.ts"],"sourcesContent":["import { describe, expect, it } from \"vitest\";\nimport {\n\tpartitionBareMetal,\n\tplatformToNativePlatform,\n\tresolvedWithOnlyServices,\n} from \"./bare-metal-partition.js\";\nimport { resolve } from \"./resolver.js\";\n\ndescribe(\"bare-metal partition\", () => {\n\tit(\"platformToNativePlatform maps platform to native\", () => {\n\t\texpect(platformToNativePlatform(\"linux/amd64\")).toBe(\"linux\");\n\t\texpect(platformToNativePlatform(\"linux/arm64\")).toBe(\"linux\");\n\t\texpect(platformToNativePlatform(\"windows/amd64\")).toBe(\"windows\");\n\t\texpect(platformToNativePlatform(\"macos/arm64\")).toBe(\"macos\");\n\t});\n\n\tit(\"partitions redis as native on Linux when redis has native recipe\", () => {\n\t\tconst resolved = resolve({\n\t\t\tservices: [\"redis\"],\n\t\t\tskillPacks: [],\n\t\t\tproxy: \"none\",\n\t\t\tgpu: false,\n\t\t\tplatform: \"linux/amd64\",\n\t\t\tmonitoring: false,\n\t\t});\n\t\tconst result = partitionBareMetal(resolved, \"linux/amd64\");\n\t\texpect(result.nativeIds.has(\"redis\")).toBe(true);\n\t\tconst redisNative = result.nativeServices.find((s) => s.definition.id === \"redis\");\n\t\texpect(redisNative).toBeDefined();\n\t\t// Mandatory services (convex, mission-control, tailscale) are Docker-only\n\t\texpect(result.dockerOnlyServices.length).toBeGreaterThanOrEqual(0);\n\t});\n\n\tit(\"partitions n8n as Docker-only (no native recipe)\", () => {\n\t\tconst resolved = resolve({\n\t\t\tservices: [\"n8n\"],\n\t\t\tskillPacks: [],\n\t\t\tproxy: \"none\",\n\t\t\tgpu: false,\n\t\t\tplatform: \"linux/amd64\",\n\t\t\tmonitoring: false,\n\t\t});\n\t\tconst result = partitionBareMetal(resolved, \"linux/amd64\");\n\t\texpect(result.nativeIds.has(\"n8n\")).toBe(false);\n\t\texpect(result.nativeServices.length).toBe(0);\n\t\texpect(result.dockerOnlyServices.length).toBeGreaterThanOrEqual(1);\n\t\texpect(result.dockerOnlyServices.some((s) => s.definition.id === \"n8n\")).toBe(true);\n\t});\n\n\tit(\"resolvedWithOnlyServices filters to given service list\", () => {\n\t\tconst resolved = resolve({\n\t\t\tservices: [\"redis\", \"n8n\"],\n\t\t\tskillPacks: [],\n\t\t\tproxy: \"none\",\n\t\t\tgpu: false,\n\t\t\tplatform: \"linux/amd64\",\n\t\t\tmonitoring: false,\n\t\t});\n\t\tconst partition = partitionBareMetal(resolved, \"linux/amd64\");\n\t\tconst filtered = resolvedWithOnlyServices(resolved, partition.dockerOnlyServices);\n\t\texpect(filtered.services.every((s) => s.definition.id !== \"redis\")).toBe(true);\n\t\texpect(filtered.services.some((s) => s.definition.id === \"n8n\")).toBe(true);\n\t});\n});\n"],"mappings":";;;;AAQAA,sBAAAA,SAAS,8BAA8B;AACtC,uBAAA,GAAG,0DAA0D;AAC5D,wBAAA,aAAOC,6BAAAA,yBAAyB,cAAc,CAAC,CAAC,KAAK,QAAQ;AAC7D,wBAAA,aAAOA,6BAAAA,yBAAyB,cAAc,CAAC,CAAC,KAAK,QAAQ;AAC7D,wBAAA,aAAOA,6BAAAA,yBAAyB,gBAAgB,CAAC,CAAC,KAAK,UAAU;AACjE,wBAAA,aAAOA,6BAAAA,yBAAyB,cAAc,CAAC,CAAC,KAAK,QAAQ;GAC5D;AAEF,uBAAA,GAAG,0EAA0E;EAS5E,MAAM,SAASC,6BAAAA,mBAREC,iBAAAA,QAAQ;GACxB,UAAU,CAAC,QAAQ;GACnB,YAAY,EAAE;GACd,OAAO;GACP,KAAK;GACL,UAAU;GACV,YAAY;GACZ,CAAC,EAC0C,cAAc;AAC1D,wBAAA,aAAO,OAAO,UAAU,IAAI,QAAQ,CAAC,CAAC,KAAK,KAAK;AAEhD,wBAAA,aADoB,OAAO,eAAe,MAAM,MAAM,EAAE,WAAW,OAAO,QAAQ,CAC/D,CAAC,aAAa;AAEjC,wBAAA,aAAO,OAAO,mBAAmB,OAAO,CAAC,uBAAuB,EAAE;GACjE;AAEF,uBAAA,GAAG,0DAA0D;EAS5D,MAAM,SAASD,6BAAAA,mBAREC,iBAAAA,QAAQ;GACxB,UAAU,CAAC,MAAM;GACjB,YAAY,EAAE;GACd,OAAO;GACP,KAAK;GACL,UAAU;GACV,YAAY;GACZ,CAAC,EAC0C,cAAc;AAC1D,wBAAA,aAAO,OAAO,UAAU,IAAI,MAAM,CAAC,CAAC,KAAK,MAAM;AAC/C,wBAAA,aAAO,OAAO,eAAe,OAAO,CAAC,KAAK,EAAE;AAC5C,wBAAA,aAAO,OAAO,mBAAmB,OAAO,CAAC,uBAAuB,EAAE;AAClE,wBAAA,aAAO,OAAO,mBAAmB,MAAM,MAAM,EAAE,WAAW,OAAO,MAAM,CAAC,CAAC,KAAK,KAAK;GAClF;AAEF,uBAAA,GAAG,gEAAgE;EAClE,MAAM,WAAWA,iBAAAA,QAAQ;GACxB,UAAU,CAAC,SAAS,MAAM;GAC1B,YAAY,EAAE;GACd,OAAO;GACP,KAAK;GACL,UAAU;GACV,YAAY;GACZ,CAAC;EAEF,MAAM,WAAWC,6BAAAA,yBAAyB,UADxBF,6BAAAA,mBAAmB,UAAU,cAAc,CACC,mBAAmB;AACjF,wBAAA,aAAO,SAAS,SAAS,OAAO,MAAM,EAAE,WAAW,OAAO,QAAQ,CAAC,CAAC,KAAK,KAAK;AAC9E,wBAAA,aAAO,SAAS,SAAS,MAAM,MAAM,EAAE,WAAW,OAAO,MAAM,CAAC,CAAC,KAAK,KAAK;GAC1E;EACD"}
|
|
@@ -1,6 +1,6 @@
|
|
|
1
|
-
import { n as describe, r as it, t as globalExpect } from "./
|
|
2
|
-
import { partitionBareMetal, platformToNativePlatform, resolvedWithOnlyServices } from "./bare-metal-partition.mjs";
|
|
1
|
+
import { n as describe, r as it, t as globalExpect } from "./test.CTcmp4Su-ClCHJ3FA.mjs";
|
|
3
2
|
import { resolve } from "./resolver.mjs";
|
|
3
|
+
import { partitionBareMetal, platformToNativePlatform, resolvedWithOnlyServices } from "./bare-metal-partition.mjs";
|
|
4
4
|
//#region src/bare-metal-partition.test.ts
|
|
5
5
|
describe("bare-metal partition", () => {
|
|
6
6
|
it("platformToNativePlatform maps platform to native", () => {
|
|
@@ -19,9 +19,8 @@ describe("bare-metal partition", () => {
|
|
|
19
19
|
monitoring: false
|
|
20
20
|
}), "linux/amd64");
|
|
21
21
|
globalExpect(result.nativeIds.has("redis")).toBe(true);
|
|
22
|
-
globalExpect(result.nativeServices.
|
|
23
|
-
globalExpect(result.
|
|
24
|
-
globalExpect(result.dockerOnlyServices.length).toBe(0);
|
|
22
|
+
globalExpect(result.nativeServices.find((s) => s.definition.id === "redis")).toBeDefined();
|
|
23
|
+
globalExpect(result.dockerOnlyServices.length).toBeGreaterThanOrEqual(0);
|
|
25
24
|
});
|
|
26
25
|
it("partitions n8n as Docker-only (no native recipe)", () => {
|
|
27
26
|
const result = partitionBareMetal(resolve({
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"bare-metal-partition.test.mjs","names":[],"sources":["../src/bare-metal-partition.test.ts"],"sourcesContent":["import { describe, expect, it } from \"vitest\";\nimport {\n\tpartitionBareMetal,\n\tplatformToNativePlatform,\n\tresolvedWithOnlyServices,\n} from \"./bare-metal-partition.js\";\nimport { resolve } from \"./resolver.js\";\n\ndescribe(\"bare-metal partition\", () => {\n\tit(\"platformToNativePlatform maps platform to native\", () => {\n\t\texpect(platformToNativePlatform(\"linux/amd64\")).toBe(\"linux\");\n\t\texpect(platformToNativePlatform(\"linux/arm64\")).toBe(\"linux\");\n\t\texpect(platformToNativePlatform(\"windows/amd64\")).toBe(\"windows\");\n\t\texpect(platformToNativePlatform(\"macos/arm64\")).toBe(\"macos\");\n\t});\n\n\tit(\"partitions redis as native on Linux when redis has native recipe\", () => {\n\t\tconst resolved = resolve({\n\t\t\tservices: [\"redis\"],\n\t\t\tskillPacks: [],\n\t\t\tproxy: \"none\",\n\t\t\tgpu: false,\n\t\t\tplatform: \"linux/amd64\",\n\t\t\tmonitoring: false,\n\t\t});\n\t\tconst result = partitionBareMetal(resolved, \"linux/amd64\");\n\t\texpect(result.nativeIds.has(\"redis\")).toBe(true);\n\t\
|
|
1
|
+
{"version":3,"file":"bare-metal-partition.test.mjs","names":[],"sources":["../src/bare-metal-partition.test.ts"],"sourcesContent":["import { describe, expect, it } from \"vitest\";\nimport {\n\tpartitionBareMetal,\n\tplatformToNativePlatform,\n\tresolvedWithOnlyServices,\n} from \"./bare-metal-partition.js\";\nimport { resolve } from \"./resolver.js\";\n\ndescribe(\"bare-metal partition\", () => {\n\tit(\"platformToNativePlatform maps platform to native\", () => {\n\t\texpect(platformToNativePlatform(\"linux/amd64\")).toBe(\"linux\");\n\t\texpect(platformToNativePlatform(\"linux/arm64\")).toBe(\"linux\");\n\t\texpect(platformToNativePlatform(\"windows/amd64\")).toBe(\"windows\");\n\t\texpect(platformToNativePlatform(\"macos/arm64\")).toBe(\"macos\");\n\t});\n\n\tit(\"partitions redis as native on Linux when redis has native recipe\", () => {\n\t\tconst resolved = resolve({\n\t\t\tservices: [\"redis\"],\n\t\t\tskillPacks: [],\n\t\t\tproxy: \"none\",\n\t\t\tgpu: false,\n\t\t\tplatform: \"linux/amd64\",\n\t\t\tmonitoring: false,\n\t\t});\n\t\tconst result = partitionBareMetal(resolved, \"linux/amd64\");\n\t\texpect(result.nativeIds.has(\"redis\")).toBe(true);\n\t\tconst redisNative = result.nativeServices.find((s) => s.definition.id === \"redis\");\n\t\texpect(redisNative).toBeDefined();\n\t\t// Mandatory services (convex, mission-control, tailscale) are Docker-only\n\t\texpect(result.dockerOnlyServices.length).toBeGreaterThanOrEqual(0);\n\t});\n\n\tit(\"partitions n8n as Docker-only (no native recipe)\", () => {\n\t\tconst resolved = resolve({\n\t\t\tservices: [\"n8n\"],\n\t\t\tskillPacks: [],\n\t\t\tproxy: \"none\",\n\t\t\tgpu: false,\n\t\t\tplatform: \"linux/amd64\",\n\t\t\tmonitoring: false,\n\t\t});\n\t\tconst result = partitionBareMetal(resolved, \"linux/amd64\");\n\t\texpect(result.nativeIds.has(\"n8n\")).toBe(false);\n\t\texpect(result.nativeServices.length).toBe(0);\n\t\texpect(result.dockerOnlyServices.length).toBeGreaterThanOrEqual(1);\n\t\texpect(result.dockerOnlyServices.some((s) => s.definition.id === \"n8n\")).toBe(true);\n\t});\n\n\tit(\"resolvedWithOnlyServices filters to given service list\", () => {\n\t\tconst resolved = resolve({\n\t\t\tservices: [\"redis\", \"n8n\"],\n\t\t\tskillPacks: [],\n\t\t\tproxy: \"none\",\n\t\t\tgpu: false,\n\t\t\tplatform: \"linux/amd64\",\n\t\t\tmonitoring: false,\n\t\t});\n\t\tconst partition = partitionBareMetal(resolved, \"linux/amd64\");\n\t\tconst filtered = resolvedWithOnlyServices(resolved, partition.dockerOnlyServices);\n\t\texpect(filtered.services.every((s) => s.definition.id !== \"redis\")).toBe(true);\n\t\texpect(filtered.services.some((s) => s.definition.id === \"n8n\")).toBe(true);\n\t});\n});\n"],"mappings":";;;;AAQA,SAAS,8BAA8B;AACtC,IAAG,0DAA0D;AAC5D,eAAO,yBAAyB,cAAc,CAAC,CAAC,KAAK,QAAQ;AAC7D,eAAO,yBAAyB,cAAc,CAAC,CAAC,KAAK,QAAQ;AAC7D,eAAO,yBAAyB,gBAAgB,CAAC,CAAC,KAAK,UAAU;AACjE,eAAO,yBAAyB,cAAc,CAAC,CAAC,KAAK,QAAQ;GAC5D;AAEF,IAAG,0EAA0E;EAS5E,MAAM,SAAS,mBARE,QAAQ;GACxB,UAAU,CAAC,QAAQ;GACnB,YAAY,EAAE;GACd,OAAO;GACP,KAAK;GACL,UAAU;GACV,YAAY;GACZ,CAAC,EAC0C,cAAc;AAC1D,eAAO,OAAO,UAAU,IAAI,QAAQ,CAAC,CAAC,KAAK,KAAK;AAEhD,eADoB,OAAO,eAAe,MAAM,MAAM,EAAE,WAAW,OAAO,QAAQ,CAC/D,CAAC,aAAa;AAEjC,eAAO,OAAO,mBAAmB,OAAO,CAAC,uBAAuB,EAAE;GACjE;AAEF,IAAG,0DAA0D;EAS5D,MAAM,SAAS,mBARE,QAAQ;GACxB,UAAU,CAAC,MAAM;GACjB,YAAY,EAAE;GACd,OAAO;GACP,KAAK;GACL,UAAU;GACV,YAAY;GACZ,CAAC,EAC0C,cAAc;AAC1D,eAAO,OAAO,UAAU,IAAI,MAAM,CAAC,CAAC,KAAK,MAAM;AAC/C,eAAO,OAAO,eAAe,OAAO,CAAC,KAAK,EAAE;AAC5C,eAAO,OAAO,mBAAmB,OAAO,CAAC,uBAAuB,EAAE;AAClE,eAAO,OAAO,mBAAmB,MAAM,MAAM,EAAE,WAAW,OAAO,MAAM,CAAC,CAAC,KAAK,KAAK;GAClF;AAEF,IAAG,gEAAgE;EAClE,MAAM,WAAW,QAAQ;GACxB,UAAU,CAAC,SAAS,MAAM;GAC1B,YAAY,EAAE;GACd,OAAO;GACP,KAAK;GACL,UAAU;GACV,YAAY;GACZ,CAAC;EAEF,MAAM,WAAW,yBAAyB,UADxB,mBAAmB,UAAU,cAAc,CACC,mBAAmB;AACjF,eAAO,SAAS,SAAS,OAAO,MAAM,EAAE,WAAW,OAAO,QAAQ,CAAC,CAAC,KAAK,KAAK;AAC9E,eAAO,SAAS,SAAS,MAAM,MAAM,EAAE,WAAW,OAAO,MAAM,CAAC,CAAC,KAAK,KAAK;GAC1E;EACD"}
|
package/dist/composer.cjs
CHANGED
|
@@ -59,6 +59,10 @@ const CATEGORY_PROFILE_MAP = {
|
|
|
59
59
|
communication: {
|
|
60
60
|
file: "docker-compose.communication.yml",
|
|
61
61
|
profile: "communication"
|
|
62
|
+
},
|
|
63
|
+
"saas-boilerplate": {
|
|
64
|
+
file: "docker-compose.saas.yml",
|
|
65
|
+
profile: "saas"
|
|
62
66
|
}
|
|
63
67
|
};
|
|
64
68
|
const YAML_OPTIONS = {
|
|
@@ -174,7 +178,15 @@ function buildGatewayServices(resolved, options, dependsOn) {
|
|
|
174
178
|
function buildCompanionService(def, resolved, options, allVolumes) {
|
|
175
179
|
const svc = {};
|
|
176
180
|
const volumeNames = [];
|
|
177
|
-
|
|
181
|
+
if (def.gitSource && def.buildContext) {
|
|
182
|
+
const subdir = def.gitSource.subdirectory || ".";
|
|
183
|
+
const ctxPath = def.buildContext.context || ".";
|
|
184
|
+
const buildBlock = { context: subdir === "." ? `./repos/${def.id}/${ctxPath}` : `./repos/${def.id}/${subdir}/${ctxPath}` };
|
|
185
|
+
if (def.buildContext.dockerfile) buildBlock.dockerfile = def.buildContext.dockerfile;
|
|
186
|
+
if (def.buildContext.args && Object.keys(def.buildContext.args).length > 0) buildBlock.args = def.buildContext.args;
|
|
187
|
+
if (def.buildContext.target) buildBlock.target = def.buildContext.target;
|
|
188
|
+
svc.build = buildBlock;
|
|
189
|
+
} else svc.image = `${def.image}:${def.imageTag}`;
|
|
178
190
|
if (def.environment.length > 0) {
|
|
179
191
|
const env = {};
|
|
180
192
|
for (const e of def.environment) env[e.key] = e.secret ? `\${${e.key}}` : e.defaultValue;
|
|
@@ -469,7 +481,11 @@ function composeMultiFile(resolved, options) {
|
|
|
469
481
|
};
|
|
470
482
|
}
|
|
471
483
|
//#endregion
|
|
484
|
+
exports.YAML_OPTIONS = YAML_OPTIONS;
|
|
485
|
+
exports.buildCompanionService = buildCompanionService;
|
|
486
|
+
exports.buildPostgresSetup = buildPostgresSetup;
|
|
472
487
|
exports.compose = compose;
|
|
473
488
|
exports.composeMultiFile = composeMultiFile;
|
|
489
|
+
exports.quotedStr = quotedStr;
|
|
474
490
|
|
|
475
491
|
//# sourceMappingURL=composer.cjs.map
|