mirror of
https://github.com/openclaw/openclaw.git
synced 2026-06-22 22:58:07 +00:00
709 lines
24 KiB
TypeScript
709 lines
24 KiB
TypeScript
// Plugins Assertions tests cover plugins assertions script behavior.
|
|
import { spawn, spawnSync } from "node:child_process";
|
|
import {
|
|
chmodSync,
|
|
existsSync,
|
|
mkdirSync,
|
|
mkdtempSync,
|
|
readFileSync,
|
|
rmSync,
|
|
writeFileSync,
|
|
} from "node:fs";
|
|
import { createServer } from "node:http";
|
|
import { tmpdir } from "node:os";
|
|
import path from "node:path";
|
|
import { describe, expect, it } from "vitest";
|
|
import { createBoundedChildOutput } from "../helpers/bounded-child-output.js";
|
|
|
|
const ASSERTIONS_SCRIPT = "scripts/e2e/lib/plugins/assertions.mjs";
|
|
|
|
function shellQuote(value: string): string {
|
|
return `'${value.replace(/'/gu, `'\\''`)}'`;
|
|
}
|
|
|
|
function writeJson(filePath: string, value: unknown) {
|
|
mkdirSync(path.dirname(filePath), { recursive: true });
|
|
writeFileSync(filePath, `${JSON.stringify(value, null, 2)}\n`, "utf8");
|
|
}
|
|
|
|
function runAssertionAsync(args: string[], env: NodeJS.ProcessEnv) {
|
|
return new Promise<{ status: number | null; stdout: string; stderr: string }>(
|
|
(resolve, reject) => {
|
|
const child = spawn(process.execPath, [ASSERTIONS_SCRIPT, ...args], {
|
|
env: { ...process.env, ...env },
|
|
stdio: ["ignore", "pipe", "pipe"],
|
|
});
|
|
const stdout = createBoundedChildOutput();
|
|
const stderr = createBoundedChildOutput();
|
|
const timeout = setTimeout(() => {
|
|
child.kill("SIGKILL");
|
|
reject(new Error(`assertion helper did not exit: ${args.join(" ")}`));
|
|
}, 2_000);
|
|
timeout.unref();
|
|
|
|
child.stdout.setEncoding("utf8");
|
|
child.stderr.setEncoding("utf8");
|
|
child.stdout.on("data", (chunk) => {
|
|
stdout.append(chunk);
|
|
});
|
|
child.stderr.on("data", (chunk) => {
|
|
stderr.append(chunk);
|
|
});
|
|
child.on("error", (error) => {
|
|
clearTimeout(timeout);
|
|
reject(error);
|
|
});
|
|
child.on("close", (status) => {
|
|
clearTimeout(timeout);
|
|
resolve({ status, stdout: stdout.text(), stderr: stderr.text() });
|
|
});
|
|
},
|
|
);
|
|
}
|
|
|
|
function writeFixtureServerShims(binDir: string, pidPath: string): void {
|
|
mkdirSync(binDir, { recursive: true });
|
|
writeFileSync(
|
|
path.join(binDir, "node"),
|
|
[
|
|
"#!/bin/bash",
|
|
'printf "%s\\n" "$$" >"$OPENCLAW_TEST_FIXTURE_SERVER_PID"',
|
|
"trap 'exit 0' TERM",
|
|
"while true; do /bin/sleep 1; done",
|
|
"",
|
|
].join("\n"),
|
|
);
|
|
writeFileSync(path.join(binDir, "sleep"), "#!/bin/bash\nexit 0\n");
|
|
chmodSync(path.join(binDir, "node"), 0o755);
|
|
chmodSync(path.join(binDir, "sleep"), 0o755);
|
|
writeFileSync(pidPath, "");
|
|
}
|
|
|
|
function writeCrashingFixtureServerShim(binDir: string): void {
|
|
mkdirSync(binDir, { recursive: true });
|
|
writeFileSync(
|
|
path.join(binDir, "node"),
|
|
[
|
|
"#!/bin/bash",
|
|
'printf "DO_NOT_DUMP_PLUGIN_FIXTURE_PREFIX\\n"',
|
|
'printf "%2048s" "" | tr " " x',
|
|
'printf "\\nPLUGIN_FIXTURE_TAIL_MARKER\\n"',
|
|
"exit 1",
|
|
"",
|
|
].join("\n"),
|
|
);
|
|
writeFileSync(path.join(binDir, "sleep"), "#!/bin/bash\nexit 0\n");
|
|
chmodSync(path.join(binDir, "node"), 0o755);
|
|
chmodSync(path.join(binDir, "sleep"), 0o755);
|
|
}
|
|
|
|
function isProcessAlive(pid: number): boolean {
|
|
try {
|
|
process.kill(pid, 0);
|
|
return true;
|
|
} catch {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
function waitForDead(pid: number, timeoutMs = 2_000): void {
|
|
const startedAt = Date.now();
|
|
while (isProcessAlive(pid)) {
|
|
if (Date.now() - startedAt > timeoutMs) {
|
|
throw new Error(`pid ${pid} is still alive`);
|
|
}
|
|
Atomics.wait(new Int32Array(new SharedArrayBuffer(4)), 0, 0, 20);
|
|
}
|
|
}
|
|
|
|
function runPluginsSweepShell(script: string, env: NodeJS.ProcessEnv = {}) {
|
|
return spawnSync("/bin/bash", ["-c", script], {
|
|
cwd: process.cwd(),
|
|
encoding: "utf8",
|
|
env: { ...process.env, ...env },
|
|
});
|
|
}
|
|
|
|
describe("plugins Docker assertions", () => {
|
|
it("rejects loose ClawHub preflight limits instead of parsing prefixes", () => {
|
|
const timeoutResult = spawnSync(process.execPath, [ASSERTIONS_SCRIPT, "clawhub-preflight"], {
|
|
encoding: "utf8",
|
|
env: {
|
|
...process.env,
|
|
CLAWHUB_PLUGIN_SPEC: "clawhub:@openclaw/kitchen-sink",
|
|
OPENCLAW_PLUGINS_E2E_CLAWHUB_PREFLIGHT_TIMEOUT_MS: "1e3",
|
|
},
|
|
});
|
|
expect(timeoutResult.status).not.toBe(0);
|
|
expect(timeoutResult.stderr).toContain(
|
|
"invalid OPENCLAW_PLUGINS_E2E_CLAWHUB_PREFLIGHT_TIMEOUT_MS: 1e3",
|
|
);
|
|
|
|
const bodyLimitResult = spawnSync(process.execPath, [ASSERTIONS_SCRIPT, "clawhub-preflight"], {
|
|
encoding: "utf8",
|
|
env: {
|
|
...process.env,
|
|
CLAWHUB_PLUGIN_SPEC: "clawhub:@openclaw/kitchen-sink",
|
|
OPENCLAW_PLUGINS_E2E_CLAWHUB_PREFLIGHT_BODY_MAX_BYTES: "1000bytes",
|
|
},
|
|
});
|
|
expect(bodyLimitResult.status).not.toBe(0);
|
|
expect(bodyLimitResult.stderr).toContain(
|
|
"invalid OPENCLAW_PLUGINS_E2E_CLAWHUB_PREFLIGHT_BODY_MAX_BYTES: 1000bytes",
|
|
);
|
|
});
|
|
|
|
it("keeps sweep artifact paths aligned with the assertion scratch root", () => {
|
|
const scripts = [
|
|
"scripts/e2e/lib/plugins/sweep.sh",
|
|
"scripts/e2e/lib/plugins/marketplace.sh",
|
|
"scripts/e2e/lib/plugins/clawhub.sh",
|
|
];
|
|
|
|
for (const scriptPath of scripts) {
|
|
const script = readFileSync(scriptPath, "utf8");
|
|
const scriptWithoutDefaultScratch = script.replace(
|
|
'mktemp -d "/tmp/openclaw-plugins.XXXXXX"',
|
|
"",
|
|
);
|
|
expect(script).toContain("OPENCLAW_PLUGINS_TMP_DIR");
|
|
expect(scriptWithoutDefaultScratch).not.toMatch(
|
|
/\/tmp\/(?:plugins|marketplace|demo-plugin|is-number|openclaw-plugin|openclaw-clawhub)/,
|
|
);
|
|
}
|
|
});
|
|
|
|
it("cleans the default plugin sweep scratch root", () => {
|
|
const root = mkdtempSync(path.join(tmpdir(), "openclaw-plugin-sweep-cleanup-"));
|
|
const marker = path.join(root, "scratch-path.txt");
|
|
try {
|
|
const result = runPluginsSweepShell(
|
|
`
|
|
set -euo pipefail
|
|
export OPENCLAW_PLUGINS_SWEEP_SOURCE_ONLY=1
|
|
source scripts/e2e/lib/plugins/sweep.sh
|
|
printf '%s\\n' "$OPENCLAW_PLUGINS_TMP_DIR" > "$MARKER"
|
|
test -d "$OPENCLAW_PLUGINS_TMP_DIR"
|
|
cleanup_openclaw_plugins_sweep
|
|
test ! -e "$OPENCLAW_PLUGINS_TMP_DIR"
|
|
`,
|
|
{ MARKER: marker },
|
|
);
|
|
|
|
expect(result.stdout).toBe("");
|
|
expect(result.stderr).toBe("");
|
|
expect(result.status).toBe(0);
|
|
const scratchRoot = readFileSync(marker, "utf8").trim();
|
|
expect(scratchRoot).toContain("/tmp/openclaw-plugins.");
|
|
expect(existsSync(scratchRoot)).toBe(false);
|
|
} finally {
|
|
rmSync(root, { force: true, recursive: true });
|
|
}
|
|
});
|
|
|
|
it("preserves caller-provided plugin sweep scratch roots", () => {
|
|
const root = mkdtempSync(path.join(tmpdir(), "openclaw-plugin-sweep-caller-"));
|
|
const scratchRoot = path.join(root, "scratch");
|
|
try {
|
|
const result = runPluginsSweepShell(
|
|
`
|
|
set -euo pipefail
|
|
export OPENCLAW_PLUGINS_SWEEP_SOURCE_ONLY=1
|
|
export OPENCLAW_PLUGINS_TMP_DIR="$SCRATCH_ROOT"
|
|
source scripts/e2e/lib/plugins/sweep.sh
|
|
test -d "$OPENCLAW_PLUGINS_TMP_DIR"
|
|
cleanup_openclaw_plugins_sweep
|
|
test -d "$OPENCLAW_PLUGINS_TMP_DIR"
|
|
`,
|
|
{ SCRATCH_ROOT: scratchRoot },
|
|
);
|
|
|
|
expect(result.stdout).toBe("");
|
|
expect(result.stderr).toBe("");
|
|
expect(result.status).toBe(0);
|
|
expect(existsSync(scratchRoot)).toBe(true);
|
|
} finally {
|
|
rmSync(root, { force: true, recursive: true });
|
|
}
|
|
});
|
|
|
|
it("scans plugin assertion logs without echoing whole files on failure", async () => {
|
|
const root = mkdtempSync(path.join(tmpdir(), "openclaw-plugin-update-log-"));
|
|
try {
|
|
const passRoot = path.join(root, "pass");
|
|
mkdirSync(passRoot, { recursive: true });
|
|
writeFileSync(
|
|
path.join(passRoot, "plugins-dir-update.log"),
|
|
`Skipping "demo-plugin-dir" (source: path).\n${"x".repeat(256 * 1024)}`,
|
|
"utf8",
|
|
);
|
|
const pass = await runAssertionAsync(["plugin-dir-update-skipped"], {
|
|
OPENCLAW_PLUGINS_TMP_DIR: passRoot,
|
|
});
|
|
expect(pass.status).toBe(0);
|
|
|
|
const failRoot = path.join(root, "fail");
|
|
mkdirSync(failRoot, { recursive: true });
|
|
writeFileSync(
|
|
path.join(failRoot, "plugins-dir-update.log"),
|
|
`${"x".repeat(256 * 1024)}\nmissing marker tail`,
|
|
"utf8",
|
|
);
|
|
const fail = await runAssertionAsync(["plugin-dir-update-skipped"], {
|
|
OPENCLAW_PLUGINS_TMP_DIR: failRoot,
|
|
});
|
|
expect(fail.status).toBe(1);
|
|
expect(fail.stderr).toContain("Output tail:");
|
|
expect(fail.stderr).toContain("missing marker tail");
|
|
expect(fail.stderr.length).toBeLessThan(20 * 1024);
|
|
|
|
const invalidRoot = path.join(root, "invalid");
|
|
const invalidHome = path.join(root, "home");
|
|
mkdirSync(invalidRoot, { recursive: true });
|
|
mkdirSync(invalidHome, { recursive: true });
|
|
writeFileSync(
|
|
path.join(invalidRoot, "plugins-invalid-openclaw-extensions.log"),
|
|
`openclaw.extensions[1]\n${"x".repeat(256 * 1024)}\nmissing validation tail`,
|
|
"utf8",
|
|
);
|
|
writeJson(path.join(invalidRoot, "plugins-invalid-openclaw-extensions-list.json"), {
|
|
plugins: [],
|
|
});
|
|
const invalid = await runAssertionAsync(["invalid-openclaw-extensions"], {
|
|
HOME: invalidHome,
|
|
OPENCLAW_PLUGINS_TMP_DIR: invalidRoot,
|
|
});
|
|
expect(invalid.status).toBe(1);
|
|
expect(invalid.stderr).toContain("malformed metadata install output");
|
|
expect(invalid.stderr).toContain("missing validation tail");
|
|
expect(invalid.stderr.length).toBeLessThan(20 * 1024);
|
|
} finally {
|
|
rmSync(root, { recursive: true, force: true });
|
|
}
|
|
});
|
|
|
|
it("cleans npm fixture registry children when readiness times out", () => {
|
|
const root = mkdtempSync(path.join(tmpdir(), "openclaw-plugin-npm-fixture-cleanup-"));
|
|
try {
|
|
const binDir = path.join(root, "bin");
|
|
const fixtureDir = path.join(root, "fixture");
|
|
const cleanupPath = path.join(root, "caller-cleanup");
|
|
const pidPath = path.join(root, "server.pid");
|
|
mkdirSync(fixtureDir);
|
|
writeFixtureServerShims(binDir, pidPath);
|
|
|
|
const result = spawnSync(
|
|
"/bin/bash",
|
|
[
|
|
"-c",
|
|
[
|
|
"set -euo pipefail",
|
|
"source scripts/e2e/lib/plugins/fixtures.sh",
|
|
"set +e",
|
|
`( set -e; trap 'printf caller-cleanup > ${shellQuote(cleanupPath)}' EXIT; start_npm_fixture_registry fixture-pkg 1.0.0 ${shellQuote(path.join(root, "fixture.tgz"))} ${shellQuote(fixtureDir)} )`,
|
|
'status="$?"',
|
|
"set -e",
|
|
'[ "$status" != "0" ]',
|
|
].join("\n"),
|
|
],
|
|
{
|
|
cwd: process.cwd(),
|
|
encoding: "utf8",
|
|
env: {
|
|
...process.env,
|
|
OPENCLAW_TEST_FIXTURE_SERVER_PID: pidPath,
|
|
PATH: `${binDir}${path.delimiter}/usr/bin${path.delimiter}/bin`,
|
|
},
|
|
},
|
|
);
|
|
|
|
expect(result.status, result.stderr || result.stdout).toBe(0);
|
|
const pid = Number(readFileSync(pidPath, "utf8"));
|
|
expect(Number.isInteger(pid)).toBe(true);
|
|
waitForDead(pid);
|
|
expect(readFileSync(cleanupPath, "utf8")).toBe("caller-cleanup");
|
|
} finally {
|
|
rmSync(root, { force: true, recursive: true });
|
|
}
|
|
});
|
|
|
|
it("bounds npm fixture registry logs when readiness fails", () => {
|
|
const root = mkdtempSync(path.join(tmpdir(), "openclaw-plugin-npm-fixture-log-"));
|
|
try {
|
|
const binDir = path.join(root, "bin");
|
|
const fixtureDir = path.join(root, "fixture");
|
|
mkdirSync(fixtureDir);
|
|
writeCrashingFixtureServerShim(binDir);
|
|
|
|
const result = spawnSync(
|
|
"/bin/bash",
|
|
[
|
|
"-c",
|
|
[
|
|
"set -euo pipefail",
|
|
"source scripts/e2e/lib/plugins/fixtures.sh",
|
|
"set +e",
|
|
`start_npm_fixture_registry fixture-pkg 1.0.0 ${shellQuote(path.join(root, "fixture.tgz"))} ${shellQuote(fixtureDir)}`,
|
|
'status="$?"',
|
|
"set -e",
|
|
'printf "status=%s\\n" "$status"',
|
|
'[ "$status" != "0" ]',
|
|
].join("\n"),
|
|
],
|
|
{
|
|
cwd: process.cwd(),
|
|
encoding: "utf8",
|
|
env: {
|
|
...process.env,
|
|
OPENCLAW_DOCKER_E2E_LOG_PRINT_BYTES: "80",
|
|
PATH: `${binDir}${path.delimiter}/usr/bin${path.delimiter}/bin`,
|
|
},
|
|
},
|
|
);
|
|
|
|
expect(result.status, result.stderr || result.stdout).toBe(0);
|
|
expect(result.stdout).toContain("truncated: showing last 80");
|
|
expect(result.stdout).toContain("PLUGIN_FIXTURE_TAIL_MARKER");
|
|
expect(result.stdout).not.toContain("DO_NOT_DUMP_PLUGIN_FIXTURE_PREFIX");
|
|
} finally {
|
|
rmSync(root, { force: true, recursive: true });
|
|
}
|
|
});
|
|
|
|
it("cleans ClawHub fixture children when readiness times out", () => {
|
|
const root = mkdtempSync(path.join(tmpdir(), "openclaw-plugin-clawhub-fixture-cleanup-"));
|
|
try {
|
|
const binDir = path.join(root, "bin");
|
|
const cleanupPath = path.join(root, "caller-cleanup");
|
|
const tmpDir = path.join(root, "scratch");
|
|
const pidPath = path.join(root, "server.pid");
|
|
mkdirSync(tmpDir);
|
|
writeFixtureServerShims(binDir, pidPath);
|
|
|
|
const result = spawnSync(
|
|
"/bin/bash",
|
|
[
|
|
"-c",
|
|
[
|
|
"set -euo pipefail",
|
|
"source scripts/e2e/lib/plugins/fixtures.sh",
|
|
"source scripts/e2e/lib/plugins/clawhub.sh",
|
|
"set +e",
|
|
`( set -e; trap 'printf caller-cleanup > ${shellQuote(cleanupPath)}' EXIT; run_plugins_clawhub_scenario )`,
|
|
'status="$?"',
|
|
"set -e",
|
|
'[ "$status" != "0" ]',
|
|
].join("\n"),
|
|
],
|
|
{
|
|
cwd: process.cwd(),
|
|
encoding: "utf8",
|
|
env: {
|
|
...process.env,
|
|
OPENCLAW_PLUGINS_E2E_LIVE_CLAWHUB: "0",
|
|
OPENCLAW_PLUGINS_TMP_DIR: tmpDir,
|
|
OPENCLAW_TEST_FIXTURE_SERVER_PID: pidPath,
|
|
PATH: `${binDir}${path.delimiter}/usr/bin${path.delimiter}/bin`,
|
|
},
|
|
},
|
|
);
|
|
|
|
expect(result.status, result.stderr || result.stdout).toBe(0);
|
|
const pid = Number(readFileSync(pidPath, "utf8"));
|
|
expect(Number.isInteger(pid)).toBe(true);
|
|
waitForDead(pid);
|
|
expect(readFileSync(cleanupPath, "utf8")).toBe("caller-cleanup");
|
|
} finally {
|
|
rmSync(root, { force: true, recursive: true });
|
|
}
|
|
});
|
|
|
|
it("bounds ClawHub fixture server logs when readiness fails", () => {
|
|
const root = mkdtempSync(path.join(tmpdir(), "openclaw-plugin-clawhub-fixture-log-"));
|
|
try {
|
|
const binDir = path.join(root, "bin");
|
|
const tmpDir = path.join(root, "scratch");
|
|
mkdirSync(tmpDir);
|
|
writeCrashingFixtureServerShim(binDir);
|
|
|
|
const result = spawnSync(
|
|
"/bin/bash",
|
|
[
|
|
"-c",
|
|
[
|
|
"set -euo pipefail",
|
|
"source scripts/e2e/lib/plugins/fixtures.sh",
|
|
"source scripts/e2e/lib/plugins/clawhub.sh",
|
|
"set +e",
|
|
"run_plugins_clawhub_scenario",
|
|
'status="$?"',
|
|
"set -e",
|
|
'printf "status=%s\\n" "$status"',
|
|
'[ "$status" != "0" ]',
|
|
].join("\n"),
|
|
],
|
|
{
|
|
cwd: process.cwd(),
|
|
encoding: "utf8",
|
|
env: {
|
|
...process.env,
|
|
OPENCLAW_DOCKER_E2E_LOG_PRINT_BYTES: "80",
|
|
OPENCLAW_PLUGINS_E2E_LIVE_CLAWHUB: "0",
|
|
OPENCLAW_PLUGINS_TMP_DIR: tmpDir,
|
|
PATH: `${binDir}${path.delimiter}/usr/bin${path.delimiter}/bin`,
|
|
},
|
|
},
|
|
);
|
|
|
|
expect(result.status, result.stderr || result.stdout).toBe(0);
|
|
expect(result.stdout).toContain("truncated: showing last 80");
|
|
expect(result.stdout).toContain("PLUGIN_FIXTURE_TAIL_MARKER");
|
|
expect(result.stdout).not.toContain("DO_NOT_DUMP_PLUGIN_FIXTURE_PREFIX");
|
|
} finally {
|
|
rmSync(root, { force: true, recursive: true });
|
|
}
|
|
});
|
|
|
|
it("uses the configured scratch root and resolves Windows home-relative install paths", () => {
|
|
const root = mkdtempSync(path.join(tmpdir(), "openclaw-plugins-assertions-"));
|
|
const home = path.join(root, "home");
|
|
const scratchRoot = path.join(root, "scratch");
|
|
const installPath = path.join(home, "managed-plugin");
|
|
mkdirSync(installPath, { recursive: true });
|
|
|
|
try {
|
|
writeJson(path.join(scratchRoot, "plugins2.json"), {
|
|
plugins: [{ id: "demo-plugin-tgz", status: "loaded" }],
|
|
});
|
|
writeJson(path.join(scratchRoot, "plugins2-inspect.json"), {
|
|
gatewayMethods: ["demo.tgz"],
|
|
});
|
|
writeJson(path.join(home, ".openclaw", "plugins", "installs.json"), {
|
|
installRecords: {
|
|
"demo-plugin-tgz": {
|
|
source: "archive",
|
|
installPath: String.raw`~\managed-plugin`,
|
|
},
|
|
},
|
|
});
|
|
|
|
const result = spawnSync(process.execPath, [ASSERTIONS_SCRIPT, "plugin-tgz"], {
|
|
encoding: "utf8",
|
|
env: {
|
|
...process.env,
|
|
HOME: home,
|
|
OPENCLAW_PLUGINS_E2E_CLAWHUB_PREFLIGHT_TIMEOUT_MS: "1e3",
|
|
OPENCLAW_PLUGINS_TMP_DIR: scratchRoot,
|
|
},
|
|
});
|
|
|
|
expect(result.status).toBe(0);
|
|
} finally {
|
|
rmSync(root, { force: true, recursive: true });
|
|
}
|
|
});
|
|
|
|
it("compares local plugin source paths by canonical path", () => {
|
|
const root = mkdtempSync(path.join(tmpdir(), "openclaw-plugins-assertions-"));
|
|
const home = path.join(root, "home");
|
|
const scratchRoot = path.join(root, "scratch");
|
|
const sourceParent = path.join(root, "source");
|
|
const sourcePath = `${sourceParent}//plugin`;
|
|
const normalizedSourcePath = path.join(sourceParent, "plugin");
|
|
const installPath = path.join(home, ".openclaw", "extensions", "demo-plugin-dir");
|
|
mkdirSync(sourcePath, { recursive: true });
|
|
mkdirSync(installPath, { recursive: true });
|
|
|
|
try {
|
|
writeJson(path.join(scratchRoot, "plugins3.json"), {
|
|
plugins: [{ id: "demo-plugin-dir", status: "loaded" }],
|
|
});
|
|
writeJson(path.join(scratchRoot, "plugins3-inspect.json"), {
|
|
gatewayMethods: ["demo.dir"],
|
|
});
|
|
writeJson(path.join(home, ".openclaw", "plugins", "installs.json"), {
|
|
installRecords: {
|
|
"demo-plugin-dir": {
|
|
source: "path",
|
|
sourcePath: normalizedSourcePath,
|
|
installPath,
|
|
},
|
|
},
|
|
});
|
|
|
|
const result = spawnSync(process.execPath, [ASSERTIONS_SCRIPT, "plugin-dir", sourcePath], {
|
|
encoding: "utf8",
|
|
env: {
|
|
...process.env,
|
|
HOME: home,
|
|
OPENCLAW_PLUGINS_TMP_DIR: scratchRoot,
|
|
},
|
|
});
|
|
|
|
expect(result.status).toBe(0);
|
|
} finally {
|
|
rmSync(root, { force: true, recursive: true });
|
|
}
|
|
});
|
|
|
|
it("still requires archive managed install directories to be removed", () => {
|
|
const root = mkdtempSync(path.join(tmpdir(), "openclaw-plugins-assertions-"));
|
|
const home = path.join(root, "home");
|
|
const scratchRoot = path.join(root, "scratch");
|
|
const installPath = path.join(home, ".openclaw", "extensions", "demo-plugin-tgz");
|
|
mkdirSync(installPath, { recursive: true });
|
|
|
|
try {
|
|
writeJson(path.join(scratchRoot, "plugins2-uninstalled.json"), { plugins: [] });
|
|
writeFileSync(path.join(scratchRoot, "plugins2-install-path.txt"), installPath, "utf8");
|
|
writeJson(path.join(home, ".openclaw", "plugins", "installs.json"), {
|
|
installRecords: {},
|
|
});
|
|
|
|
const result = spawnSync(process.execPath, [ASSERTIONS_SCRIPT, "plugin-tgz-removed"], {
|
|
encoding: "utf8",
|
|
env: {
|
|
...process.env,
|
|
HOME: home,
|
|
OPENCLAW_PLUGINS_TMP_DIR: scratchRoot,
|
|
},
|
|
});
|
|
|
|
expect(result.status).not.toBe(0);
|
|
expect(result.stderr).toContain("managed install path still exists after uninstall");
|
|
} finally {
|
|
rmSync(root, { force: true, recursive: true });
|
|
}
|
|
});
|
|
|
|
it("rejects unreadable config during plugin uninstall proof", () => {
|
|
const root = mkdtempSync(path.join(tmpdir(), "openclaw-plugins-assertions-"));
|
|
const home = path.join(root, "home");
|
|
const scratchRoot = path.join(root, "scratch");
|
|
const removedInstallPath = path.join(home, ".openclaw", "extensions", "demo-plugin-tgz");
|
|
|
|
try {
|
|
writeJson(path.join(scratchRoot, "plugins2-uninstalled.json"), { plugins: [] });
|
|
writeFileSync(
|
|
path.join(scratchRoot, "plugins2-install-path.txt"),
|
|
removedInstallPath,
|
|
"utf8",
|
|
);
|
|
writeJson(path.join(home, ".openclaw", "plugins", "installs.json"), {
|
|
installRecords: {},
|
|
});
|
|
writeFileSync(path.join(home, ".openclaw", "openclaw.json"), "{ malformed\n", "utf8");
|
|
|
|
const result = spawnSync(process.execPath, [ASSERTIONS_SCRIPT, "plugin-tgz-removed"], {
|
|
encoding: "utf8",
|
|
env: {
|
|
...process.env,
|
|
HOME: home,
|
|
OPENCLAW_PLUGINS_TMP_DIR: scratchRoot,
|
|
},
|
|
});
|
|
|
|
expect(result.status).not.toBe(0);
|
|
expect(result.stderr).toContain("failed to read OpenClaw config");
|
|
} finally {
|
|
rmSync(root, { force: true, recursive: true });
|
|
}
|
|
});
|
|
|
|
it("times out stalled ClawHub package metadata requests", async () => {
|
|
const server = createServer((_request, _response) => {});
|
|
await new Promise<void>((resolve) => {
|
|
server.listen(0, "127.0.0.1", resolve);
|
|
});
|
|
|
|
try {
|
|
const address = server.address();
|
|
if (!address || typeof address === "string") {
|
|
throw new Error("expected TCP server address");
|
|
}
|
|
const result = await runAssertionAsync(["clawhub-preflight"], {
|
|
CLAWHUB_PLUGIN_ID: "openclaw-kitchen-sink-fixture",
|
|
CLAWHUB_PLUGIN_SPEC: "clawhub:@openclaw/kitchen-sink",
|
|
OPENCLAW_CLAWHUB_URL: `http://127.0.0.1:${address.port}`,
|
|
OPENCLAW_PLUGINS_E2E_CLAWHUB_PREFLIGHT_TIMEOUT_MS: "25",
|
|
});
|
|
|
|
expect(result.status).not.toBe(0);
|
|
expect(result.stderr).toContain(
|
|
"ClawHub package preflight for @openclaw/kitchen-sink timed out after 25ms",
|
|
);
|
|
} finally {
|
|
await new Promise<void>((resolve) => {
|
|
server.close(() => resolve());
|
|
});
|
|
}
|
|
});
|
|
|
|
it("times out stalled ClawHub package metadata bodies", async () => {
|
|
const server = createServer((_request, response) => {
|
|
response.writeHead(200, { "content-type": "application/json" });
|
|
response.flushHeaders();
|
|
response.write("{");
|
|
});
|
|
await new Promise<void>((resolve) => {
|
|
server.listen(0, "127.0.0.1", resolve);
|
|
});
|
|
|
|
try {
|
|
const address = server.address();
|
|
if (!address || typeof address === "string") {
|
|
throw new Error("expected TCP server address");
|
|
}
|
|
const result = await runAssertionAsync(["clawhub-preflight"], {
|
|
CLAWHUB_PLUGIN_ID: "openclaw-kitchen-sink-fixture",
|
|
CLAWHUB_PLUGIN_SPEC: "clawhub:@openclaw/kitchen-sink",
|
|
OPENCLAW_CLAWHUB_URL: `http://127.0.0.1:${address.port}`,
|
|
OPENCLAW_PLUGINS_E2E_CLAWHUB_PREFLIGHT_TIMEOUT_MS: "75",
|
|
});
|
|
|
|
expect(result.status).not.toBe(0);
|
|
expect(result.stderr).toContain(
|
|
"ClawHub package preflight response for @openclaw/kitchen-sink timed out after 75ms",
|
|
);
|
|
} finally {
|
|
await new Promise<void>((resolve) => {
|
|
server.close(() => resolve());
|
|
});
|
|
}
|
|
});
|
|
|
|
it("bounds ClawHub package metadata response bodies", async () => {
|
|
const server = createServer((_request, response) => {
|
|
response.writeHead(500, { "content-type": "text/plain" });
|
|
response.end("x".repeat(128));
|
|
});
|
|
await new Promise<void>((resolve) => {
|
|
server.listen(0, "127.0.0.1", resolve);
|
|
});
|
|
|
|
try {
|
|
const address = server.address();
|
|
if (!address || typeof address === "string") {
|
|
throw new Error("expected TCP server address");
|
|
}
|
|
const result = await runAssertionAsync(["clawhub-preflight"], {
|
|
CLAWHUB_PLUGIN_ID: "openclaw-kitchen-sink-fixture",
|
|
CLAWHUB_PLUGIN_SPEC: "clawhub:@openclaw/kitchen-sink",
|
|
OPENCLAW_CLAWHUB_URL: `http://127.0.0.1:${address.port}`,
|
|
OPENCLAW_PLUGINS_E2E_CLAWHUB_PREFLIGHT_BODY_MAX_BYTES: "16",
|
|
OPENCLAW_PLUGINS_E2E_CLAWHUB_PREFLIGHT_TIMEOUT_MS: "1000",
|
|
});
|
|
|
|
expect(result.status).not.toBe(0);
|
|
expect(result.stderr).toContain(
|
|
"ClawHub package preflight response for @openclaw/kitchen-sink response body exceeded 16 bytes",
|
|
);
|
|
expect(result.stderr).not.toContain("xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx");
|
|
} finally {
|
|
await new Promise<void>((resolve) => {
|
|
server.close(() => resolve());
|
|
});
|
|
}
|
|
});
|
|
});
|