Files
openclaw/packages/gateway-protocol/src/exec-approvals-validators.test.ts
Jesse Merhi c9707ab635 fix(exec): rebuild command authorization on the Tree-sitter command planner
Replace the exec approval parser/planner path with Tree-sitter-backed authorization planning, carrying planner decisions through node and gateway execution.

This keeps unpersistable shell shapes one-shot, adds typed `unavailableDecisions` for approval prompts, and refreshes coverage for allowlist matching, command rendering, durable allow-always persistence, and host approval paths.

Verification:
- GitHub PR checks for ce2381192d: CLEAN, 142 success, 32 skipped, 0 failed, 0 pending.
- /Users/jmerhi/.nvm/versions/node/v24.12.0/bin/node scripts/plugin-sdk-surface-report.mjs --check
- /Users/jmerhi/.nvm/versions/node/v24.12.0/bin/node scripts/run-vitest.mjs test/scripts/plugin-sdk-surface-report.test.ts --reporter=verbose
- Focused exec approval suite: 13 files, 467 tests.
2026-06-18 10:12:48 +10:00

133 lines
3.3 KiB
TypeScript

// Gateway Protocol tests cover exec approvals validators behavior.
import { describe, expect, it } from "vitest";
import {
validateExecApprovalRequestParams,
validateExecApprovalsNodeSetParams,
validateExecApprovalsSetParams,
} from "./index.js";
/**
* Exec approval validator regressions for gateway and node-scoped policy
* writes. The fixtures pin runtime-owned allowlist metadata and command-span
* bounds because those contracts are consumed by approval UI and replay logic.
*/
describe("exec approvals protocol validators", () => {
it("accepts runtime-owned allowlist metadata on gateway and node set payloads", () => {
const file = {
version: 1 as const,
agents: {
main: {
allowlist: [
{
id: "entry-1",
pattern: "cmd:allow-always:abcdef",
source: "allow-always" as const,
commandText: "python3 -c 'print(123)'",
argPattern: "-c *",
lastUsedAt: 1775154056736,
lastUsedCommand: "python3 -c 'print(123)'",
lastResolvedPath: "/usr/bin/python3",
},
],
},
},
};
expect(validateExecApprovalsSetParams({ file, baseHash: "abc123" })).toBe(true);
expect(
validateExecApprovalsNodeSetParams({
nodeId: "node-1",
file,
baseHash: "abc123",
}),
).toBe(true);
});
it("rejects unknown allowlist metadata", () => {
expect(
validateExecApprovalsSetParams({
file: {
version: 1,
agents: {
main: {
allowlist: [
{
pattern: "/usr/bin/python3",
source: "unknown-source",
},
],
},
},
},
baseHash: "abc123",
}),
).toBe(false);
expect(
validateExecApprovalsSetParams({
file: {
version: 1,
agents: {
main: {
allowlist: [
{
pattern: "/usr/bin/python3",
randomMetadata: true,
},
],
},
},
},
baseHash: "abc123",
}),
).toBe(false);
});
it("requires command spans to have non-negative starts and positive exclusive ends", () => {
expect(
validateExecApprovalRequestParams({
command: "echo hi",
commandSpans: [{ startIndex: 0, endIndex: 4 }],
}),
).toBe(true);
expect(
validateExecApprovalRequestParams({
command: "echo hi",
commandSpans: [{ startIndex: 0, endIndex: 0 }],
}),
).toBe(false);
expect(
validateExecApprovalRequestParams({
command: "echo hi",
commandSpans: [{ startIndex: -1, endIndex: 4 }],
}),
).toBe(false);
});
it("accepts only optional unavailable approval decisions", () => {
expect(
validateExecApprovalRequestParams({
command: "echo hi",
unavailableDecisions: ["allow-always"],
}),
).toBe(true);
for (const unavailableDecisions of [
[],
["allow-always", "allow-always"],
["allow-once"],
["deny"],
]) {
expect(
validateExecApprovalRequestParams({
command: "echo hi",
unavailableDecisions,
}),
).toBe(false);
}
});
});