mirror of
https://github.com/openclaw/openclaw.git
synced 2026-06-13 01:31:18 +08:00
277 lines
7.9 KiB
TypeScript
277 lines
7.9 KiB
TypeScript
// Codex App Server Protocol Source tests cover codex app server protocol source script behavior.
|
|
import fs from "node:fs";
|
|
import path from "node:path";
|
|
import { afterEach, describe, expect, it } from "vitest";
|
|
import {
|
|
buildCodexProtocolExportArgs,
|
|
canonicalizeCodexAppServerProtocolJson,
|
|
formatCodexAppServerProtocolJsonText,
|
|
resolveCodexAppServerProtocolSource,
|
|
resolveCodexProtocolCargoTargetDir,
|
|
resolveCodexProtocolMinFreeBytes,
|
|
resolveCodexProtocolPnpmCommand,
|
|
validateCodexProtocolGenerationHeadroom,
|
|
} from "../../scripts/lib/codex-app-server-protocol-source.js";
|
|
import { createScriptTestHarness } from "./test-helpers.js";
|
|
|
|
const { createTempDir } = createScriptTestHarness();
|
|
const originalOpenClawCodexRepo = process.env.OPENCLAW_CODEX_REPO;
|
|
|
|
afterEach(() => {
|
|
if (originalOpenClawCodexRepo === undefined) {
|
|
delete process.env.OPENCLAW_CODEX_REPO;
|
|
} else {
|
|
process.env.OPENCLAW_CODEX_REPO = originalOpenClawCodexRepo;
|
|
}
|
|
});
|
|
|
|
describe("codex app-server protocol source resolver", () => {
|
|
it("uses the app-server protocol export binary instead of compiling the full codex cli", () => {
|
|
expect(buildCodexProtocolExportArgs("/codex/codex-rs/Cargo.toml", "/tmp/protocol")).toEqual([
|
|
"run",
|
|
"--manifest-path",
|
|
"/codex/codex-rs/Cargo.toml",
|
|
"-p",
|
|
"codex-app-server-protocol",
|
|
"--bin",
|
|
"export",
|
|
"--",
|
|
"--out",
|
|
"/tmp/protocol",
|
|
"--experimental",
|
|
]);
|
|
});
|
|
|
|
it("fails before cargo protocol generation when local disk headroom is too low", () => {
|
|
expect(() =>
|
|
validateCodexProtocolGenerationHeadroom({
|
|
freeBytes: 6 * 1024 * 1024 * 1024,
|
|
minFreeBytes: 10 * 1024 * 1024 * 1024,
|
|
pathLabel: "/repo",
|
|
}),
|
|
).toThrow(/Run this check on Crabbox\/Testbox/);
|
|
});
|
|
|
|
it("allows an explicit local disk headroom override", () => {
|
|
expect(resolveCodexProtocolMinFreeBytes({ OPENCLAW_CODEX_PROTOCOL_MIN_FREE_BYTES: "0" })).toBe(
|
|
0,
|
|
);
|
|
expect(() =>
|
|
validateCodexProtocolGenerationHeadroom({
|
|
freeBytes: 1,
|
|
minFreeBytes: 0,
|
|
pathLabel: "/repo",
|
|
}),
|
|
).not.toThrow();
|
|
});
|
|
|
|
it("rejects malformed local disk headroom overrides", () => {
|
|
expect(() =>
|
|
resolveCodexProtocolMinFreeBytes({ OPENCLAW_CODEX_PROTOCOL_MIN_FREE_BYTES: "nope" }),
|
|
).toThrow(/non-negative byte count/);
|
|
});
|
|
|
|
it("checks the Codex workspace target dir by default", () => {
|
|
expect(resolveCodexProtocolCargoTargetDir("/codex", {})).toBe(
|
|
path.join("/codex", "codex-rs", "target"),
|
|
);
|
|
});
|
|
|
|
it("checks an explicit Cargo target dir override", () => {
|
|
expect(
|
|
resolveCodexProtocolCargoTargetDir("/codex", { CARGO_TARGET_DIR: "/cache/target" }),
|
|
).toBe(path.resolve("/cache/target"));
|
|
});
|
|
|
|
it("resolves relative Cargo target dir overrides from the Codex checkout", () => {
|
|
expect(resolveCodexProtocolCargoTargetDir("/codex", { CARGO_TARGET_DIR: "target-cache" })).toBe(
|
|
path.join("/codex", "target-cache"),
|
|
);
|
|
});
|
|
|
|
it("checks Cargo's build target dir override", () => {
|
|
expect(
|
|
resolveCodexProtocolCargoTargetDir("/codex", {
|
|
CARGO_BUILD_TARGET_DIR: "/cache/build-target",
|
|
}),
|
|
).toBe(path.resolve("/cache/build-target"));
|
|
});
|
|
|
|
it("prefers Cargo's target dir override over the build config env override", () => {
|
|
expect(
|
|
resolveCodexProtocolCargoTargetDir("/codex", {
|
|
CARGO_BUILD_TARGET_DIR: "/cache/build-target",
|
|
CARGO_TARGET_DIR: "/cache/target",
|
|
}),
|
|
).toBe(path.resolve("/cache/target"));
|
|
});
|
|
|
|
it("wraps Windows pnpm formatting through cmd.exe without shell mode", () => {
|
|
expect(
|
|
resolveCodexProtocolPnpmCommand(
|
|
["exec", "oxfmt", "--write", "--threads=1", String.raw`C:\tmp\generated types`],
|
|
{
|
|
comSpec: String.raw`C:\Windows\System32\cmd.exe`,
|
|
npmExecPath: String.raw`C:\Program Files\nodejs\pnpm.cmd`,
|
|
platform: "win32",
|
|
},
|
|
),
|
|
).toEqual({
|
|
args: [
|
|
"/d",
|
|
"/s",
|
|
"/c",
|
|
String.raw`""C:\Program Files\nodejs\pnpm.cmd" exec oxfmt --write --threads=1 "C:\tmp\generated types""`,
|
|
],
|
|
command: String.raw`C:\Windows\System32\cmd.exe`,
|
|
shell: false,
|
|
windowsVerbatimArguments: true,
|
|
});
|
|
});
|
|
|
|
it("uses OPENCLAW_CODEX_REPO when provided", async () => {
|
|
const root = createTempDir("openclaw-protocol-source-root-");
|
|
const codexRepo = createTempDir("openclaw-protocol-source-codex-");
|
|
createProtocolSchema(codexRepo);
|
|
process.env.OPENCLAW_CODEX_REPO = codexRepo;
|
|
|
|
await expect(resolveCodexAppServerProtocolSource(root)).resolves.toEqual({
|
|
codexRepo,
|
|
sourceRoot: path.join(codexRepo, "codex-rs/app-server-protocol/schema"),
|
|
});
|
|
});
|
|
|
|
it("finds the primary checkout sibling from a git worktree", async () => {
|
|
const parentDir = createTempDir("openclaw-protocol-source-parent-");
|
|
const primaryOpenClaw = path.join(parentDir, "openclaw");
|
|
const codexRepo = path.join(parentDir, "codex");
|
|
const worktreeRoot = createTempDir("openclaw-protocol-source-worktree-");
|
|
fs.mkdirSync(path.join(primaryOpenClaw, ".git", "worktrees", "codex-harness"), {
|
|
recursive: true,
|
|
});
|
|
fs.mkdirSync(worktreeRoot, { recursive: true });
|
|
fs.writeFileSync(
|
|
path.join(worktreeRoot, ".git"),
|
|
`gitdir: ${path.join(primaryOpenClaw, ".git", "worktrees", "codex-harness")}\n`,
|
|
);
|
|
createProtocolSchema(codexRepo);
|
|
delete process.env.OPENCLAW_CODEX_REPO;
|
|
|
|
await expect(resolveCodexAppServerProtocolSource(worktreeRoot)).resolves.toEqual({
|
|
codexRepo,
|
|
sourceRoot: path.join(codexRepo, "codex-rs/app-server-protocol/schema"),
|
|
});
|
|
});
|
|
});
|
|
|
|
describe("Codex app-server protocol JSON canonicalizer", () => {
|
|
it("sorts object keys recursively before formatting", () => {
|
|
const source = JSON.stringify({
|
|
z: {
|
|
d: 1,
|
|
b: {
|
|
y: 2,
|
|
x: 3,
|
|
},
|
|
},
|
|
a: [
|
|
{
|
|
z: 4,
|
|
a: {
|
|
c: 5,
|
|
b: 6,
|
|
},
|
|
},
|
|
],
|
|
});
|
|
|
|
expect(formatCodexAppServerProtocolJsonText(source)).toBe(`{
|
|
"a": [
|
|
{
|
|
"a": {
|
|
"b": 6,
|
|
"c": 5
|
|
},
|
|
"z": 4
|
|
}
|
|
],
|
|
"z": {
|
|
"b": {
|
|
"x": 3,
|
|
"y": 2
|
|
},
|
|
"d": 1
|
|
}
|
|
}
|
|
`);
|
|
});
|
|
|
|
it("sorts typed-object arrays only for order-insensitive schema keywords", () => {
|
|
expect(
|
|
canonicalizeCodexAppServerProtocolJson({
|
|
anyOf: [
|
|
{ z: 1, type: "string" },
|
|
{ type: "integer", a: 2 },
|
|
],
|
|
enum: [
|
|
{ z: 1, type: "z" },
|
|
{ type: "a", a: 2 },
|
|
],
|
|
mixed: [{ type: "b" }, "item", { type: "a" }],
|
|
oneOf: [
|
|
{ type: "object", z: true },
|
|
{ a: true, type: "array" },
|
|
{ type: "object", z: false },
|
|
],
|
|
prefixItems: [
|
|
{ z: 1, type: "string" },
|
|
{ type: "number", a: 2 },
|
|
],
|
|
required: [
|
|
{ z: 1, type: "z" },
|
|
{ type: "a", a: 2 },
|
|
],
|
|
typed: [
|
|
{ type: "beta", z: 1 },
|
|
{ type: "alpha", z: 2 },
|
|
{ type: "beta", z: 3 },
|
|
],
|
|
}),
|
|
).toEqual({
|
|
anyOf: [
|
|
{ a: 2, type: "integer" },
|
|
{ type: "string", z: 1 },
|
|
],
|
|
enum: [
|
|
{ a: 2, type: "a" },
|
|
{ type: "z", z: 1 },
|
|
],
|
|
mixed: [{ type: "b" }, "item", { type: "a" }],
|
|
oneOf: [
|
|
{ a: true, type: "array" },
|
|
{ type: "object", z: true },
|
|
{ type: "object", z: false },
|
|
],
|
|
prefixItems: [
|
|
{ type: "string", z: 1 },
|
|
{ a: 2, type: "number" },
|
|
],
|
|
required: [
|
|
{ a: 2, type: "a" },
|
|
{ type: "z", z: 1 },
|
|
],
|
|
typed: [
|
|
{ type: "beta", z: 1 },
|
|
{ type: "alpha", z: 2 },
|
|
{ type: "beta", z: 3 },
|
|
],
|
|
});
|
|
});
|
|
});
|
|
|
|
function createProtocolSchema(codexRepo: string): void {
|
|
fs.mkdirSync(path.join(codexRepo, "codex-rs/app-server-protocol/schema/typescript"), {
|
|
recursive: true,
|
|
});
|
|
}
|