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

推荐订阅源

人人都是产品经理
人人都是产品经理
W
WeLiveSecurity
Recorded Future
Recorded Future
P
Privacy & Cybersecurity Law Blog
V
Vulnerabilities – Threatpost
C
Cybersecurity and Infrastructure Security Agency CISA
G
GRAHAM CLULEY
S
Securelist
让小产品的独立变现更简单 - ezindie.com
让小产品的独立变现更简单 - ezindie.com
小众软件
小众软件
The Hacker News
The Hacker News
The Cloudflare Blog
D
Darknet – Hacking Tools, Hacker News & Cyber Security
V
V2EX
C
Cisco Blogs
Cisco Talos Blog
Cisco Talos Blog
腾讯CDC
Recent Announcements
Recent Announcements
Jina AI
Jina AI
K
Kaspersky official blog
The GitHub Blog
The GitHub Blog
云风的 BLOG
云风的 BLOG
酷 壳 – CoolShell
酷 壳 – CoolShell
GbyAI
GbyAI
F
Fortinet All Blogs
T
ThreatConnect
S
Schneier on Security
罗磊的独立博客
Y
Y Combinator Blog
C
Check Point Blog
T
The Exploit Database - CXSecurity.com
宝玉的分享
宝玉的分享
aimingoo的专栏
aimingoo的专栏
CTFtime.org: upcoming CTF events
CTFtime.org: upcoming CTF events
I
Intezer
F
Full Disclosure
T
Troy Hunt's Blog
OSCHINA 社区最新新闻
OSCHINA 社区最新新闻
WordPress大学
WordPress大学
Application and Cybersecurity Blog
Application and Cybersecurity Blog
V
V2EX - 技术
C
Comments on: Blog
T
Tenable Blog
Project Zero
Project Zero
H
Help Net Security
A
Arctic Wolf
Google DeepMind News
Google DeepMind News
NISL@THU
NISL@THU
博客园 - 【当耐特】
F
Fox-IT International blog

Recent Commits to openclaw:main

fix: avoid gateway startup event-loop stalls test(codex): type forced-tool request mock test(codex): avoid startup cleanup socket flake fix(gateway): pin relative state dir at startup · openclaw/openclaw@2e5be0c fix(whatsapp): persist inbound delivery in plugin state (#85506) · openclaw/openclaw@b47bace test(codex): make sandbox cleanup proof deterministic fix(cron): route topic targets through channel plugins · openclaw/openclaw@9175491 fix(agents): simplify subagent completion handoff ci(release): isolate npm publish concurrency · openclaw/openclaw@9c26b87 ci(release): allow beta publish after npm preflight · openclaw/openclaw@0e37263 fix(release): allow large beta smoke run lists ci(release): retry child workflow polling · openclaw/openclaw@c689f71 ci(release): poll child workflows through actions api · openclaw/openclaw@e5dab55 fix(bootstrap): guard bootstrap name checks against undefined names (… · openclaw/openclaw@25fa46b fix(cli): waitForever must keep the event loop alive (#85694) · openclaw/openclaw@eca9645 fix(cli-output): ignore cumulative usage from result events in stream… · openclaw/openclaw@84229d9 fix(secrets): show irreversible warning after interactive apply confi… · openclaw/openclaw@bb52b54 fix(agents/harness): pass CLI runtime aliases through to PI in select… · openclaw/openclaw@2c3b7ea docs: add bug-sweep changelog entries fix(release): run npm shims on Windows · openclaw/openclaw@1f32a48 fix(ci): repair crabbox hydrate replay (#85706) · openclaw/openclaw@0d7d99b feat(auth): support named model login profiles fix(gateway): restore WebChat image understanding routing · openclaw/openclaw@55a0c9b fix(anthropic): migrate 1M context to GA handling fix(scripts): resolve Crabbox shims on Windows · openclaw/openclaw@5c535df test(ci): harden installer smoke coverage · openclaw/openclaw@68bcd4e fix(status): bound deep docker audit probes (#85476) · openclaw/openclaw@f7c05dc docs: expand meeting notes docs · openclaw/openclaw@a7e0fa0 fix(cli): validate tasks audit limit (#84901) · openclaw/openclaw@44d470f fix(twitch): preserve newer message handler during cleanup (#85425) · openclaw/openclaw@71ddc01 fix(ci): require live docker credentials by resource · openclaw/openclaw@1e21121 fix(diagnostics): drop snake case otel ids (#72645) · openclaw/openclaw@e0bafc5 fix(ci): clear signal and docs guard blockers (#85693) · openclaw/openclaw@3a1d4dd fix(ci): require factory auth for droid live docker · openclaw/openclaw@cc6c372 fix: honor disabled synthetic auth lookup · openclaw/openclaw@a4a1abb fix(scripts): repair live docker auth shellcheck · openclaw/openclaw@4e34ac4 fix(docker): avoid printing gateway token · openclaw/openclaw@5db773f fix(cli): keep logs follow on live gateway state · openclaw/openclaw@6e3b318 fix(e2e): prefer x64 MinGit on Windows · openclaw/openclaw@15d9134 feat: add meeting notes plugin docs: update changelog for memory artifacts (#85060) (thanks @brokema… · openclaw/openclaw@9e55383 fix(memory-lancedb): expose public memory artifacts · openclaw/openclaw@aac1abe fix(memory): preserve sidecar capability hooks · openclaw/openclaw@e6288ca fix(e2e): scrub Windows update config on PowerShell 5.1 · openclaw/openclaw@6657b49 docs: absorb documentation PR sweep · openclaw/openclaw@2c536a8 fix(agents): stabilize Linux fallback tests · openclaw/openclaw@6b04170 fix(codex): preserve native web search action metadata (#85378) · openclaw/openclaw@bcf756c test: refresh Codex prompt snapshots · openclaw/openclaw@492d656 docs: absorb hook and subagent guidance PRs fix(agents): audit tool policy blocks (#85673) · openclaw/openclaw@09dd051 ci: fix plugin npm bundled dependency install · openclaw/openclaw@d485464 feat(diagnostics): classify skill and tool usage (#80370) docs(skills): clarify control ui recording proof (#85568) · openclaw/openclaw@0b476b9 test(agents): repair main failure fixtures · openclaw/openclaw@c29967b feat(diagnostics): trace gateway secret preparation (#83019) · openclaw/openclaw@4f0c902 fix(diagnostics): harden observability exports and smokes (#85371) · openclaw/openclaw@7f05be0 fix(stepfun): drop stale auth choice metadata · openclaw/openclaw@0b2ab6c test(e2e): sample kitchen sink RSS on Windows · openclaw/openclaw@73c1e37 test(plugins): fail gauntlet on load diagnostics fix(build): preserve tsdown heap floor · openclaw/openclaw@9ff1a43 fix(tools): honor config apiKey in media tool preflight (#85570) · openclaw/openclaw@31c269f fix(e2e): support macOS script wrappers fix(ci): scope changed shrinkwrap checks · openclaw/openclaw@743fd4c chore(ui): refresh fa control ui locale chore(ui): refresh nl control ui locale · openclaw/openclaw@908464b chore(ui): refresh vi control ui locale · openclaw/openclaw@62b75f4 chore(ui): refresh th control ui locale · openclaw/openclaw@fc4ba31 chore(ui): refresh id control ui locale · openclaw/openclaw@5b1bdd1 chore(ui): refresh pl control ui locale · openclaw/openclaw@534d4b1 chore(ui): refresh uk control ui locale · openclaw/openclaw@055c3bd chore(ui): refresh ar control ui locale · openclaw/openclaw@89c5a68 chore(ui): refresh it control ui locale · openclaw/openclaw@44ca805 chore(ui): refresh tr control ui locale chore(ui): refresh fr control ui locale · openclaw/openclaw@2240b0e chore(ui): refresh ko control ui locale · openclaw/openclaw@5fa250b chore(ui): refresh ja-JP control ui locale · openclaw/openclaw@f4ea401 chore(ui): refresh es control ui locale · openclaw/openclaw@751dde0 chore(ui): refresh de control ui locale · openclaw/openclaw@72a9b5b chore(ui): refresh pt-BR control ui locale · openclaw/openclaw@501b6e0 chore(ui): refresh zh-CN control ui locale chore(ui): refresh zh-TW control ui locale · openclaw/openclaw@88dee79 Add Slack approval QA checkpoints (#85141) fix(agents): mirror internal ui message tool replies (#85564) perf(whatsapp): narrow runtime setter entry (#85589) · openclaw/openclaw@463929d fix: smooth chat focus mode layout · openclaw/openclaw@bb5abef Fix Telegram missing harness spool poison (#85605) · openclaw/openclaw@b745082 fix(session): surface previous-transcript archive failures on /new ro… · openclaw/openclaw@679a46d Policy: add agent workspace conformance checks (#85096) · openclaw/openclaw@a94f344 fix(installer): fail failed Windows git builds · openclaw/openclaw@2edd6e2 fix(codex): honor node exec policy for native surfaces (#85534) · openclaw/openclaw@e0405ec fix(qa-lab): stabilize codex runtime parity fixtures fix(github): preserve sufficient proof against negative relabel (#85567) · openclaw/openclaw@6b52dff docs: add changelog for context pressure preflight · openclaw/openclaw@5ca734f Fix context pressure preflight for tool-heavy sessions (#85541) · openclaw/openclaw@c08400e fix(doctor): classify codex asset notice as info (#85119) · openclaw/openclaw@959b935 fix(ui): keep chat picker search current (#85547) fix: preserve message-tool delivery evidence · openclaw/openclaw@f022b05 fix(codex): add API key paste auth (#85533) fix: satisfy openai video test typecheck · openclaw/openclaw@743caed fix: route openai video edits to edits endpoint · openclaw/openclaw@6c3fcb8
test(codex): avoid forced-tool turn flake · openclaw/openclaw@8d1ab83
steipete · 2026-05-23 · via Recent Commits to openclaw:main

@@ -526,6 +526,35 @@ function createNamedDynamicTool(

526526

};

527527

}

528528529+

async function buildDynamicToolsForTest(

530+

params: EmbeddedRunAttemptParams,

531+

workspaceDir: string,

532+

options: Partial<

533+

Pick<

534+

Parameters<typeof testing.buildDynamicTools>[0],

535+

"forceHeartbeatTool" | "ignoreRuntimePlan"

536+

>

537+

> = {},

538+

) {

539+

const sandboxSessionKey = params.sessionKey;

540+

if (!sandboxSessionKey) {

541+

throw new Error("createParams must provide a sessionKey for Codex dynamic tool tests.");

542+

}

543+

return testing.buildDynamicTools({

544+

params,

545+

resolvedWorkspace: workspaceDir,

546+

effectiveWorkspace: workspaceDir,

547+

sandboxSessionKey,

548+

sandbox: { enabled: false, backendId: "docker" } as never,

549+

nativeToolSurfaceEnabled: true,

550+

runAbortController: new AbortController(),

551+

sessionAgentId: "main",

552+

pluginConfig: {},

553+

onYieldDetected: () => undefined,

554+

...options,

555+

});

556+

}

557+529558

type RuntimeDynamicToolForTest = Parameters<

530559

typeof createCodexDynamicToolBridge

531560

>[0]["tools"][number];

@@ -2236,34 +2265,19 @@ describe("runCodexAppServerAttempt", () => {

22362265

createRuntimeDynamicTool("message"),

22372266

createRuntimeDynamicTool("music_generate"),

22382267

]);

2239-

const harness = createStartedThreadHarness(async (method) => {

2240-

if (method === "turn/start") {

2241-

await new Promise((resolve) => setTimeout(resolve, 5));

2242-

return turnStartResult();

2243-

}

2244-

return undefined;

2245-

});

2246-

const params = createParams(

2247-

path.join(tempDir, "session.jsonl"),

2248-

path.join(tempDir, "workspace"),

2249-

);

2268+

const workspaceDir = path.join(tempDir, "workspace");

2269+

const params = createParams(path.join(tempDir, "session.jsonl"), workspaceDir);

22502270

params.disableTools = false;

22512271

params.runtimePlan = createCodexRuntimePlanFixture();

22522272

params.sourceReplyDeliveryMode = "message_tool_only";

22532273

params.toolsAllow = ["music_generate"];

225422742255-

const run = runCodexAppServerAttempt(params, {

2256-

pluginConfig: { appServer: { mode: "yolo" } },

2257-

});

2258-

await harness.waitForMethod("turn/start", 120_000);

2259-

await harness.completeTurn({ threadId: "thread-1", turnId: "turn-1" });

2260-

await run;

2261-2262-

const startRequest = harness.requests.find((entry) => entry.method === "thread/start");

2263-

const dynamicToolNames =

2264-

(

2265-

startRequest?.params as { dynamicTools?: Array<{ name?: string }> } | undefined

2266-

)?.dynamicTools?.map((tool) => tool.name) ?? [];

2275+

const dynamicToolNames = (

2276+

await buildDynamicToolsForTest(params, workspaceDir, {

2277+

forceHeartbeatTool: true,

2278+

ignoreRuntimePlan: true,

2279+

})

2280+

).map((tool) => tool.name);

2267228122682282

expect(dynamicToolNames).toContain("message");

22692283

expect(dynamicToolNames).toContain("music_generate");

@@ -2277,28 +2291,19 @@ describe("runCodexAppServerAttempt", () => {

22772291

? [createRuntimeDynamicTool("heartbeat_respond")]

22782292

: []),

22792293

]);

2280-

const harness = createStartedThreadHarness();

2281-

const params = createParams(

2282-

path.join(tempDir, "session.jsonl"),

2283-

path.join(tempDir, "workspace"),

2284-

);

2294+

const workspaceDir = path.join(tempDir, "workspace");

2295+

const params = createParams(path.join(tempDir, "session.jsonl"), workspaceDir);

22852296

params.disableTools = false;

22862297

params.runtimePlan = createCodexRuntimePlanFixture();

22872298

params.sourceReplyDeliveryMode = "message_tool_only";

22882299

params.toolsAllow = [];

228923002290-

const run = runCodexAppServerAttempt(params, {

2291-

pluginConfig: { appServer: { mode: "yolo" } },

2292-

});

2293-

await harness.waitForMethod("turn/start", 120_000);

2294-

await harness.completeTurn({ threadId: "thread-1", turnId: "turn-1" });

2295-

await run;

2296-2297-

const startRequest = harness.requests.find((entry) => entry.method === "thread/start");

2298-

const dynamicToolNames =

2299-

(

2300-

startRequest?.params as { dynamicTools?: Array<{ name?: string }> } | undefined

2301-

)?.dynamicTools?.map((tool) => tool.name) ?? [];

2301+

const dynamicToolNames = (

2302+

await buildDynamicToolsForTest(params, workspaceDir, {

2303+

forceHeartbeatTool: true,

2304+

ignoreRuntimePlan: true,

2305+

})

2306+

).map((tool) => tool.name);

2302230723032308

expect(dynamicToolNames).toEqual(["message"]);

23042309

});

@@ -2310,27 +2315,18 @@ describe("runCodexAppServerAttempt", () => {

23102315

? [createRuntimeDynamicTool("heartbeat_respond")]

23112316

: []),

23122317

]);

2313-

const harness = createStartedThreadHarness();

2314-

const params = createParams(

2315-

path.join(tempDir, "session.jsonl"),

2316-

path.join(tempDir, "workspace"),

2317-

);

2318+

const workspaceDir = path.join(tempDir, "workspace");

2319+

const params = createParams(path.join(tempDir, "session.jsonl"), workspaceDir);

23182320

params.disableTools = false;

23192321

params.runtimePlan = createCodexRuntimePlanFixture();

23202322

params.toolsAllow = ["message"];

232123232322-

const run = runCodexAppServerAttempt(params, {

2323-

pluginConfig: { appServer: { mode: "yolo" } },

2324-

});

2325-

await harness.waitForMethod("turn/start", 120_000);

2326-

await harness.completeTurn({ threadId: "thread-1", turnId: "turn-1" });

2327-

await run;

2328-2329-

const startRequest = harness.requests.find((entry) => entry.method === "thread/start");

2330-

const dynamicToolNames =

2331-

(

2332-

startRequest?.params as { dynamicTools?: Array<{ name?: string }> } | undefined

2333-

)?.dynamicTools?.map((tool) => tool.name) ?? [];

2324+

const dynamicToolNames = (

2325+

await buildDynamicToolsForTest(params, workspaceDir, {

2326+

forceHeartbeatTool: true,

2327+

ignoreRuntimePlan: true,

2328+

})

2329+

).map((tool) => tool.name);

2334233023352331

expect(dynamicToolNames).toEqual(["message"]);

23362332

});

@@ -2343,33 +2339,50 @@ describe("runCodexAppServerAttempt", () => {

23432339

createRuntimeDynamicTool("sessions_spawn"),

23442340

createRuntimeDynamicTool("sessions_yield"),

23452341

]);

2346-

const harness = createStartedThreadHarness();

2347-

const params = createParams(

2348-

path.join(tempDir, "session.jsonl"),

2349-

path.join(tempDir, "workspace"),

2350-

);

2342+

const workspaceDir = path.join(tempDir, "workspace");

2343+

const params = createParams(path.join(tempDir, "session.jsonl"), workspaceDir);

23512344

params.disableTools = false;

23522345

params.runtimePlan = createCodexRuntimePlanFixture();

23532346

params.sourceReplyDeliveryMode = "message_tool_only";

2347+

const dynamicTools = await buildDynamicToolsForTest(params, workspaceDir, {

2348+

forceHeartbeatTool: true,

2349+

ignoreRuntimePlan: true,

2350+

});

2351+

const toolBridge = createCodexDynamicToolBridge({

2352+

tools: dynamicTools,

2353+

signal: new AbortController().signal,

2354+

directToolNames: ["message"],

2355+

});

2356+

const request = vi.fn(async (method: string) => {

2357+

if (method === "thread/start") {

2358+

return threadStartResult();

2359+

}

2360+

throw new Error(`unexpected method: ${method}`);

2361+

});

235423622355-

const run = runCodexAppServerAttempt(params, {

2356-

pluginConfig: { appServer: { mode: "yolo", codeModeOnly: true } },

2363+

await startOrResumeThread({

2364+

client: { request } as never,

2365+

params,

2366+

cwd: workspaceDir,

2367+

dynamicTools: toolBridge.specs,

2368+

appServer: { ...createThreadLifecycleAppServerOptions(), codeModeOnly: true },

2369+

nativeCodeModeEnabled: true,

2370+

nativeCodeModeOnlyEnabled: true,

2371+

userMcpServersEnabled: true,

2372+

environmentSelection: [],

23572373

});

2358-

await harness.waitForMethod("turn/start", 120_000);

2359-

await harness.completeTurn({ threadId: "thread-1", turnId: "turn-1" });

2360-

await run;

236123742362-

const startRequest = harness.requests.find((entry) => entry.method === "thread/start");

2363-

const dynamicTools =

2364-

(startRequest?.params as { dynamicTools?: Array<Record<string, unknown>> } | undefined)

2375+

const startRequest = request.mock.calls.find(([method]) => method === "thread/start");

2376+

const startDynamicTools =

2377+

(startRequest?.[1] as { dynamicTools?: Array<Record<string, unknown>> } | undefined)

23652378

?.dynamicTools ?? [];

2366-

const startConfig = (startRequest?.params as { config?: Record<string, unknown> } | undefined)

2379+

const startConfig = (startRequest?.[1] as { config?: Record<string, unknown> } | undefined)

23672380

?.config;

2368-

const message = dynamicTools.find((tool) => tool.name === "message");

2369-

const webSearch = dynamicTools.find((tool) => tool.name === "web_search");

2370-

const heartbeat = dynamicTools.find((tool) => tool.name === "heartbeat_respond");

2371-

const sessionsSpawn = dynamicTools.find((tool) => tool.name === "sessions_spawn");

2372-

const sessionsYield = dynamicTools.find((tool) => tool.name === "sessions_yield");

2381+

const message = startDynamicTools.find((tool) => tool.name === "message");

2382+

const webSearch = startDynamicTools.find((tool) => tool.name === "web_search");

2383+

const heartbeat = startDynamicTools.find((tool) => tool.name === "heartbeat_respond");

2384+

const sessionsSpawn = startDynamicTools.find((tool) => tool.name === "sessions_spawn");

2385+

const sessionsYield = startDynamicTools.find((tool) => tool.name === "sessions_yield");

2373238623742387

expect(message).not.toHaveProperty("namespace");

23752388

expect(message).not.toHaveProperty("deferLoading");