mirror of
https://github.com/openclaw/openclaw.git
synced 2026-05-30 07:13:39 +00:00
* refactor: share talk event metric extraction * refactor: reuse shared coercion helpers * refactor: reuse shared primitive guards * refactor: reuse shared record guard * refactor: reuse shared primitive helpers * refactor: reuse shared string guards * refactor: reuse shared non-empty string guard * refactor: share plugin primitive coercion helpers * refactor: reuse plugin coercion helpers * refactor: reuse plugin coercion helpers in more plugins * refactor: reuse channel coercion helpers * refactor: reuse monitor coercion helpers * refactor: reuse provider coercion helpers * refactor: reuse core coercion helpers * refactor: reuse runtime coercion helpers * refactor: reuse helper coercion in codex paths * refactor: reuse helper coercion in runtime paths * refactor: reuse codex app-server coercion helpers * refactor: reuse codex record helpers * refactor: reuse migration and qa record helpers * refactor: reuse feishu and core helper guards * refactor: reuse browser and policy coercion helpers * refactor: reuse memory wiki record helper * refactor: share boolean coercion helpers * refactor: reuse finite number coercion * refactor: reuse trimmed string list helpers * refactor: reuse string list normalization * refactor: reuse remaining string list helpers * refactor: reuse string entry normalizer * refactor: share sorted string helpers * refactor: share string list normalization * test: preserve command registry browser imports * refactor: reuse trimmed list helpers * refactor: reuse string dedupe helpers * refactor: reuse local dedupe helpers * refactor: reuse more string dedupe helpers * refactor: reuse command string dedupe helpers * refactor: dedupe memory path lists with helper * refactor: expose string dedupe helpers to plugins * refactor: reuse core string dedupe helpers * refactor: reuse shared unique value helpers * refactor: reuse unique helpers in agent utilities * refactor: reuse unique helpers in config plumbing * refactor: reuse unique helpers in extensions * refactor: reuse unique helpers in core utilities * refactor: reuse unique helpers in qa plugins * refactor: reuse unique helpers in memory plugins * refactor: reuse unique helpers in channel plugins * refactor: reuse unique helpers in core tails * refactor: reuse unique helper in comfy workflow * refactor: reuse unique helpers in test utilities * refactor: expose unique value helper to plugins * refactor: reuse unique helpers for numeric lists * refactor: replace index dedupe filters * refactor: reuse string entry normalization * refactor: reuse string normalization in plugin helpers * refactor: reuse string normalization in extension helpers * refactor: reuse string normalization in channel parsers * refactor: reuse string normalization in memory search * refactor: reuse string normalization in provider parsers * refactor: reuse string normalization in qa helpers * refactor: reuse string normalization in infra parsers * refactor: reuse string normalization in messaging parsers * refactor: reuse string normalization in core parsers * refactor: reuse string normalization in extension parsers * refactor: reuse string normalization in remaining parsers * refactor: reuse string normalization in final parser spots * refactor: reuse string normalization in qa media helpers * refactor: reuse normalization in provider and media lists * refactor: reuse normalization for remaining set filters * refactor: reuse normalization in policy allowlists * refactor: reuse normalization in session and owner lists * refactor: centralize primitive string lists * refactor: reuse lowercase entry helpers * refactor: reuse sorted string helpers * refactor: reuse unique trimmed helpers * refactor: reuse string normalization helpers * refactor: reuse catalog string helpers * refactor: reuse remaining string helpers * refactor: simplify remaining list normalization * refactor: reuse codex auth order normalization * chore: refresh plugin sdk api baseline * fix: make shared string sorting deterministic * chore: refresh plugin sdk api baseline * fix: align host env security ordering
267 lines
8.1 KiB
TypeScript
267 lines
8.1 KiB
TypeScript
import fs from "node:fs";
|
|
import path from "node:path";
|
|
import type { Command } from "commander";
|
|
import { tryReadJsonSync } from "../infra/json-files.js";
|
|
import { findBundledPluginSource } from "../plugins/bundled-sources.js";
|
|
import { loadPluginManifest } from "../plugins/manifest.js";
|
|
import {
|
|
listOfficialExternalPluginCatalogEntries,
|
|
resolveOfficialExternalPluginId,
|
|
resolveOfficialExternalPluginInstall,
|
|
} from "../plugins/official-external-plugin-catalog.js";
|
|
import { normalizeStringEntries } from "../shared/string-normalization.js";
|
|
import { resolveUserPath } from "../utils.js";
|
|
import { parseNpmPrefixSpec, resolveFileNpmSpecToLocalPath } from "./plugins-command-helpers.js";
|
|
|
|
type PluginInstallInvalidConfigPolicy = "deny" | "allow-plugin-recovery";
|
|
|
|
export type PluginInstallRequestContext = {
|
|
rawSpec: string;
|
|
normalizedSpec: string;
|
|
resolvedPath?: string;
|
|
marketplace?: string;
|
|
bundledPluginId?: string;
|
|
allowInvalidConfigRecovery?: boolean;
|
|
};
|
|
|
|
type PluginInstallRequestResolution =
|
|
| { ok: true; request: PluginInstallRequestContext }
|
|
| { ok: false; error: string };
|
|
|
|
function isPluginInstallCommand(commandPath: string[]): boolean {
|
|
return commandPath[0] === "plugins" && commandPath[1] === "install";
|
|
}
|
|
|
|
function readBundledInstallRecoveryMetadata(rootDir: string): {
|
|
pluginId?: string;
|
|
allowInvalidConfigRecovery: boolean;
|
|
} {
|
|
const packageJsonPath = path.join(rootDir, "package.json");
|
|
if (!fs.existsSync(packageJsonPath)) {
|
|
return { allowInvalidConfigRecovery: false };
|
|
}
|
|
const manifest = loadPluginManifest(rootDir, false);
|
|
const pluginId = manifest.ok ? manifest.manifest.id : undefined;
|
|
const parsed = tryReadJsonSync<{
|
|
openclaw?: {
|
|
install?: {
|
|
allowInvalidConfigRecovery?: boolean;
|
|
};
|
|
};
|
|
}>(packageJsonPath);
|
|
return {
|
|
...(pluginId ? { pluginId } : {}),
|
|
allowInvalidConfigRecovery: parsed?.openclaw?.install?.allowInvalidConfigRecovery === true,
|
|
};
|
|
}
|
|
|
|
function resolveBundledInstallRecoveryMetadata(
|
|
request: Pick<
|
|
PluginInstallRequestContext,
|
|
"rawSpec" | "normalizedSpec" | "resolvedPath" | "marketplace"
|
|
>,
|
|
): {
|
|
pluginId?: string;
|
|
allowInvalidConfigRecovery?: boolean;
|
|
} {
|
|
if (request.marketplace) {
|
|
return {};
|
|
}
|
|
if (request.resolvedPath && fs.existsSync(path.join(request.resolvedPath, "package.json"))) {
|
|
const direct = readBundledInstallRecoveryMetadata(request.resolvedPath);
|
|
if (direct.pluginId || direct.allowInvalidConfigRecovery) {
|
|
return direct;
|
|
}
|
|
}
|
|
const rawNpmPrefixSpec = parseNpmPrefixSpec(request.rawSpec);
|
|
const normalizedNpmPrefixSpec = parseNpmPrefixSpec(request.normalizedSpec);
|
|
for (const value of [
|
|
request.rawSpec.trim(),
|
|
request.normalizedSpec.trim(),
|
|
rawNpmPrefixSpec ?? "",
|
|
normalizedNpmPrefixSpec ?? "",
|
|
]) {
|
|
if (!value) {
|
|
continue;
|
|
}
|
|
const bundled = findBundledPluginSource({
|
|
lookup: { kind: "npmSpec", value },
|
|
});
|
|
if (!bundled) {
|
|
continue;
|
|
}
|
|
const recovered = readBundledInstallRecoveryMetadata(bundled.localPath);
|
|
return {
|
|
pluginId: recovered.pluginId ?? bundled.pluginId,
|
|
allowInvalidConfigRecovery: recovered.allowInvalidConfigRecovery,
|
|
};
|
|
}
|
|
return {};
|
|
}
|
|
|
|
function resolveOfficialExternalInstallRecoveryMetadata(
|
|
request: Pick<PluginInstallRequestContext, "rawSpec" | "normalizedSpec" | "marketplace">,
|
|
): {
|
|
pluginId?: string;
|
|
allowInvalidConfigRecovery?: boolean;
|
|
} {
|
|
if (request.marketplace) {
|
|
return {};
|
|
}
|
|
const rawNpmPrefixSpec = parseNpmPrefixSpec(request.rawSpec);
|
|
const normalizedNpmPrefixSpec = parseNpmPrefixSpec(request.normalizedSpec);
|
|
const values = new Set(
|
|
normalizeStringEntries([
|
|
request.rawSpec,
|
|
request.normalizedSpec,
|
|
rawNpmPrefixSpec ?? "",
|
|
normalizedNpmPrefixSpec ?? "",
|
|
]),
|
|
);
|
|
if (values.size === 0) {
|
|
return {};
|
|
}
|
|
for (const entry of listOfficialExternalPluginCatalogEntries()) {
|
|
const install = resolveOfficialExternalPluginInstall(entry);
|
|
const npmSpec = install?.npmSpec?.trim() || entry.name?.trim();
|
|
if (!npmSpec || !values.has(npmSpec)) {
|
|
continue;
|
|
}
|
|
const pluginId = resolveOfficialExternalPluginId(entry);
|
|
return {
|
|
...(pluginId ? { pluginId } : {}),
|
|
allowInvalidConfigRecovery: install?.allowInvalidConfigRecovery === true,
|
|
};
|
|
}
|
|
return {};
|
|
}
|
|
|
|
function resolvePluginInstallArgvTokens(commandPath: string[], argv: string[]): string[] {
|
|
const args = argv.slice(2);
|
|
let cursor = 0;
|
|
for (const segment of commandPath) {
|
|
while (cursor < args.length && args[cursor] !== segment) {
|
|
cursor += 1;
|
|
}
|
|
if (cursor >= args.length) {
|
|
return [];
|
|
}
|
|
cursor += 1;
|
|
}
|
|
return args.slice(cursor);
|
|
}
|
|
|
|
function resolvePluginInstallArgvRequest(commandPath: string[], argv: string[]) {
|
|
if (!isPluginInstallCommand(commandPath)) {
|
|
return null;
|
|
}
|
|
const tokens = resolvePluginInstallArgvTokens(commandPath, argv);
|
|
let rawSpec: string | null = null;
|
|
let marketplace: string | undefined;
|
|
for (let index = 0; index < tokens.length; index += 1) {
|
|
const token = tokens[index];
|
|
if (token.startsWith("--marketplace=")) {
|
|
marketplace = token.slice("--marketplace=".length);
|
|
continue;
|
|
}
|
|
if (token === "--marketplace") {
|
|
const value = tokens[index + 1];
|
|
if (typeof value === "string") {
|
|
marketplace = value;
|
|
index += 1;
|
|
}
|
|
continue;
|
|
}
|
|
if (token.startsWith("-")) {
|
|
continue;
|
|
}
|
|
rawSpec ??= token;
|
|
}
|
|
return rawSpec ? { rawSpec, marketplace } : null;
|
|
}
|
|
|
|
export function resolvePluginInstallRequestContext(params: {
|
|
rawSpec: string;
|
|
marketplace?: string;
|
|
}): PluginInstallRequestResolution {
|
|
if (params.marketplace) {
|
|
return {
|
|
ok: true,
|
|
request: {
|
|
rawSpec: params.rawSpec,
|
|
normalizedSpec: params.rawSpec,
|
|
marketplace: params.marketplace,
|
|
},
|
|
};
|
|
}
|
|
const fileSpec = resolveFileNpmSpecToLocalPath(params.rawSpec);
|
|
if (fileSpec && !fileSpec.ok) {
|
|
return {
|
|
ok: false,
|
|
error: fileSpec.error,
|
|
};
|
|
}
|
|
const normalizedSpec = fileSpec && fileSpec.ok ? fileSpec.path : params.rawSpec;
|
|
const bundledRecovered = resolveBundledInstallRecoveryMetadata({
|
|
rawSpec: params.rawSpec,
|
|
normalizedSpec,
|
|
resolvedPath: resolveUserPath(normalizedSpec),
|
|
marketplace: params.marketplace,
|
|
});
|
|
const officialRecovered = resolveOfficialExternalInstallRecoveryMetadata({
|
|
rawSpec: params.rawSpec,
|
|
normalizedSpec,
|
|
marketplace: params.marketplace,
|
|
});
|
|
const recovered =
|
|
officialRecovered.pluginId || officialRecovered.allowInvalidConfigRecovery !== undefined
|
|
? officialRecovered
|
|
: bundledRecovered;
|
|
return {
|
|
ok: true,
|
|
request: {
|
|
rawSpec: params.rawSpec,
|
|
normalizedSpec,
|
|
resolvedPath: resolveUserPath(normalizedSpec),
|
|
...(recovered.pluginId ? { bundledPluginId: recovered.pluginId } : {}),
|
|
...(recovered.allowInvalidConfigRecovery !== undefined
|
|
? { allowInvalidConfigRecovery: recovered.allowInvalidConfigRecovery }
|
|
: {}),
|
|
},
|
|
};
|
|
}
|
|
|
|
export function resolvePluginInstallPreactionRequest(params: {
|
|
actionCommand: Command;
|
|
commandPath: string[];
|
|
argv: string[];
|
|
}): PluginInstallRequestContext | null {
|
|
if (!isPluginInstallCommand(params.commandPath)) {
|
|
return null;
|
|
}
|
|
const argvRequest = resolvePluginInstallArgvRequest(params.commandPath, params.argv);
|
|
const opts = params.actionCommand.opts<Record<string, unknown>>();
|
|
const marketplace =
|
|
(typeof opts.marketplace === "string" && opts.marketplace.trim()
|
|
? opts.marketplace
|
|
: argvRequest?.marketplace) || undefined;
|
|
const rawSpec =
|
|
(typeof params.actionCommand.processedArgs?.[0] === "string"
|
|
? params.actionCommand.processedArgs[0]
|
|
: argvRequest?.rawSpec) ?? null;
|
|
if (!rawSpec) {
|
|
return null;
|
|
}
|
|
const request = resolvePluginInstallRequestContext({ rawSpec, marketplace });
|
|
return request.ok ? request.request : null;
|
|
}
|
|
|
|
export function resolvePluginInstallInvalidConfigPolicy(
|
|
request: PluginInstallRequestContext | null,
|
|
): PluginInstallInvalidConfigPolicy {
|
|
if (!request) {
|
|
return "deny";
|
|
}
|
|
return request.allowInvalidConfigRecovery === true ? "allow-plugin-recovery" : "deny";
|
|
}
|