mirror of
https://github.com/openclaw/openclaw.git
synced 2026-04-06 06:41:08 +00:00
214 lines
6.3 KiB
TypeScript
214 lines
6.3 KiB
TypeScript
import fs from "node:fs";
|
|
import path from "node:path";
|
|
import type { Command } from "commander";
|
|
import { findBundledPluginSource } from "../plugins/bundled-sources.js";
|
|
import { loadPluginManifest } from "../plugins/manifest.js";
|
|
import { resolveUserPath } from "../utils.js";
|
|
import { resolveFileNpmSpecToLocalPath } from "./plugins-command-helpers.js";
|
|
|
|
export type PluginInstallInvalidConfigPolicy = "deny" | "allow-bundled-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;
|
|
try {
|
|
const parsed = JSON.parse(fs.readFileSync(packageJsonPath, "utf8")) as {
|
|
openclaw?: {
|
|
install?: {
|
|
allowInvalidConfigRecovery?: boolean;
|
|
};
|
|
};
|
|
};
|
|
return {
|
|
...(pluginId ? { pluginId } : {}),
|
|
allowInvalidConfigRecovery: parsed.openclaw?.install?.allowInvalidConfigRecovery === true,
|
|
};
|
|
} catch {
|
|
return {
|
|
...(pluginId ? { pluginId } : {}),
|
|
allowInvalidConfigRecovery: false,
|
|
};
|
|
}
|
|
}
|
|
|
|
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;
|
|
}
|
|
}
|
|
for (const value of [request.rawSpec.trim(), request.normalizedSpec.trim()]) {
|
|
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 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 recovered = resolveBundledInstallRecoveryMetadata({
|
|
rawSpec: params.rawSpec,
|
|
normalizedSpec,
|
|
resolvedPath: resolveUserPath(normalizedSpec),
|
|
marketplace: params.marketplace,
|
|
});
|
|
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-bundled-recovery" : "deny";
|
|
}
|