mirror of
http://10.0.2.1:3031/sauer/claude-code.git
synced 2026-06-30 21:26:58 +10:00
135 lines
18 KiB
TypeScript
135 lines
18 KiB
TypeScript
|
|
/**
|
||
|
|
* Post-install/post-enable config prompt.
|
||
|
|
*
|
||
|
|
* Given a LoadedPlugin, checks both the top-level manifest.userConfig and the
|
||
|
|
* channel-specific userConfig. Walks PluginOptionsDialog through each
|
||
|
|
* unconfigured item, saving via the appropriate storage function. Calls
|
||
|
|
* onDone('skipped') immediately if nothing needs filling.
|
||
|
|
*/
|
||
|
|
|
||
|
|
import * as React from 'react';
|
||
|
|
import type { LoadedPlugin } from '../../types/plugin.js';
|
||
|
|
import { errorMessage } from '../../utils/errors.js';
|
||
|
|
import { loadMcpServerUserConfig, saveMcpServerUserConfig } from '../../utils/plugins/mcpbHandler.js';
|
||
|
|
import { getUnconfiguredChannels, type UnconfiguredChannel } from '../../utils/plugins/mcpPluginIntegration.js';
|
||
|
|
import { loadAllPlugins } from '../../utils/plugins/pluginLoader.js';
|
||
|
|
import { getUnconfiguredOptions, loadPluginOptions, type PluginOptionSchema, type PluginOptionValues, savePluginOptions } from '../../utils/plugins/pluginOptionsStorage.js';
|
||
|
|
import { PluginOptionsDialog } from './PluginOptionsDialog.js';
|
||
|
|
|
||
|
|
/**
|
||
|
|
* Post-install lookup: return the LoadedPlugin for the just-installed
|
||
|
|
* pluginId so the caller can divert to PluginOptionsFlow. Returns undefined
|
||
|
|
* if the plugin somehow didn't make it into the fresh load — callers treat
|
||
|
|
* undefined as "carry on closing."
|
||
|
|
*
|
||
|
|
* Install should have cleared caches already; loadAllPlugins reads fresh.
|
||
|
|
*/
|
||
|
|
export async function findPluginOptionsTarget(pluginId: string): Promise<LoadedPlugin | undefined> {
|
||
|
|
const {
|
||
|
|
enabled,
|
||
|
|
disabled
|
||
|
|
} = await loadAllPlugins();
|
||
|
|
return [...enabled, ...disabled].find(p => p.repository === pluginId || p.source === pluginId);
|
||
|
|
}
|
||
|
|
|
||
|
|
/**
|
||
|
|
* A single dialog step in the walk. Top-level options and channels both
|
||
|
|
* collapse to this shape — the only difference is which save function runs.
|
||
|
|
*/
|
||
|
|
type ConfigStep = {
|
||
|
|
key: string;
|
||
|
|
title: string;
|
||
|
|
subtitle: string;
|
||
|
|
schema: PluginOptionSchema;
|
||
|
|
/** Returns any already-saved values so PluginOptionsDialog can pre-fill and
|
||
|
|
* skip unchanged sensitive fields on reconfigure. */
|
||
|
|
load: () => PluginOptionValues | undefined;
|
||
|
|
save: (values: PluginOptionValues) => void;
|
||
|
|
};
|
||
|
|
type Props = {
|
||
|
|
plugin: LoadedPlugin;
|
||
|
|
/** `name@marketplace` — the savePluginOptions / saveMcpServerUserConfig key. */
|
||
|
|
pluginId: string;
|
||
|
|
/**
|
||
|
|
* `configured` = user filled all fields. `skipped` = nothing needed
|
||
|
|
* configuring, or user hit cancel. `error` = save threw.
|
||
|
|
*/
|
||
|
|
onDone: (outcome: 'configured' | 'skipped' | 'error', detail?: string) => void;
|
||
|
|
};
|
||
|
|
export function PluginOptionsFlow({
|
||
|
|
plugin,
|
||
|
|
pluginId,
|
||
|
|
onDone
|
||
|
|
}: Props): React.ReactNode {
|
||
|
|
// Build the step list once at mount. Re-calling after a save would drop the
|
||
|
|
// item we just configured.
|
||
|
|
const [steps] = React.useState<ConfigStep[]>(() => {
|
||
|
|
const result: ConfigStep[] = [];
|
||
|
|
|
||
|
|
// Top-level manifest.userConfig
|
||
|
|
const unconfigured = getUnconfiguredOptions(plugin);
|
||
|
|
if (Object.keys(unconfigured).length > 0) {
|
||
|
|
result.push({
|
||
|
|
key: 'top-level',
|
||
|
|
title: `Configure ${plugin.name}`,
|
||
|
|
subtitle: 'Plugin options',
|
||
|
|
schema: unconfigured,
|
||
|
|
load: () => loadPluginOptions(pluginId),
|
||
|
|
save: values => savePluginOptions(pluginId, values, plugin.manifest.userConfig!)
|
||
|
|
});
|
||
|
|
}
|
||
|
|
|
||
|
|
// Per-channel userConfig (assistant-mode channels)
|
||
|
|
const channels: UnconfiguredChannel[] = getUnconfiguredChannels(plugin);
|
||
|
|
for (const channel of channels) {
|
||
|
|
result.push({
|
||
|
|
key: `channel:${channel.server}`,
|
||
|
|
title: `Configure ${channel.displayName}`,
|
||
|
|
subtitle: `Plugin: ${plugin.name}`,
|
||
|
|
schema: channel.configSchema,
|
||
|
|
load: () => loadMcpServerUserConfig(pluginId, channel.server) ?? undefined,
|
||
|
|
save: values_0 => saveMcpServerUserConfig(pluginId, channel.server, values_0, channel.configSchema)
|
||
|
|
});
|
||
|
|
}
|
||
|
|
return result;
|
||
|
|
});
|
||
|
|
const [index, setIndex] = React.useState(0);
|
||
|
|
|
||
|
|
// Latest-ref: lets the effect close over the current onDone without
|
||
|
|
// re-running when the parent re-renders.
|
||
|
|
const onDoneRef = React.useRef(onDone);
|
||
|
|
onDoneRef.current = onDone;
|
||
|
|
|
||
|
|
// Nothing to configure → tell the caller and render nothing. Effect,
|
||
|
|
// not inline call: calling setState in the parent during our render
|
||
|
|
// is a React rules-of-hooks violation.
|
||
|
|
React.useEffect(() => {
|
||
|
|
if (steps.length === 0) {
|
||
|
|
onDoneRef.current('skipped');
|
||
|
|
}
|
||
|
|
}, [steps.length]);
|
||
|
|
if (steps.length === 0) {
|
||
|
|
return null;
|
||
|
|
}
|
||
|
|
const current = steps[index]!;
|
||
|
|
function handleSave(values_1: PluginOptionValues): void {
|
||
|
|
try {
|
||
|
|
current.save(values_1);
|
||
|
|
} catch (err) {
|
||
|
|
onDone('error', errorMessage(err));
|
||
|
|
return;
|
||
|
|
}
|
||
|
|
const next = index + 1;
|
||
|
|
if (next < steps.length) {
|
||
|
|
setIndex(next);
|
||
|
|
} else {
|
||
|
|
onDone('configured');
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
// key forces a remount when advancing to the next step — React would
|
||
|
|
// otherwise reuse the instance and carry PluginOptionsDialog's
|
||
|
|
// internal useState (field index, typed values) over.
|
||
|
|
return <PluginOptionsDialog key={current.key} title={current.title} subtitle={current.subtitle} configSchema={current.schema} initialValues={current.load()} onSave={handleSave} onCancel={() => onDone('skipped')} />;
|
||
|
|
}
|
||
|
|
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsIkxvYWRlZFBsdWdpbiIsImVycm9yTWVzc2FnZSIsImxvYWRNY3BTZXJ2ZXJVc2VyQ29uZmlnIiwic2F2ZU1jcFNlcnZlclVzZXJDb25maWciLCJnZXRVbmNvbmZpZ3VyZWRDaGFubmVscyIsIlVuY29uZmlndXJlZENoYW5uZWwiLCJsb2FkQWxsUGx1Z2lucyIsImdldFVuY29uZmlndXJlZE9wdGlvbnMiLCJsb2FkUGx1Z2luT3B0aW9ucyIsIlBsdWdpbk9wdGlvblNjaGVtYSIsIlBsdWdpbk9wdGlvblZhbHVlcyIsInNhdmVQbHVnaW5PcHRpb25zIiwiUGx1Z2luT3B0aW9uc0RpYWxvZyIsImZpbmRQbHVnaW5PcHRpb25zVGFyZ2V0IiwicGx1Z2luSWQiLCJQcm9taXNlIiwiZW5hYmxlZCIsImRpc2FibGVkIiwiZmluZCIsInAiLCJyZXBvc2l0b3J5Iiwic291cmNlIiwiQ29uZmlnU3RlcCIsImtleSIsInRpdGxlIiwic3VidGl0bGUiLCJzY2hlbWEiLCJsb2FkIiwic2F2ZSIsInZhbHVlcyIsIlByb3BzIiwicGx1Z2luIiwib25Eb25lIiwib3V0Y29tZSIsImRldGFpbCIsIlBsdWdpbk9wdGlvbnNGbG93IiwiUmVhY3ROb2RlIiwic3RlcHMiLCJ1c2VTdGF0ZSIsInJlc3VsdCIsInVuY29uZmlndXJlZCIsIk9iamVjdCIsImtleXMiLCJsZW5ndGgiLCJwdXNoIiwibmFtZSIsIm1hbmlmZXN0IiwidXNlckNvbmZpZyIsImNoYW5uZWxzIiwiY2hhbm5lbCIsInNlcnZlciIsImRpc3BsYXlOYW1lIiwiY29uZmlnU2NoZW1hIiwidW5kZWZpbmVkIiwiaW5kZXgiLCJzZXRJbmRleCIsIm9uRG9uZVJlZiIsInVzZVJlZiIsImN1cnJlbnQiLCJ1c2VFZmZlY3QiLCJoYW5kbGVTYXZlIiwiZXJyIiwibmV4dCJdLCJzb3VyY2VzIjpbIlBsdWdpbk9wdGlvbnNGbG93LnRzeCJdLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIFBvc3QtaW5zdGFsbC9wb3N0LWVuYWJsZSBjb25maWcgcHJvbXB0LlxuICpcbiAqIEdpdmVuIGEgTG9hZGVkUGx1Z2luLCBjaGVja3MgYm90aCB0aGUgdG9wLWxldmVsIG1hbmlmZXN0LnVzZXJDb25maWcgYW5kIHRoZVxuICogY2hhbm5lbC1zcGVjaWZpYyB1c2VyQ29uZmlnLiBXYWxrcyBQbHVnaW5PcHRpb25zRGlhbG9nIHRocm91Z2ggZWFjaFxuICogdW5jb25maWd1cmVkIGl0ZW0sIHNhdmluZyB2aWEgdGhlIGFwcHJvcHJpYXRlIHN0b3JhZ2UgZnVuY3Rpb24uIENhbGxzXG4gKiBvbkRvbmUoJ3NraXBwZWQnKSBpbW1lZGlhdGVseSBpZiBub3RoaW5nIG5lZWRzIGZpbGxpbmcuXG4gKi9cblxuaW1wb3J0ICogYXMgUmVhY3QgZnJvbSAncmVhY3QnXG5pbXBvcnQgdHlwZSB7IExvYWRlZFBsdWdpbiB9IGZyb20gJy4uLy4uL3R5cGVzL3BsdWdpbi5qcydcbmltcG9ydCB7IGVycm9yTWVzc2FnZSB9IGZyb20gJy4uLy4uL3V0aWxzL2Vycm9ycy5qcydcbmltcG9ydCB7XG4gIGxvYWRNY3BTZXJ2ZXJVc2VyQ29uZmlnLFxuICBzYXZlTWNwU2VydmVyVXNlckNvbmZpZyxcbn0gZnJvbSAnLi4vLi4vdXRpbHMvcGx1Z2lucy9tY3BiSGFuZGxlci5qcydcbmltcG9ydCB7XG4gIGdldFVuY29uZmlndXJlZENoYW5uZWxzLFxuICB0eXBlIFVuY29uZmlndXJlZENoYW5uZWwsXG59IGZyb20gJy4uLy4uL3V0aWxzL3BsdWdpbnMvbWNwUGx1Z2luSW50ZWdyYXRpb24uanMnXG5pbXBvcnQgeyBsb2FkQWxsUGx1Z2lucyB9IGZyb20gJy4uLy4uL3V0aWxzL3BsdWdpbnMvcGx1Z2luTG9hZGVyLmpzJ1xuaW1wb3J0IHtcbiAgZ2V0VW5jb25maWd1cmVkT3B0aW9ucyxcbiAgbG9hZFBsdWdpbk9wdGlvbnMsXG4gIHR5cGUgUGx1Z2luT3B0aW9uU2NoZW1hLFxuICB0eXBlIFBsdWdpbk9wdGlvblZhbHVlcyxcbiAgc2F2ZVBsdWdpbk9wdGlvbnMsXG59IGZyb20gJy4uLy4uL3V0aWxzL3BsdWdpbnMvcGx1Z2luT3B0aW9uc1N0b3JhZ2UuanMnXG5pbXBvcnQgeyBQbHVnaW5PcHRpb25zRGlhbG9nIH0gZnJvbSAnLi9QbHVnaW5PcHRpb25zRGlhbG9nLmpzJ1xuXG4vKipcbiAqIFBvc3QtaW5zdGFsbCBsb29rdXA6IHJldHVybiB0aGUgTG9hZGVkUGx1Z2luIGZvciB0aGUganVzdC1pbnN0YWxsZWRcbiAqIHBsdWdpbklkIHNvIHRoZSBjYWxsZXIgY2FuIGRpdmVydCB0byBQbHVnaW5PcHRpb25zRmxvdy4gUmV0dXJucyB1bmRlZmluZWRcbiAqIGlmIHRoZSBwbHVnaW4gc29tZWhvdyBkaWRuJ3QgbWFrZSBpdCBpbnRvIHRoZSBmcmVzaCBsb2FkIOKAlCBjYWxsZXJzIHRyZWF0XG4gKiB1bmRlZmluZWQgYXMgXCJjYXJyeSBvbiBjbG9zaW5nLlwiXG4gKlxuICogSW5zdGFsbCBzaG91bGQgaGF2ZSBjbGVhcmVkIGNhY2hlcyBhbHJlYWR5OyBsb2FkQWxsUGx1Z2lucyByZWFkcyBmcmVzaC5cbiAqL1xuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIGZpbmRQbHVnaW5PcHRpb25zVGFyZ2V0KFxuICBwbHVnaW5JZDogc3RyaW5nLFxuKTogUHJvbWlzZTxMb2FkZWRQbHVnaW4gfCB1bmRlZmluZWQ+IHtcbiAgY29uc3QgeyBlbmFibGVkLCBkaXNhYmxlZCB9ID0gYXdhaXQgbG9hZEFsbFBsdWdpbnMoKVxuICByZXR1cm4gWy4uLmVuYWJsZWQsIC4uLmRpc2FibGVkXS5maW5kKFxuICAgIHAgPT4gcC5yZXBvc2l0b3J5ID09PSBwbHVnaW5JZCB8fCBwLnNvdXJjZSA9PT0gcGx1Z2luSWQsXG4gIClcbn1cblxuLyoqXG4gKiBBIHNpbmdsZSBkaWFsb2cgc3RlcCBpbiB0aGUgd2Fsay4gVG9wLWxldmVsIG9wdGlvbnMgYW5kIGNoYW5uZWxzIGJvdGhcbiAqIGNvbGxhcHNlIHRvIHRoaXMgc2hhcGUg4oCUIHRoZSBvbmx5IGRpZmZlcmVuY2UgaXMgd2hpY2ggc2F2ZSBmdW5jdGlvbiBydW5zLlxuICovXG50eXBlIENvbmZpZ1N0ZXAgPSB7XG4gIGtleTogc3RyaW5nXG4gIHRpdGxlOiBzdHJpbmdcbiAgc3VidGl0bGU6IHN0cmluZ1xuICBzY2hlbWE6IFBsdWdpbk9wdGlvblNjaGVtYVxuICAvKiogUmV0dXJucyBhbnkgYWxyZWFkeS1zYXZlZCB2YWx1ZXMgc28gUGx1Z2luT3B0aW9uc0RpYWxvZyBjYW4gcHJlLWZpbGwgYW5kXG4gICAqICBza2lwIHVuY2hhbmdlZCBzZW5zaXRpdmUgZmllbGRzIG9uIHJlY29uZmlndXJlLiAqL1xuICBsb2FkOiAoKSA9PiBQbHVnaW5PcHRpb25WYWx1ZXMgfCB1bmRlZmluZWRcbiAgc2F2ZTogKHZ
|