Files
openclaw/src/agents/models-config.write-serialization.test.ts
Peter Steinberger 538605ff44 [codex] Extract filesystem safety primitives (#77918)
* refactor: extract filesystem safety primitives

* refactor: use fs-safe for file access helpers

* refactor: reuse fs-safe for media reads

* refactor: use fs-safe for image reads

* refactor: reuse fs-safe in qqbot media opener

* refactor: reuse fs-safe for local media checks

* refactor: consume cleaner fs-safe api

* refactor: align fs-safe json option names

* fix: preserve fs-safe migration contracts

* refactor: use fs-safe primitive subpaths

* refactor: use grouped fs-safe subpaths

* refactor: align fs-safe api usage

* refactor: adapt private state store api

* chore: refresh proof gate

* refactor: follow fs-safe json api split

* refactor: follow reduced fs-safe surface

* build: default fs-safe python helper off

* fix: preserve fs-safe plugin sdk aliases

* refactor: consolidate fs-safe usage

* refactor: unify fs-safe store usage

* refactor: trim fs-safe temp workspace usage

* refactor: hide low-level fs-safe primitives

* build: use published fs-safe package

* fix: preserve outbound recovery durability after rebase

* chore: refresh pr checks
2026-05-06 02:15:17 +01:00

288 lines
11 KiB
TypeScript

import fs from "node:fs/promises";
import path from "node:path";
import { beforeAll, beforeEach, describe, expect, it, vi } from "vitest";
import { resolveInstalledPluginIndexPolicyHash } from "../plugins/installed-plugin-index-policy.js";
import type { PluginMetadataSnapshot } from "../plugins/plugin-metadata-snapshot.js";
import { resolveDefaultAgentDir } from "./agent-scope.js";
import {
CUSTOM_PROXY_MODELS_CONFIG,
installModelsConfigTestHooks,
withModelsTempHome,
} from "./models-config.e2e-harness.js";
import { readGeneratedModelsJson } from "./models-config.test-utils.js";
const planOpenClawModelsJsonMock = vi.fn();
const writePrivateStoreTextWriteMock = vi.fn();
let actualPrivateFileStore:
| typeof import("../infra/private-file-store.js").privateFileStore
| undefined;
installModelsConfigTestHooks();
let ensureOpenClawModelsJson: typeof import("./models-config.js").ensureOpenClawModelsJson;
let clearCurrentPluginMetadataSnapshot: typeof import("../plugins/current-plugin-metadata-snapshot.js").clearCurrentPluginMetadataSnapshot;
let setCurrentPluginMetadataSnapshot: typeof import("../plugins/current-plugin-metadata-snapshot.js").setCurrentPluginMetadataSnapshot;
function createPluginMetadataSnapshot(workspaceDir: string): PluginMetadataSnapshot {
const policyHash = resolveInstalledPluginIndexPolicyHash({});
return {
policyHash,
workspaceDir,
index: {
version: 1,
hostContractVersion: "test",
compatRegistryVersion: "test",
migrationVersion: 1,
policyHash,
generatedAtMs: 1,
installRecords: {},
plugins: [],
diagnostics: [],
},
registryDiagnostics: [],
manifestRegistry: { plugins: [], diagnostics: [] },
plugins: [],
diagnostics: [],
byPluginId: new Map(),
normalizePluginId: (pluginId) => pluginId,
owners: {
channels: new Map(),
channelConfigs: new Map(),
providers: new Map(),
modelCatalogProviders: new Map(),
cliBackends: new Map(),
setupProviders: new Map(),
commandAliases: new Map(),
contracts: new Map(),
},
metrics: {
registrySnapshotMs: 0,
manifestRegistryMs: 0,
ownerMapsMs: 0,
totalMs: 0,
indexPluginCount: 0,
manifestPluginCount: 0,
},
};
}
beforeAll(async () => {
vi.doMock("./models-config.plan.js", () => ({
planOpenClawModelsJson: (...args: unknown[]) => planOpenClawModelsJsonMock(...args),
}));
vi.doMock("../infra/private-file-store.js", async () => {
const actual = await vi.importActual<typeof import("../infra/private-file-store.js")>(
"../infra/private-file-store.js",
);
actualPrivateFileStore = actual.privateFileStore;
return {
...actual,
privateFileStore: (rootDir: string) => {
const store = actual.privateFileStore(rootDir);
return {
...store,
writeText: (relativePath: string, content: string | Uint8Array) =>
writePrivateStoreTextWriteMock({
rootDir,
filePath: path.join(rootDir, relativePath),
content,
}),
};
},
};
});
({ ensureOpenClawModelsJson } = await import("./models-config.js"));
({ clearCurrentPluginMetadataSnapshot, setCurrentPluginMetadataSnapshot } =
await import("../plugins/current-plugin-metadata-snapshot.js"));
});
beforeEach(() => {
clearCurrentPluginMetadataSnapshot();
writePrivateStoreTextWriteMock
.mockReset()
.mockImplementation(
async (params: { filePath: string; rootDir: string; content: string | Uint8Array }) => {
if (!actualPrivateFileStore) {
throw new Error("private file store mock not initialized");
}
return await actualPrivateFileStore(params.rootDir).writeText(
path.basename(params.filePath),
params.content,
);
},
);
planOpenClawModelsJsonMock
.mockReset()
.mockImplementation(async (params: { cfg?: typeof CUSTOM_PROXY_MODELS_CONFIG }) => ({
action: "write",
contents: `${JSON.stringify({ providers: params.cfg?.models?.providers ?? {} }, null, 2)}\n`,
}));
});
describe("models-config write serialization", () => {
it("does not reuse default workspace plugin metadata for explicit agent dirs without workspace", async () => {
await withModelsTempHome(async (home) => {
const snapshot = createPluginMetadataSnapshot(path.join(home, "default-workspace"));
setCurrentPluginMetadataSnapshot(snapshot, { config: {} });
const agentDir = path.join(home, "agent-non-default");
await ensureOpenClawModelsJson({}, agentDir);
expect(planOpenClawModelsJsonMock).toHaveBeenCalledWith(
expect.not.objectContaining({ pluginMetadataSnapshot: snapshot }),
);
});
});
it("reuses current plugin metadata for explicit agent dirs with matching workspace", async () => {
await withModelsTempHome(async (home) => {
const workspaceDir = path.join(home, "agent-workspace");
const snapshot = createPluginMetadataSnapshot(workspaceDir);
setCurrentPluginMetadataSnapshot(snapshot, { config: {} });
const agentDir = path.join(home, "agent-non-default");
await ensureOpenClawModelsJson({}, agentDir, { workspaceDir });
expect(planOpenClawModelsJsonMock).toHaveBeenCalledWith(
expect.objectContaining({
workspaceDir,
pluginMetadataSnapshot: snapshot,
}),
);
});
});
it("writes implicit models.json into the configured default agent dir", async () => {
await withModelsTempHome(async (home) => {
const cfg = {
agents: {
list: [{ id: "main" }, { id: "ops", default: true }],
},
};
const result = await ensureOpenClawModelsJson(cfg);
expect(result.agentDir).toBe(path.join(home, ".openclaw", "agents", "ops", "agent"));
await expect(fs.access(path.join(result.agentDir, "models.json"))).resolves.toBeUndefined();
await expect(
fs.access(path.join(home, ".openclaw", "agents", "main", "agent", "models.json")),
).rejects.toThrow();
});
});
it("does not reuse scoped startup discovery cache for a different provider scope", async () => {
await withModelsTempHome(async (home) => {
planOpenClawModelsJsonMock.mockImplementation(async () => ({ action: "skip" }));
const agentDir = path.join(home, "agent");
await ensureOpenClawModelsJson({}, agentDir, {
providerDiscoveryProviderIds: ["openai"],
providerDiscoveryTimeoutMs: 5000,
});
await ensureOpenClawModelsJson({}, agentDir, {
providerDiscoveryProviderIds: ["anthropic"],
providerDiscoveryTimeoutMs: 5000,
});
expect(planOpenClawModelsJsonMock).toHaveBeenCalledTimes(2);
expect(planOpenClawModelsJsonMock).toHaveBeenLastCalledWith(
expect.objectContaining({
providerDiscoveryProviderIds: ["anthropic"],
providerDiscoveryTimeoutMs: 5000,
}),
);
});
});
it("keeps the ready cache warm after models.json is written", async () => {
await withModelsTempHome(async () => {
await ensureOpenClawModelsJson(CUSTOM_PROXY_MODELS_CONFIG);
await ensureOpenClawModelsJson(CUSTOM_PROXY_MODELS_CONFIG);
expect(planOpenClawModelsJsonMock).toHaveBeenCalledTimes(1);
});
});
it("invalidates the ready cache when models.json changes externally", async () => {
await withModelsTempHome(async () => {
await ensureOpenClawModelsJson(CUSTOM_PROXY_MODELS_CONFIG);
await ensureOpenClawModelsJson(CUSTOM_PROXY_MODELS_CONFIG);
const modelPath = path.join(resolveDefaultAgentDir({}), "models.json");
await fs.writeFile(modelPath, `${JSON.stringify({ external: true })}\n`, "utf8");
const externalMtime = new Date(Date.now() + 2000);
await fs.utimes(modelPath, externalMtime, externalMtime);
await ensureOpenClawModelsJson(CUSTOM_PROXY_MODELS_CONFIG);
expect(planOpenClawModelsJsonMock).toHaveBeenCalledTimes(2);
});
});
it("keeps distinct config fingerprints cached without evicting each other", async () => {
await withModelsTempHome(async () => {
planOpenClawModelsJsonMock.mockImplementation(async () => ({ action: "noop" }));
const first = structuredClone(CUSTOM_PROXY_MODELS_CONFIG);
const second = structuredClone(CUSTOM_PROXY_MODELS_CONFIG);
first.agents = { defaults: { model: "openai/gpt-5.4" } };
second.agents = { defaults: { model: "anthropic/claude-sonnet-4-5" } };
await ensureOpenClawModelsJson(first);
await ensureOpenClawModelsJson(second);
await ensureOpenClawModelsJson(first);
expect(planOpenClawModelsJsonMock).toHaveBeenCalledTimes(2);
});
});
it("serializes concurrent models.json writes to avoid overlap", async () => {
await withModelsTempHome(async () => {
const first = structuredClone(CUSTOM_PROXY_MODELS_CONFIG);
const second = structuredClone(CUSTOM_PROXY_MODELS_CONFIG);
const firstModel = first.models?.providers?.["custom-proxy"]?.models?.[0];
const secondModel = second.models?.providers?.["custom-proxy"]?.models?.[0];
if (!firstModel || !secondModel) {
throw new Error("custom-proxy fixture missing expected model entries");
}
firstModel.name = "Proxy A";
secondModel.name = "Proxy B with longer name";
let inFlightWrites = 0;
let maxInFlightWrites = 0;
writePrivateStoreTextWriteMock.mockImplementation(
async (params: { filePath: string; rootDir: string; content: string | Uint8Array }) => {
const isModelsWrite = path.basename(params.filePath) === "models.json";
if (isModelsWrite) {
inFlightWrites += 1;
if (inFlightWrites > maxInFlightWrites) {
maxInFlightWrites = inFlightWrites;
}
await new Promise((resolve) => setTimeout(resolve, 10));
}
try {
if (!actualPrivateFileStore) {
throw new Error("private file store mock not initialized");
}
return await actualPrivateFileStore(params.rootDir).writeText(
path.basename(params.filePath),
params.content,
);
} finally {
if (isModelsWrite) {
inFlightWrites -= 1;
}
}
},
);
await Promise.all([ensureOpenClawModelsJson(first), ensureOpenClawModelsJson(second)]);
expect(maxInFlightWrites).toBe(1);
const parsed = await readGeneratedModelsJson<{
providers: { "custom-proxy"?: { models?: Array<{ name?: string }> } };
}>();
expect(["Proxy A", "Proxy B with longer name"]).toContain(
parsed.providers["custom-proxy"]?.models?.[0]?.name,
);
});
}, 60_000);
});