mirror of
https://github.com/openclaw/openclaw.git
synced 2026-03-12 23:40:45 +00:00
Verified: - pnpm install --frozen-lockfile - pnpm build - pnpm check - pnpm test Co-authored-by: JustasMonkev <59362982+JustasMonkev@users.noreply.github.com> Co-authored-by: Tak Hoffman <781889+Takhoffman@users.noreply.github.com>
238 lines
6.3 KiB
TypeScript
238 lines
6.3 KiB
TypeScript
import fs from "node:fs/promises";
|
|
import path from "node:path";
|
|
import type { OpenClawConfig } from "../config/config.js";
|
|
import type { PluginInstallRecord } from "../config/types.plugins.js";
|
|
import { resolvePluginInstallDir } from "./install.js";
|
|
import { defaultSlotIdForKey } from "./slots.js";
|
|
|
|
export type UninstallActions = {
|
|
entry: boolean;
|
|
install: boolean;
|
|
allowlist: boolean;
|
|
loadPath: boolean;
|
|
memorySlot: boolean;
|
|
directory: boolean;
|
|
};
|
|
|
|
export type UninstallPluginResult =
|
|
| {
|
|
ok: true;
|
|
config: OpenClawConfig;
|
|
pluginId: string;
|
|
actions: UninstallActions;
|
|
warnings: string[];
|
|
}
|
|
| { ok: false; error: string };
|
|
|
|
export function resolveUninstallDirectoryTarget(params: {
|
|
pluginId: string;
|
|
hasInstall: boolean;
|
|
installRecord?: PluginInstallRecord;
|
|
extensionsDir?: string;
|
|
}): string | null {
|
|
if (!params.hasInstall) {
|
|
return null;
|
|
}
|
|
|
|
if (params.installRecord?.source === "path") {
|
|
return null;
|
|
}
|
|
|
|
let defaultPath: string;
|
|
try {
|
|
defaultPath = resolvePluginInstallDir(params.pluginId, params.extensionsDir);
|
|
} catch {
|
|
return null;
|
|
}
|
|
|
|
const configuredPath = params.installRecord?.installPath;
|
|
if (!configuredPath) {
|
|
return defaultPath;
|
|
}
|
|
|
|
if (path.resolve(configuredPath) === path.resolve(defaultPath)) {
|
|
return configuredPath;
|
|
}
|
|
|
|
// Never trust configured installPath blindly for recursive deletes.
|
|
return defaultPath;
|
|
}
|
|
|
|
/**
|
|
* Remove plugin references from config (pure config mutation).
|
|
* Returns a new config with the plugin removed from entries, installs, allow, load.paths, and slots.
|
|
*/
|
|
export function removePluginFromConfig(
|
|
cfg: OpenClawConfig,
|
|
pluginId: string,
|
|
): { config: OpenClawConfig; actions: Omit<UninstallActions, "directory"> } {
|
|
const actions: Omit<UninstallActions, "directory"> = {
|
|
entry: false,
|
|
install: false,
|
|
allowlist: false,
|
|
loadPath: false,
|
|
memorySlot: false,
|
|
};
|
|
|
|
const pluginsConfig = cfg.plugins ?? {};
|
|
|
|
// Remove from entries
|
|
let entries = pluginsConfig.entries;
|
|
if (entries && pluginId in entries) {
|
|
const { [pluginId]: _, ...rest } = entries;
|
|
entries = Object.keys(rest).length > 0 ? rest : undefined;
|
|
actions.entry = true;
|
|
}
|
|
|
|
// Remove from installs
|
|
let installs = pluginsConfig.installs;
|
|
const installRecord = installs?.[pluginId];
|
|
if (installs && pluginId in installs) {
|
|
const { [pluginId]: _, ...rest } = installs;
|
|
installs = Object.keys(rest).length > 0 ? rest : undefined;
|
|
actions.install = true;
|
|
}
|
|
|
|
// Remove from allowlist
|
|
let allow = pluginsConfig.allow;
|
|
if (Array.isArray(allow) && allow.includes(pluginId)) {
|
|
allow = allow.filter((id) => id !== pluginId);
|
|
if (allow.length === 0) {
|
|
allow = undefined;
|
|
}
|
|
actions.allowlist = true;
|
|
}
|
|
|
|
// Remove linked path from load.paths (for source === "path" plugins)
|
|
let load = pluginsConfig.load;
|
|
if (installRecord?.source === "path" && installRecord.sourcePath) {
|
|
const sourcePath = installRecord.sourcePath;
|
|
const loadPaths = load?.paths;
|
|
if (Array.isArray(loadPaths) && loadPaths.includes(sourcePath)) {
|
|
const nextLoadPaths = loadPaths.filter((p) => p !== sourcePath);
|
|
load = nextLoadPaths.length > 0 ? { ...load, paths: nextLoadPaths } : undefined;
|
|
actions.loadPath = true;
|
|
}
|
|
}
|
|
|
|
// Reset memory slot if this plugin was selected
|
|
let slots = pluginsConfig.slots;
|
|
if (slots?.memory === pluginId) {
|
|
slots = {
|
|
...slots,
|
|
memory: defaultSlotIdForKey("memory"),
|
|
};
|
|
actions.memorySlot = true;
|
|
}
|
|
if (slots && Object.keys(slots).length === 0) {
|
|
slots = undefined;
|
|
}
|
|
|
|
const newPlugins = {
|
|
...pluginsConfig,
|
|
entries,
|
|
installs,
|
|
allow,
|
|
load,
|
|
slots,
|
|
};
|
|
|
|
// Clean up undefined properties from newPlugins
|
|
const cleanedPlugins: typeof newPlugins = { ...newPlugins };
|
|
if (cleanedPlugins.entries === undefined) {
|
|
delete cleanedPlugins.entries;
|
|
}
|
|
if (cleanedPlugins.installs === undefined) {
|
|
delete cleanedPlugins.installs;
|
|
}
|
|
if (cleanedPlugins.allow === undefined) {
|
|
delete cleanedPlugins.allow;
|
|
}
|
|
if (cleanedPlugins.load === undefined) {
|
|
delete cleanedPlugins.load;
|
|
}
|
|
if (cleanedPlugins.slots === undefined) {
|
|
delete cleanedPlugins.slots;
|
|
}
|
|
|
|
const config: OpenClawConfig = {
|
|
...cfg,
|
|
plugins: Object.keys(cleanedPlugins).length > 0 ? cleanedPlugins : undefined,
|
|
};
|
|
|
|
return { config, actions };
|
|
}
|
|
|
|
export type UninstallPluginParams = {
|
|
config: OpenClawConfig;
|
|
pluginId: string;
|
|
deleteFiles?: boolean;
|
|
extensionsDir?: string;
|
|
};
|
|
|
|
/**
|
|
* Uninstall a plugin by removing it from config and optionally deleting installed files.
|
|
* Linked plugins (source === "path") never have their source directory deleted.
|
|
*/
|
|
export async function uninstallPlugin(
|
|
params: UninstallPluginParams,
|
|
): Promise<UninstallPluginResult> {
|
|
const { config, pluginId, deleteFiles = true, extensionsDir } = params;
|
|
|
|
// Validate plugin exists
|
|
const hasEntry = pluginId in (config.plugins?.entries ?? {});
|
|
const hasInstall = pluginId in (config.plugins?.installs ?? {});
|
|
|
|
if (!hasEntry && !hasInstall) {
|
|
return { ok: false, error: `Plugin not found: ${pluginId}` };
|
|
}
|
|
|
|
const installRecord = config.plugins?.installs?.[pluginId];
|
|
const isLinked = installRecord?.source === "path";
|
|
|
|
// Remove from config
|
|
const { config: newConfig, actions: configActions } = removePluginFromConfig(config, pluginId);
|
|
|
|
const actions: UninstallActions = {
|
|
...configActions,
|
|
directory: false,
|
|
};
|
|
const warnings: string[] = [];
|
|
|
|
const deleteTarget =
|
|
deleteFiles && !isLinked
|
|
? resolveUninstallDirectoryTarget({
|
|
pluginId,
|
|
hasInstall,
|
|
installRecord,
|
|
extensionsDir,
|
|
})
|
|
: null;
|
|
|
|
// Delete installed directory if requested and safe.
|
|
if (deleteTarget) {
|
|
const existed =
|
|
(await fs
|
|
.access(deleteTarget)
|
|
.then(() => true)
|
|
.catch(() => false)) ?? false;
|
|
try {
|
|
await fs.rm(deleteTarget, { recursive: true, force: true });
|
|
actions.directory = existed;
|
|
} catch (error) {
|
|
warnings.push(
|
|
`Failed to remove plugin directory ${deleteTarget}: ${error instanceof Error ? error.message : String(error)}`,
|
|
);
|
|
// Directory deletion failure is not fatal; config is the source of truth.
|
|
}
|
|
}
|
|
|
|
return {
|
|
ok: true,
|
|
config: newConfig,
|
|
pluginId,
|
|
actions,
|
|
warnings,
|
|
};
|
|
}
|