import { spawnSync } from "node:child_process"; import { chmod, copyFile, mkdir, mkdtemp, readFile, writeFile } from "node:fs/promises"; import { tmpdir } from "node:os"; import { join, resolve } from "node:path"; import { fileURLToPath } from "node:url"; import { describe, expect, it } from "vitest"; const repoRoot = resolve(fileURLToPath(new URL(".", import.meta.url)), ".."); type DockerSetupSandbox = { rootDir: string; scriptPath: string; logPath: string; binDir: string; }; async function writeDockerStub(binDir: string, logPath: string) { const stub = `#!/usr/bin/env bash set -euo pipefail log="$DOCKER_STUB_LOG" if [[ "\${1:-}" == "compose" && "\${2:-}" == "version" ]]; then exit 0 fi if [[ "\${1:-}" == "build" ]]; then echo "build $*" >>"$log" exit 0 fi if [[ "\${1:-}" == "compose" ]]; then echo "compose $*" >>"$log" exit 0 fi echo "unknown $*" >>"$log" exit 0 `; await mkdir(binDir, { recursive: true }); await writeFile(join(binDir, "docker"), stub, { mode: 0o755 }); await writeFile(logPath, ""); } async function createDockerSetupSandbox(): Promise { const rootDir = await mkdtemp(join(tmpdir(), "openclaw-docker-setup-")); const scriptPath = join(rootDir, "docker-setup.sh"); const dockerfilePath = join(rootDir, "Dockerfile"); const composePath = join(rootDir, "docker-compose.yml"); const binDir = join(rootDir, "bin"); const logPath = join(rootDir, "docker-stub.log"); await copyFile(join(repoRoot, "docker-setup.sh"), scriptPath); await chmod(scriptPath, 0o755); await writeFile(dockerfilePath, "FROM scratch\n"); await writeFile( composePath, "services:\n openclaw-gateway:\n image: noop\n openclaw-cli:\n image: noop\n", ); await writeDockerStub(binDir, logPath); return { rootDir, scriptPath, logPath, binDir }; } function createEnv( sandbox: DockerSetupSandbox, overrides: Record = {}, ): NodeJS.ProcessEnv { return { ...process.env, PATH: `${sandbox.binDir}:${process.env.PATH ?? ""}`, DOCKER_STUB_LOG: sandbox.logPath, OPENCLAW_GATEWAY_TOKEN: "test-token", OPENCLAW_CONFIG_DIR: join(sandbox.rootDir, "config"), OPENCLAW_WORKSPACE_DIR: join(sandbox.rootDir, "openclaw"), ...overrides, }; } function resolveBashForCompatCheck(): string | null { for (const candidate of ["/bin/bash", "bash"]) { const probe = spawnSync(candidate, ["-c", "exit 0"], { encoding: "utf8" }); if (!probe.error && probe.status === 0) { return candidate; } } return null; } describe("docker-setup.sh", () => { it("handles env defaults, home-volume mounts, and apt build args", async () => { const sandbox = await createDockerSetupSandbox(); const result = spawnSync("bash", [sandbox.scriptPath], { cwd: sandbox.rootDir, env: createEnv(sandbox, { OPENCLAW_DOCKER_APT_PACKAGES: "ffmpeg build-essential", OPENCLAW_EXTRA_MOUNTS: undefined, OPENCLAW_HOME_VOLUME: "openclaw-home", }), stdio: ["ignore", "ignore", "pipe"], }); expect(result.status).toBe(0); const envFile = await readFile(join(sandbox.rootDir, ".env"), "utf8"); expect(envFile).toContain("OPENCLAW_DOCKER_APT_PACKAGES=ffmpeg build-essential"); expect(envFile).toContain("OPENCLAW_EXTRA_MOUNTS="); expect(envFile).toContain("OPENCLAW_HOME_VOLUME=openclaw-home"); const extraCompose = await readFile(join(sandbox.rootDir, "docker-compose.extra.yml"), "utf8"); expect(extraCompose).toContain("openclaw-home:/home/node"); expect(extraCompose).toContain("volumes:"); expect(extraCompose).toContain("openclaw-home:"); const log = await readFile(sandbox.logPath, "utf8"); expect(log).toContain("--build-arg OPENCLAW_DOCKER_APT_PACKAGES=ffmpeg build-essential"); }); it("avoids associative arrays so the script remains Bash 3.2-compatible", async () => { const script = await readFile(join(repoRoot, "docker-setup.sh"), "utf8"); expect(script).not.toMatch(/^\s*declare -A\b/m); const systemBash = resolveBashForCompatCheck(); if (!systemBash) { return; } const assocCheck = spawnSync(systemBash, ["-c", "declare -A _t=()"], { encoding: "utf8", }); if (assocCheck.status === 0 || assocCheck.status === null) { // Skip runtime check when system bash supports associative arrays // (not Bash 3.2) or when /bin/bash is unavailable (e.g. Windows). return; } const syntaxCheck = spawnSync(systemBash, ["-n", join(repoRoot, "docker-setup.sh")], { encoding: "utf8", }); expect(syntaxCheck.status).toBe(0); expect(syntaxCheck.stderr).not.toContain("declare: -A: invalid option"); }); it("keeps docker-compose gateway command in sync", async () => { const compose = await readFile(join(repoRoot, "docker-compose.yml"), "utf8"); expect(compose).not.toContain("gateway-daemon"); expect(compose).toContain('"gateway"'); }); });