553 lines
17 KiB
TypeScript
553 lines
17 KiB
TypeScript
import type { Api, Model } from "@mariozechner/pi-ai";
|
|
import type { AuthStorage, ModelRegistry } from "@mariozechner/pi-coding-agent";
|
|
import type { OpenClawConfig } from "../../config/config.js";
|
|
import type { ModelDefinitionConfig } from "../../config/types.js";
|
|
import {
|
|
prepareProviderDynamicModel,
|
|
resolveProviderRuntimePlugin,
|
|
runProviderDynamicModel,
|
|
normalizeProviderResolvedModelWithPlugin,
|
|
} from "../../plugins/provider-runtime.js";
|
|
import { resolveOpenClawAgentDir } from "../agent-paths.js";
|
|
import { DEFAULT_CONTEXT_TOKENS } from "../defaults.js";
|
|
import { buildModelAliasLines } from "../model-alias-lines.js";
|
|
import { isSecretRefHeaderValueMarker } from "../model-auth-markers.js";
|
|
import { normalizeModelCompat } from "../model-compat.js";
|
|
import { findNormalizedProviderValue, normalizeProviderId } from "../model-selection.js";
|
|
import {
|
|
buildSuppressedBuiltInModelError,
|
|
shouldSuppressBuiltInModel,
|
|
} from "../model-suppression.js";
|
|
import { discoverAuthStorage, discoverModels } from "../pi-model-discovery.js";
|
|
import { normalizeResolvedProviderModel } from "./model.provider-normalization.js";
|
|
|
|
type InlineModelEntry = ModelDefinitionConfig & {
|
|
provider: string;
|
|
baseUrl?: string;
|
|
headers?: Record<string, string>;
|
|
};
|
|
type InlineProviderConfig = {
|
|
baseUrl?: string;
|
|
api?: ModelDefinitionConfig["api"];
|
|
models?: ModelDefinitionConfig[];
|
|
headers?: unknown;
|
|
};
|
|
|
|
function sanitizeModelHeaders(
|
|
headers: unknown,
|
|
opts?: { stripSecretRefMarkers?: boolean },
|
|
): Record<string, string> | undefined {
|
|
if (!headers || typeof headers !== "object" || Array.isArray(headers)) {
|
|
return undefined;
|
|
}
|
|
const next: Record<string, string> = {};
|
|
for (const [headerName, headerValue] of Object.entries(headers)) {
|
|
if (typeof headerValue !== "string") {
|
|
continue;
|
|
}
|
|
if (opts?.stripSecretRefMarkers && isSecretRefHeaderValueMarker(headerValue)) {
|
|
continue;
|
|
}
|
|
next[headerName] = headerValue;
|
|
}
|
|
return Object.keys(next).length > 0 ? next : undefined;
|
|
}
|
|
|
|
function normalizeResolvedModel(params: {
|
|
provider: string;
|
|
model: Model<Api>;
|
|
cfg?: OpenClawConfig;
|
|
agentDir?: string;
|
|
}): Model<Api> {
|
|
const pluginNormalized = normalizeProviderResolvedModelWithPlugin({
|
|
provider: params.provider,
|
|
config: params.cfg,
|
|
context: {
|
|
config: params.cfg,
|
|
agentDir: params.agentDir,
|
|
provider: params.provider,
|
|
modelId: params.model.id,
|
|
model: params.model,
|
|
},
|
|
});
|
|
if (pluginNormalized) {
|
|
return normalizeModelCompat(pluginNormalized);
|
|
}
|
|
return normalizeResolvedProviderModel(params);
|
|
}
|
|
|
|
export { buildModelAliasLines };
|
|
|
|
function resolveConfiguredProviderConfig(
|
|
cfg: OpenClawConfig | undefined,
|
|
provider: string,
|
|
): InlineProviderConfig | undefined {
|
|
const configuredProviders = cfg?.models?.providers;
|
|
if (!configuredProviders) {
|
|
return undefined;
|
|
}
|
|
const exactProviderConfig = configuredProviders[provider];
|
|
if (exactProviderConfig) {
|
|
return exactProviderConfig;
|
|
}
|
|
return findNormalizedProviderValue(configuredProviders, provider);
|
|
}
|
|
|
|
function applyConfiguredProviderOverrides(params: {
|
|
discoveredModel: Model<Api>;
|
|
providerConfig?: InlineProviderConfig;
|
|
modelId: string;
|
|
}): Model<Api> {
|
|
const { discoveredModel, providerConfig, modelId } = params;
|
|
if (!providerConfig) {
|
|
return {
|
|
...discoveredModel,
|
|
// Discovered models originate from models.json and may contain persistence markers.
|
|
headers: sanitizeModelHeaders(discoveredModel.headers, { stripSecretRefMarkers: true }),
|
|
};
|
|
}
|
|
const configuredModel = providerConfig.models?.find((candidate) => candidate.id === modelId);
|
|
const discoveredHeaders = sanitizeModelHeaders(discoveredModel.headers, {
|
|
stripSecretRefMarkers: true,
|
|
});
|
|
const providerHeaders = sanitizeModelHeaders(providerConfig.headers, {
|
|
stripSecretRefMarkers: true,
|
|
});
|
|
const configuredHeaders = sanitizeModelHeaders(configuredModel?.headers, {
|
|
stripSecretRefMarkers: true,
|
|
});
|
|
if (!configuredModel && !providerConfig.baseUrl && !providerConfig.api && !providerHeaders) {
|
|
return {
|
|
...discoveredModel,
|
|
headers: discoveredHeaders,
|
|
};
|
|
}
|
|
const resolvedInput = configuredModel?.input ?? discoveredModel.input;
|
|
const normalizedInput =
|
|
Array.isArray(resolvedInput) && resolvedInput.length > 0
|
|
? resolvedInput.filter((item) => item === "text" || item === "image")
|
|
: (["text"] as Array<"text" | "image">);
|
|
|
|
return {
|
|
...discoveredModel,
|
|
api: configuredModel?.api ?? providerConfig.api ?? discoveredModel.api,
|
|
baseUrl: providerConfig.baseUrl ?? discoveredModel.baseUrl,
|
|
reasoning: configuredModel?.reasoning ?? discoveredModel.reasoning,
|
|
input: normalizedInput,
|
|
cost: configuredModel?.cost ?? discoveredModel.cost,
|
|
contextWindow: configuredModel?.contextWindow ?? discoveredModel.contextWindow,
|
|
maxTokens: configuredModel?.maxTokens ?? discoveredModel.maxTokens,
|
|
headers:
|
|
discoveredHeaders || providerHeaders || configuredHeaders
|
|
? {
|
|
...discoveredHeaders,
|
|
...providerHeaders,
|
|
...configuredHeaders,
|
|
}
|
|
: undefined,
|
|
compat: configuredModel?.compat ?? discoveredModel.compat,
|
|
};
|
|
}
|
|
|
|
export function buildInlineProviderModels(
|
|
providers: Record<string, InlineProviderConfig>,
|
|
): InlineModelEntry[] {
|
|
return Object.entries(providers).flatMap(([providerId, entry]) => {
|
|
const trimmed = providerId.trim();
|
|
if (!trimmed) {
|
|
return [];
|
|
}
|
|
const providerHeaders = sanitizeModelHeaders(entry?.headers, {
|
|
stripSecretRefMarkers: true,
|
|
});
|
|
return (entry?.models ?? []).map((model) => ({
|
|
...model,
|
|
provider: trimmed,
|
|
baseUrl: entry?.baseUrl,
|
|
api: model.api ?? entry?.api,
|
|
headers: (() => {
|
|
const modelHeaders = sanitizeModelHeaders((model as InlineModelEntry).headers, {
|
|
stripSecretRefMarkers: true,
|
|
});
|
|
if (!providerHeaders && !modelHeaders) {
|
|
return undefined;
|
|
}
|
|
return {
|
|
...providerHeaders,
|
|
...modelHeaders,
|
|
};
|
|
})(),
|
|
}));
|
|
});
|
|
}
|
|
|
|
function resolveExplicitModelWithRegistry(params: {
|
|
provider: string;
|
|
modelId: string;
|
|
modelRegistry: ModelRegistry;
|
|
cfg?: OpenClawConfig;
|
|
agentDir?: string;
|
|
}): { kind: "resolved"; model: Model<Api> } | { kind: "suppressed" } | undefined {
|
|
const { provider, modelId, modelRegistry, cfg, agentDir } = params;
|
|
if (shouldSuppressBuiltInModel({ provider, id: modelId })) {
|
|
return { kind: "suppressed" };
|
|
}
|
|
const providerConfig = resolveConfiguredProviderConfig(cfg, provider);
|
|
const model = modelRegistry.find(provider, modelId) as Model<Api> | null;
|
|
|
|
if (model) {
|
|
return {
|
|
kind: "resolved",
|
|
model: normalizeResolvedModel({
|
|
provider,
|
|
cfg,
|
|
agentDir,
|
|
model: applyConfiguredProviderOverrides({
|
|
discoveredModel: model,
|
|
providerConfig,
|
|
modelId,
|
|
}),
|
|
}),
|
|
};
|
|
}
|
|
|
|
const providers = cfg?.models?.providers ?? {};
|
|
const inlineModels = buildInlineProviderModels(providers);
|
|
const normalizedProvider = normalizeProviderId(provider);
|
|
const inlineMatch = inlineModels.find(
|
|
(entry) => normalizeProviderId(entry.provider) === normalizedProvider && entry.id === modelId,
|
|
);
|
|
if (inlineMatch?.api) {
|
|
return {
|
|
kind: "resolved",
|
|
model: normalizeResolvedModel({
|
|
provider,
|
|
cfg,
|
|
agentDir,
|
|
model: inlineMatch as Model<Api>,
|
|
}),
|
|
};
|
|
}
|
|
|
|
return undefined;
|
|
}
|
|
|
|
function preferResolvedModel(
|
|
discoveredModel: Model<Api> | undefined,
|
|
dynamicModel: Model<Api> | undefined,
|
|
): Model<Api> | undefined {
|
|
if (!dynamicModel) {
|
|
return discoveredModel;
|
|
}
|
|
if (!discoveredModel) {
|
|
return dynamicModel;
|
|
}
|
|
const dynamicContextWindow = dynamicModel.contextWindow ?? 0;
|
|
const discoveredContextWindow = discoveredModel.contextWindow ?? 0;
|
|
if (dynamicContextWindow > discoveredContextWindow) {
|
|
return dynamicModel;
|
|
}
|
|
if (dynamicContextWindow < discoveredContextWindow) {
|
|
return discoveredModel;
|
|
}
|
|
return dynamicModel;
|
|
}
|
|
|
|
const OPENAI_CODEX_DYNAMIC_OVERRIDE_MODELS = new Set(["gpt-5.4"]);
|
|
const OPENAI_CODEX_DYNAMIC_OVERRIDE_TEMPLATES: Record<string, readonly string[]> = {
|
|
"gpt-5.4": ["gpt-5.3-codex", "gpt-5.2-codex"],
|
|
};
|
|
|
|
function shouldPreferDynamicModelOverride(params: { provider: string; modelId: string }): boolean {
|
|
return (
|
|
normalizeProviderId(params.provider) === "openai-codex" &&
|
|
OPENAI_CODEX_DYNAMIC_OVERRIDE_MODELS.has(params.modelId)
|
|
);
|
|
}
|
|
|
|
function hasDynamicOverrideTemplate(params: {
|
|
provider: string;
|
|
modelId: string;
|
|
modelRegistry: ModelRegistry;
|
|
}): boolean {
|
|
if (!shouldPreferDynamicModelOverride(params)) {
|
|
return false;
|
|
}
|
|
const templateIds = OPENAI_CODEX_DYNAMIC_OVERRIDE_TEMPLATES[params.modelId];
|
|
if (!templateIds?.length) {
|
|
return false;
|
|
}
|
|
return templateIds.some((templateId) => params.modelRegistry.find(params.provider, templateId));
|
|
}
|
|
|
|
function preserveDiscoveredTransportMetadata(params: {
|
|
discoveredModel: Model<Api> | undefined;
|
|
dynamicModel: Model<Api> | undefined;
|
|
providerConfig?: InlineProviderConfig;
|
|
modelId: string;
|
|
}): Model<Api> | undefined {
|
|
const { discoveredModel, dynamicModel, providerConfig, modelId } = params;
|
|
if (!discoveredModel || !dynamicModel) {
|
|
return dynamicModel;
|
|
}
|
|
const configuredModel = providerConfig?.models?.find((candidate) => candidate.id === modelId);
|
|
const discoveredHeaders = sanitizeModelHeaders(discoveredModel.headers, {
|
|
stripSecretRefMarkers: true,
|
|
});
|
|
const dynamicHeaders = sanitizeModelHeaders(dynamicModel.headers, {
|
|
stripSecretRefMarkers: true,
|
|
});
|
|
return {
|
|
...dynamicModel,
|
|
api: configuredModel?.api ?? providerConfig?.api ?? discoveredModel.api ?? dynamicModel.api,
|
|
baseUrl: providerConfig?.baseUrl ?? discoveredModel.baseUrl ?? dynamicModel.baseUrl,
|
|
input: configuredModel?.input ?? discoveredModel.input ?? dynamicModel.input,
|
|
compat: configuredModel?.compat ?? discoveredModel.compat ?? dynamicModel.compat,
|
|
maxTokens: configuredModel?.maxTokens ?? discoveredModel.maxTokens ?? dynamicModel.maxTokens,
|
|
cost: discoveredModel.cost ?? dynamicModel.cost,
|
|
headers:
|
|
discoveredHeaders || dynamicHeaders
|
|
? {
|
|
...dynamicHeaders,
|
|
...discoveredHeaders,
|
|
}
|
|
: undefined,
|
|
};
|
|
}
|
|
|
|
export function resolveModelWithRegistry(params: {
|
|
provider: string;
|
|
modelId: string;
|
|
modelRegistry: ModelRegistry;
|
|
cfg?: OpenClawConfig;
|
|
agentDir?: string;
|
|
}): Model<Api> | undefined {
|
|
const explicitModel = resolveExplicitModelWithRegistry(params);
|
|
if (explicitModel?.kind === "suppressed") {
|
|
return undefined;
|
|
}
|
|
const shouldCompareDynamicOverride =
|
|
shouldPreferDynamicModelOverride(params) && hasDynamicOverrideTemplate(params);
|
|
if (explicitModel?.kind === "resolved" && !shouldCompareDynamicOverride) {
|
|
return explicitModel.model;
|
|
}
|
|
|
|
const { provider, modelId, cfg, modelRegistry, agentDir } = params;
|
|
const providerConfig = resolveConfiguredProviderConfig(cfg, provider);
|
|
const pluginDynamicModel = runProviderDynamicModel({
|
|
provider,
|
|
config: cfg,
|
|
context: {
|
|
config: cfg,
|
|
agentDir,
|
|
provider,
|
|
modelId,
|
|
modelRegistry,
|
|
providerConfig,
|
|
},
|
|
});
|
|
const configuredDynamicModel = pluginDynamicModel
|
|
? applyConfiguredProviderOverrides({
|
|
discoveredModel: pluginDynamicModel,
|
|
providerConfig,
|
|
modelId,
|
|
})
|
|
: undefined;
|
|
const discoveredResolvedModel =
|
|
explicitModel?.kind === "resolved" ? explicitModel.model : undefined;
|
|
const dynamicModelForComparison = shouldCompareDynamicOverride
|
|
? preserveDiscoveredTransportMetadata({
|
|
discoveredModel: discoveredResolvedModel,
|
|
dynamicModel: configuredDynamicModel,
|
|
providerConfig,
|
|
modelId,
|
|
})
|
|
: configuredDynamicModel;
|
|
const preferredModel = preferResolvedModel(discoveredResolvedModel, dynamicModelForComparison);
|
|
if (preferredModel) {
|
|
if (preferredModel === explicitModel?.model) {
|
|
return preferredModel;
|
|
}
|
|
return normalizeResolvedModel({
|
|
provider,
|
|
cfg,
|
|
agentDir,
|
|
model: preferredModel,
|
|
});
|
|
}
|
|
|
|
const configuredModel = providerConfig?.models?.find((candidate) => candidate.id === modelId);
|
|
const providerHeaders = sanitizeModelHeaders(providerConfig?.headers, {
|
|
stripSecretRefMarkers: true,
|
|
});
|
|
const modelHeaders = sanitizeModelHeaders(configuredModel?.headers, {
|
|
stripSecretRefMarkers: true,
|
|
});
|
|
if (providerConfig || modelId.startsWith("mock-")) {
|
|
return normalizeResolvedModel({
|
|
provider,
|
|
cfg,
|
|
agentDir,
|
|
model: {
|
|
id: modelId,
|
|
name: modelId,
|
|
api: providerConfig?.api ?? "openai-responses",
|
|
provider,
|
|
baseUrl: providerConfig?.baseUrl,
|
|
reasoning: configuredModel?.reasoning ?? false,
|
|
input: ["text"],
|
|
cost: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0 },
|
|
contextWindow:
|
|
configuredModel?.contextWindow ??
|
|
providerConfig?.models?.[0]?.contextWindow ??
|
|
DEFAULT_CONTEXT_TOKENS,
|
|
maxTokens:
|
|
configuredModel?.maxTokens ??
|
|
providerConfig?.models?.[0]?.maxTokens ??
|
|
DEFAULT_CONTEXT_TOKENS,
|
|
headers:
|
|
providerHeaders || modelHeaders ? { ...providerHeaders, ...modelHeaders } : undefined,
|
|
} as Model<Api>,
|
|
});
|
|
}
|
|
|
|
return undefined;
|
|
}
|
|
|
|
export function resolveModel(
|
|
provider: string,
|
|
modelId: string,
|
|
agentDir?: string,
|
|
cfg?: OpenClawConfig,
|
|
): {
|
|
model?: Model<Api>;
|
|
error?: string;
|
|
authStorage: AuthStorage;
|
|
modelRegistry: ModelRegistry;
|
|
} {
|
|
const resolvedAgentDir = agentDir ?? resolveOpenClawAgentDir();
|
|
const authStorage = discoverAuthStorage(resolvedAgentDir);
|
|
const modelRegistry = discoverModels(authStorage, resolvedAgentDir);
|
|
const model = resolveModelWithRegistry({
|
|
provider,
|
|
modelId,
|
|
modelRegistry,
|
|
cfg,
|
|
agentDir: resolvedAgentDir,
|
|
});
|
|
if (model) {
|
|
return { model, authStorage, modelRegistry };
|
|
}
|
|
|
|
return {
|
|
error: buildUnknownModelError(provider, modelId),
|
|
authStorage,
|
|
modelRegistry,
|
|
};
|
|
}
|
|
|
|
export async function resolveModelAsync(
|
|
provider: string,
|
|
modelId: string,
|
|
agentDir?: string,
|
|
cfg?: OpenClawConfig,
|
|
): Promise<{
|
|
model?: Model<Api>;
|
|
error?: string;
|
|
authStorage: AuthStorage;
|
|
modelRegistry: ModelRegistry;
|
|
}> {
|
|
const resolvedAgentDir = agentDir ?? resolveOpenClawAgentDir();
|
|
const authStorage = discoverAuthStorage(resolvedAgentDir);
|
|
const modelRegistry = discoverModels(authStorage, resolvedAgentDir);
|
|
const explicitModel = resolveExplicitModelWithRegistry({
|
|
provider,
|
|
modelId,
|
|
modelRegistry,
|
|
cfg,
|
|
agentDir: resolvedAgentDir,
|
|
});
|
|
if (explicitModel?.kind === "suppressed") {
|
|
return {
|
|
error: buildUnknownModelError(provider, modelId),
|
|
authStorage,
|
|
modelRegistry,
|
|
};
|
|
}
|
|
const maybePrepareDynamicModel = async () => {
|
|
const providerPlugin = resolveProviderRuntimePlugin({
|
|
provider,
|
|
config: cfg,
|
|
});
|
|
if (providerPlugin?.prepareDynamicModel) {
|
|
await prepareProviderDynamicModel({
|
|
provider,
|
|
config: cfg,
|
|
context: {
|
|
config: cfg,
|
|
agentDir: resolvedAgentDir,
|
|
provider,
|
|
modelId,
|
|
modelRegistry,
|
|
providerConfig: resolveConfiguredProviderConfig(cfg, provider),
|
|
},
|
|
});
|
|
}
|
|
};
|
|
if (
|
|
!explicitModel ||
|
|
(explicitModel.kind === "resolved" &&
|
|
hasDynamicOverrideTemplate({ provider, modelId, modelRegistry }))
|
|
) {
|
|
await maybePrepareDynamicModel();
|
|
}
|
|
const model = resolveModelWithRegistry({
|
|
provider,
|
|
modelId,
|
|
modelRegistry,
|
|
cfg,
|
|
agentDir: resolvedAgentDir,
|
|
});
|
|
if (model) {
|
|
return { model, authStorage, modelRegistry };
|
|
}
|
|
|
|
return {
|
|
error: buildUnknownModelError(provider, modelId),
|
|
authStorage,
|
|
modelRegistry,
|
|
};
|
|
}
|
|
|
|
/**
|
|
* Build a more helpful error when the model is not found.
|
|
*
|
|
* Local providers (ollama, vllm) need a dummy API key to be registered.
|
|
* Users often configure `agents.defaults.model.primary: "ollama/…"` but
|
|
* forget to set `OLLAMA_API_KEY`, resulting in a confusing "Unknown model"
|
|
* error. This detects known providers that require opt-in auth and adds
|
|
* a hint.
|
|
*
|
|
* See: https://github.com/openclaw/openclaw/issues/17328
|
|
*/
|
|
const LOCAL_PROVIDER_HINTS: Record<string, string> = {
|
|
ollama:
|
|
"Ollama requires authentication to be registered as a provider. " +
|
|
'Set OLLAMA_API_KEY="ollama-local" (any value works) or run "openclaw configure". ' +
|
|
"See: https://docs.openclaw.ai/providers/ollama",
|
|
vllm:
|
|
"vLLM requires authentication to be registered as a provider. " +
|
|
'Set VLLM_API_KEY (any value works) or run "openclaw configure". ' +
|
|
"See: https://docs.openclaw.ai/providers/vllm",
|
|
};
|
|
|
|
function buildUnknownModelError(provider: string, modelId: string): string {
|
|
const suppressed = buildSuppressedBuiltInModelError({ provider, id: modelId });
|
|
if (suppressed) {
|
|
return suppressed;
|
|
}
|
|
const base = `Unknown model: ${provider}/${modelId}`;
|
|
const hint = LOCAL_PROVIDER_HINTS[provider.toLowerCase()];
|
|
return hint ? `${base}. ${hint}` : base;
|
|
}
|