Files
2026-04-27 12:35:58 +01:00

117 lines
4.0 KiB
TypeScript

import {
resolveAgentEffectiveModelPrimary,
resolveDefaultAgentId,
setAgentEffectiveModelPrimary,
} from "openclaw/plugin-sdk/agent-runtime";
import type { MigrationItem, MigrationProviderContext } from "openclaw/plugin-sdk/plugin-entry";
import { readString } from "./helpers.js";
import {
HERMES_REASON_ALREADY_CONFIGURED,
HERMES_REASON_CONFIG_RUNTIME_UNAVAILABLE,
HERMES_REASON_DEFAULT_MODEL_CONFIGURED,
hermesItemConflict,
hermesItemError,
hermesItemSkipped,
readHermesModelDetails,
} from "./items.js";
export function resolveHermesModelRef(config: Record<string, unknown>): string | undefined {
const model = config.model;
if (typeof model === "string" && model.trim()) {
const rawModel = model.trim();
const provider = readString(config.provider);
if (provider && !rawModel.includes("/")) {
return `${provider}/${rawModel}`;
}
return rawModel;
}
if (model && typeof model === "object" && !Array.isArray(model)) {
const modelRecord = model as Record<string, unknown>;
const rawModel = readString(modelRecord.default) ?? readString(modelRecord.model);
const provider = readString(modelRecord.provider);
if (rawModel && provider && !rawModel.includes("/")) {
return `${provider}/${rawModel}`;
}
return rawModel;
}
const rootModel = readString(config.default_model) ?? readString(config.model_name);
const rootProvider = readString(config.provider);
if (rootModel && rootProvider && !rootModel.includes("/")) {
return `${rootProvider}/${rootModel}`;
}
return rootModel;
}
function resolveDefaultAgentModelState(config: MigrationProviderContext["config"]): {
agentId: string;
effectivePrimary?: string;
} {
const agentId = resolveDefaultAgentId(config);
const effectivePrimary = resolveAgentEffectiveModelPrimary(config, agentId);
return {
agentId,
effectivePrimary,
};
}
export function resolveCurrentModelRef(ctx: MigrationProviderContext): string | undefined {
return resolveDefaultAgentModelState(ctx.config).effectivePrimary;
}
class ModelApplyAbortError extends Error {
constructor(
readonly status: "conflict" | "skipped",
readonly reason: string,
) {
super(reason);
this.name = "ModelApplyAbortError";
}
}
export async function applyModelItem(
ctx: MigrationProviderContext,
item: MigrationItem,
): Promise<MigrationItem> {
const details = readHermesModelDetails(item);
if (!details || item.status !== "planned") {
return item;
}
try {
const configApi = ctx.runtime?.config;
if (!configApi?.current || !configApi.mutateConfigFile) {
return hermesItemError(item, HERMES_REASON_CONFIG_RUNTIME_UNAVAILABLE);
}
const currentState = resolveDefaultAgentModelState(
configApi.current() as MigrationProviderContext["config"],
);
if (currentState.effectivePrimary === details.model) {
return hermesItemSkipped(item, HERMES_REASON_ALREADY_CONFIGURED);
}
if (currentState.effectivePrimary && !ctx.overwrite) {
return hermesItemConflict(item, HERMES_REASON_DEFAULT_MODEL_CONFIGURED);
}
await configApi.mutateConfigFile({
base: "runtime",
afterWrite: { mode: "auto" },
mutate(draft) {
const mutationState = resolveDefaultAgentModelState(draft);
if (mutationState.effectivePrimary === details.model) {
throw new ModelApplyAbortError("skipped", HERMES_REASON_ALREADY_CONFIGURED);
}
if (mutationState.effectivePrimary && !ctx.overwrite) {
throw new ModelApplyAbortError("conflict", HERMES_REASON_DEFAULT_MODEL_CONFIGURED);
}
setAgentEffectiveModelPrimary(draft, mutationState.agentId, details.model);
},
});
return { ...item, status: "migrated" };
} catch (err) {
if (err instanceof ModelApplyAbortError) {
return err.status === "conflict"
? hermesItemConflict(item, err.reason)
: hermesItemSkipped(item, err.reason);
}
return hermesItemError(item, err instanceof Error ? err.message : String(err));
}
}