mirror of
https://github.com/openclaw/openclaw.git
synced 2026-06-02 13:35:14 +00:00
* refactor: extract agent core package Introduce packages/agent-core as the OpenClaw-owned home for reusable agent loop, harness, session, prompt, and runtime dependency contracts. * refactor: extract shared llm runtime Move provider model registries, stream wrappers, OAuth helpers, and LLM utilities into src/llm with plugin-sdk barrels instead of depending on the old embedded runtime layout. * refactor: remove pi runtime internals Rename remaining Pi-shaped agent surfaces to OpenClaw agent runtime names, delete obsolete Pi docs and package graph checks, and add the third-party notice for incorporated code. * refactor: tighten agent session runtime Make agent-core/runtime dependencies explicit, consolidate compaction and session transcript helpers, and move model/session helpers behind OpenClaw-owned contracts. * refactor: remove static model and pi auth paths Drop static model catalogs and Pi auth bridges, move model/provider facts to manifest-owned runtime contracts, and harden internal embedded-agent utilities. * refactor: remove legacy provider compat paths * docs: remove agent parity notes * fix: skip provider wildcard metadata parsing * refactor: share session extension sdk loading * refactor: inline acpx proxy error formatter * refactor: fold edit recovery into edit tool * fix: accept extension batch separator * test: align startup provider plugin expectations * fix: restore provider-scoped release discovery * test: align static asset packaging expectations * fix: run static provider catalogs during scoped discovery * fix: add provider entry catalogs for scoped live discovery * fix: load lightweight provider catalog entries * fix: refresh provider-scoped plugin metadata * fix: keep provider catalog entries on release live path * fix: keep static manifest models in release live checks * fix: harden release model discovery * fix: reduce OpenAI live cache probe reasoning * fix: disable OpenAI cache probe reasoning * ci: extend OpenAI gateway live timeout * fix: extend live gateway model budget * fix: stabilize release validation regressions * fix: honor provider aliases in model rows * fix: stabilize release validation lanes * fix: stabilize release memory qa * ci: stabilize release validation lanes * ci: prefer ipv4 for live docker node calls * fix: restore shared tool-call stream wrapper * ci: remove legacy pi test shard alias * fix: clean up embedded agent test drift * fix: stabilize runtime alias status * fix: clean up embedded agent ci drift * fix: restore release ci invariants * fix: clean up post-rebase runtime drift * fix: restore release ci checks * fix: restore release ci after rebase * fix: remove stale pi runtime path * test: align compaction runtime expectations * test: update plugin prerelease expectations * fix: handle claude live tool approvals * fix: stabilize release validation gates * fix: finish agent runtime import * test: finish post-rebase agent runtime mocks * fix: keep codex compaction native * fix: stabilize codex app-server hook tests * test: isolate codex diagnostic active run * test: remove codex diagnostic completion race # Conflicts: # extensions/codex/src/app-server/run-attempt.test.ts * ci: fix full release manifest performance run id * refactor: narrow llm plugin sdk boundary * chore: drop generated google boundary stamps * fix: repair rebase fallout * fix: clean up rebased runtime references * fix: decode codex jwt payloads as base64url * fix: preserve shipped pi runtime alias * fix: add scoped sdk virtual modules * fix: decode llm codex oauth jwt as base64url * fix: avoid stale vertex adc negative cache * fix: harden tool arg decoding and codeql path * fix: keep vertex adc negative checks live * refactor: consolidate codex jwt and edit helpers * fix: await codex oauth node runtime imports * fix: preserve sdk tool and notice contracts * fix: preserve shipped compat config boundaries * fix: align codex oauth callback host * fix: terminate agent-core loop streams on failure * fix: keep codex oauth callback alive during fallback * ci: include session tools in critical codeql scans * fix: keep Cloudflare Anthropic provider auth header * docs: redirect legacy pi runtime pages * fix: honor bundled web provider compat discovery * fix: protect session output spill files * fix: keep legacy agent dir env blocked * fix: contain auto-discovered skill symlinks * fix: harden agent core sdk proxy surfaces * fix: restore approval reaction sdk compat * fix: keep live docker runs bounded * fix: keep codex oauth redirect host aligned * fix: resolve post-rebase agent runtime drift * fix: redact anthropic oauth parse failures * fix: preserve responses strict tool shaping * fix: repair agent runtime rebase cleanup * docs: redirect retired parity pages * fix: bound auto-discovered resources to roots * fix: repair post-rebase agent test drift * fix: preserve bundled provider allowlist migration * fix: preserve manifest-owned provider aliases * fix: declare photon image dependency * fix: keep provider headers out of proxy body * fix: preserve shipped env aliases * fix: refresh control ui i18n generated state * fix: quote read fallback paths * fix: preview edits through configured backend * test: satisfy core test typecheck * fix: preserve ZAI usage auth fallback * test: repair codex diagnostic test * fix: repair agent runtime rebase drift * test: finish embedded runner import rename * fix: repair agent runtime rebase integrations * test: align compaction oauth fallback expectations * fix: allow sdk-auth session models * fix: update doctor tool schema import * fix: preserve bedrock plugin region * fix: stream harmony-like prose immediately * ci: include session runtime in codeql shards * fix: repair latest rebase integrations * fix: honor explicit codex websocket transport * fix: keep openai-compatible credentials provider-scoped * fix: refresh sdk api baseline after rebase * fix: route cli runtime aliases through openclaw harness * test: rename stale harness mock expectation * test: rename embedded agent overflow calls * test: clean embedded auth test wording * test: use openclaw stream types in deepinfra cache test * fix: refresh sdk api baseline on latest main * fix: honor bundled discovery compat allowlists * fix: refresh sdk api baseline after latest rebase * fix: remove stale rebase imports * test: rename stale model catalog mock * test: mock renamed doctor runtime modules * fix: map canonical kimi env auth * fix: use internal model registry in bench script * fix: migrate deepinfra provider catalog entry * fix: enforce builtin tool suppression * fix: route compaction auth and proxy payloads safely * refactor: prune unused llm registry leftovers * test: update codex hooks session import * test: fix model picker ci coverage * test: align model picker auth mock types
604 lines
20 KiB
TypeScript
604 lines
20 KiB
TypeScript
import fs from "node:fs";
|
|
import path from "node:path";
|
|
import { resolveAgentModelFallbackValues } from "../config/model-input.js";
|
|
import { hasSessionAutoModelFallbackProvenance } from "../config/sessions/model-override-provenance.js";
|
|
export { hasSessionAutoModelFallbackProvenance } from "../config/sessions/model-override-provenance.js";
|
|
import type { SessionEntry } from "../config/sessions/types.js";
|
|
import type { AgentDefaultsConfig } from "../config/types.agent-defaults.js";
|
|
import type { AgentModelConfig } from "../config/types.agents-shared.js";
|
|
import type { AgentConfig } from "../config/types.agents.js";
|
|
import type { OpenClawConfig } from "../config/types.js";
|
|
import { isPathInside } from "../infra/path-guards.js";
|
|
import {
|
|
isSubagentSessionKey,
|
|
normalizeAgentId,
|
|
parseAgentSessionKey,
|
|
resolveAgentIdFromSessionKey,
|
|
} from "../routing/session-key.js";
|
|
import {
|
|
lowercasePreservingWhitespace,
|
|
normalizeLowercaseStringOrEmpty,
|
|
normalizeOptionalString,
|
|
resolvePrimaryStringValue,
|
|
} from "../shared/string-coerce.js";
|
|
import { resolveUserPath } from "../utils.js";
|
|
import {
|
|
listAgentIds,
|
|
resolveAgentConfig,
|
|
resolveAgentWorkspaceDir,
|
|
resolveDefaultAgentId,
|
|
} from "./agent-scope-config.js";
|
|
import { resolveEffectiveAgentSkillFilter } from "./skills/agent-filter.js";
|
|
export {
|
|
listAgentEntries,
|
|
listAgentIds,
|
|
resolveAgentConfig,
|
|
resolveAgentContextLimits,
|
|
resolveAgentDir,
|
|
resolveDefaultAgentDir,
|
|
resolveAgentWorkspaceDir,
|
|
resolveDefaultAgentId,
|
|
type ResolvedAgentConfig,
|
|
} from "./agent-scope-config.js";
|
|
|
|
/** Strip null bytes from paths to prevent ENOTDIR errors. */
|
|
function stripNullBytes(s: string): string {
|
|
// eslint-disable-next-line no-control-regex
|
|
return s.replace(/\0/g, "");
|
|
}
|
|
|
|
const AUTO_FALLBACK_PRIMARY_PROBE_INTERVAL_MS = 5 * 60 * 1000;
|
|
const AUTO_FALLBACK_PRIMARY_PROBE_MAX_KEYS = 4096;
|
|
const autoFallbackPrimaryProbeState = new Map<string, number>();
|
|
|
|
function autoFallbackPrimaryProbeStateKey(params: {
|
|
sessionKey?: string | null;
|
|
primaryProvider: string;
|
|
primaryModel: string;
|
|
}): string {
|
|
return [
|
|
normalizeOptionalString(params.sessionKey) ?? "",
|
|
`${params.primaryProvider}/${params.primaryModel}`,
|
|
].join("\0");
|
|
}
|
|
|
|
function pruneAutoFallbackPrimaryProbeState(params: {
|
|
state: Map<string, number>;
|
|
now: number;
|
|
minIntervalMs: number;
|
|
maxKeys?: number;
|
|
}): void {
|
|
const maxKeys = Math.max(1, Math.trunc(params.maxKeys ?? AUTO_FALLBACK_PRIMARY_PROBE_MAX_KEYS));
|
|
const staleBefore = params.now - params.minIntervalMs;
|
|
for (const [key, lastProbeAt] of params.state) {
|
|
if (!Number.isFinite(lastProbeAt) || lastProbeAt < staleBefore) {
|
|
params.state.delete(key);
|
|
}
|
|
}
|
|
if (params.state.size <= maxKeys) {
|
|
return;
|
|
}
|
|
const removeCount = params.state.size - maxKeys;
|
|
let removed = 0;
|
|
for (const key of params.state.keys()) {
|
|
params.state.delete(key);
|
|
removed += 1;
|
|
if (removed >= removeCount) {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
export type AutoFallbackPrimaryProbe = {
|
|
provider: string;
|
|
model: string;
|
|
fallbackProvider: string;
|
|
fallbackModel: string;
|
|
fallbackAuthProfileId?: string;
|
|
fallbackAuthProfileIdSource?: "auto" | "user";
|
|
};
|
|
|
|
export function resolveAutoFallbackPrimaryProbe(params: {
|
|
entry:
|
|
| Pick<
|
|
SessionEntry,
|
|
| "providerOverride"
|
|
| "modelOverride"
|
|
| "modelOverrideSource"
|
|
| "modelOverrideFallbackOriginProvider"
|
|
| "modelOverrideFallbackOriginModel"
|
|
| "authProfileOverride"
|
|
| "authProfileOverrideSource"
|
|
| "authProfileOverrideCompactionCount"
|
|
>
|
|
| null
|
|
| undefined;
|
|
sessionKey?: string | null;
|
|
primaryProvider: string;
|
|
primaryModel: string;
|
|
now?: number;
|
|
minIntervalMs?: number;
|
|
maxTrackedProbeKeys?: number;
|
|
probeState?: Map<string, number>;
|
|
}): AutoFallbackPrimaryProbe | undefined {
|
|
const entry = params.entry;
|
|
if (!entry) {
|
|
return undefined;
|
|
}
|
|
const recoveredAutoFallbackOverride =
|
|
entry.modelOverrideSource === undefined && hasSessionAutoModelFallbackProvenance(entry);
|
|
if (entry.modelOverrideSource !== "auto" && !recoveredAutoFallbackOverride) {
|
|
return undefined;
|
|
}
|
|
|
|
const originProvider = normalizeOptionalString(entry.modelOverrideFallbackOriginProvider);
|
|
const originModel = normalizeOptionalString(entry.modelOverrideFallbackOriginModel);
|
|
const overrideProvider = normalizeOptionalString(entry.providerOverride);
|
|
const overrideModel = normalizeOptionalString(entry.modelOverride);
|
|
const primaryProvider = normalizeOptionalString(params.primaryProvider);
|
|
const primaryModel = normalizeOptionalString(params.primaryModel);
|
|
if (!originProvider || !originModel || !overrideProvider || !overrideModel) {
|
|
return undefined;
|
|
}
|
|
if (!primaryProvider || !primaryModel) {
|
|
return undefined;
|
|
}
|
|
if (originProvider !== primaryProvider || originModel !== primaryModel) {
|
|
return undefined;
|
|
}
|
|
if (overrideProvider === originProvider && overrideModel === originModel) {
|
|
return undefined;
|
|
}
|
|
|
|
const now = params.now ?? Date.now();
|
|
const minIntervalMs = params.minIntervalMs ?? AUTO_FALLBACK_PRIMARY_PROBE_INTERVAL_MS;
|
|
const state = params.probeState ?? autoFallbackPrimaryProbeState;
|
|
pruneAutoFallbackPrimaryProbeState({
|
|
state,
|
|
now,
|
|
minIntervalMs,
|
|
maxKeys: params.maxTrackedProbeKeys,
|
|
});
|
|
const key = autoFallbackPrimaryProbeStateKey({
|
|
sessionKey: params.sessionKey,
|
|
primaryProvider: originProvider,
|
|
primaryModel: originModel,
|
|
});
|
|
const lastProbeAt = state.get(key);
|
|
if (
|
|
typeof lastProbeAt === "number" &&
|
|
Number.isFinite(lastProbeAt) &&
|
|
now - lastProbeAt < minIntervalMs
|
|
) {
|
|
return undefined;
|
|
}
|
|
const fallbackAuthProfileId = normalizeOptionalString(entry.authProfileOverride);
|
|
const fallbackAuthProfileIdSource =
|
|
entry.authProfileOverrideSource ??
|
|
(entry.authProfileOverrideCompactionCount !== undefined ? "auto" : undefined);
|
|
return {
|
|
provider: originProvider,
|
|
model: originModel,
|
|
fallbackProvider: overrideProvider,
|
|
fallbackModel: overrideModel,
|
|
...(fallbackAuthProfileId
|
|
? {
|
|
fallbackAuthProfileId,
|
|
...(fallbackAuthProfileIdSource ? { fallbackAuthProfileIdSource } : {}),
|
|
}
|
|
: {}),
|
|
};
|
|
}
|
|
|
|
export function markAutoFallbackPrimaryProbe(params: {
|
|
probe: AutoFallbackPrimaryProbe;
|
|
sessionKey?: string | null;
|
|
now?: number;
|
|
minIntervalMs?: number;
|
|
maxTrackedProbeKeys?: number;
|
|
probeState?: Map<string, number>;
|
|
}): void {
|
|
const now = params.now ?? Date.now();
|
|
const minIntervalMs = params.minIntervalMs ?? AUTO_FALLBACK_PRIMARY_PROBE_INTERVAL_MS;
|
|
const state = params.probeState ?? autoFallbackPrimaryProbeState;
|
|
pruneAutoFallbackPrimaryProbeState({
|
|
state,
|
|
now,
|
|
minIntervalMs,
|
|
maxKeys: params.maxTrackedProbeKeys,
|
|
});
|
|
const key = autoFallbackPrimaryProbeStateKey({
|
|
sessionKey: params.sessionKey,
|
|
primaryProvider: params.probe.provider,
|
|
primaryModel: params.probe.model,
|
|
});
|
|
state.set(key, now);
|
|
pruneAutoFallbackPrimaryProbeState({
|
|
state,
|
|
now,
|
|
minIntervalMs,
|
|
maxKeys: params.maxTrackedProbeKeys,
|
|
});
|
|
}
|
|
|
|
export function entryMatchesAutoFallbackPrimaryProbe(
|
|
entry:
|
|
| Pick<
|
|
SessionEntry,
|
|
| "providerOverride"
|
|
| "modelOverride"
|
|
| "modelOverrideSource"
|
|
| "modelOverrideFallbackOriginProvider"
|
|
| "modelOverrideFallbackOriginModel"
|
|
>
|
|
| null
|
|
| undefined,
|
|
probe: AutoFallbackPrimaryProbe,
|
|
): boolean {
|
|
if (!entry) {
|
|
return false;
|
|
}
|
|
const recoveredAutoFallbackOverride =
|
|
entry.modelOverrideSource === undefined && hasSessionAutoModelFallbackProvenance(entry);
|
|
if (entry.modelOverrideSource !== "auto" && !recoveredAutoFallbackOverride) {
|
|
return false;
|
|
}
|
|
return (
|
|
normalizeOptionalString(entry.providerOverride) === probe.fallbackProvider &&
|
|
normalizeOptionalString(entry.modelOverride) === probe.fallbackModel &&
|
|
normalizeOptionalString(entry.modelOverrideFallbackOriginProvider) === probe.provider &&
|
|
normalizeOptionalString(entry.modelOverrideFallbackOriginModel) === probe.model
|
|
);
|
|
}
|
|
|
|
export function clearAutoFallbackPrimaryProbeSelection(
|
|
entry: SessionEntry,
|
|
now = Date.now(),
|
|
): void {
|
|
delete entry.providerOverride;
|
|
delete entry.modelOverride;
|
|
delete entry.modelOverrideSource;
|
|
delete entry.modelOverrideFallbackOriginProvider;
|
|
delete entry.modelOverrideFallbackOriginModel;
|
|
if (
|
|
entry.authProfileOverrideSource === "auto" ||
|
|
(entry.authProfileOverrideSource === undefined &&
|
|
entry.authProfileOverrideCompactionCount !== undefined)
|
|
) {
|
|
delete entry.authProfileOverride;
|
|
delete entry.authProfileOverrideSource;
|
|
delete entry.authProfileOverrideCompactionCount;
|
|
}
|
|
delete entry.fallbackNoticeSelectedModel;
|
|
delete entry.fallbackNoticeActiveModel;
|
|
delete entry.fallbackNoticeReason;
|
|
entry.updatedAt = now;
|
|
}
|
|
|
|
export { resolveAgentIdFromSessionKey };
|
|
|
|
export function resolveSessionAgentIds(params: {
|
|
sessionKey?: string;
|
|
config?: OpenClawConfig;
|
|
agentId?: string;
|
|
}): {
|
|
defaultAgentId: string;
|
|
sessionAgentId: string;
|
|
} {
|
|
const defaultAgentId = resolveDefaultAgentId(params.config ?? {});
|
|
const explicitAgentIdRaw = normalizeLowercaseStringOrEmpty(params.agentId);
|
|
const explicitAgentId = explicitAgentIdRaw ? normalizeAgentId(explicitAgentIdRaw) : null;
|
|
const sessionKey = params.sessionKey?.trim();
|
|
const normalizedSessionKey = sessionKey ? normalizeLowercaseStringOrEmpty(sessionKey) : undefined;
|
|
const parsed = normalizedSessionKey ? parseAgentSessionKey(normalizedSessionKey) : null;
|
|
const sessionAgentId =
|
|
explicitAgentId ?? (parsed?.agentId ? normalizeAgentId(parsed.agentId) : defaultAgentId);
|
|
return { defaultAgentId, sessionAgentId };
|
|
}
|
|
|
|
export function resolveSessionAgentId(params: {
|
|
sessionKey?: string;
|
|
config?: OpenClawConfig;
|
|
}): string {
|
|
return resolveSessionAgentIds(params).sessionAgentId;
|
|
}
|
|
|
|
export function resolveAgentExecutionContract(
|
|
cfg: OpenClawConfig | undefined,
|
|
agentId?: string | null,
|
|
): NonNullable<NonNullable<AgentDefaultsConfig["embeddedAgent"]>["executionContract"]> | undefined {
|
|
const defaultContract = cfg?.agents?.defaults?.embeddedAgent?.executionContract;
|
|
if (!cfg || !agentId) {
|
|
return defaultContract;
|
|
}
|
|
const agentConfig = resolveAgentConfig(cfg, agentId);
|
|
const agentContract = agentConfig?.embeddedAgent?.executionContract;
|
|
return agentContract ?? defaultContract;
|
|
}
|
|
|
|
export function resolveAgentSkillsFilter(
|
|
cfg: OpenClawConfig,
|
|
agentId: string,
|
|
): string[] | undefined {
|
|
return resolveEffectiveAgentSkillFilter(cfg, agentId);
|
|
}
|
|
|
|
export function resolveAgentExplicitModelPrimary(
|
|
cfg: OpenClawConfig,
|
|
agentId: string,
|
|
): string | undefined {
|
|
const raw = resolveAgentConfig(cfg, agentId)?.model;
|
|
return resolvePrimaryStringValue(raw);
|
|
}
|
|
|
|
export function resolveAgentEffectiveModelPrimary(
|
|
cfg: OpenClawConfig,
|
|
agentId: string,
|
|
): string | undefined {
|
|
return (
|
|
resolveAgentExplicitModelPrimary(cfg, agentId) ??
|
|
resolvePrimaryStringValue(cfg.agents?.defaults?.model)
|
|
);
|
|
}
|
|
|
|
function findMutableAgentEntry(cfg: OpenClawConfig, agentId: string): AgentConfig | undefined {
|
|
const id = normalizeAgentId(agentId);
|
|
return cfg.agents?.list?.find((entry) => normalizeAgentId(entry?.id) === id);
|
|
}
|
|
|
|
function updateAgentModelPrimary(
|
|
existing: AgentModelConfig | undefined,
|
|
primary: string,
|
|
): AgentModelConfig {
|
|
if (existing && typeof existing === "object" && !Array.isArray(existing)) {
|
|
return { ...existing, primary };
|
|
}
|
|
return primary;
|
|
}
|
|
|
|
export type AgentModelPrimaryWriteTarget = "agent" | "defaults";
|
|
|
|
export function setAgentEffectiveModelPrimary(
|
|
cfg: OpenClawConfig,
|
|
agentId: string,
|
|
primary: string,
|
|
): AgentModelPrimaryWriteTarget {
|
|
const id = normalizeAgentId(agentId);
|
|
if (resolveAgentExplicitModelPrimary(cfg, id)) {
|
|
const entry = findMutableAgentEntry(cfg, id);
|
|
if (entry) {
|
|
entry.model = updateAgentModelPrimary(entry.model, primary);
|
|
return "agent";
|
|
}
|
|
}
|
|
cfg.agents ??= {};
|
|
cfg.agents.defaults ??= {};
|
|
cfg.agents.defaults.model = updateAgentModelPrimary(cfg.agents.defaults.model, primary);
|
|
return "defaults";
|
|
}
|
|
|
|
/** @deprecated Prefer explicit/effective helpers at new call sites. */
|
|
export function resolveAgentModelPrimary(cfg: OpenClawConfig, agentId: string): string | undefined {
|
|
return resolveAgentExplicitModelPrimary(cfg, agentId);
|
|
}
|
|
|
|
export function resolveAgentModelFallbacksOverride(
|
|
cfg: OpenClawConfig,
|
|
agentId: string,
|
|
): string[] | undefined {
|
|
return resolveSelectedModelFallbacksOverride(resolveAgentConfig(cfg, agentId)?.model);
|
|
}
|
|
|
|
function resolveSelectedModelFallbacksOverride(
|
|
raw: AgentModelConfig | undefined,
|
|
): string[] | undefined {
|
|
if (!raw) {
|
|
return undefined;
|
|
}
|
|
if (typeof raw === "string") {
|
|
return resolvePrimaryStringValue(raw) ? [] : undefined;
|
|
}
|
|
// Important: treat an explicitly provided empty array as an override to disable global fallbacks.
|
|
if (!Object.hasOwn(raw, "fallbacks")) {
|
|
return Object.hasOwn(raw, "primary") && resolvePrimaryStringValue(raw) ? [] : undefined;
|
|
}
|
|
return Array.isArray(raw.fallbacks) ? raw.fallbacks : undefined;
|
|
}
|
|
|
|
function resolveFirstModelFallbacksOverride(
|
|
candidates: Array<AgentModelConfig | undefined>,
|
|
): string[] | undefined {
|
|
for (const candidate of candidates) {
|
|
const fallbackOverride = resolveSelectedModelFallbacksOverride(candidate);
|
|
if (fallbackOverride !== undefined) {
|
|
return fallbackOverride;
|
|
}
|
|
}
|
|
return undefined;
|
|
}
|
|
|
|
export type SubagentModelConfigSelectionSource = "subagent" | "agent" | "default-subagent";
|
|
|
|
export type SubagentModelConfigSelectionResult = {
|
|
raw: AgentModelConfig;
|
|
source: SubagentModelConfigSelectionSource;
|
|
};
|
|
|
|
export function resolveSubagentModelConfigSelectionResult(params: {
|
|
cfg: OpenClawConfig;
|
|
agentId?: string;
|
|
agentConfigOverride?: Pick<AgentConfig, "model" | "subagents">;
|
|
}): SubagentModelConfigSelectionResult | undefined {
|
|
const agentConfig =
|
|
params.agentConfigOverride ??
|
|
(params.agentId ? resolveAgentConfig(params.cfg, params.agentId) : undefined);
|
|
const candidates: SubagentModelConfigSelectionResult[] = [
|
|
...(agentConfig?.subagents?.model
|
|
? [{ raw: agentConfig.subagents.model, source: "subagent" as const }]
|
|
: []),
|
|
...(agentConfig?.model ? [{ raw: agentConfig.model, source: "agent" as const }] : []),
|
|
...(params.cfg.agents?.defaults?.subagents?.model
|
|
? [
|
|
{
|
|
raw: params.cfg.agents.defaults.subagents.model,
|
|
source: "default-subagent" as const,
|
|
},
|
|
]
|
|
: []),
|
|
];
|
|
return candidates.find((candidate) => resolvePrimaryStringValue(candidate.raw));
|
|
}
|
|
|
|
export function resolveSubagentModelConfigSelection(params: {
|
|
cfg: OpenClawConfig;
|
|
agentId?: string;
|
|
agentConfigOverride?: Pick<AgentConfig, "model" | "subagents">;
|
|
}): AgentModelConfig | undefined {
|
|
return resolveSubagentModelConfigSelectionResult(params)?.raw;
|
|
}
|
|
|
|
export function resolveSubagentModelFallbacksOverride(
|
|
cfg: OpenClawConfig,
|
|
agentId: string,
|
|
): string[] | undefined {
|
|
const agentConfig = resolveAgentConfig(cfg, agentId);
|
|
const subagentFallbacks = resolveSelectedModelFallbacksOverride(agentConfig?.subagents?.model);
|
|
if (subagentFallbacks !== undefined) {
|
|
return subagentFallbacks;
|
|
}
|
|
const selection = resolveSubagentModelConfigSelectionResult({ cfg, agentId });
|
|
if (selection?.source === "agent") {
|
|
return resolveSelectedModelFallbacksOverride(agentConfig?.model);
|
|
}
|
|
if (selection?.source === "default-subagent") {
|
|
return resolveSelectedModelFallbacksOverride(cfg.agents?.defaults?.subagents?.model);
|
|
}
|
|
return undefined;
|
|
}
|
|
|
|
function resolveSubagentSpawnModelFallbacksOverride(
|
|
cfg: OpenClawConfig,
|
|
agentId: string,
|
|
): string[] | undefined {
|
|
const agentConfig = resolveAgentConfig(cfg, agentId);
|
|
return resolveFirstModelFallbacksOverride([
|
|
agentConfig?.subagents?.model,
|
|
cfg.agents?.defaults?.subagents?.model,
|
|
agentConfig?.model,
|
|
]);
|
|
}
|
|
|
|
export function resolveFallbackAgentId(params: {
|
|
agentId?: string | null;
|
|
sessionKey?: string | null;
|
|
}): string {
|
|
const explicitAgentId = normalizeOptionalString(params.agentId) ?? "";
|
|
if (explicitAgentId) {
|
|
return normalizeAgentId(explicitAgentId);
|
|
}
|
|
return resolveAgentIdFromSessionKey(params.sessionKey);
|
|
}
|
|
|
|
export function resolveRunModelFallbacksOverride(params: {
|
|
cfg: OpenClawConfig | undefined;
|
|
agentId?: string | null;
|
|
sessionKey?: string | null;
|
|
}): string[] | undefined {
|
|
if (!params.cfg) {
|
|
return undefined;
|
|
}
|
|
return resolveAgentModelFallbacksOverride(
|
|
params.cfg,
|
|
resolveFallbackAgentId({ agentId: params.agentId, sessionKey: params.sessionKey }),
|
|
);
|
|
}
|
|
|
|
export function hasConfiguredModelFallbacks(params: {
|
|
cfg: OpenClawConfig | undefined;
|
|
agentId?: string | null;
|
|
sessionKey?: string | null;
|
|
}): boolean {
|
|
const fallbacksOverride = resolveRunModelFallbacksOverride(params);
|
|
const defaultFallbacks = resolveAgentModelFallbackValues(params.cfg?.agents?.defaults?.model);
|
|
return (fallbacksOverride ?? defaultFallbacks).length > 0;
|
|
}
|
|
|
|
export function resolveEffectiveModelFallbacks(params: {
|
|
cfg: OpenClawConfig;
|
|
agentId: string;
|
|
sessionKey?: string | null;
|
|
hasSessionModelOverride: boolean;
|
|
modelOverrideSource?: "auto" | "user";
|
|
hasAutoFallbackProvenance?: boolean;
|
|
}): string[] | undefined {
|
|
const agentFallbacksOverride = resolveAgentModelFallbacksOverride(params.cfg, params.agentId);
|
|
if (!params.hasSessionModelOverride) {
|
|
return agentFallbacksOverride;
|
|
}
|
|
const canUseConfiguredFallbacks =
|
|
params.modelOverrideSource === "auto" ||
|
|
(params.modelOverrideSource === undefined && params.hasAutoFallbackProvenance === true);
|
|
if (!canUseConfiguredFallbacks) {
|
|
return [];
|
|
}
|
|
const subagentFallbacksOverride = isSubagentSessionKey(params.sessionKey)
|
|
? resolveSubagentSpawnModelFallbacksOverride(params.cfg, params.agentId)
|
|
: undefined;
|
|
if (subagentFallbacksOverride !== undefined) {
|
|
return subagentFallbacksOverride;
|
|
}
|
|
const defaultFallbacks = resolveAgentModelFallbackValues(params.cfg.agents?.defaults?.model);
|
|
return agentFallbacksOverride ?? defaultFallbacks;
|
|
}
|
|
|
|
function normalizePathForComparison(input: string): string {
|
|
const resolved = path.resolve(stripNullBytes(resolveUserPath(input)));
|
|
let normalized = resolved;
|
|
// Prefer realpath when available to normalize aliases/symlinks (for example /tmp -> /private/tmp)
|
|
// and canonical path case without forcing case-folding on case-sensitive macOS volumes.
|
|
try {
|
|
normalized = fs.realpathSync.native(resolved);
|
|
} catch {
|
|
// Keep lexical path for non-existent directories.
|
|
}
|
|
if (process.platform === "win32") {
|
|
return lowercasePreservingWhitespace(normalized);
|
|
}
|
|
return normalized;
|
|
}
|
|
|
|
export function resolveAgentIdsByWorkspacePath(
|
|
cfg: OpenClawConfig,
|
|
workspacePath: string,
|
|
): string[] {
|
|
const normalizedWorkspacePath = normalizePathForComparison(workspacePath);
|
|
const ids = listAgentIds(cfg);
|
|
const matches: Array<{ id: string; workspaceDir: string; order: number }> = [];
|
|
|
|
for (let index = 0; index < ids.length; index += 1) {
|
|
const id = ids[index];
|
|
const workspaceDir = normalizePathForComparison(resolveAgentWorkspaceDir(cfg, id));
|
|
if (!isPathInside(workspaceDir, normalizedWorkspacePath)) {
|
|
continue;
|
|
}
|
|
matches.push({ id, workspaceDir, order: index });
|
|
}
|
|
|
|
matches.sort((left, right) => {
|
|
const workspaceLengthDelta = right.workspaceDir.length - left.workspaceDir.length;
|
|
if (workspaceLengthDelta !== 0) {
|
|
return workspaceLengthDelta;
|
|
}
|
|
return left.order - right.order;
|
|
});
|
|
|
|
return matches.map((entry) => entry.id);
|
|
}
|
|
|
|
export function resolveAgentIdByWorkspacePath(
|
|
cfg: OpenClawConfig,
|
|
workspacePath: string,
|
|
): string | undefined {
|
|
return resolveAgentIdsByWorkspacePath(cfg, workspacePath)[0];
|
|
}
|