mirror of
https://github.com/openclaw/openclaw.git
synced 2026-02-19 18:39:20 -05:00
495 lines
14 KiB
TypeScript
495 lines
14 KiB
TypeScript
import type { OpenClawConfig } from "../config/config.js";
|
|
import { resolveAgentModelPrimary } from "./agent-scope.js";
|
|
import { DEFAULT_MODEL, DEFAULT_PROVIDER } from "./defaults.js";
|
|
import type { ModelCatalogEntry } from "./model-catalog.js";
|
|
import { normalizeGoogleModelId } from "./models-config.providers.js";
|
|
|
|
export type ModelRef = {
|
|
provider: string;
|
|
model: string;
|
|
};
|
|
|
|
export type ThinkLevel = "off" | "minimal" | "low" | "medium" | "high" | "xhigh";
|
|
|
|
export type ModelAliasIndex = {
|
|
byAlias: Map<string, { alias: string; ref: ModelRef }>;
|
|
byKey: Map<string, string[]>;
|
|
};
|
|
|
|
const ANTHROPIC_MODEL_ALIASES: Record<string, string> = {
|
|
"opus-4.6": "claude-opus-4-6",
|
|
"opus-4.5": "claude-opus-4-5",
|
|
"sonnet-4.6": "claude-sonnet-4-6",
|
|
"sonnet-4.5": "claude-sonnet-4-5",
|
|
};
|
|
const OPENAI_CODEX_OAUTH_MODEL_PREFIXES = ["gpt-5.3-codex"] as const;
|
|
|
|
function normalizeAliasKey(value: string): string {
|
|
return value.trim().toLowerCase();
|
|
}
|
|
|
|
export function modelKey(provider: string, model: string) {
|
|
return `${provider}/${model}`;
|
|
}
|
|
|
|
export function normalizeProviderId(provider: string): string {
|
|
const normalized = provider.trim().toLowerCase();
|
|
if (normalized === "z.ai" || normalized === "z-ai") {
|
|
return "zai";
|
|
}
|
|
if (normalized === "opencode-zen") {
|
|
return "opencode";
|
|
}
|
|
if (normalized === "qwen") {
|
|
return "qwen-portal";
|
|
}
|
|
if (normalized === "kimi-code") {
|
|
return "kimi-coding";
|
|
}
|
|
return normalized;
|
|
}
|
|
|
|
export function findNormalizedProviderValue<T>(
|
|
entries: Record<string, T> | undefined,
|
|
provider: string,
|
|
): T | undefined {
|
|
if (!entries) {
|
|
return undefined;
|
|
}
|
|
const providerKey = normalizeProviderId(provider);
|
|
for (const [key, value] of Object.entries(entries)) {
|
|
if (normalizeProviderId(key) === providerKey) {
|
|
return value;
|
|
}
|
|
}
|
|
return undefined;
|
|
}
|
|
|
|
export function findNormalizedProviderKey(
|
|
entries: Record<string, unknown> | undefined,
|
|
provider: string,
|
|
): string | undefined {
|
|
if (!entries) {
|
|
return undefined;
|
|
}
|
|
const providerKey = normalizeProviderId(provider);
|
|
return Object.keys(entries).find((key) => normalizeProviderId(key) === providerKey);
|
|
}
|
|
|
|
export function isCliProvider(provider: string, cfg?: OpenClawConfig): boolean {
|
|
const normalized = normalizeProviderId(provider);
|
|
if (normalized === "claude-cli") {
|
|
return true;
|
|
}
|
|
if (normalized === "codex-cli") {
|
|
return true;
|
|
}
|
|
const backends = cfg?.agents?.defaults?.cliBackends ?? {};
|
|
return Object.keys(backends).some((key) => normalizeProviderId(key) === normalized);
|
|
}
|
|
|
|
function normalizeAnthropicModelId(model: string): string {
|
|
const trimmed = model.trim();
|
|
if (!trimmed) {
|
|
return trimmed;
|
|
}
|
|
const lower = trimmed.toLowerCase();
|
|
return ANTHROPIC_MODEL_ALIASES[lower] ?? trimmed;
|
|
}
|
|
|
|
function normalizeProviderModelId(provider: string, model: string): string {
|
|
if (provider === "anthropic") {
|
|
return normalizeAnthropicModelId(model);
|
|
}
|
|
if (provider === "google") {
|
|
return normalizeGoogleModelId(model);
|
|
}
|
|
return model;
|
|
}
|
|
|
|
function shouldUseOpenAICodexProvider(provider: string, model: string): boolean {
|
|
if (provider !== "openai") {
|
|
return false;
|
|
}
|
|
const normalized = model.trim().toLowerCase();
|
|
if (!normalized) {
|
|
return false;
|
|
}
|
|
return OPENAI_CODEX_OAUTH_MODEL_PREFIXES.some(
|
|
(prefix) => normalized === prefix || normalized.startsWith(`${prefix}-`),
|
|
);
|
|
}
|
|
|
|
export function normalizeModelRef(provider: string, model: string): ModelRef {
|
|
const normalizedProvider = normalizeProviderId(provider);
|
|
const normalizedModel = normalizeProviderModelId(normalizedProvider, model.trim());
|
|
if (shouldUseOpenAICodexProvider(normalizedProvider, normalizedModel)) {
|
|
return { provider: "openai-codex", model: normalizedModel };
|
|
}
|
|
return { provider: normalizedProvider, model: normalizedModel };
|
|
}
|
|
|
|
export function parseModelRef(raw: string, defaultProvider: string): ModelRef | null {
|
|
const trimmed = raw.trim();
|
|
if (!trimmed) {
|
|
return null;
|
|
}
|
|
const slash = trimmed.indexOf("/");
|
|
if (slash === -1) {
|
|
return normalizeModelRef(defaultProvider, trimmed);
|
|
}
|
|
const providerRaw = trimmed.slice(0, slash).trim();
|
|
const model = trimmed.slice(slash + 1).trim();
|
|
if (!providerRaw || !model) {
|
|
return null;
|
|
}
|
|
return normalizeModelRef(providerRaw, model);
|
|
}
|
|
|
|
export function resolveAllowlistModelKey(raw: string, defaultProvider: string): string | null {
|
|
const parsed = parseModelRef(raw, defaultProvider);
|
|
if (!parsed) {
|
|
return null;
|
|
}
|
|
return modelKey(parsed.provider, parsed.model);
|
|
}
|
|
|
|
export function buildConfiguredAllowlistKeys(params: {
|
|
cfg: OpenClawConfig | undefined;
|
|
defaultProvider: string;
|
|
}): Set<string> | null {
|
|
const rawAllowlist = Object.keys(params.cfg?.agents?.defaults?.models ?? {});
|
|
if (rawAllowlist.length === 0) {
|
|
return null;
|
|
}
|
|
|
|
const keys = new Set<string>();
|
|
for (const raw of rawAllowlist) {
|
|
const key = resolveAllowlistModelKey(String(raw ?? ""), params.defaultProvider);
|
|
if (key) {
|
|
keys.add(key);
|
|
}
|
|
}
|
|
return keys.size > 0 ? keys : null;
|
|
}
|
|
|
|
export function buildModelAliasIndex(params: {
|
|
cfg: OpenClawConfig;
|
|
defaultProvider: string;
|
|
}): ModelAliasIndex {
|
|
const byAlias = new Map<string, { alias: string; ref: ModelRef }>();
|
|
const byKey = new Map<string, string[]>();
|
|
|
|
const rawModels = params.cfg.agents?.defaults?.models ?? {};
|
|
for (const [keyRaw, entryRaw] of Object.entries(rawModels)) {
|
|
const parsed = parseModelRef(String(keyRaw ?? ""), params.defaultProvider);
|
|
if (!parsed) {
|
|
continue;
|
|
}
|
|
const alias = String((entryRaw as { alias?: string } | undefined)?.alias ?? "").trim();
|
|
if (!alias) {
|
|
continue;
|
|
}
|
|
const aliasKey = normalizeAliasKey(alias);
|
|
byAlias.set(aliasKey, { alias, ref: parsed });
|
|
const key = modelKey(parsed.provider, parsed.model);
|
|
const existing = byKey.get(key) ?? [];
|
|
existing.push(alias);
|
|
byKey.set(key, existing);
|
|
}
|
|
|
|
return { byAlias, byKey };
|
|
}
|
|
|
|
export function resolveModelRefFromString(params: {
|
|
raw: string;
|
|
defaultProvider: string;
|
|
aliasIndex?: ModelAliasIndex;
|
|
}): { ref: ModelRef; alias?: string } | null {
|
|
const trimmed = params.raw.trim();
|
|
if (!trimmed) {
|
|
return null;
|
|
}
|
|
if (!trimmed.includes("/")) {
|
|
const aliasKey = normalizeAliasKey(trimmed);
|
|
const aliasMatch = params.aliasIndex?.byAlias.get(aliasKey);
|
|
if (aliasMatch) {
|
|
return { ref: aliasMatch.ref, alias: aliasMatch.alias };
|
|
}
|
|
}
|
|
const parsed = parseModelRef(trimmed, params.defaultProvider);
|
|
if (!parsed) {
|
|
return null;
|
|
}
|
|
return { ref: parsed };
|
|
}
|
|
|
|
export function resolveConfiguredModelRef(params: {
|
|
cfg: OpenClawConfig;
|
|
defaultProvider: string;
|
|
defaultModel: string;
|
|
}): ModelRef {
|
|
const rawModel = (() => {
|
|
const raw = params.cfg.agents?.defaults?.model as { primary?: string } | string | undefined;
|
|
if (typeof raw === "string") {
|
|
return raw.trim();
|
|
}
|
|
return raw?.primary?.trim() ?? "";
|
|
})();
|
|
if (rawModel) {
|
|
const trimmed = rawModel.trim();
|
|
const aliasIndex = buildModelAliasIndex({
|
|
cfg: params.cfg,
|
|
defaultProvider: params.defaultProvider,
|
|
});
|
|
if (!trimmed.includes("/")) {
|
|
const aliasKey = normalizeAliasKey(trimmed);
|
|
const aliasMatch = aliasIndex.byAlias.get(aliasKey);
|
|
if (aliasMatch) {
|
|
return aliasMatch.ref;
|
|
}
|
|
|
|
// Default to anthropic if no provider is specified, but warn as this is deprecated.
|
|
console.warn(
|
|
`[openclaw] Model "${trimmed}" specified without provider. Falling back to "anthropic/${trimmed}". Please use "anthropic/${trimmed}" in your config.`,
|
|
);
|
|
return { provider: "anthropic", model: trimmed };
|
|
}
|
|
|
|
const resolved = resolveModelRefFromString({
|
|
raw: trimmed,
|
|
defaultProvider: params.defaultProvider,
|
|
aliasIndex,
|
|
});
|
|
if (resolved) {
|
|
return resolved.ref;
|
|
}
|
|
}
|
|
return { provider: params.defaultProvider, model: params.defaultModel };
|
|
}
|
|
|
|
export function resolveDefaultModelForAgent(params: {
|
|
cfg: OpenClawConfig;
|
|
agentId?: string;
|
|
}): ModelRef {
|
|
const agentModelOverride = params.agentId
|
|
? resolveAgentModelPrimary(params.cfg, params.agentId)
|
|
: undefined;
|
|
const cfg =
|
|
agentModelOverride && agentModelOverride.length > 0
|
|
? {
|
|
...params.cfg,
|
|
agents: {
|
|
...params.cfg.agents,
|
|
defaults: {
|
|
...params.cfg.agents?.defaults,
|
|
model: {
|
|
...(typeof params.cfg.agents?.defaults?.model === "object"
|
|
? params.cfg.agents.defaults.model
|
|
: undefined),
|
|
primary: agentModelOverride,
|
|
},
|
|
},
|
|
},
|
|
}
|
|
: params.cfg;
|
|
return resolveConfiguredModelRef({
|
|
cfg,
|
|
defaultProvider: DEFAULT_PROVIDER,
|
|
defaultModel: DEFAULT_MODEL,
|
|
});
|
|
}
|
|
|
|
export function buildAllowedModelSet(params: {
|
|
cfg: OpenClawConfig;
|
|
catalog: ModelCatalogEntry[];
|
|
defaultProvider: string;
|
|
defaultModel?: string;
|
|
}): {
|
|
allowAny: boolean;
|
|
allowedCatalog: ModelCatalogEntry[];
|
|
allowedKeys: Set<string>;
|
|
} {
|
|
const rawAllowlist = (() => {
|
|
const modelMap = params.cfg.agents?.defaults?.models ?? {};
|
|
return Object.keys(modelMap);
|
|
})();
|
|
const allowAny = rawAllowlist.length === 0;
|
|
const defaultModel = params.defaultModel?.trim();
|
|
const defaultKey =
|
|
defaultModel && params.defaultProvider
|
|
? modelKey(params.defaultProvider, defaultModel)
|
|
: undefined;
|
|
const catalogKeys = new Set(params.catalog.map((entry) => modelKey(entry.provider, entry.id)));
|
|
|
|
if (allowAny) {
|
|
if (defaultKey) {
|
|
catalogKeys.add(defaultKey);
|
|
}
|
|
return {
|
|
allowAny: true,
|
|
allowedCatalog: params.catalog,
|
|
allowedKeys: catalogKeys,
|
|
};
|
|
}
|
|
|
|
const allowedKeys = new Set<string>();
|
|
const configuredProviders = (params.cfg.models?.providers ?? {}) as Record<string, unknown>;
|
|
for (const raw of rawAllowlist) {
|
|
const parsed = parseModelRef(String(raw), params.defaultProvider);
|
|
if (!parsed) {
|
|
continue;
|
|
}
|
|
const key = modelKey(parsed.provider, parsed.model);
|
|
const providerKey = normalizeProviderId(parsed.provider);
|
|
if (isCliProvider(parsed.provider, params.cfg)) {
|
|
allowedKeys.add(key);
|
|
} else if (catalogKeys.has(key)) {
|
|
allowedKeys.add(key);
|
|
} else if (configuredProviders[providerKey] != null) {
|
|
// Explicitly configured providers should be allowlist-able even when
|
|
// they don't exist in the curated model catalog.
|
|
allowedKeys.add(key);
|
|
}
|
|
}
|
|
|
|
if (defaultKey) {
|
|
allowedKeys.add(defaultKey);
|
|
}
|
|
|
|
const allowedCatalog = params.catalog.filter((entry) =>
|
|
allowedKeys.has(modelKey(entry.provider, entry.id)),
|
|
);
|
|
|
|
if (allowedCatalog.length === 0 && allowedKeys.size === 0) {
|
|
if (defaultKey) {
|
|
catalogKeys.add(defaultKey);
|
|
}
|
|
return {
|
|
allowAny: true,
|
|
allowedCatalog: params.catalog,
|
|
allowedKeys: catalogKeys,
|
|
};
|
|
}
|
|
|
|
return { allowAny: false, allowedCatalog, allowedKeys };
|
|
}
|
|
|
|
export type ModelRefStatus = {
|
|
key: string;
|
|
inCatalog: boolean;
|
|
allowAny: boolean;
|
|
allowed: boolean;
|
|
};
|
|
|
|
export function getModelRefStatus(params: {
|
|
cfg: OpenClawConfig;
|
|
catalog: ModelCatalogEntry[];
|
|
ref: ModelRef;
|
|
defaultProvider: string;
|
|
defaultModel?: string;
|
|
}): ModelRefStatus {
|
|
const allowed = buildAllowedModelSet({
|
|
cfg: params.cfg,
|
|
catalog: params.catalog,
|
|
defaultProvider: params.defaultProvider,
|
|
defaultModel: params.defaultModel,
|
|
});
|
|
const key = modelKey(params.ref.provider, params.ref.model);
|
|
return {
|
|
key,
|
|
inCatalog: params.catalog.some((entry) => modelKey(entry.provider, entry.id) === key),
|
|
allowAny: allowed.allowAny,
|
|
allowed: allowed.allowAny || allowed.allowedKeys.has(key),
|
|
};
|
|
}
|
|
|
|
export function resolveAllowedModelRef(params: {
|
|
cfg: OpenClawConfig;
|
|
catalog: ModelCatalogEntry[];
|
|
raw: string;
|
|
defaultProvider: string;
|
|
defaultModel?: string;
|
|
}):
|
|
| { ref: ModelRef; key: string }
|
|
| {
|
|
error: string;
|
|
} {
|
|
const trimmed = params.raw.trim();
|
|
if (!trimmed) {
|
|
return { error: "invalid model: empty" };
|
|
}
|
|
|
|
const aliasIndex = buildModelAliasIndex({
|
|
cfg: params.cfg,
|
|
defaultProvider: params.defaultProvider,
|
|
});
|
|
const resolved = resolveModelRefFromString({
|
|
raw: trimmed,
|
|
defaultProvider: params.defaultProvider,
|
|
aliasIndex,
|
|
});
|
|
if (!resolved) {
|
|
return { error: `invalid model: ${trimmed}` };
|
|
}
|
|
|
|
const status = getModelRefStatus({
|
|
cfg: params.cfg,
|
|
catalog: params.catalog,
|
|
ref: resolved.ref,
|
|
defaultProvider: params.defaultProvider,
|
|
defaultModel: params.defaultModel,
|
|
});
|
|
if (!status.allowed) {
|
|
return { error: `model not allowed: ${status.key}` };
|
|
}
|
|
|
|
return { ref: resolved.ref, key: status.key };
|
|
}
|
|
|
|
export function resolveThinkingDefault(params: {
|
|
cfg: OpenClawConfig;
|
|
provider: string;
|
|
model: string;
|
|
catalog?: ModelCatalogEntry[];
|
|
}): ThinkLevel {
|
|
const configured = params.cfg.agents?.defaults?.thinkingDefault;
|
|
if (configured) {
|
|
return configured;
|
|
}
|
|
const candidate = params.catalog?.find(
|
|
(entry) => entry.provider === params.provider && entry.id === params.model,
|
|
);
|
|
if (candidate?.reasoning) {
|
|
return "low";
|
|
}
|
|
return "off";
|
|
}
|
|
|
|
/**
|
|
* Resolve the model configured for Gmail hook processing.
|
|
* Returns null if hooks.gmail.model is not set.
|
|
*/
|
|
export function resolveHooksGmailModel(params: {
|
|
cfg: OpenClawConfig;
|
|
defaultProvider: string;
|
|
}): ModelRef | null {
|
|
const hooksModel = params.cfg.hooks?.gmail?.model;
|
|
if (!hooksModel?.trim()) {
|
|
return null;
|
|
}
|
|
|
|
const aliasIndex = buildModelAliasIndex({
|
|
cfg: params.cfg,
|
|
defaultProvider: params.defaultProvider,
|
|
});
|
|
|
|
const resolved = resolveModelRefFromString({
|
|
raw: hooksModel,
|
|
defaultProvider: params.defaultProvider,
|
|
aliasIndex,
|
|
});
|
|
|
|
return resolved?.ref ?? null;
|
|
}
|