mirror of
https://github.com/openclaw/openclaw.git
synced 2026-05-27 17:55:58 +00:00
139 lines
3.6 KiB
TypeScript
139 lines
3.6 KiB
TypeScript
import { execFileSync } from "node:child_process";
|
|
import { chmodSync, mkdtempSync, rmSync, writeFileSync } from "node:fs";
|
|
import { tmpdir } from "node:os";
|
|
import path from "node:path";
|
|
import { afterEach, describe, expect, it } from "vitest";
|
|
|
|
const tempDirCleanups: Array<() => void> = [];
|
|
|
|
function makeTempBin(prefix: string) {
|
|
const dir = mkdtempSync(path.join(tmpdir(), prefix));
|
|
tempDirCleanups.push(() => {
|
|
rmSync(dir, { force: true, recursive: true });
|
|
});
|
|
return dir;
|
|
}
|
|
|
|
function writeExecutable(filePath: string, contents: string) {
|
|
writeFileSync(filePath, contents, "utf8");
|
|
chmodSync(filePath, 0o755);
|
|
}
|
|
|
|
function resolveDockerRunArgs(pathPrefix: string) {
|
|
const script = [
|
|
"source scripts/lib/live-docker-auth.sh",
|
|
"ARGS=()",
|
|
"openclaw_live_init_docker_run_args ARGS 42s",
|
|
"printf '%s\\n' \"${ARGS[@]}\"",
|
|
].join("\n");
|
|
|
|
const output = execFileSync("/bin/bash", ["-c", script], {
|
|
cwd: process.cwd(),
|
|
encoding: "utf8",
|
|
env: {
|
|
...process.env,
|
|
PATH: pathPrefix,
|
|
},
|
|
}).trimEnd();
|
|
return output ? output.split("\n") : [];
|
|
}
|
|
|
|
function failDockerRunArgs(pathPrefix: string) {
|
|
const script = [
|
|
"source scripts/lib/live-docker-auth.sh",
|
|
"ARGS=()",
|
|
"openclaw_live_init_docker_run_args ARGS 42s",
|
|
].join("\n");
|
|
|
|
try {
|
|
execFileSync("/bin/bash", ["-c", script], {
|
|
cwd: process.cwd(),
|
|
encoding: "utf8",
|
|
env: {
|
|
...process.env,
|
|
PATH: pathPrefix,
|
|
},
|
|
stdio: ["ignore", "pipe", "pipe"],
|
|
});
|
|
} catch (error) {
|
|
return error as { status?: number; stderr?: Buffer | string };
|
|
}
|
|
throw new Error("Expected live Docker run arg initialization to fail");
|
|
}
|
|
|
|
afterEach(() => {
|
|
while (tempDirCleanups.length > 0) {
|
|
tempDirCleanups.pop()!();
|
|
}
|
|
});
|
|
|
|
describe("scripts/lib/live-docker-auth.sh", () => {
|
|
it("adds a kill-after grace period when timeout supports it", () => {
|
|
const binDir = makeTempBin("openclaw-live-docker-auth-gnu-");
|
|
writeExecutable(
|
|
path.join(binDir, "timeout"),
|
|
[
|
|
"#!/bin/sh",
|
|
'if [ "$1" = "--kill-after=1s" ] && [ "$2" = "1s" ] && [ "$3" = "true" ]; then',
|
|
" exit 0",
|
|
"fi",
|
|
"exit 64",
|
|
"",
|
|
].join("\n"),
|
|
);
|
|
|
|
expect(resolveDockerRunArgs(binDir)).toEqual([
|
|
"timeout",
|
|
"--kill-after=30s",
|
|
"42s",
|
|
"docker",
|
|
"run",
|
|
]);
|
|
});
|
|
|
|
it("falls back to plain timeout when kill-after is unavailable", () => {
|
|
const binDir = makeTempBin("openclaw-live-docker-auth-plain-");
|
|
writeExecutable(
|
|
path.join(binDir, "timeout"),
|
|
["#!/bin/sh", 'if [ "$1" = "--kill-after=1s" ]; then', " exit 1", "fi", "exit 0", ""].join(
|
|
"\n",
|
|
),
|
|
);
|
|
|
|
expect(resolveDockerRunArgs(binDir)).toEqual(["timeout", "42s", "docker", "run"]);
|
|
});
|
|
|
|
it("uses gtimeout when timeout is unavailable", () => {
|
|
const binDir = makeTempBin("openclaw-live-docker-auth-gtimeout-");
|
|
writeExecutable(
|
|
path.join(binDir, "gtimeout"),
|
|
[
|
|
"#!/bin/sh",
|
|
'if [ "$1" = "--kill-after=1s" ] && [ "$2" = "1s" ] && [ "$3" = "true" ]; then',
|
|
" exit 0",
|
|
"fi",
|
|
"exit 64",
|
|
"",
|
|
].join("\n"),
|
|
);
|
|
|
|
expect(resolveDockerRunArgs(binDir)).toEqual([
|
|
"gtimeout",
|
|
"--kill-after=30s",
|
|
"42s",
|
|
"docker",
|
|
"run",
|
|
]);
|
|
});
|
|
|
|
it("fails fast when timeout is unavailable", () => {
|
|
const binDir = makeTempBin("openclaw-live-docker-auth-no-timeout-");
|
|
const error = failDockerRunArgs(binDir);
|
|
|
|
expect(error.status).toBe(127);
|
|
expect(String(error.stderr)).toContain(
|
|
"timeout command not found; cannot bound live Docker run after 42s",
|
|
);
|
|
});
|
|
});
|