* feat(ui): add chat infrastructure modules (slice 1 of dashboard-v2) New self-contained chat modules extracted from dashboard-v2-structure: - chat/slash-commands.ts: slash command definitions and completions - chat/slash-command-executor.ts: execute slash commands via gateway RPC - chat/slash-command-executor.node.test.ts: test coverage - chat/speech.ts: speech-to-text (STT) support - chat/input-history.ts: per-session input history navigation - chat/pinned-messages.ts: pinned message management - chat/deleted-messages.ts: deleted message tracking - chat/export.ts: shared exportChatMarkdown helper - chat-export.ts: re-export shim for backwards compat Gateway fix: - Restore usage/cost stripping in chat.history sanitization - Add test coverage for sanitization behavior These modules are additive and tree-shaken — no existing code imports them yet. They will be wired in subsequent slices. * feat(ui): add utilities, theming, and i18n updates (slice 2 of dashboard-v2) UI utilities and theming improvements extracted from dashboard-v2-structure: Icons & formatting: - icons.ts: expanded icon set for new dashboard views - format.ts: date/number formatting helpers - tool-labels.ts: human-readable tool name mappings Theming: - theme.ts: enhanced theme resolution and system theme support - theme-transition.ts: simplified transition logic - storage.ts: theme parsing improvements for settings persistence Navigation & types: - navigation.ts: extended tab definitions for dashboard-v2 - app-view-state.ts: expanded view state management - types.ts: new type definitions (HealthSummary, ModelCatalogEntry, etc.) Components: - components/dashboard-header.ts: reusable header component i18n: - Updated en, pt-BR, zh-CN, zh-TW locales with new dashboard strings All changes are additive or backwards-compatible. Build passes. Part of #36853. * feat(ui): dashboard-v2 views refactor (slice 3 of dashboard-v2) Complete views refactor from dashboard-v2-structure, building on slice 1 (chat infra, #41497) and slice 2 (utilities/theming, #41500). Core app wiring: - app.ts: updated host component with new state properties - app-render.ts: refactored render pipeline for new dashboard layout - app-render.helpers.ts: extracted render helpers - app-settings.ts: theme listener lifecycle fix, cron runs on tab load - app-gateway.ts: refactored chat event handling - app-chat.ts: slash command integration New views: - views/command-palette.ts: command palette (Cmd+K) - views/login-gate.ts: authentication gate - views/bottom-tabs.ts: mobile tab navigation - views/overview-*.ts: modular overview dashboard (cards, attention, event log, hints, log tail, quick actions) - views/agents-panels-overview.ts: agent overview panel Refactored views: - views/chat.ts: major refactor with STT, slash commands, search, export, pinned messages, input history - views/config.ts: restructured config management - views/agents.ts: streamlined agent management - views/overview.ts: modular composition from sub-views - views/sessions.ts: enhanced session management Controllers: - controllers/health.ts: new health check controller - controllers/models.ts: new model catalog controller - controllers/agents.ts: tools catalog improvements - controllers/config.ts: config form enhancements Tests & infrastructure: - Updated test helpers, browser tests, node tests - vite.config.ts: build configuration updates - markdown.ts: rendering improvements Build passes ✅ | 44 files | +6,626/-1,499 Part of #36853. Depends on #41497 and #41500. * UI: fix chat review follow-ups * fix(ui): repair chat clear and attachment regressions * fix(ui): address remaining chat review comments * fix(ui): address review follow-ups * fix(ui): replay queued local slash commands * fix(ui): repair control-ui type drift * fix(ui): restore control UI styling * feat(ui): enhance layout and styling for config and topbar components - Updated grid layout for the config layout to allow full-width usage. - Introduced new styles for top tabs and search components to improve usability. - Added theme mode toggle styling for better visual integration. - Implemented tests for layout and theme mode components to ensure proper rendering and functionality. * feat(ui): add config file opening functionality and enhance styles - Implemented a new handler to open the configuration file using the default application based on the operating system. - Updated various CSS styles across components for improved visual consistency and usability, including adjustments to padding, margins, and font sizes. - Introduced new styles for the data table and sidebar components to enhance layout and interaction. - Added tests for the collapsed navigation rail to ensure proper functionality in different states. * refactor(ui): update CSS styles for improved layout and consistency - Simplified font-body declaration in base.css for cleaner code. - Adjusted transition properties in components.css for better readability. - Added new .workspace-link class in components.css for enhanced link styling. - Changed config layout from grid to flex in config.css for better responsiveness. - Updated related tests to reflect layout changes in config-layout.browser.test.ts. * feat(ui): enhance theme handling and loading states in chat interface - Updated CSS to support new theme mode attributes for better styling consistency across light and dark themes. - Introduced loading skeletons in the chat view to improve user experience during data fetching. - Refactored command palette to manage focus more effectively, enhancing accessibility. - Added tests for the appearance theme picker and loading states to ensure proper rendering and functionality. * refactor(ui): streamline ephemeral state management in chat and config views - Introduced interfaces for ephemeral state in chat and config views to encapsulate related variables. - Refactored state management to utilize a single object for better organization and maintainability. - Removed legacy state variables and updated related functions to reference the new state structure. - Enhanced readability and consistency across the codebase by standardizing state handling. * chore: remove test files to reduce PR scope * fix(ui): resolve type errors in debug props and chat search * refactor(ui): remove stream mode functionality across various components - Eliminated stream mode related translations and CSS styles to streamline the user interface. - Updated multiple components to remove references to stream mode, enhancing code clarity and maintainability. - Adjusted rendering logic in views to ensure consistent behavior without stream mode. - Improved overall readability by cleaning up unused variables and props. * fix(ui): add msg-meta CSS and fix rebase type errors * fix(ui): add CSS for chat footer action buttons (TTS, delete) and msg-meta * feat(ui): add delete confirmation with remember-decision checkbox * fix(ui): delete confirmation with remember, attention icon sizing * fix(ui): open delete confirm popover to the left (not clipped) * fix(ui): show all nav items in collapsed sidebar, remove gap * fix(ui): address P1/P2 review feedback — session queue clear, kill scope, palette guard, stop button * fix(ui): address Greptile re-review — kill scope, queue flush, idle handling, parallel fetch - SECURITY: /kill <target> now enforces session tree scope (not just /kill all) - /kill reports idle sessions gracefully instead of throwing - Queue continues draining after local slash commands - /model fetches sessions.list + models.list in parallel (perf fix) * fix(ui): style update banner close button — SVG stroke + sizing * fix(ui): update layout styles for sidebar and content spacing * UI: restore colon slash command parsing * UI: restore slash command session queries * Refactor thinking resolution: Introduce resolveThinkingDefaultForModel function and update model-selection to utilize it. Add tests for new functionality in thinking.test.ts. * fix(ui): constrain welcome state logo size, add missing CSS for new session view --------- Co-authored-by: Vincent Koc <vincentkoc@ieee.org>
723 lines
22 KiB
TypeScript
723 lines
22 KiB
TypeScript
import { resolveThinkingDefaultForModel } from "../auto-reply/thinking.js";
|
|
import type { OpenClawConfig } from "../config/config.js";
|
|
import {
|
|
resolveAgentModelFallbackValues,
|
|
resolveAgentModelPrimaryValue,
|
|
toAgentModelListLike,
|
|
} from "../config/model-input.js";
|
|
import { createSubsystemLogger } from "../logging/subsystem.js";
|
|
import { sanitizeForLog } from "../terminal/ansi.js";
|
|
import {
|
|
resolveAgentConfig,
|
|
resolveAgentEffectiveModelPrimary,
|
|
resolveAgentModelFallbacksOverride,
|
|
} from "./agent-scope.js";
|
|
import { DEFAULT_MODEL, DEFAULT_PROVIDER } from "./defaults.js";
|
|
import type { ModelCatalogEntry } from "./model-catalog.js";
|
|
import { splitTrailingAuthProfile } from "./model-ref-profile.js";
|
|
import { normalizeGoogleModelId } from "./models-config.providers.js";
|
|
|
|
const log = createSubsystemLogger("model-selection");
|
|
|
|
export type ModelRef = {
|
|
provider: string;
|
|
model: string;
|
|
};
|
|
|
|
export type ThinkLevel = "off" | "minimal" | "low" | "medium" | "high" | "xhigh" | "adaptive";
|
|
|
|
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",
|
|
};
|
|
|
|
function normalizeAliasKey(value: string): string {
|
|
return value.trim().toLowerCase();
|
|
}
|
|
|
|
export function modelKey(provider: string, model: string) {
|
|
const providerId = provider.trim();
|
|
const modelId = model.trim();
|
|
if (!providerId) {
|
|
return modelId;
|
|
}
|
|
if (!modelId) {
|
|
return providerId;
|
|
}
|
|
return modelId.toLowerCase().startsWith(`${providerId.toLowerCase()}/`)
|
|
? modelId
|
|
: `${providerId}/${modelId}`;
|
|
}
|
|
|
|
export function legacyModelKey(provider: string, model: string): string | null {
|
|
const providerId = provider.trim();
|
|
const modelId = model.trim();
|
|
if (!providerId || !modelId) {
|
|
return null;
|
|
}
|
|
const rawKey = `${providerId}/${modelId}`;
|
|
const canonicalKey = modelKey(providerId, modelId);
|
|
return rawKey === canonicalKey ? null : rawKey;
|
|
}
|
|
|
|
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 === "opencode-go-auth") {
|
|
return "opencode-go";
|
|
}
|
|
if (normalized === "qwen") {
|
|
return "qwen-portal";
|
|
}
|
|
if (normalized === "kimi-code") {
|
|
return "kimi-coding";
|
|
}
|
|
if (normalized === "bedrock" || normalized === "aws-bedrock") {
|
|
return "amazon-bedrock";
|
|
}
|
|
// Backward compatibility for older provider naming.
|
|
if (normalized === "bytedance" || normalized === "doubao") {
|
|
return "volcengine";
|
|
}
|
|
return normalized;
|
|
}
|
|
|
|
/** Normalize provider ID for auth lookup. Coding-plan variants share auth with base. */
|
|
export function normalizeProviderIdForAuth(provider: string): string {
|
|
const normalized = normalizeProviderId(provider);
|
|
if (normalized === "volcengine-plan") {
|
|
return "volcengine";
|
|
}
|
|
if (normalized === "byteplus-plan") {
|
|
return "byteplus";
|
|
}
|
|
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 === "vercel-ai-gateway" && !model.includes("/")) {
|
|
// Allow Vercel-specific Claude refs without an upstream prefix.
|
|
const normalizedAnthropicModel = normalizeAnthropicModelId(model);
|
|
if (normalizedAnthropicModel.startsWith("claude-")) {
|
|
return `anthropic/${normalizedAnthropicModel}`;
|
|
}
|
|
}
|
|
if (provider === "google") {
|
|
return normalizeGoogleModelId(model);
|
|
}
|
|
// OpenRouter-native models (e.g. "openrouter/aurora-alpha") need the full
|
|
// "openrouter/<name>" as the model ID sent to the API. Models from external
|
|
// providers already contain a slash (e.g. "anthropic/claude-sonnet-4-5") and
|
|
// are passed through as-is (#12924).
|
|
if (provider === "openrouter" && !model.includes("/")) {
|
|
return `openrouter/${model}`;
|
|
}
|
|
return model;
|
|
}
|
|
|
|
export function normalizeModelRef(provider: string, model: string): ModelRef {
|
|
const normalizedProvider = normalizeProviderId(provider);
|
|
const normalizedModel = normalizeProviderModelId(normalizedProvider, model.trim());
|
|
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 inferUniqueProviderFromConfiguredModels(params: {
|
|
cfg: OpenClawConfig;
|
|
model: string;
|
|
}): string | undefined {
|
|
const model = params.model.trim();
|
|
if (!model) {
|
|
return undefined;
|
|
}
|
|
const configuredModels = params.cfg.agents?.defaults?.models;
|
|
if (!configuredModels) {
|
|
return undefined;
|
|
}
|
|
const normalized = model.toLowerCase();
|
|
const providers = new Set<string>();
|
|
for (const key of Object.keys(configuredModels)) {
|
|
const ref = key.trim();
|
|
if (!ref || !ref.includes("/")) {
|
|
continue;
|
|
}
|
|
const parsed = parseModelRef(ref, DEFAULT_PROVIDER);
|
|
if (!parsed) {
|
|
continue;
|
|
}
|
|
if (parsed.model === model || parsed.model.toLowerCase() === normalized) {
|
|
providers.add(parsed.provider);
|
|
if (providers.size > 1) {
|
|
return undefined;
|
|
}
|
|
}
|
|
}
|
|
if (providers.size !== 1) {
|
|
return undefined;
|
|
}
|
|
return providers.values().next().value;
|
|
}
|
|
|
|
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 { model } = splitTrailingAuthProfile(params.raw);
|
|
if (!model) {
|
|
return null;
|
|
}
|
|
if (!model.includes("/")) {
|
|
const aliasKey = normalizeAliasKey(model);
|
|
const aliasMatch = params.aliasIndex?.byAlias.get(aliasKey);
|
|
if (aliasMatch) {
|
|
return { ref: aliasMatch.ref, alias: aliasMatch.alias };
|
|
}
|
|
}
|
|
const parsed = parseModelRef(model, params.defaultProvider);
|
|
if (!parsed) {
|
|
return null;
|
|
}
|
|
return { ref: parsed };
|
|
}
|
|
|
|
export function resolveConfiguredModelRef(params: {
|
|
cfg: OpenClawConfig;
|
|
defaultProvider: string;
|
|
defaultModel: string;
|
|
}): ModelRef {
|
|
const rawModel = resolveAgentModelPrimaryValue(params.cfg.agents?.defaults?.model) ?? "";
|
|
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.
|
|
const safeTrimmed = sanitizeForLog(trimmed);
|
|
log.warn(
|
|
`Model "${safeTrimmed}" specified without provider. Falling back to "anthropic/${safeTrimmed}". Please use "anthropic/${safeTrimmed}" in your config.`,
|
|
);
|
|
return { provider: "anthropic", model: trimmed };
|
|
}
|
|
|
|
const resolved = resolveModelRefFromString({
|
|
raw: trimmed,
|
|
defaultProvider: params.defaultProvider,
|
|
aliasIndex,
|
|
});
|
|
if (resolved) {
|
|
return resolved.ref;
|
|
}
|
|
|
|
// User specified a model but it could not be resolved — warn before falling back.
|
|
const safe = sanitizeForLog(trimmed);
|
|
const safeFallback = sanitizeForLog(`${params.defaultProvider}/${params.defaultModel}`);
|
|
log.warn(`Model "${safe}" could not be resolved. Falling back to default "${safeFallback}".`);
|
|
}
|
|
// Before falling back to the hardcoded default, check if the default provider
|
|
// is actually available. If it isn't but other providers are configured, prefer
|
|
// the first configured provider's first model to avoid reporting a stale default
|
|
// from a removed provider. (See #38880)
|
|
const configuredProviders = params.cfg.models?.providers;
|
|
if (configuredProviders && typeof configuredProviders === "object") {
|
|
const hasDefaultProvider = Boolean(configuredProviders[params.defaultProvider]);
|
|
if (!hasDefaultProvider) {
|
|
const availableProvider = Object.entries(configuredProviders).find(
|
|
([, providerCfg]) =>
|
|
providerCfg &&
|
|
Array.isArray(providerCfg.models) &&
|
|
providerCfg.models.length > 0 &&
|
|
providerCfg.models[0]?.id,
|
|
);
|
|
if (availableProvider) {
|
|
const [providerName, providerCfg] = availableProvider;
|
|
const firstModel = providerCfg.models[0];
|
|
return { provider: providerName, model: firstModel.id };
|
|
}
|
|
}
|
|
}
|
|
return { provider: params.defaultProvider, model: params.defaultModel };
|
|
}
|
|
|
|
export function resolveDefaultModelForAgent(params: {
|
|
cfg: OpenClawConfig;
|
|
agentId?: string;
|
|
}): ModelRef {
|
|
const agentModelOverride = params.agentId
|
|
? resolveAgentEffectiveModelPrimary(params.cfg, params.agentId)
|
|
: undefined;
|
|
const cfg =
|
|
agentModelOverride && agentModelOverride.length > 0
|
|
? {
|
|
...params.cfg,
|
|
agents: {
|
|
...params.cfg.agents,
|
|
defaults: {
|
|
...params.cfg.agents?.defaults,
|
|
model: {
|
|
...toAgentModelListLike(params.cfg.agents?.defaults?.model),
|
|
primary: agentModelOverride,
|
|
},
|
|
},
|
|
},
|
|
}
|
|
: params.cfg;
|
|
return resolveConfiguredModelRef({
|
|
cfg,
|
|
defaultProvider: DEFAULT_PROVIDER,
|
|
defaultModel: DEFAULT_MODEL,
|
|
});
|
|
}
|
|
|
|
function resolveAllowedFallbacks(params: { cfg: OpenClawConfig; agentId?: string }): string[] {
|
|
if (params.agentId) {
|
|
const override = resolveAgentModelFallbacksOverride(params.cfg, params.agentId);
|
|
if (override !== undefined) {
|
|
return override;
|
|
}
|
|
}
|
|
return resolveAgentModelFallbackValues(params.cfg.agents?.defaults?.model);
|
|
}
|
|
|
|
export function resolveSubagentConfiguredModelSelection(params: {
|
|
cfg: OpenClawConfig;
|
|
agentId: string;
|
|
}): string | undefined {
|
|
const agentConfig = resolveAgentConfig(params.cfg, params.agentId);
|
|
return (
|
|
normalizeModelSelection(agentConfig?.subagents?.model) ??
|
|
normalizeModelSelection(params.cfg.agents?.defaults?.subagents?.model) ??
|
|
normalizeModelSelection(agentConfig?.model)
|
|
);
|
|
}
|
|
|
|
export function resolveSubagentSpawnModelSelection(params: {
|
|
cfg: OpenClawConfig;
|
|
agentId: string;
|
|
modelOverride?: unknown;
|
|
}): string {
|
|
const runtimeDefault = resolveDefaultModelForAgent({
|
|
cfg: params.cfg,
|
|
agentId: params.agentId,
|
|
});
|
|
return (
|
|
normalizeModelSelection(params.modelOverride) ??
|
|
resolveSubagentConfiguredModelSelection({
|
|
cfg: params.cfg,
|
|
agentId: params.agentId,
|
|
}) ??
|
|
normalizeModelSelection(resolveAgentModelPrimaryValue(params.cfg.agents?.defaults?.model)) ??
|
|
`${runtimeDefault.provider}/${runtimeDefault.model}`
|
|
);
|
|
}
|
|
|
|
export function buildAllowedModelSet(params: {
|
|
cfg: OpenClawConfig;
|
|
catalog: ModelCatalogEntry[];
|
|
defaultProvider: string;
|
|
defaultModel?: string;
|
|
agentId?: 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 defaultRef =
|
|
defaultModel && params.defaultProvider
|
|
? parseModelRef(defaultModel, params.defaultProvider)
|
|
: null;
|
|
const defaultKey = defaultRef ? modelKey(defaultRef.provider, defaultRef.model) : 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 syntheticCatalogEntries = new Map<string, ModelCatalogEntry>();
|
|
for (const raw of rawAllowlist) {
|
|
const parsed = parseModelRef(String(raw), params.defaultProvider);
|
|
if (!parsed) {
|
|
continue;
|
|
}
|
|
const key = modelKey(parsed.provider, parsed.model);
|
|
// Explicit allowlist entries are always trusted, even when bundled catalog
|
|
// data is stale and does not include the configured model yet.
|
|
allowedKeys.add(key);
|
|
|
|
if (!catalogKeys.has(key) && !syntheticCatalogEntries.has(key)) {
|
|
syntheticCatalogEntries.set(key, {
|
|
id: parsed.model,
|
|
name: parsed.model,
|
|
provider: parsed.provider,
|
|
});
|
|
}
|
|
}
|
|
|
|
for (const fallback of resolveAllowedFallbacks({
|
|
cfg: params.cfg,
|
|
agentId: params.agentId,
|
|
})) {
|
|
const parsed = parseModelRef(String(fallback), params.defaultProvider);
|
|
if (parsed) {
|
|
const key = modelKey(parsed.provider, parsed.model);
|
|
allowedKeys.add(key);
|
|
|
|
if (!catalogKeys.has(key) && !syntheticCatalogEntries.has(key)) {
|
|
syntheticCatalogEntries.set(key, {
|
|
id: parsed.model,
|
|
name: parsed.model,
|
|
provider: parsed.provider,
|
|
});
|
|
}
|
|
}
|
|
}
|
|
|
|
if (defaultKey) {
|
|
allowedKeys.add(defaultKey);
|
|
}
|
|
|
|
const allowedCatalog = [
|
|
...params.catalog.filter((entry) => allowedKeys.has(modelKey(entry.provider, entry.id))),
|
|
...syntheticCatalogEntries.values(),
|
|
];
|
|
|
|
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 _normalizedProvider = normalizeProviderId(params.provider);
|
|
const _modelLower = params.model.toLowerCase();
|
|
const configuredModels = params.cfg.agents?.defaults?.models;
|
|
const canonicalKey = modelKey(params.provider, params.model);
|
|
const legacyKey = legacyModelKey(params.provider, params.model);
|
|
const perModelThinking =
|
|
configuredModels?.[canonicalKey]?.params?.thinking ??
|
|
(legacyKey ? configuredModels?.[legacyKey]?.params?.thinking : undefined);
|
|
if (
|
|
perModelThinking === "off" ||
|
|
perModelThinking === "minimal" ||
|
|
perModelThinking === "low" ||
|
|
perModelThinking === "medium" ||
|
|
perModelThinking === "high" ||
|
|
perModelThinking === "xhigh" ||
|
|
perModelThinking === "adaptive"
|
|
) {
|
|
return perModelThinking;
|
|
}
|
|
const configured = params.cfg.agents?.defaults?.thinkingDefault;
|
|
if (configured) {
|
|
return configured;
|
|
}
|
|
return resolveThinkingDefaultForModel({
|
|
provider: params.provider,
|
|
model: params.model,
|
|
catalog: params.catalog,
|
|
});
|
|
}
|
|
|
|
/** Default reasoning level when session/directive do not set it: "on" if model supports reasoning, else "off". */
|
|
export function resolveReasoningDefault(params: {
|
|
provider: string;
|
|
model: string;
|
|
catalog?: ModelCatalogEntry[];
|
|
}): "on" | "off" {
|
|
const key = modelKey(params.provider, params.model);
|
|
const candidate = params.catalog?.find(
|
|
(entry) =>
|
|
(entry.provider === params.provider && entry.id === params.model) ||
|
|
(entry.provider === key && entry.id === params.model),
|
|
);
|
|
return candidate?.reasoning === true ? "on" : "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;
|
|
}
|
|
|
|
/**
|
|
* Normalize a model selection value (string or `{primary?: string}`) to a
|
|
* plain trimmed string. Returns `undefined` when the input is empty/missing.
|
|
* Shared by sessions-spawn and cron isolated-agent model resolution.
|
|
*/
|
|
export function normalizeModelSelection(value: unknown): string | undefined {
|
|
if (typeof value === "string") {
|
|
const trimmed = value.trim();
|
|
return trimmed || undefined;
|
|
}
|
|
if (!value || typeof value !== "object") {
|
|
return undefined;
|
|
}
|
|
const primary = (value as { primary?: unknown }).primary;
|
|
if (typeof primary === "string" && primary.trim()) {
|
|
return primary.trim();
|
|
}
|
|
return undefined;
|
|
}
|