Files
openclaw/src/plugins/provider-wizard.ts
2026-04-07 10:37:39 +01:00

314 lines
9.4 KiB
TypeScript

import { DEFAULT_PROVIDER } from "../agents/defaults.js";
import { normalizeProviderId } from "../agents/model-selection.js";
import type { OpenClawConfig } from "../config/config.js";
import {
normalizeOptionalLowercaseString,
normalizeOptionalString,
} from "../shared/string-coerce.js";
import type { WizardPrompter } from "../wizard/prompts.js";
import { resolvePluginProviders } from "./providers.runtime.js";
import type {
ProviderAuthMethod,
ProviderPlugin,
ProviderPluginWizardModelPicker,
ProviderPluginWizardSetup,
} from "./types.js";
export const PROVIDER_PLUGIN_CHOICE_PREFIX = "provider-plugin:";
export type ProviderWizardOption = {
value: string;
label: string;
hint?: string;
groupId: string;
groupLabel: string;
groupHint?: string;
onboardingScopes?: Array<"text-inference" | "image-generation">;
assistantPriority?: number;
assistantVisibility?: "visible" | "manual-only";
};
export type ProviderModelPickerEntry = {
value: string;
label: string;
hint?: string;
};
function resolveWizardSetupChoiceId(
provider: ProviderPlugin,
wizard: ProviderPluginWizardSetup,
): string {
const explicit = normalizeOptionalString(wizard.choiceId);
if (explicit) {
return explicit;
}
const explicitMethodId = normalizeOptionalString(wizard.methodId);
if (explicitMethodId) {
return buildProviderPluginMethodChoice(provider.id, explicitMethodId);
}
if (provider.auth.length === 1) {
return provider.id;
}
return buildProviderPluginMethodChoice(provider.id, provider.auth[0]?.id ?? "default");
}
function resolveMethodById(
provider: ProviderPlugin,
methodId?: string,
): ProviderAuthMethod | undefined {
const normalizedMethodId = normalizeOptionalLowercaseString(methodId);
if (!normalizedMethodId) {
return provider.auth[0];
}
return provider.auth.find(
(method) => normalizeOptionalLowercaseString(method.id) === normalizedMethodId,
);
}
function listMethodWizardSetups(provider: ProviderPlugin): Array<{
method: ProviderAuthMethod;
wizard: ProviderPluginWizardSetup;
}> {
return provider.auth
.map((method) => (method.wizard ? { method, wizard: method.wizard } : null))
.filter((entry): entry is { method: ProviderAuthMethod; wizard: ProviderPluginWizardSetup } =>
Boolean(entry),
);
}
function buildSetupOptionForMethod(params: {
provider: ProviderPlugin;
wizard: ProviderPluginWizardSetup;
method: ProviderAuthMethod;
value: string;
}): ProviderWizardOption {
const normalizedGroupId = normalizeOptionalString(params.wizard.groupId) || params.provider.id;
return {
value: normalizeOptionalString(params.value) ?? "",
label:
normalizeOptionalString(params.wizard.choiceLabel) ||
(params.provider.auth.length === 1 ? params.provider.label : params.method.label),
hint: normalizeOptionalString(params.wizard.choiceHint) || params.method.hint,
groupId: normalizedGroupId,
groupLabel: normalizeOptionalString(params.wizard.groupLabel) || params.provider.label,
groupHint: normalizeOptionalString(params.wizard.groupHint),
...(params.wizard.onboardingScopes ? { onboardingScopes: params.wizard.onboardingScopes } : {}),
...(typeof params.wizard.assistantPriority === "number" &&
Number.isFinite(params.wizard.assistantPriority)
? { assistantPriority: params.wizard.assistantPriority }
: {}),
...(params.wizard.assistantVisibility
? { assistantVisibility: params.wizard.assistantVisibility }
: {}),
};
}
export function buildProviderPluginMethodChoice(providerId: string, methodId: string): string {
return `${PROVIDER_PLUGIN_CHOICE_PREFIX}${normalizeOptionalString(providerId) ?? ""}:${normalizeOptionalString(methodId) ?? ""}`;
}
function resolveProviderWizardProviders(params: {
config?: OpenClawConfig;
workspaceDir?: string;
env?: NodeJS.ProcessEnv;
}): ProviderPlugin[] {
return resolvePluginProviders({
config: params.config,
workspaceDir: params.workspaceDir,
env: params.env,
mode: "setup",
});
}
export function resolveProviderWizardOptions(params: {
config?: OpenClawConfig;
workspaceDir?: string;
env?: NodeJS.ProcessEnv;
}): ProviderWizardOption[] {
const providers = resolveProviderWizardProviders(params);
const options: ProviderWizardOption[] = [];
for (const provider of providers) {
const methodSetups = listMethodWizardSetups(provider);
for (const { method, wizard } of methodSetups) {
options.push(
buildSetupOptionForMethod({
provider,
wizard,
method,
value:
normalizeOptionalString(wizard.choiceId) ||
buildProviderPluginMethodChoice(provider.id, method.id),
}),
);
}
if (methodSetups.length > 0) {
continue;
}
const setup = provider.wizard?.setup;
if (!setup) {
continue;
}
const explicitMethod = resolveMethodById(provider, setup.methodId);
if (explicitMethod) {
options.push(
buildSetupOptionForMethod({
provider,
wizard: setup,
method: explicitMethod,
value: resolveWizardSetupChoiceId(provider, setup),
}),
);
continue;
}
for (const method of provider.auth) {
options.push(
buildSetupOptionForMethod({
provider,
wizard: setup,
method,
value: buildProviderPluginMethodChoice(provider.id, method.id),
}),
);
}
}
return options;
}
function resolveModelPickerChoiceValue(
provider: ProviderPlugin,
modelPicker: ProviderPluginWizardModelPicker,
): string {
const explicitMethodId = normalizeOptionalString(modelPicker.methodId);
if (explicitMethodId) {
return buildProviderPluginMethodChoice(provider.id, explicitMethodId);
}
if (provider.auth.length === 1) {
return provider.id;
}
return buildProviderPluginMethodChoice(provider.id, provider.auth[0]?.id ?? "default");
}
export function resolveProviderModelPickerEntries(params: {
config?: OpenClawConfig;
workspaceDir?: string;
env?: NodeJS.ProcessEnv;
}): ProviderModelPickerEntry[] {
const providers = resolveProviderWizardProviders(params);
const entries: ProviderModelPickerEntry[] = [];
for (const provider of providers) {
const modelPicker = provider.wizard?.modelPicker;
if (!modelPicker) {
continue;
}
entries.push({
value: resolveModelPickerChoiceValue(provider, modelPicker),
label: normalizeOptionalString(modelPicker.label) || `${provider.label} (custom)`,
hint: normalizeOptionalString(modelPicker.hint),
});
}
return entries;
}
export function resolveProviderPluginChoice(params: {
providers: ProviderPlugin[];
choice: string;
}): {
provider: ProviderPlugin;
method: ProviderAuthMethod;
wizard?: ProviderPluginWizardSetup;
} | null {
const choice = normalizeOptionalString(params.choice) ?? "";
if (!choice) {
return null;
}
if (choice.startsWith(PROVIDER_PLUGIN_CHOICE_PREFIX)) {
const payload = choice.slice(PROVIDER_PLUGIN_CHOICE_PREFIX.length);
const separator = payload.indexOf(":");
const providerId = separator >= 0 ? payload.slice(0, separator) : payload;
const methodId = separator >= 0 ? payload.slice(separator + 1) : undefined;
const provider = params.providers.find(
(entry) => normalizeProviderId(entry.id) === normalizeProviderId(providerId),
);
if (!provider) {
return null;
}
const method = resolveMethodById(provider, methodId);
return method ? { provider, method } : null;
}
for (const provider of params.providers) {
for (const { method, wizard } of listMethodWizardSetups(provider)) {
const choiceId =
normalizeOptionalString(wizard.choiceId) ||
buildProviderPluginMethodChoice(provider.id, method.id);
if ((normalizeOptionalString(choiceId) ?? "") === choice) {
return { provider, method, wizard };
}
}
const setup = provider.wizard?.setup;
if (setup) {
const setupChoiceId = resolveWizardSetupChoiceId(provider, setup);
if ((normalizeOptionalString(setupChoiceId) ?? "") === choice) {
const method = resolveMethodById(provider, setup.methodId);
if (method) {
return { provider, method, wizard: setup };
}
}
}
if (
normalizeProviderId(provider.id) === normalizeProviderId(choice) &&
provider.auth.length > 0
) {
return { provider, method: provider.auth[0] };
}
}
return null;
}
export async function runProviderModelSelectedHook(params: {
config: OpenClawConfig;
model: string;
prompter: WizardPrompter;
agentDir?: string;
workspaceDir?: string;
env?: NodeJS.ProcessEnv;
}): Promise<void> {
const rawModel = params.model.trim();
if (!rawModel) {
return;
}
const slashIndex = rawModel.indexOf("/");
const selectedProviderId =
slashIndex === -1
? DEFAULT_PROVIDER
: normalizeProviderId(rawModel.slice(0, slashIndex).trim());
if (!selectedProviderId || (slashIndex !== -1 && !rawModel.slice(slashIndex + 1).trim())) {
return;
}
const providers = resolveProviderWizardProviders({
config: params.config,
workspaceDir: params.workspaceDir,
env: params.env,
});
const provider = providers.find((entry) => normalizeProviderId(entry.id) === selectedProviderId);
if (!provider?.onModelSelected) {
return;
}
await provider.onModelSelected({
config: params.config,
model: params.model,
prompter: params.prompter,
agentDir: params.agentDir,
workspaceDir: params.workspaceDir,
});
}