Files
openclaw/extensions/migrate-hermes/provider.secret-failure.test.ts
2026-06-04 21:02:07 -04:00

184 lines
5.5 KiB
TypeScript

// Migrate Hermes tests cover provider.secret failure plugin behavior.
import fs from "node:fs/promises";
import os from "node:os";
import path from "node:path";
import { resolveAuthStorePathForDisplay } from "openclaw/plugin-sdk/agent-runtime";
import type { MigrationProviderContext } from "openclaw/plugin-sdk/plugin-entry";
import type { OpenClawConfig } from "openclaw/plugin-sdk/provider-auth";
import { afterEach, describe, expect, it, vi } from "vitest";
import { HERMES_REASON_AUTH_PROFILE_WRITE_FAILED } from "./items.js";
const mocks = vi.hoisted(() => ({
updateAuthProfileStoreWithLock: vi.fn(async () => null),
}));
vi.mock("openclaw/plugin-sdk/provider-auth", async (importOriginal) => ({
...(await importOriginal<typeof import("openclaw/plugin-sdk/provider-auth")>()),
updateAuthProfileStoreWithLock: mocks.updateAuthProfileStoreWithLock,
}));
const { buildHermesMigrationProvider } = await import("./provider.js");
const tempRoots = new Set<string>();
const logger = {
info() {},
warn() {},
error() {},
debug() {},
};
async function makeTempRoot() {
const root = await fs.mkdtemp(path.join(os.tmpdir(), "openclaw-hermes-secret-failure-"));
tempRoots.add(root);
return root;
}
async function writeFile(filePath: string, content: string) {
await fs.mkdir(path.dirname(filePath), { recursive: true });
await fs.writeFile(filePath, content, "utf8");
}
function makeContext(params: {
source: string;
stateDir: string;
workspaceDir: string;
reportDir: string;
}): MigrationProviderContext {
return {
config: {
agents: {
defaults: {
workspace: params.workspaceDir,
},
},
} as OpenClawConfig,
stateDir: params.stateDir,
source: params.source,
includeSecrets: true,
overwrite: true,
reportDir: params.reportDir,
logger,
};
}
function fakeJwt(payload: Record<string, unknown>): string {
const header = Buffer.from(JSON.stringify({ alg: "none", typ: "JWT" })).toString("base64url");
const body = Buffer.from(JSON.stringify(payload)).toString("base64url");
return `${header}.${body}.signature`;
}
function authProfileTarget(agentDir: string, profileId: string): string {
return `${resolveAuthStorePathForDisplay(agentDir)}#${profileId}`;
}
describe("Hermes migration provider secret write failures", () => {
afterEach(async () => {
for (const root of tempRoots) {
await fs.rm(root, { force: true, recursive: true });
}
tempRoots.clear();
mocks.updateAuthProfileStoreWithLock.mockClear();
});
it("reports an error when a secret auth-profile write fails", async () => {
const root = await makeTempRoot();
const source = path.join(root, "hermes");
const workspaceDir = path.join(root, "workspace");
const stateDir = path.join(root, "state");
await writeFile(path.join(source, ".env"), "OPENAI_API_KEY=sk-hermes\n");
const provider = buildHermesMigrationProvider();
const result = await provider.apply(
makeContext({
source,
stateDir,
workspaceDir,
reportDir: path.join(root, "report"),
}),
);
expect(result.items).toEqual([
{
id: "secret:openai",
kind: "secret",
action: "create",
source: path.join(source, ".env"),
target: authProfileTarget(
path.join(stateDir, "agents", "main", "agent"),
"openai:hermes-import",
),
status: "error",
sensitive: true,
reason: HERMES_REASON_AUTH_PROFILE_WRITE_FAILED,
details: {
envVar: "OPENAI_API_KEY",
provider: "openai",
profileId: "openai:hermes-import",
},
},
]);
expect(result.summary.errors).toBe(1);
expect(result.summary.migrated).toBe(0);
});
it("reports an error when an OAuth auth-profile write fails", async () => {
const root = await makeTempRoot();
const source = path.join(root, "hermes");
const workspaceDir = path.join(root, "workspace");
const stateDir = path.join(root, "state");
const accessToken = fakeJwt({
exp: Math.floor(Date.now() / 1000) + 3600,
"https://api.openai.com/profile": { email: "codex@example.test" },
"https://api.openai.com/auth": {
chatgpt_account_id: "acct_fail",
chatgpt_plan_type: "plus",
},
});
await writeFile(path.join(source, "auth.json"), "{}");
const opencodeAuthPath = path.join(root, ".local", "share", "opencode", "auth.json");
await writeFile(
opencodeAuthPath,
JSON.stringify({
openai: {
type: "oauth",
access: accessToken,
refresh: "refresh-fail-token",
},
}),
);
const provider = buildHermesMigrationProvider();
const result = await provider.apply(
makeContext({
source,
stateDir,
workspaceDir,
reportDir: path.join(root, "report"),
}),
);
expect(result.items).toEqual([
expect.objectContaining({
id: "auth:openai",
kind: "auth",
action: "create",
source: opencodeAuthPath,
target: authProfileTarget(
path.join(stateDir, "agents", "main", "agent"),
"openai:account-acct_fail",
),
status: "error",
sensitive: true,
reason: HERMES_REASON_AUTH_PROFILE_WRITE_FAILED,
details: expect.objectContaining({
provider: "openai",
profileId: "openai:account-acct_fail",
sourceProfileId: "openai:account-acct_fail",
}),
}),
]);
expect(result.summary.errors).toBe(1);
expect(result.summary.migrated).toBe(0);
});
});