mirror of
https://github.com/openclaw/openclaw.git
synced 2026-05-06 09:10:45 +00:00
308 lines
9.7 KiB
TypeScript
308 lines
9.7 KiB
TypeScript
import {
|
|
applyMigrationConfigPatchItem,
|
|
applyMigrationManualItem,
|
|
createMigrationConfigPatchItem,
|
|
createMigrationManualItem,
|
|
hasMigrationConfigPatchConflict,
|
|
} from "openclaw/plugin-sdk/migration";
|
|
import type { MigrationItem, MigrationProviderContext } from "openclaw/plugin-sdk/plugin-entry";
|
|
import { childRecord, isRecord, readString, readStringArray } from "./helpers.js";
|
|
|
|
type HermesProviderConfig = {
|
|
id: string;
|
|
baseUrl?: string;
|
|
apiKeyEnv?: string;
|
|
models: string[];
|
|
};
|
|
|
|
function envKeyForProvider(providerId: string): string {
|
|
return `${providerId.toUpperCase().replaceAll(/[^A-Z0-9]/gu, "_")}_API_KEY`;
|
|
}
|
|
|
|
function splitProviderModel(modelRef: string | undefined): { provider?: string; model?: string } {
|
|
if (!modelRef) {
|
|
return {};
|
|
}
|
|
const slash = modelRef.indexOf("/");
|
|
if (slash > 0 && slash < modelRef.length - 1) {
|
|
return { provider: modelRef.slice(0, slash), model: modelRef.slice(slash + 1) };
|
|
}
|
|
return { model: modelRef };
|
|
}
|
|
|
|
function modelDefinition(modelId: string, baseUrl?: string): Record<string, unknown> {
|
|
return {
|
|
id: modelId,
|
|
name: modelId,
|
|
api: baseUrl ? "openai-completions" : "openai-responses",
|
|
reasoning: false,
|
|
input: ["text"],
|
|
cost: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0 },
|
|
contextWindow: 128_000,
|
|
maxTokens: 8192,
|
|
...(baseUrl ? { baseUrl } : {}),
|
|
metadataSource: "models-add",
|
|
};
|
|
}
|
|
|
|
function providerConfig(entry: HermesProviderConfig): Record<string, unknown> {
|
|
const models = entry.models.length > 0 ? entry.models : [`${entry.id}/default`];
|
|
return {
|
|
baseUrl: entry.baseUrl ?? "",
|
|
...(entry.apiKeyEnv
|
|
? { apiKey: { source: "env", provider: "default", id: entry.apiKeyEnv } }
|
|
: {}),
|
|
api: "openai-completions",
|
|
models: models.map((modelId) => modelDefinition(modelId, entry.baseUrl)),
|
|
};
|
|
}
|
|
|
|
export function collectHermesProviders(
|
|
config: Record<string, unknown>,
|
|
modelRef?: string,
|
|
): HermesProviderConfig[] {
|
|
const collected: HermesProviderConfig[] = [];
|
|
for (const [id, raw] of Object.entries(childRecord(config, "providers"))) {
|
|
if (!isRecord(raw)) {
|
|
continue;
|
|
}
|
|
const baseUrl =
|
|
readString(raw.base_url) ??
|
|
readString(raw.baseUrl) ??
|
|
readString(raw.url) ??
|
|
readString(raw.api);
|
|
const apiKeyEnv =
|
|
readString(raw.api_key_env) ??
|
|
readString(raw.apiKeyEnv) ??
|
|
readString(raw.env) ??
|
|
envKeyForProvider(id);
|
|
const models = [
|
|
...readStringArray(raw.models),
|
|
...Object.keys(childRecord(raw, "models")),
|
|
readString(raw.model),
|
|
].filter((value): value is string => Boolean(value));
|
|
collected.push({ id, baseUrl, apiKeyEnv, models: [...new Set(models)] });
|
|
}
|
|
|
|
const customProviders = config.custom_providers;
|
|
if (Array.isArray(customProviders)) {
|
|
for (const raw of customProviders) {
|
|
if (!isRecord(raw)) {
|
|
continue;
|
|
}
|
|
const id = readString(raw.name) ?? readString(raw.id);
|
|
if (!id) {
|
|
continue;
|
|
}
|
|
const baseUrl = readString(raw.base_url) ?? readString(raw.baseUrl) ?? readString(raw.url);
|
|
const apiKeyEnv = readString(raw.api_key_env) ?? readString(raw.apiKeyEnv);
|
|
const models = [
|
|
...readStringArray(raw.models),
|
|
...Object.keys(childRecord(raw, "models")),
|
|
readString(raw.model),
|
|
].filter((value): value is string => Boolean(value));
|
|
collected.push({ id, baseUrl, apiKeyEnv, models: [...new Set(models)] });
|
|
}
|
|
}
|
|
|
|
const defaultRef = splitProviderModel(modelRef);
|
|
if (defaultRef.provider && !collected.some((entry) => entry.id === defaultRef.provider)) {
|
|
collected.push({
|
|
id: defaultRef.provider,
|
|
apiKeyEnv: envKeyForProvider(defaultRef.provider),
|
|
models: defaultRef.model ? [defaultRef.model] : [],
|
|
});
|
|
}
|
|
return collected;
|
|
}
|
|
|
|
function mapMcpServers(raw: unknown): Record<string, unknown> | undefined {
|
|
if (!isRecord(raw)) {
|
|
return undefined;
|
|
}
|
|
const mapped: Record<string, unknown> = {};
|
|
for (const [name, value] of Object.entries(raw)) {
|
|
if (!isRecord(value)) {
|
|
continue;
|
|
}
|
|
const next: Record<string, unknown> = {};
|
|
for (const key of [
|
|
"command",
|
|
"args",
|
|
"env",
|
|
"cwd",
|
|
"workingDirectory",
|
|
"url",
|
|
"transport",
|
|
"headers",
|
|
"connectionTimeoutMs",
|
|
]) {
|
|
if (value[key] !== undefined) {
|
|
next[key] = value[key];
|
|
}
|
|
}
|
|
if (Object.keys(next).length > 0) {
|
|
mapped[name] = next;
|
|
}
|
|
}
|
|
return Object.keys(mapped).length > 0 ? mapped : undefined;
|
|
}
|
|
|
|
function mapSkillEntries(config: Record<string, unknown>): Record<string, unknown> | undefined {
|
|
const entries: Record<string, unknown> = {};
|
|
for (const [skillKey, value] of Object.entries(
|
|
childRecord(childRecord(config, "skills"), "config"),
|
|
)) {
|
|
if (isRecord(value)) {
|
|
entries[skillKey] = { config: value };
|
|
}
|
|
}
|
|
return Object.keys(entries).length > 0 ? entries : undefined;
|
|
}
|
|
|
|
export function buildConfigItems(params: {
|
|
ctx: MigrationProviderContext;
|
|
config: Record<string, unknown>;
|
|
modelRef?: string;
|
|
hasMemoryFiles?: boolean;
|
|
}): MigrationItem[] {
|
|
const items: MigrationItem[] = [];
|
|
const memory = childRecord(params.config, "memory");
|
|
const memoryProvider = readString(memory.provider);
|
|
|
|
if (params.hasMemoryFiles || memoryProvider) {
|
|
items.push(
|
|
createMigrationConfigPatchItem({
|
|
id: "config:memory",
|
|
target: "memory",
|
|
path: ["memory"],
|
|
value: { backend: "builtin" },
|
|
message: "Use OpenClaw built-in file memory for imported Hermes memory files.",
|
|
conflict:
|
|
!params.ctx.overwrite &&
|
|
hasMigrationConfigPatchConflict(params.ctx.config, ["memory"], { backend: true }),
|
|
}),
|
|
);
|
|
items.push(
|
|
createMigrationConfigPatchItem({
|
|
id: "config:memory-plugin-slot",
|
|
target: "plugins.slots",
|
|
path: ["plugins", "slots"],
|
|
value: { memory: "memory-core" },
|
|
message: "Select the default OpenClaw memory plugin for imported file memory.",
|
|
conflict:
|
|
!params.ctx.overwrite &&
|
|
hasMigrationConfigPatchConflict(params.ctx.config, ["plugins", "slots"], {
|
|
memory: true,
|
|
}),
|
|
}),
|
|
);
|
|
}
|
|
|
|
if (memoryProvider === "honcho") {
|
|
const value = {
|
|
honcho: {
|
|
enabled: true,
|
|
config: childRecord(memory, "honcho"),
|
|
},
|
|
};
|
|
items.push(
|
|
createMigrationConfigPatchItem({
|
|
id: "config:memory-plugin:honcho",
|
|
target: "plugins.entries.honcho",
|
|
path: ["plugins", "entries"],
|
|
value,
|
|
message: "Preserve Hermes Honcho memory settings as a plugin entry for manual activation.",
|
|
conflict:
|
|
!params.ctx.overwrite &&
|
|
hasMigrationConfigPatchConflict(params.ctx.config, ["plugins", "entries"], value),
|
|
}),
|
|
);
|
|
items.push(
|
|
createMigrationManualItem({
|
|
id: "manual:memory-provider:honcho",
|
|
source: "config.yaml:memory.provider",
|
|
message:
|
|
"Hermes used Honcho memory. OpenClaw keeps built-in memory selected until the matching plugin is installed and reviewed.",
|
|
recommendation:
|
|
"Install or review the Honcho memory plugin before selecting it for plugins.slots.memory.",
|
|
}),
|
|
);
|
|
} else if (memoryProvider && !["builtin", "file", "files"].includes(memoryProvider)) {
|
|
items.push(
|
|
createMigrationManualItem({
|
|
id: `manual:memory-provider:${memoryProvider}`,
|
|
source: "config.yaml:memory.provider",
|
|
message: `Hermes memory provider "${memoryProvider}" does not have a known OpenClaw mapping.`,
|
|
recommendation: "Install or configure an equivalent OpenClaw memory plugin manually.",
|
|
}),
|
|
);
|
|
}
|
|
|
|
const providers = collectHermesProviders(params.config, params.modelRef);
|
|
if (providers.length > 0) {
|
|
const value = Object.fromEntries(providers.map((entry) => [entry.id, providerConfig(entry)]));
|
|
items.push(
|
|
createMigrationConfigPatchItem({
|
|
id: "config:model-providers",
|
|
target: "models.providers",
|
|
path: ["models", "providers"],
|
|
value,
|
|
message: "Import Hermes provider and custom endpoint config.",
|
|
conflict:
|
|
!params.ctx.overwrite &&
|
|
hasMigrationConfigPatchConflict(params.ctx.config, ["models", "providers"], value),
|
|
}),
|
|
);
|
|
}
|
|
|
|
const mcpConfig = params.config.mcp;
|
|
const rawMcpServers =
|
|
params.config.mcp_servers ??
|
|
(isRecord(mcpConfig) && isRecord(mcpConfig.servers) ? mcpConfig.servers : mcpConfig);
|
|
const mcpServers = mapMcpServers(rawMcpServers);
|
|
if (mcpServers) {
|
|
items.push(
|
|
createMigrationConfigPatchItem({
|
|
id: "config:mcp-servers",
|
|
target: "mcp.servers",
|
|
path: ["mcp", "servers"],
|
|
value: mcpServers,
|
|
message: "Import Hermes MCP server definitions.",
|
|
conflict:
|
|
!params.ctx.overwrite &&
|
|
hasMigrationConfigPatchConflict(params.ctx.config, ["mcp", "servers"], mcpServers),
|
|
}),
|
|
);
|
|
}
|
|
|
|
const skillEntries = mapSkillEntries(params.config);
|
|
if (skillEntries) {
|
|
items.push(
|
|
createMigrationConfigPatchItem({
|
|
id: "config:skill-entries",
|
|
target: "skills.entries",
|
|
path: ["skills", "entries"],
|
|
value: skillEntries,
|
|
message: "Import Hermes skill config values.",
|
|
conflict:
|
|
!params.ctx.overwrite &&
|
|
hasMigrationConfigPatchConflict(params.ctx.config, ["skills", "entries"], skillEntries),
|
|
}),
|
|
);
|
|
}
|
|
|
|
return items;
|
|
}
|
|
|
|
export async function applyConfigItem(
|
|
ctx: MigrationProviderContext,
|
|
item: MigrationItem,
|
|
): Promise<MigrationItem> {
|
|
return applyMigrationConfigPatchItem(ctx, item);
|
|
}
|
|
|
|
export function applyManualItem(item: MigrationItem): MigrationItem {
|
|
return applyMigrationManualItem(item);
|
|
}
|