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

推荐订阅源

Google DeepMind News
Google DeepMind News
F
Fortinet All Blogs
阮一峰的网络日志
阮一峰的网络日志
Apple Machine Learning Research
Apple Machine Learning Research
爱范儿
爱范儿
WordPress大学
WordPress大学
让小产品的独立变现更简单 - ezindie.com
让小产品的独立变现更简单 - ezindie.com
J
Java Code Geeks
罗磊的独立博客
S
SegmentFault 最新的问题
V
V2EX
V
Visual Studio Blog
钛媒体:引领未来商业与生活新知
钛媒体:引领未来商业与生活新知
美团技术团队
博客园 - 三生石上(FineUI控件)
Stack Overflow Blog
Stack Overflow Blog
Y
Y Combinator Blog
MyScale Blog
MyScale Blog
D
Docker
Google DeepMind News
Google DeepMind News
Blog — PlanetScale
Blog — PlanetScale
M
Microsoft Research Blog - Microsoft Research
Martin Fowler
Martin Fowler
S
Secure Thoughts
B
Blog
cs.CL updates on arXiv.org
cs.CL updates on arXiv.org
www.infosecurity-magazine.com
www.infosecurity-magazine.com
Recent Announcements
Recent Announcements
MongoDB | Blog
MongoDB | Blog
C
Cisco Blogs
C
CERT Recently Published Vulnerability Notes
T
True Tiger Recordings
GbyAI
GbyAI
P
Proofpoint News Feed
P
Privacy International News Feed
Jina AI
Jina AI
The Cloudflare Blog
I
Intezer
AWS News Blog
AWS News Blog
Hacker News - Newest:
Hacker News - Newest: "LLM"
S
Security Archives - TechRepublic
NISL@THU
NISL@THU
The Register - Security
The Register - Security
Recent Commits to openclaw:main
Recent Commits to openclaw:main
P
Palo Alto Networks Blog
S
Schneier on Security
L
LINUX DO - 热门话题
C
CXSECURITY Database RSS Feed - CXSecurity.com
Security Latest
Security Latest
C
Cybersecurity and Infrastructure Security Agency CISA

Recent Commits to openclaw:main

test(qa-lab): report scenario pack coverage fix(plugins): drop stale tlon tool contract fix(installer): prefer tar for portable Node extraction fix(codex): deliver native subagent completions feat: add context-engine host capability requirements (#84994) · openclaw/openclaw@cff5244 fix(release): keep shrinkwrap pinned to pnpm lock · openclaw/openclaw@9d24fde fix: surface plan updates as status notices · openclaw/openclaw@dc04503 test(google): narrow web search fake timers · openclaw/openclaw@fe7d13c fix(installer): extract portable Node with ZipFile · openclaw/openclaw@ffa6cd8 fix(gateway): defer provider auth prewarm after startup (#85369) · openclaw/openclaw@69255f8 fix(talk): stabilize realtime voice consults · openclaw/openclaw@683ad75 test(qa): tolerate slow gateway rpc startup · openclaw/openclaw@29118a0 fix(diagnostics): surface async queue drops chore(diagnostics): refresh plugin sdk baseline · openclaw/openclaw@ab684f5 fix(diagnostics): bound diagnostic buffers · openclaw/openclaw@bdcaac0 fix(installer): copy portable Node into place · openclaw/openclaw@c21ca88 fix(cli): recover replaced device approvals (#85342) · openclaw/openclaw@6ea907c test(release): align prerelease validation · openclaw/openclaw@0def3e2 fix(installer): install portable Node directory atomically · openclaw/openclaw@2890b1a fix(runtime-llm): avoid duplicate provider prefix in allowlist diagno… · openclaw/openclaw@937a756 fix(gateway): include openclaw bin in service PATH (#84475) · openclaw/openclaw@66d1d13 fix(gateway): handle concurrent launchd bootstrap restart race (#84722) · openclaw/openclaw@ba86716 feat: support pi and opencode autoreview engines · openclaw/openclaw@31a189d ci(package): gate acceptance on package integrity · openclaw/openclaw@5275929 ci(release): bypass pnpm for tsdown package build · openclaw/openclaw@fea89cd test(release): align prerelease validation baselines · openclaw/openclaw@04ebdc6 ci(release): harden docker package build · openclaw/openclaw@7b1fbe1 fix(codex): skip native web search transcript mirroring (#85346) · openclaw/openclaw@c3531fc fix(gateway): harden launchd reload handoff race recovery (#84641) · openclaw/openclaw@fc7a531 fix: honor per-model provider transport overrides (#80488) fix(skills): dedupe shared-directory watchers across agent workspaces… · openclaw/openclaw@3e94290 fix(skills): type watcher mock calls in dedupe regression tests · openclaw/openclaw@bb73f0a fix(skills): document watcher edge cases, add teardown/rebuild tests,… · openclaw/openclaw@19ff77e fix(infra): allow macos browser open over ssh env (#85340) · openclaw/openclaw@47d66fe fix(update): preserve package service state during cutover (#83026) · openclaw/openclaw@a15797a fix(gateway): broadcast agent-run error payloads (#85355) · openclaw/openclaw@07e61fc test(e2e): avoid synthetic channel config in plugin smoke fix(cli): suppress systemd hints for live gateway (#85336) · openclaw/openclaw@a00c583 fix(cli): honor agent for model auth logout (#85326) · openclaw/openclaw@fc47c1f fix(gateway): eager-load lifecycle runtime to survive in-place upgrad… · openclaw/openclaw@4a91385 fix(doctor): point codex asset warning at migrate plan (#85324) fix(update): harden managed handoff cwd (#83875) · openclaw/openclaw@1bafc23 docs(release): prepare 2026.5.21 notes ci(crabbox): harden docker hydration refactor(crabbox): parse provider list from binary help instead of ha… test(plugins): run kitchen sink rpc lane without tsx test(plugins): keep rpc source walk on source call gateway test(plugins): add kitchen sink rpc docker lane · openclaw/openclaw@6f6da5f test(qa-lab): add bus tool trace scenario · openclaw/openclaw@2b39613 fix(cron): classify network retry errors (#85344) fix(installer): bootstrap portable Windows Node · openclaw/openclaw@3551e98 fix(ui): move chat session search into picker (#85303) · openclaw/openclaw@1fdc73a fix: include plugin shrinkwraps in dependency reports · openclaw/openclaw@82f69a2 test: update shrinkwrap packaging expectations · openclaw/openclaw@a1b05aa fix: honor shrinkwrap when bundling plugin deps fix: make bundled plugin packages portable fix: opt acpx out of bundled runtime deps · openclaw/openclaw@9914e25 chore: refresh shrinkwrap metadata fix: publish explicit plugin bundled dependencies · openclaw/openclaw@976da39 fix: opt codex out of bundled runtime deps · openclaw/openclaw@fcecbd8 chore: refresh shrinkwrap for Testbox npm · openclaw/openclaw@b2dc449 fix: honor overrides in npm shrinkwrap generation · openclaw/openclaw@0d28040 chore: harden npm shrinkwrap release path chore: add shrinkwrap to plugin npm packages · openclaw/openclaw@b6c8807 fix: cover plugin package locks in dependency review · openclaw/openclaw@bfa5b39 test: refresh shrinkwrap after rebase · openclaw/openclaw@8b0537c fix: keep bundled plugin peers nested · openclaw/openclaw@86faf65 feat: bundle plugin npm dependencies · openclaw/openclaw@de022bb fix: limit subagent bootstrap defaults · openclaw/openclaw@56308a7 feat: update autoreview engine coverage · openclaw/openclaw@ab1fedb fix(messages): strip unsupported citation markers (#85204) (thanks @n… · openclaw/openclaw@0a95e53 test(qa-lab): report live transport coverage lanes · openclaw/openclaw@fda0baf fix(gateway): close child ACP sessions on parent reset/delete · openclaw/openclaw@136c927 fix: preserve Google Gemini 3 cron thinking (#85300) docs(skills): exclude SDK boundary bug sweeps · openclaw/openclaw@85e468d feat(plugin-sdk): add generic channel poll sender (#85299) · openclaw/openclaw@c9a0f03 fix(agents): preserve OpenAI reasoning token usage · openclaw/openclaw@0ddf51c test(e2e): harden plugin smoke cleanup fix(plugins): resolve native plugin sdk aliases (#85298) · openclaw/openclaw@6b1c868 fix(update): keep service logs out of json output · openclaw/openclaw@03f61cd fix(agent): retry transient gateway handshake closes · openclaw/openclaw@ff79299 fix(codex): keep interrupted turns visible-answer eligible (#84494) · openclaw/openclaw@8523e09 test(agents): narrow bundle mcp e2e setup · openclaw/openclaw@6bd430e test: add mocked Control UI E2E tests and playwright for local verifi… fix(code-mode): return structured worker error codes · openclaw/openclaw@edab653 fix: land code-mode structured worker errors (#83444) (thanks @Kaspre) · openclaw/openclaw@70dd315 fix: break plugin metadata snapshot cycle · openclaw/openclaw@4ee8a2a fix: preserve route-bound direct thread events · openclaw/openclaw@0d8c9ca test node exec event wake metadata · openclaw/openclaw@37207c6 fix heartbeat event routing for main-scoped DMs fix: route direct thread event wakes to main DMs · openclaw/openclaw@0acfb7b test: align exec event routing proof (#83743) (thanks @Kaspre) · openclaw/openclaw@7b48956 test(plugins): retry bundled smoke health probes test(gateway): bind auth-free websocket harness to loopback · openclaw/openclaw@2b1c01f test(plugins): keep npm peer prune mock directory-safe · openclaw/openclaw@a12e302 chore(ui): refresh fa control ui locale fix(ci): allow release update restarts · openclaw/openclaw@b859654 chore(ui): refresh nl control ui locale · openclaw/openclaw@cc6d222 chore(ui): refresh vi control ui locale · openclaw/openclaw@b59ab5b chore(ui): refresh th control ui locale · openclaw/openclaw@f483f59
fix(codex): preserve nested tool-result middleware output · openclaw/openclaw@a4bea46
vincentkoc · 2026-05-17 · via Recent Commits to openclaw:main

@@ -167,6 +167,291 @@ describe("createAgentToolResultMiddlewareRunner", () => {

167167

});

168168

});

169169170+

it("sanitizes incoming details before failing closed on uncoercible content", async () => {

171+

const details: Record<string, unknown> = {

172+

ok: true,

173+

callback: () => 1,

174+

};

175+

details.self = details;

176+

let observedDetails: unknown;

177+

const runner = createAgentToolResultMiddlewareRunner({ runtime: "codex" }, [

178+

(event) => {

179+

observedDetails = event.result.details;

180+

return undefined;

181+

},

182+

]);

183+184+

const result = await runner.applyToolResultMiddleware({

185+

toolCallId: "call-1",

186+

toolName: "message",

187+

args: {},

188+

result: {

189+

content: [{ type: "unknown", payload: "raw" } as never],

190+

details,

191+

},

192+

});

193+194+

expect(result.details).toEqual({ status: "error", middlewareError: true });

195+

expect(observedDetails).toEqual({ ok: true });

196+

});

197+198+

it("coerces incoming nested toolResult content before middleware validation", async () => {

199+

const runner = createAgentToolResultMiddlewareRunner({ runtime: "codex" }, [() => undefined]);

200+201+

const result = await runner.applyToolResultMiddleware({

202+

toolCallId: "call-1",

203+

toolName: "message",

204+

args: {},

205+

result: {

206+

content: [

207+

{

208+

type: "toolResult",

209+

toolUseId: "call-1",

210+

content: [

211+

{ type: "text", text: "sent message id msg_123" },

212+

{ type: "text", text: "status delivered" },

213+

],

214+

} as never,

215+

],

216+

details: { status: "sent", messageId: "msg_123" },

217+

},

218+

});

219+220+

expect(result.content).toEqual([

221+

{

222+

type: "text",

223+

text: "sent message id msg_123\nstatus delivered",

224+

},

225+

]);

226+

expect(result.details).toEqual({ status: "sent", messageId: "msg_123" });

227+

});

228+229+

it("coerces nested tool_result blocks returned by middleware", async () => {

230+

const runner = createAgentToolResultMiddlewareRunner({ runtime: "codex" }, [

231+

() => ({

232+

result: {

233+

content: [

234+

{

235+

type: "tool_result",

236+

content: {

237+

message: "message delivered",

238+

id: "msg_456",

239+

},

240+

} as never,

241+

],

242+

details: { status: "sent" },

243+

},

244+

}),

245+

]);

246+247+

const result = await runner.applyToolResultMiddleware({

248+

toolCallId: "call-1",

249+

toolName: "message",

250+

args: {},

251+

result: { content: [{ type: "text", text: "raw" }], details: {} },

252+

});

253+254+

expect(result.content).toEqual([{ type: "text", text: "message delivered" }]);

255+

expect(result.details).toEqual({ status: "sent" });

256+

});

257+258+

it("does not coerce tool/function call blocks as middleware results", async () => {

259+

const runner = createAgentToolResultMiddlewareRunner({ runtime: "codex" }, [

260+

() => ({

261+

result: {

262+

content: [

263+

{

264+

type: "function",

265+

name: "send_message",

266+

arguments: { text: "raw" },

267+

} as never,

268+

],

269+

details: {},

270+

},

271+

}),

272+

]);

273+274+

const result = await runner.applyToolResultMiddleware({

275+

toolCallId: "call-1",

276+

toolName: "message",

277+

args: {},

278+

result: { content: [{ type: "text", text: "raw" }], details: {} },

279+

});

280+281+

expect(result.details).toEqual({ status: "error", middlewareError: true });

282+

});

283+284+

it("bounds nested toolResult content before flattening", async () => {

285+

const runner = createAgentToolResultMiddlewareRunner({ runtime: "codex" }, [() => undefined]);

286+287+

const result = await runner.applyToolResultMiddleware({

288+

toolCallId: "call-1",

289+

toolName: "message",

290+

args: {},

291+

result: {

292+

content: [

293+

{

294+

type: "toolResult",

295+

toolUseId: "call-1",

296+

content: [

297+

...Array.from({ length: 200 }, () => ({

298+

type: "text",

299+

text: "x".repeat(600),

300+

})),

301+

{ type: "text", text: "late chunk" },

302+

],

303+

} as never,

304+

],

305+

details: {},

306+

},

307+

});

308+309+

const content = result.content[0];

310+

if (content?.type !== "text") {

311+

throw new Error("expected flattened text content");

312+

}

313+

expect(content.text.length).toBeLessThanOrEqual(100_000);

314+

expect(content.text).not.toContain("late chunk");

315+

});

316+317+

it("preserves nested image toolResult content without stringifying data", async () => {

318+

const runner = createAgentToolResultMiddlewareRunner({ runtime: "codex" }, [() => undefined]);

319+320+

const result = await runner.applyToolResultMiddleware({

321+

toolCallId: "call-1",

322+

toolName: "vision",

323+

args: {},

324+

result: {

325+

content: [

326+

{

327+

type: "toolResult",

328+

toolUseId: "call-1",

329+

content: [{ type: "image", mimeType: "image/png", data: "base64-image" }],

330+

} as never,

331+

],

332+

details: {},

333+

},

334+

});

335+336+

expect(result.content).toEqual([

337+

{ type: "image", mimeType: "image/png", data: "base64-image" },

338+

]);

339+

});

340+341+

it("preserves mixed nested text and image toolResult content", async () => {

342+

const runner = createAgentToolResultMiddlewareRunner({ runtime: "codex" }, [() => undefined]);

343+344+

const result = await runner.applyToolResultMiddleware({

345+

toolCallId: "call-1",

346+

toolName: "screenshot",

347+

args: {},

348+

result: {

349+

content: [

350+

{

351+

type: "toolResult",

352+

toolUseId: "call-1",

353+

content: [

354+

{ type: "text", text: "captured screenshot" },

355+

{ type: "image", mimeType: "image/png", data: "base64-image" },

356+

],

357+

} as never,

358+

],

359+

details: {},

360+

},

361+

});

362+363+

expect(result.content).toEqual([

364+

{ type: "text", text: "captured screenshot" },

365+

{ type: "image", mimeType: "image/png", data: "base64-image" },

366+

]);

367+

});

368+369+

it("preserves images from deeper nested toolResult content", async () => {

370+

const runner = createAgentToolResultMiddlewareRunner({ runtime: "codex" }, [() => undefined]);

371+372+

const result = await runner.applyToolResultMiddleware({

373+

toolCallId: "call-1",

374+

toolName: "screenshot",

375+

args: {},

376+

result: {

377+

content: [

378+

{

379+

type: "toolResult",

380+

toolUseId: "call-1",

381+

content: [

382+

{

383+

type: "tool_result",

384+

content: [

385+

{ type: "text", text: "captured screenshot" },

386+

{ type: "image", mimeType: "image/png", data: "base64-image" },

387+

],

388+

},

389+

],

390+

} as never,

391+

],

392+

details: {},

393+

},

394+

});

395+396+

expect(result.content).toEqual([

397+

{ type: "text", text: "captured screenshot" },

398+

{ type: "image", mimeType: "image/png", data: "base64-image" },

399+

]);

400+

});

401+402+

it("preserves interleaved nested text and image order", async () => {

403+

const runner = createAgentToolResultMiddlewareRunner({ runtime: "codex" }, [() => undefined]);

404+405+

const result = await runner.applyToolResultMiddleware({

406+

toolCallId: "call-1",

407+

toolName: "screenshot",

408+

args: {},

409+

result: {

410+

content: [

411+

{

412+

type: "toolResult",

413+

toolUseId: "call-1",

414+

content: [

415+

{ type: "text", text: "first caption" },

416+

{ type: "image", mimeType: "image/png", data: "image-one" },

417+

{ type: "text", text: "second caption" },

418+

{ type: "image", mimeType: "image/png", data: "image-two" },

419+

],

420+

} as never,

421+

],

422+

details: {},

423+

},

424+

});

425+426+

expect(result.content).toEqual([

427+

{ type: "text", text: "first caption" },

428+

{ type: "image", mimeType: "image/png", data: "image-one" },

429+

{ type: "text", text: "second caption" },

430+

{ type: "image", mimeType: "image/png", data: "image-two" },

431+

]);

432+

});

433+434+

it("fails closed instead of recursing forever on cyclic nested content", async () => {

435+

const nested: Record<string, unknown> = {

436+

type: "toolResult",

437+

content: [],

438+

};

439+

nested.content = [nested];

440+

const runner = createAgentToolResultMiddlewareRunner({ runtime: "codex" }, [() => undefined]);

441+442+

const result = await runner.applyToolResultMiddleware({

443+

toolCallId: "call-1",

444+

toolName: "message",

445+

args: {},

446+

result: {

447+

content: [nested as never],

448+

details: {},

449+

},

450+

});

451+452+

expect(result.details).toEqual({ status: "error", middlewareError: true });

453+

});

454+170455

it("sanitizes incoming function/symbol/bigint values in details", async () => {

171456

const runner = createAgentToolResultMiddlewareRunner({ runtime: "codex" }, [() => undefined]);

172457