import { spawnSync } from "node:child_process"; import { describe, expect, it } from "vitest"; import { DEFAULT_RESOURCE_LIMITS } from "../../scripts/lib/docker-e2e-plan.mjs"; import { canStartSchedulerLane, describeDockerSchedulerLimits, dockerPreflightContainerNames, parseDockerAllCliArgs, } from "../../scripts/test-docker-all.mjs"; const limits = { resourceLimits: { docker: 2, npm: 2, }, weightLimit: 2, }; function activePool({ count = 0, resources = {}, weight = 0, }: { count?: number; resources?: Record; weight?: number; } = {}) { return { count, resources: new Map(Object.entries(resources)), weight, }; } describe("scripts/test-docker-all scheduler", () => { it("parses the supported CLI options", () => { expect(parseDockerAllCliArgs([])).toEqual({ help: false, planJson: false, }); expect(parseDockerAllCliArgs(["--plan-json"])).toEqual({ help: false, planJson: true, }); expect(parseDockerAllCliArgs(["--help"])).toEqual({ help: true, planJson: false, }); }); it("prints CLI help without a stack trace", () => { const result = spawnSync(process.execPath, ["scripts/test-docker-all.mjs", "--help"], { cwd: process.cwd(), encoding: "utf8", }); expect(result.status).toBe(0); expect(result.stderr).toBe(""); expect(result.stdout).toContain("Usage: node scripts/test-docker-all.mjs [--plan-json]"); expect(result.stdout).toContain("OPENCLAW_DOCKER_ALL_* env vars"); }); it("rejects unknown CLI options without a stack trace", () => { const result = spawnSync(process.execPath, ["scripts/test-docker-all.mjs", "--bogus"], { cwd: process.cwd(), encoding: "utf8", }); expect(result.status).toBe(1); expect(result.stdout).toBe(""); expect(result.stderr).toContain("unknown argument: --bogus"); expect(result.stderr).toContain("Usage: node scripts/test-docker-all.mjs [--plan-json]"); expect(result.stderr).not.toContain("at "); }); it("allows an overweight lane to start alone under low parallelism", () => { expect( canStartSchedulerLane( { name: "install-e2e", resources: ["npm"], weight: 4, }, activePool(), 2, limits, ), ).toBe(true); }); it("does not co-schedule another lane while an overweight lane is active", () => { expect( canStartSchedulerLane( { name: "package-update", resources: ["npm"], weight: 1, }, activePool({ count: 1, resources: { docker: 4, npm: 4, }, weight: 4, }), 2, limits, ), ).toBe(false); }); it("can co-schedule the split installer provider lanes", () => { expect( canStartSchedulerLane( { name: "install-e2e-anthropic", resources: ["npm", "service"], weight: 3, }, activePool({ count: 1, resources: { docker: 3, npm: 3, service: 3, }, weight: 3, }), 10, { resourceLimits: { docker: 10, npm: 10, service: 7, }, weightLimit: 10, }, ), ).toBe(true); }); it("preserves the parallelism count cap", () => { expect( canStartSchedulerLane( { name: "package-update", resources: ["npm"], weight: 1, }, activePool({ count: 2, resources: { docker: 1, npm: 1, }, weight: 1, }), 2, limits, ), ).toBe(false); }); it("keeps resource and weight limits as co-scheduling limits", () => { expect( canStartSchedulerLane( { name: "npm-smoke", resources: ["npm"], weight: 1, }, activePool({ count: 1, resources: { docker: 1, npm: 1, }, weight: 1, }), 2, limits, ), ).toBe(true); expect( canStartSchedulerLane( { name: "npm-heavy", resources: ["npm"], weight: 2, }, activePool({ count: 1, resources: { docker: 1, npm: 1, }, weight: 1, }), 2, limits, ), ).toBe(false); }); it("serializes live OpenAI Docker lanes by default", () => { expect(DEFAULT_RESOURCE_LIMITS["live:openai"]).toBe(1); }); it("cleans stale stopped containers from all named Docker E2E lanes", () => { expect( dockerPreflightContainerNames(` openclaw-gateway-e2e-123 Exited (1) 2 minutes ago openclaw-config-reload-e2e-234 Created openclaw-plugin-binding-command-escape-e2e-345 Dead openclaw-kitchen-sink-rpc-e2e-456 Exited (137) 10 seconds ago openclaw-openwebui-gateway-567 Exited (1) 3 minutes ago openclaw-openwebui-678 Created openclaw-not-an-e2e-container Exited (1) 2 minutes ago postgres Created `), ).toEqual([ "openclaw-gateway-e2e-123", "openclaw-config-reload-e2e-234", "openclaw-plugin-binding-command-escape-e2e-345", "openclaw-kitchen-sink-rpc-e2e-456", "openclaw-openwebui-gateway-567", "openclaw-openwebui-678", ]); }); it("describes effective scheduler limits for operator errors", () => { expect(describeDockerSchedulerLimits(2, limits)).toBe( "parallelism=2 weightLimit=2 resources=docker=2 npm=2", ); }); });