惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

N
News and Events Feed by Topic
Malwarebytes
Malwarebytes
Threat Intelligence Blog | Flashpoint
Threat Intelligence Blog | Flashpoint
C
Cybersecurity and Infrastructure Security Agency CISA
F
Future of Privacy Forum
C
Cisco Blogs
T
The Exploit Database - CXSecurity.com
A
Arctic Wolf
S
Securelist
K
Kaspersky official blog
S
Schneier on Security
T
ThreatConnect
T
Tenable Blog
Spread Privacy
Spread Privacy
T
True Tiger Recordings
AWS News Blog
AWS News Blog
F
Fox-IT International blog
量子位
T
Threatpost
V
Vulnerabilities – Threatpost
C
CERT Recently Published Vulnerability Notes
Cisco Talos Blog
Cisco Talos Blog
GbyAI
GbyAI
宝玉的分享
宝玉的分享
腾讯CDC
G
Google Developers Blog
aimingoo的专栏
aimingoo的专栏
Cyberwarzone
Cyberwarzone
有赞技术团队
有赞技术团队
S
SegmentFault 最新的问题
OSCHINA 社区最新新闻
OSCHINA 社区最新新闻
V
Visual Studio Blog
U
Unit 42
雷峰网
雷峰网
cs.CV updates on arXiv.org
cs.CV updates on arXiv.org
Simon Willison's Weblog
Simon Willison's Weblog
O
OpenAI News
freeCodeCamp Programming Tutorials: Python, JavaScript, Git & More
The GitHub Blog
The GitHub Blog
The Register - Security
The Register - Security
MyScale Blog
MyScale Blog
小众软件
小众软件
A
About on SuperTechFans
Last Week in AI
Last Week in AI
Y
Y Combinator Blog
博客园 - 三生石上(FineUI控件)
美团技术团队
Google Online Security Blog
Google Online Security Blog
P
Proofpoint News Feed
MongoDB | Blog
MongoDB | Blog

Recent Commits to openclaw:main

test(agents): keep runtime-plan provider mock current fix(scripts): launch env package scripts on Windows · openclaw/openclaw@4d4ce9e fix(test): make import timing scripts Windows-safe · openclaw/openclaw@8ae9977 fix(telegram): transient Telegram pairing prompts (#85555) · openclaw/openclaw@8209426 fix(test): make max Vitest scripts Windows-safe · openclaw/openclaw@b681d5d fix(doctor): migrate Feishu account bot names (#86081) · openclaw/openclaw@9e8cc7e fix(scripts): prefilter conflict marker scans docs: add ClawSweeper review policy to AGENTS (#86197) · openclaw/openclaw@242e876 fix(installer): avoid before with npm release-age configs (#85491) · openclaw/openclaw@4742db6 fix(e2e): retry Windows kitchen sink probes · openclaw/openclaw@3e275a5 fix(installer): install node with apk on alpine fix(installer): detect musl linux shells · openclaw/openclaw@acfed37 perf(plugins,gateway): thread metadata snapshot + discovery through h… · openclaw/openclaw@8ccb11c fix(ui): split control ui runtime chunks · openclaw/openclaw@8bf4f7d refactor(config): extract GoogleChat schema into zod-schema.providers… · openclaw/openclaw@fe34141 fix(update): suppress internal handoff version warnings · openclaw/openclaw@6cc8244 test(e2e): select installable bundled plugins · openclaw/openclaw@0acc3e3 fix(scripts): harden Windows native opus install · openclaw/openclaw@43252c8 fix(agents): match runtime policy entries when session provider is em… fix(scripts): harden Windows generated formatting · openclaw/openclaw@0a98559 fix(mcp): bound tools/list during catalog discovery (#85063) · openclaw/openclaw@07f500a fix(test): focus plugin binding Docker smoke · openclaw/openclaw@dfa1a51 test(e2e): fail release memory indexing errors test(daemon): fail launchd integration bootstrap errors · openclaw/openclaw@af07769 feat(imessage): support thumb approval reactions (#85952) · openclaw/openclaw@5c7980f fix(crabbox): default macos aws runs on demand fix(scripts): preserve test passthrough args · openclaw/openclaw@e4332f7 fix(e2e): harden Windows plugin assertions fix(test): mount upgrade survivor helper · openclaw/openclaw@5f03154 fix(android): prevent stale chat during session switches fix(android): keep permission setup action visible · openclaw/openclaw@94bc18a style(android): sharpen voice mode surfaces · openclaw/openclaw@c452510 fix(android): hide internal chat content blocks · openclaw/openclaw@d86ed21 style(android): refine list surface spacing · openclaw/openclaw@955909c feat(android): add pair new gateway action · openclaw/openclaw@cc5eb97 Advance iMessage catchup cursor after live handling (#85475) · openclaw/openclaw@102555c fix(scripts): ignore forwarded arg separator · openclaw/openclaw@79ee70c fix(test): fail empty gateway startup samples · openclaw/openclaw@5a8ce6a fix(e2e): harden Windows kitchen sink assertions · openclaw/openclaw@87a2eba fix(e2e): harden Telegram credential paths on Windows · openclaw/openclaw@c643370 fix(android): align setup pairing scopes fix(android): complete qr setup operator handoff · openclaw/openclaw@be9bb77 fix(test): copy cleanup smoke prepare hook · openclaw/openclaw@dbc08f6 fix(secrets): allow hash in exec SecretRef ids (#86072) · openclaw/openclaw@675158c fix(media): use static image compression metadata · openclaw/openclaw@694d45e fix(release): verify large plugin npm packs fix(test): require kitchen sink diagnostic canaries · openclaw/openclaw@7e51f83 fix(scripts): harden Windows upgrade survivor recipe · openclaw/openclaw@483d7be fix(installer): count verify progress stage fix: Refine PR template for review state (#86054) fix(test): repair split agent shard runs · openclaw/openclaw@125d82c fix(codex): harden Windows protocol formatting · openclaw/openclaw@ce48e4c fix(openrouter): use endpoint context limits (#86041) · openclaw/openclaw@dd01a2e test(qa): remove brittle capability flip setup turn fix(telegram): migrate legacy cache sidecars · openclaw/openclaw@eb9b882 fix(telegram): migrate account topic cache sidecars · openclaw/openclaw@5cfb12f fix(scripts): harden Windows ZAI fallback repro · openclaw/openclaw@5be62e7 style(android): sharpen v2 screen rhythm · openclaw/openclaw@400d90a test(qa): extend capability flip setup budget · openclaw/openclaw@c91c3c6 fix(android): simplify gateway status copy fix(android): route offline voice to gateway setup fix(scripts): harden Windows control UI i18n commands · openclaw/openclaw@581c8a6 fix(android): stop operator chat subscription · openclaw/openclaw@5c15859 fix(test): fail missing kitchen sink rss samples test(qa): widen capability flip restart budget · openclaw/openclaw@c7d4e9e fix(android): smooth gateway pairing recovery · openclaw/openclaw@60e6ccd fix(test): suppress rolldown timing noise · openclaw/openclaw@6d9b388 style(android): fix talk mode ktlint formatting · openclaw/openclaw@01b284c fix(telegram): store topic cache in plugin state fix(telegram): store bot info cache in plugin state · openclaw/openclaw@2ed5296 fix(test): sync sparse AWS Crabbox runs from full checkout · openclaw/openclaw@0f82c81 fix(release): harden Windows cross-os command shims · openclaw/openclaw@7154767 fix(test): harden Docker resource ceilings test(telegram): keep startup limiter coverage focused test(telegram): isolate startup probe limiter timing · openclaw/openclaw@04d86e0 test(release): harden plugin prerelease checks · openclaw/openclaw@578e73f fix(telegram): serialize topic dispatch replies (#85709) · openclaw/openclaw@62b51a6 test(release): stabilize plugin prerelease checks · openclaw/openclaw@3679151 fix(test): fail live gateway false greens · openclaw/openclaw@295339d fix(test): build startup artifacts for smoke scripts · openclaw/openclaw@3838e45 test(telegram): wait for startup probe slots · openclaw/openclaw@0a8af67 test(codex): match sandbox exec-server yolo policy · openclaw/openclaw@783290f fix(test): fail missing explicit test targets · openclaw/openclaw@9ff4d36 test(codex): avoid full sandbox exec-server turn run · openclaw/openclaw@558c1bc fix(ci): keep Crabbox pnpm hydration shims writable · openclaw/openclaw@bca1ac0 fix(release): harden Windows release-check npm probes · openclaw/openclaw@75ac11a fix(docker): parse peer-suffixed lockfile packages · openclaw/openclaw@cf46f2e fix(docker): seed lockfile packages before prune · openclaw/openclaw@f799da0 fix(docker): seed lockfile snapshot tarballs before prune · openclaw/openclaw@2cd93f1 test(codex): type thread start mock params · openclaw/openclaw@a4ef3a2 test(codex): avoid full sandbox run in thread-start test · openclaw/openclaw@11bf642 fix(plugins): harden Windows npm package staging · openclaw/openclaw@abdd8a4 test(codex): complete sandbox turn inline · openclaw/openclaw@c14a0c6 fix(release): harden Windows npm shim verification · openclaw/openclaw@a56f452 test(release): type metadata snapshot mock params · openclaw/openclaw@f878959 test(release): finish plugin metadata prerelease sync fix(update): avoid broad tag fetches for dev updates (#84737) · openclaw/openclaw@501f2cb Fix iMessage slash command acknowledgements (#82642) · openclaw/openclaw@4d15020 test(release): align prerelease contracts · openclaw/openclaw@02f53e6 test(release): align plugin prerelease checks
fix(agents): cache fallback provider resolution · openclaw/openclaw@3c8d101
vincentkoc · 2026-05-25 · via Recent Commits to openclaw:main

@@ -6,9 +6,13 @@ import { resetLogger, setLoggerOverride } from "../logging/logger.js";

66

import { createWarnLogCapture } from "../logging/test-helpers/warn-log-capture.js";

77

import {

88

clearCurrentPluginMetadataSnapshot,

9+

resolvePluginMetadataControlPlaneFingerprint,

910

setCurrentPluginMetadataSnapshot,

1011

} from "../plugins/current-plugin-metadata-snapshot.js";

12+

import { resolveInstalledPluginIndexPolicyHash } from "../plugins/installed-plugin-index-policy.js";

13+

import type { InstalledPluginIndex } from "../plugins/installed-plugin-index.js";

1114

import { loadPluginMetadataSnapshot } from "../plugins/plugin-metadata-snapshot.js";

15+

import type { PluginMetadataSnapshot } from "../plugins/plugin-metadata-snapshot.types.js";

1216

import { CommandLaneTaskTimeoutError } from "../process/command-queue.js";

1317

import { AUTH_STORE_VERSION } from "./auth-profiles/constants.js";

1418

import type { AuthProfileStore } from "./auth-profiles/types.js";

@@ -164,10 +168,7 @@ let authTempRoot = "";

164168

let authTempCounter = 0;

165169166170

beforeAll(() => {

167-

setCurrentPluginMetadataSnapshot(loadPluginMetadataSnapshot({ config: {}, env: process.env }), {

168-

config: {},

169-

env: process.env,

170-

});

171+

setDefaultPluginMetadataSnapshot();

171172

});

172173173174

afterAll(() => {

@@ -181,6 +182,73 @@ function resetModelFallbackTestState(): void {

181182

authSourceCheckMock.hasAnyAuthProfileStoreSource.mockReset().mockReturnValue(false);

182183

}

183184185+

function setDefaultPluginMetadataSnapshot(): void {

186+

setCurrentPluginMetadataSnapshot(loadPluginMetadataSnapshot({ config: {}, env: process.env }), {

187+

config: {},

188+

env: process.env,

189+

});

190+

}

191+192+

function createModelNormalizerSnapshot(params: {

193+

manifestHash: string;

194+

prefix: string;

195+

}): PluginMetadataSnapshot {

196+

const policyHash = resolveInstalledPluginIndexPolicyHash({});

197+

const index: InstalledPluginIndex = {

198+

version: 1,

199+

hostContractVersion: "test-host",

200+

compatRegistryVersion: "test-compat",

201+

migrationVersion: 1,

202+

policyHash,

203+

generatedAtMs: 0,

204+

installRecords: {},

205+

plugins: [

206+

{

207+

pluginId: "fallback-normalizer",

208+

manifestPath: `/tmp/fallback-normalizer-${params.manifestHash}/openclaw.plugin.json`,

209+

manifestHash: params.manifestHash,

210+

source: `/tmp/fallback-normalizer-${params.manifestHash}/index.ts`,

211+

rootDir: `/tmp/fallback-normalizer-${params.manifestHash}`,

212+

origin: "global",

213+

enabled: true,

214+

startup: {

215+

sidecar: false,

216+

memory: false,

217+

deferConfiguredChannelFullLoadUntilAfterListen: false,

218+

agentHarnesses: [],

219+

},

220+

compat: [],

221+

},

222+

],

223+

diagnostics: [],

224+

};

225+

return {

226+

policyHash,

227+

configFingerprint: resolvePluginMetadataControlPlaneFingerprint(

228+

{},

229+

{

230+

env: process.env,

231+

index,

232+

policyHash,

233+

},

234+

),

235+

index,

236+

registryDiagnostics: [],

237+

plugins: [

238+

{

239+

id: "fallback-normalizer",

240+

modelIdNormalization: {

241+

providers: {

242+

demo: {

243+

prefixWhenBare: params.prefix,

244+

},

245+

},

246+

},

247+

},

248+

],

249+

} as unknown as PluginMetadataSnapshot;

250+

}

251+184252

afterEach(resetModelFallbackTestState);

185253186254

beforeEach(() => {

@@ -227,6 +295,31 @@ function makeProviderFallbackCfg(provider: string): OpenClawConfig {

227295

});

228296

}

229297298+

function makeProviderOrderFallbackCfg(

299+

entries: Array<[provider: string, model: string]>,

300+

): OpenClawConfig {

301+

return {

302+

agents: {

303+

defaults: {

304+

model: {

305+

fallbacks: [],

306+

},

307+

},

308+

},

309+

models: {

310+

providers: Object.fromEntries(

311+

entries.map(([provider, model]) => [

312+

provider,

313+

{

314+

baseUrl: `https://${provider}.example.test`,

315+

models: [{ id: model }],

316+

},

317+

]),

318+

),

319+

},

320+

} as unknown as OpenClawConfig;

321+

}

322+230323

async function withTempAuthStore<T>(

231324

store: AuthProfileStore,

232325

run: (tempDir: string) => Promise<T>,

@@ -1969,6 +2062,82 @@ describe("runWithModelFallback", () => {

19692062

]);

19702063

});

197120642065+

it("does not reuse provider-order-sensitive configured fallback candidates", () => {

2066+

const anthropicFirst = makeProviderOrderFallbackCfg([

2067+

["anthropic", "claude-sonnet-4"],

2068+

["ollama", "llama3"],

2069+

]);

2070+

const ollamaFirst = makeProviderOrderFallbackCfg([

2071+

["ollama", "llama3"],

2072+

["anthropic", "claude-sonnet-4"],

2073+

]);

2074+2075+

expect(

2076+

testing.resolveFallbackCandidates({

2077+

cfg: anthropicFirst,

2078+

provider: "",

2079+

model: "",

2080+

fallbacksOverride: [],

2081+

}),

2082+

).toEqual([{ provider: "anthropic", model: "claude-sonnet-4" }]);

2083+

expect(

2084+

testing.resolveFallbackCandidates({

2085+

cfg: ollamaFirst,

2086+

provider: "",

2087+

model: "",

2088+

fallbacksOverride: [],

2089+

}),

2090+

).toEqual([{ provider: "ollama", model: "llama3" }]);

2091+

});

2092+2093+

it("does not reuse fallback candidate cache entries across manifest normalization snapshots", () => {

2094+

const cfg = makeCfg({

2095+

agents: {

2096+

defaults: {

2097+

model: {

2098+

fallbacks: [],

2099+

},

2100+

},

2101+

},

2102+

});

2103+2104+

try {

2105+

setCurrentPluginMetadataSnapshot(

2106+

createModelNormalizerSnapshot({

2107+

manifestHash: "alpha",

2108+

prefix: "alpha",

2109+

}),

2110+

{ config: {}, env: process.env },

2111+

);

2112+

expect(

2113+

testing.resolveFallbackCandidates({

2114+

cfg,

2115+

provider: "demo",

2116+

model: "demo-model",

2117+

fallbacksOverride: [],

2118+

}),

2119+

).toEqual([{ provider: "demo", model: "alpha/demo-model" }]);

2120+2121+

setCurrentPluginMetadataSnapshot(

2122+

createModelNormalizerSnapshot({

2123+

manifestHash: "bravo",

2124+

prefix: "bravo",

2125+

}),

2126+

{ config: {}, env: process.env },

2127+

);

2128+

expect(

2129+

testing.resolveFallbackCandidates({

2130+

cfg,

2131+

provider: "demo",

2132+

model: "demo-model",

2133+

fallbacksOverride: [],

2134+

}),

2135+

).toEqual([{ provider: "demo", model: "bravo/demo-model" }]);

2136+

} finally {

2137+

setDefaultPluginMetadataSnapshot();

2138+

}

2139+

});

2140+19722141

it("defaults provider/model when missing (regression #946)", () => {

19732142

const cfg = makeCfg({

19742143

agents: {