mirror of
https://github.com/openclaw/openclaw.git
synced 2026-05-11 23:50:43 +00:00
133 lines
4.3 KiB
TypeScript
133 lines
4.3 KiB
TypeScript
import { applyMergePatch } from "../config/merge-patch.js";
|
|
import type { OpenClawConfig } from "../config/types.openclaw.js";
|
|
import { matchRootFileOpenFailure, type RootFileOpenFailure } from "../infra/boundary-file-read.js";
|
|
import { readRootJsonObjectSync } from "../infra/json-files.js";
|
|
import { normalizePluginsConfig, resolveEffectivePluginActivationState } from "./config-state.js";
|
|
import type { PluginBundleFormat } from "./manifest-types.js";
|
|
import { loadPluginManifestRegistryForPluginRegistry } from "./plugin-registry.js";
|
|
|
|
type ReadBundleJsonResult =
|
|
| { ok: true; raw: Record<string, unknown> }
|
|
| { ok: false; error: string };
|
|
|
|
export type BundleServerRuntimeSupport = {
|
|
hasSupportedServer: boolean;
|
|
supportedServerNames: string[];
|
|
unsupportedServerNames: string[];
|
|
diagnostics: string[];
|
|
};
|
|
|
|
export function readBundleJsonObject(params: {
|
|
rootDir: string;
|
|
relativePath: string;
|
|
onOpenFailure?: (failure: RootFileOpenFailure) => ReadBundleJsonResult;
|
|
}): ReadBundleJsonResult {
|
|
const result = readRootJsonObjectSync({
|
|
rootDir: params.rootDir,
|
|
relativePath: params.relativePath,
|
|
boundaryLabel: "plugin root",
|
|
rejectHardlinks: true,
|
|
});
|
|
if (result.ok) {
|
|
return { ok: true, raw: result.value };
|
|
}
|
|
if (result.reason === "open") {
|
|
return params.onOpenFailure?.(result.failure) ?? { ok: true, raw: {} };
|
|
}
|
|
return { ok: false, error: result.error };
|
|
}
|
|
|
|
export function resolveBundleJsonOpenFailure(params: {
|
|
failure: RootFileOpenFailure;
|
|
relativePath: string;
|
|
allowMissing?: boolean;
|
|
}): ReadBundleJsonResult {
|
|
return matchRootFileOpenFailure(params.failure, {
|
|
path: () => {
|
|
if (params.allowMissing) {
|
|
return { ok: true, raw: {} };
|
|
}
|
|
return { ok: false, error: `unable to read ${params.relativePath}: path` };
|
|
},
|
|
fallback: (failure) => ({
|
|
ok: false,
|
|
error: `unable to read ${params.relativePath}: ${failure.reason}`,
|
|
}),
|
|
});
|
|
}
|
|
|
|
export function inspectBundleServerRuntimeSupport<TConfig>(params: {
|
|
loaded: { config: TConfig; diagnostics: string[] };
|
|
resolveServers: (config: TConfig) => Record<string, Record<string, unknown>>;
|
|
}): BundleServerRuntimeSupport {
|
|
const supportedServerNames: string[] = [];
|
|
const unsupportedServerNames: string[] = [];
|
|
let hasSupportedServer = false;
|
|
for (const [serverName, server] of Object.entries(params.resolveServers(params.loaded.config))) {
|
|
if (typeof server.command === "string" && server.command.trim().length > 0) {
|
|
hasSupportedServer = true;
|
|
supportedServerNames.push(serverName);
|
|
continue;
|
|
}
|
|
unsupportedServerNames.push(serverName);
|
|
}
|
|
return {
|
|
hasSupportedServer,
|
|
supportedServerNames,
|
|
unsupportedServerNames,
|
|
diagnostics: params.loaded.diagnostics,
|
|
};
|
|
}
|
|
|
|
export function loadEnabledBundleConfig<TConfig, TDiagnostic>(params: {
|
|
workspaceDir: string;
|
|
cfg?: OpenClawConfig;
|
|
createEmptyConfig: () => TConfig;
|
|
loadBundleConfig: (params: {
|
|
pluginId: string;
|
|
rootDir: string;
|
|
bundleFormat: PluginBundleFormat;
|
|
}) => { config: TConfig; diagnostics: string[] };
|
|
createDiagnostic: (pluginId: string, message: string) => TDiagnostic;
|
|
}): { config: TConfig; diagnostics: TDiagnostic[] } {
|
|
const normalizedPlugins = normalizePluginsConfig(params.cfg?.plugins);
|
|
if (!normalizedPlugins.enabled) {
|
|
return { config: params.createEmptyConfig(), diagnostics: [] };
|
|
}
|
|
|
|
const registry = loadPluginManifestRegistryForPluginRegistry({
|
|
workspaceDir: params.workspaceDir,
|
|
config: params.cfg,
|
|
includeDisabled: true,
|
|
});
|
|
const diagnostics: TDiagnostic[] = [];
|
|
let merged = params.createEmptyConfig();
|
|
|
|
for (const record of registry.plugins) {
|
|
if (record.format !== "bundle" || !record.bundleFormat) {
|
|
continue;
|
|
}
|
|
const activationState = resolveEffectivePluginActivationState({
|
|
id: record.id,
|
|
origin: record.origin,
|
|
config: normalizedPlugins,
|
|
rootConfig: params.cfg,
|
|
});
|
|
if (!activationState.activated) {
|
|
continue;
|
|
}
|
|
|
|
const loaded = params.loadBundleConfig({
|
|
pluginId: record.id,
|
|
rootDir: record.rootDir,
|
|
bundleFormat: record.bundleFormat,
|
|
});
|
|
merged = applyMergePatch(merged, loaded.config) as TConfig;
|
|
for (const message of loaded.diagnostics) {
|
|
diagnostics.push(params.createDiagnostic(record.id, message));
|
|
}
|
|
}
|
|
|
|
return { config: merged, diagnostics };
|
|
}
|