Files
openclaw/extensions/migrate-hermes/provider.secret-failure.test.ts
Vincent Koc 1fc5b2b703 feat(migrations): add plugin-owned Hermes import
* feat: add migration providers

* feat: offer Hermes migration during onboarding

* feat(hermes): map imported config surfaces

* feat(onboard): require fresh migration imports

* docs(cli): clarify Hermes import coverage

* chore(migrations): rename Hermes importer package

* chore(migrations): rewire Hermes importer id

* fix(migrations): redact migration JSON details

* fix(hermes): use provider runtime for config imports

* test(hermes): cover missing source planning

---------

Co-authored-by: Peter Steinberger <steipete@gmail.com>
2026-04-27 00:34:29 -07:00

100 lines
2.7 KiB
TypeScript

import fs from "node:fs/promises";
import os from "node:os";
import path from "node:path";
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", () => ({
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,
};
}
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(
expect.arrayContaining([
expect.objectContaining({
id: "secret:openai",
status: "error",
reason: HERMES_REASON_AUTH_PROFILE_WRITE_FAILED,
}),
]),
);
expect(result.summary.errors).toBe(1);
expect(result.summary.migrated).toBe(0);
});
});