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

推荐订阅源

酷 壳 – CoolShell
酷 壳 – CoolShell
GbyAI
GbyAI
N
Netflix TechBlog - Medium
Hugging Face - Blog
Hugging Face - Blog
奇客Solidot–传递最新科技情报
奇客Solidot–传递最新科技情报
Microsoft Security Blog
Microsoft Security Blog
Engineering at Meta
Engineering at Meta
Y
Y Combinator Blog
罗磊的独立博客
云风的 BLOG
云风的 BLOG
Blog — PlanetScale
Blog — PlanetScale
Last Week in AI
Last Week in AI
MongoDB | Blog
MongoDB | Blog
Simon Willison's Weblog
Simon Willison's Weblog
The Register - Security
The Register - Security
U
Unit 42
Stack Overflow Blog
Stack Overflow Blog
Recorded Future
Recorded Future
博客园 - 三生石上(FineUI控件)
F
Future of Privacy Forum
www.infosecurity-magazine.com
www.infosecurity-magazine.com
The Hacker News
The Hacker News
C
Cybersecurity and Infrastructure Security Agency CISA
V
Visual Studio Blog
Hacker News - Newest:
Hacker News - Newest: "LLM"
V
Vulnerabilities – Threatpost
L
Lohrmann on Cybersecurity
阮一峰的网络日志
阮一峰的网络日志
A
Arctic Wolf
C
CXSECURITY Database RSS Feed - CXSecurity.com
IntelliJ IDEA : IntelliJ IDEA – the Leading IDE for Professional Development in Java and Kotlin | The JetBrains Blog
IntelliJ IDEA : IntelliJ IDEA – the Leading IDE for Professional Development in Java and Kotlin | The JetBrains Blog
cs.CV updates on arXiv.org
cs.CV updates on arXiv.org
月光博客
月光博客
C
Comments on: Blog
博客园 - Franky
PCI Perspectives
PCI Perspectives
Recent Commits to openclaw:main
Recent Commits to openclaw:main
Jina AI
Jina AI
S
Security Archives - TechRepublic
A
About on SuperTechFans
Google DeepMind News
Google DeepMind News
I
Intezer
C
CERT Recently Published Vulnerability Notes
T
True Tiger Recordings
F
Fox-IT International blog
MyScale Blog
MyScale Blog
CTFtime.org: upcoming CTF events
CTFtime.org: upcoming CTF events
T
ThreatConnect
D
DataBreaches.Net
博客园 - 聂微东

Recent Commits to openclaw:main

fix(gateway): dedupe session tool fanout · openclaw/openclaw@89a21db fix: Hook ingress token unlocks password-mode gateway auth (#86453) · openclaw/openclaw@d51f268 fix(diagnostics): reclaim wedged session lanes with a stale leaked ac… · openclaw/openclaw@6f76d9f fix: derive plugin media trust from metadata (#86410) · openclaw/openclaw@e761eb8 fix(media-understanding): normalize HEIC before image descriptions (#… · openclaw/openclaw@75c7236 fix: accept leading fuzzy Discord voice wake names (#86484) · openclaw/openclaw@8fe4f34 feat: promote provider tool call stream wrapper (#86489) fix(test): dedupe kitchen sink command assertions test: derive deprecated sdk usage guard (#86403) fix(qa): extend memory fallback Windows budget fix(ui): move control ui chunking helper out of runtime source · openclaw/openclaw@968c87d fix: quiet retained lost task noise (#86475) fix(build): keep control ui chunking out of deadcode · openclaw/openclaw@dc26069 fix: rotate realtime voice sessions on max duration · openclaw/openclaw@dc2c4aa fix(test): stream bundled plugin sweep logs · openclaw/openclaw@fc3cd49 docs: add bugfix changelog credits · openclaw/openclaw@2e7e4bc fix(models): show oauth marker auth status (#86378) · openclaw/openclaw@a6df39d fix: seed cron task progress summaries (#86313) · openclaw/openclaw@92afd8b fix(update): exclude prerelease tags from stable git channel (#86260) · openclaw/openclaw@28f169b fix(doctor): warn and continue when cron job store is unreadable (#86… fix(gateway): clear runtime config snapshot before in-process restart… · openclaw/openclaw@90caa3b fix(scripts): restore sparse crabbox changed gates · openclaw/openclaw@d270879 fix(build): support Windows UI builds · openclaw/openclaw@0bb9b42 Fix local embedding worker safety (#85348) · openclaw/openclaw@7ff29a9 fix(ui): scope chat session picker to active agent (#85965) · openclaw/openclaw@70c7d6f [codex] improve iOS realtime talk mode (#86355) · openclaw/openclaw@9ca52ce fix(scripts): dedupe docker lane resources · openclaw/openclaw@5e94469 docs: add code size guidance · openclaw/openclaw@9a60fcf fix(test): avoid source gateway import in rpc walk · openclaw/openclaw@e9b8a6e docs: add bugfix changelog entries · openclaw/openclaw@f950132 Fix heartbeat response loop guard (#86324) (#86357) · openclaw/openclaw@e2c174e fix(memory-core): filter REM dreaming candidates to light-staged entr… · openclaw/openclaw@8b42771 fix(telegram): propagate forum topic names into agent context (#86299) fix(slack): keep downloaded files out of reply media (#86318) · openclaw/openclaw@2fcd481 fix(cron): accept plus durations for one-shot jobs (#86341) · openclaw/openclaw@9239f94 fix(plugins): clear metadata memo at lifecycle boundaries · openclaw/openclaw@e7c696a chore(skills): normalize release skill routing · openclaw/openclaw@4737e19 docs(release): require early performance regression check · openclaw/openclaw@0336938 fix(qa): capture Windows gateway metrics · openclaw/openclaw@9afbfc1 feat(qa): add coverage scenario matching · openclaw/openclaw@a1fe86a fix(perf): avoid duplicate docker package ui build build: enable modern TypeScript module syntax · openclaw/openclaw@bbc1772 ci: include performance evidence in release validation fix(providers): stream ordinary tool-like prose promptly fix(perf): harden gateway restart bench exits · openclaw/openclaw@82bbcf6 fix(gateway): gate talk secret bootstrap handoff (#85690) · openclaw/openclaw@c791e42 fix: suppress async media incomplete-turn errors (#85933) · openclaw/openclaw@35dcd42 migrate auth credentials · openclaw/openclaw@f036bac fix migrate auth lint · openclaw/openclaw@50e6cb0 fix migrate supported auth imports · openclaw/openclaw@44bb2be fix migrate auth opt-out precedence · openclaw/openclaw@2016a51 honor migrate auth opt-out in plan · openclaw/openclaw@17edec7 address migrate auth review comments · openclaw/openclaw@0a98c2d fix ci blockers for migrate auth docs: add migrate auth changelog (#85667) · openclaw/openclaw@f7fcbdb fix(scripts): avoid duplicate install smoke ui build · openclaw/openclaw@b1b2841 fix(telegram): preserve inbound text entities (#83873) · openclaw/openclaw@b552919 chore: ignore Python bytecode caches · openclaw/openclaw@b6b2755 fix: make autoreview progress visible · openclaw/openclaw@236edb2 ci(release): fix plugin prerelease extension batch invocation test(telegram): provide topic cache store in message context harness · openclaw/openclaw@ff1fde1 test(agents): complete provider runtime test mocks · openclaw/openclaw@be8cd12 test(telegram): type topic cache harness store · openclaw/openclaw@84ab206 test(agents): sync provider runtime mocks · openclaw/openclaw@a289dd9 refactor: keep plain text tool-call promotion private (#86374) · openclaw/openclaw@c3ab2de fix(discord): suppress self-reply prompt echoes (#86238) docs: clarify config migration policy · openclaw/openclaw@c44367f fix(perf): fail startup bench on early gateway exit · openclaw/openclaw@a8fc28c fix: prevent plain text tool call leaks (#86222) · openclaw/openclaw@cd62780 fix: handle npm min-release-age in installers · openclaw/openclaw@316d97c fix(scripts): include ui:build in build-all full and ciArtifacts prof… · openclaw/openclaw@6704d0a fix(e2e): sample Windows kitchen sink gateway RSS · openclaw/openclaw@73189e3 fix(cron): respect isolated target and error on missing remove id (#8… · openclaw/openclaw@6709f4e fix(pi-embedded-runner): propagate trigger-derived priority to the gl… · openclaw/openclaw@0580f57 fix(cli): suppress self-update version warnings · openclaw/openclaw@e2bd20f fix: preserve webchat source reply details · openclaw/openclaw@aa50c51 docs: replace OpenClaw docs skill and add plugin permissions guide · openclaw/openclaw@0dabb70 fix(codex): preserve source reply mode for active runs (#86325) · openclaw/openclaw@b962110 fix: make compaction reinjection opt-in · openclaw/openclaw@ab910f8 fix codex usage-limit recovery copy (#86305) · openclaw/openclaw@c3c8a65 feat(ui): add ephemeral Activity tab · openclaw/openclaw@3dd0e8e fix(tests): harden native macos plugin proof · openclaw/openclaw@a5d5604 fix(commitments): serialize load-modify-save with in-process queue + … · openclaw/openclaw@d3c293d Fail Codex compaction at the Codex boundary (#85958) · openclaw/openclaw@dd47e47 fix(docker): restore config parent ownership · openclaw/openclaw@908b894 docs: clarify config default review policy (#86329) · openclaw/openclaw@3a03dd5 docs: clean changelog script entries · openclaw/openclaw@0eead19 fix(scripts): budget restart benchmark timeouts · openclaw/openclaw@5bd5509 fix: align ui vitest config assertion · openclaw/openclaw@730fd19 fix: route unit ui vitest targets narrowly · openclaw/openclaw@777402e fix: route explicit ui vitest targets narrowly · openclaw/openclaw@56a383c fix(android): harden play media permission removal fix(webchat): stabilize live transcript run state · openclaw/openclaw@119a01c fix(scripts): fail restart benchmark regressions · openclaw/openclaw@95d1b39 fix(openai): scope external codex auth to realtime fix(openai): prefer codex auth for GPT realtime · openclaw/openclaw@48c4f57 fix(openai): discover codex cli auth for provider checks · openclaw/openclaw@4656275 fix(android): keep talk mode on realtime relay · openclaw/openclaw@70614f8 test(android): add gateway connect adb probe · openclaw/openclaw@d7aa1f3 fix(android): stabilize realtime talk connection state · openclaw/openclaw@ffb02a5
fix #86077: keep fallback errors candidate scoped (#86134) · openclaw/openclaw@d6b7fe8
zhangguiping · 2026-05-25 · via Recent Commits to openclaw:main

@@ -17,16 +17,20 @@ import type { EmbeddedRunAttemptResult } from "./run/types.js";

17171818

let runEmbeddedPiAgent: typeof import("./run.js").runEmbeddedPiAgent;

1919

const DEEPSEEK_ERROR_MESSAGE = "429 deepseek rate limit";

20+

type CurrentAttemptAssistantWithError = NonNullable<

21+

EmbeddedRunAttemptResult["currentAttemptAssistant"]

22+

> & { errorMessage: string };

20232124

function isCurrentAttemptAssistant(

2225

value: unknown,

23-

): value is NonNullable<EmbeddedRunAttemptResult["currentAttemptAssistant"]> {

26+

): value is CurrentAttemptAssistantWithError {

2427

return (

2528

typeof value === "object" &&

2629

value !== null &&

2730

"provider" in value &&

2831

"model" in value &&

29-

"errorMessage" in value

32+

"errorMessage" in value &&

33+

typeof value.errorMessage === "string"

3034

);

3135

}

3236

@@ -130,16 +134,24 @@ describe("runEmbeddedPiAgent cross-provider fallback error handling", () => {

130134

});

131135132136

it("falls back to the session assistant when compaction removes the current attempt slice", async () => {

133-

setupDeepseekFallbackErrorMatchers();

134137

const getLastFormattedAssistant = captureFormattedAssistant();

138+

const sameCandidateErrorMessage = "429 current candidate rate limit";

139+

mockedIsFailoverAssistantError.mockImplementation((...args: unknown[]) => {

140+

const assistant = args[0];

141+

return isCurrentAttemptAssistant(assistant) && assistant.provider === "anthropic";

142+

});

143+

mockedIsRateLimitAssistantError.mockImplementation((...args: unknown[]) => {

144+

const assistant = args[0];

145+

return isCurrentAttemptAssistant(assistant) && assistant.provider === "anthropic";

146+

});

135147

mockedRunEmbeddedAttempt.mockResolvedValueOnce(

136148

makeAttemptResult({

137149

assistantTexts: [],

138150

lastAssistant: makeAssistantMessageFixture({

139151

stopReason: "error",

140-

errorMessage: DEEPSEEK_ERROR_MESSAGE,

141-

provider: "deepseek",

142-

model: "deepseek-chat",

152+

errorMessage: sameCandidateErrorMessage,

153+

provider: "anthropic",

154+

model: "test-model",

143155

content: [],

144156

}),

145157

currentAttemptAssistant: undefined,

@@ -152,6 +164,119 @@ describe("runEmbeddedPiAgent cross-provider fallback error handling", () => {

152164

config: makeCrossProviderFallbackConfig(),

153165

});

154166155-

await expectDeepseekFallbackError(promise, getLastFormattedAssistant);

167+

await expect(promise).rejects.toBeInstanceOf(MockedFailoverError);

168+

await expect(promise).rejects.toThrow(`anthropic/test-model: ${sameCandidateErrorMessage}`);

169+

expect(mockedIsRateLimitAssistantError).toHaveBeenCalledTimes(1);

170+

expect(getLastFormattedAssistant()).toMatchObject({

171+

provider: "anthropic",

172+

model: "test-model",

173+

errorMessage: sameCandidateErrorMessage,

174+

});

175+

});

176+177+

it("keeps PI-stamped session assistant errors for the current candidate after compaction", async () => {

178+

const getLastFormattedAssistant = captureFormattedAssistant();

179+

const sameCandidateErrorMessage = "429 current PI-stamped candidate rate limit";

180+

mockedIsFailoverAssistantError.mockImplementation((...args: unknown[]) => {

181+

const assistant = args[0];

182+

return isCurrentAttemptAssistant(assistant) && assistant.provider === "pi";

183+

});

184+

mockedIsRateLimitAssistantError.mockImplementation((...args: unknown[]) => {

185+

const assistant = args[0];

186+

return isCurrentAttemptAssistant(assistant) && assistant.provider === "pi";

187+

});

188+

mockedRunEmbeddedAttempt.mockResolvedValueOnce(

189+

makeAttemptResult({

190+

assistantTexts: [],

191+

lastAssistant: makeAssistantMessageFixture({

192+

stopReason: "error",

193+

errorMessage: sameCandidateErrorMessage,

194+

provider: "pi",

195+

model: "pi",

196+

content: [],

197+

}),

198+

currentAttemptAssistant: undefined,

199+

}),

200+

);

201+202+

const promise = runEmbeddedPiAgent({

203+

...overflowBaseRunParams,

204+

runId: "run-compaction-pi-stamped-fallback-error-context",

205+

config: makeCrossProviderFallbackConfig(),

206+

});

207+208+

await expect(promise).rejects.toBeInstanceOf(MockedFailoverError);

209+

await expect(promise).rejects.toThrow(sameCandidateErrorMessage);

210+

expect(mockedIsRateLimitAssistantError).toHaveBeenCalledTimes(1);

211+

const rateLimitCalls = mockedIsRateLimitAssistantError.mock.calls as unknown[][];

212+

expect(rateLimitCalls.at(-1)?.[0]).toMatchObject({

213+

provider: "pi",

214+

model: "pi",

215+

errorMessage: sameCandidateErrorMessage,

216+

});

217+

expect(getLastFormattedAssistant()).toMatchObject({

218+

provider: "pi",

219+

model: "pi",

220+

errorMessage: sameCandidateErrorMessage,

221+

});

222+

});

223+224+

it("does not reuse a prior provider session assistant when the current candidate times out", async () => {

225+

const getLastFormattedAssistant = captureFormattedAssistant();

226+

mockedRunEmbeddedAttempt.mockResolvedValueOnce(

227+

makeAttemptResult({

228+

assistantTexts: [],

229+

timedOut: true,

230+

lastAssistant: makeAssistantMessageFixture({

231+

stopReason: "error",

232+

errorMessage: "You exceeded your current OpenAI quota.",

233+

provider: "openai-codex",

234+

model: "gpt-5.4",

235+

content: [],

236+

}),

237+

currentAttemptAssistant: undefined,

238+

}),

239+

);

240+241+

const promise = runEmbeddedPiAgent({

242+

...overflowBaseRunParams,

243+

runId: "run-stale-session-assistant-timeout",

244+

config: makeCrossProviderFallbackConfig(),

245+

});

246+247+

await expect(promise).rejects.toBeInstanceOf(MockedFailoverError);

248+

await expect(promise).rejects.toThrow("LLM request timed out.");

249+

await expect(promise).rejects.not.toThrow("OpenAI quota");

250+

expect(getLastFormattedAssistant()).toBeUndefined();

251+

});

252+253+

it("does not reuse a prior provider session assistant for non-timeout failover", async () => {

254+

mockedIsFailoverAssistantError.mockImplementation((...args: unknown[]) => {

255+

const assistant = args[0];

256+

return isCurrentAttemptAssistant(assistant) && assistant.errorMessage.includes("quota");

257+

});

258+

const getLastFormattedAssistant = captureFormattedAssistant();

259+

mockedRunEmbeddedAttempt.mockResolvedValueOnce(

260+

makeAttemptResult({

261+

assistantTexts: [],

262+

lastAssistant: makeAssistantMessageFixture({

263+

stopReason: "error",

264+

errorMessage: "You exceeded your current OpenAI quota.",

265+

provider: "openai-codex",

266+

model: "gpt-5.4",

267+

content: [],

268+

}),

269+

currentAttemptAssistant: undefined,

270+

}),

271+

);

272+273+

await runEmbeddedPiAgent({

274+

...overflowBaseRunParams,

275+

runId: "run-stale-session-assistant-non-timeout",

276+

config: makeCrossProviderFallbackConfig(),

277+

});

278+279+

expect(mockedIsFailoverAssistantError).toHaveBeenCalledWith(undefined);

280+

expect(getLastFormattedAssistant()).toBeUndefined();

156281

});

157282

});