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

推荐订阅源

E
Exploit-DB.com RSS Feed
Last Week in AI
Last Week in AI
月光博客
月光博客
博客园 - 三生石上(FineUI控件)
爱范儿
爱范儿
奇客Solidot–传递最新科技情报
奇客Solidot–传递最新科技情报
罗磊的独立博客
S
SegmentFault 最新的问题
Jina AI
Jina AI
V
V2EX
博客园 - Franky
OSCHINA 社区最新新闻
OSCHINA 社区最新新闻
WordPress大学
WordPress大学
博客园 - 叶小钗
大猫的无限游戏
大猫的无限游戏
让小产品的独立变现更简单 - ezindie.com
让小产品的独立变现更简单 - ezindie.com
博客园_首页
P
Proofpoint News Feed
Recorded Future
Recorded Future
G
GRAHAM CLULEY
T
Tailwind CSS Blog
李成银的技术随笔
钛媒体:引领未来商业与生活新知
钛媒体:引领未来商业与生活新知
Hugging Face - Blog
Hugging Face - Blog
Apple Machine Learning Research
Apple Machine Learning Research
IT之家
IT之家
Latest news
Latest news
Recent Announcements
Recent Announcements
酷 壳 – CoolShell
酷 壳 – CoolShell
freeCodeCamp Programming Tutorials: Python, JavaScript, Git & More
博客园 - 【当耐特】
CTFtime.org: upcoming CTF events
CTFtime.org: upcoming CTF events
宝玉的分享
宝玉的分享
P
Privacy International News Feed
Scott Helme
Scott Helme
Cyberwarzone
Cyberwarzone
N
News and Events Feed by Topic
小众软件
小众软件
Stack Overflow Blog
Stack Overflow Blog
D
Darknet – Hacking Tools, Hacker News & Cyber Security
C
CXSECURITY Database RSS Feed - CXSecurity.com
F
Full Disclosure
Blog — PlanetScale
Blog — PlanetScale
P
Proofpoint News Feed
G
Google Developers Blog
博客园 - 聂微东
cs.AI updates on arXiv.org
cs.AI updates on arXiv.org
The Cloudflare Blog
T
ThreatConnect
C
Cybersecurity and Infrastructure Security Agency CISA

Recent Commits to openclaw:main

fix(scripts): harden Windows generated formatting fix(test): focus plugin binding Docker smoke 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(docker): copy prepare hook before install · openclaw/openclaw@0ba6b23 fix: share signed thinking replay policy fix(memory): strip invalid thinking signatures for signed-thinking pr… · openclaw/openclaw@41329c0 fix: preserve signed thinking tool ids · openclaw/openclaw@906476a fix(scripts): harden Windows install checks · openclaw/openclaw@d21abb8 fix(release): keep private QA markers out of bundled alias code · openclaw/openclaw@b972ac1 fix(discord): harden realtime voice wake joins · openclaw/openclaw@fdfcb07 test(parallels): harden release VM smoke isolation · openclaw/openclaw@3839b48 fix: keep blank agent allowlists fail closed (#85849) · openclaw/openclaw@0f83c93 fix: harden session allowlist glob matching (#85849) (thanks @SebTardif) · openclaw/openclaw@88aa713 fix(security): replace regex wildcard matching with linear-time glob … fix(telegram): normalize legacy action targets · openclaw/openclaw@32631eb fix: notify chat when main session recovery fails · openclaw/openclaw@cf61b87 docs(changelog): note restart recovery notice · openclaw/openclaw@ae9308b fix(minimax): normalize OAuth token expiry to absolute millisecond ti… fix(memory-wiki): show vault totals in palace summary (#85824) Isolate boot-md startup sessions (#85919) · openclaw/openclaw@8deb1ef Guard OpenAI image compression for PNG outputs (#85776) · openclaw/openclaw@d075111 fix(agents): clamp proxy completions caps after payload shaping (#85889) · openclaw/openclaw@1d1a7c2 fix: gate discord realtime voice by wake name (#85915) · openclaw/openclaw@17dcdea
fix(mcp): bound tools/list during catalog discovery (#85063) · openclaw/openclaw@07f500a
nxmxbbd · 2026-05-25 · via Recent Commits to openclaw:main

@@ -1,4 +1,8 @@

1+

import fs from "node:fs/promises";

2+

import os from "node:os";

3+

import path from "node:path";

14

import { afterEach, describe, expect, it, vi } from "vitest";

5+

import { writeExecutable } from "./bundle-mcp-shared.test-harness.js";

26

import { createBundleMcpJsonSchemaValidator } from "./pi-bundle-mcp-runtime.js";

37

import { cleanupBundleMcpHarness } from "./pi-bundle-mcp-test-harness.js";

48

import {

@@ -21,6 +25,129 @@ type RuntimeFactoryOptions = NonNullable<

2125

Parameters<typeof testing.createSessionMcpRuntimeManager>[0]

2226

>;

2327

type RuntimeFactory = NonNullable<RuntimeFactoryOptions["createRuntime"]>;

28+

const LIST_TOOLS_SERVER_LOG_TIMEOUT_MS = 2_000;

29+

const LIST_TOOLS_TEST_DEADLINE_MS = 4_000;

30+31+

async function writeListToolsMcpServer(params: {

32+

filePath: string;

33+

logPath: string;

34+

delayMs?: number;

35+

hang?: boolean;

36+

}): Promise<void> {

37+

await writeExecutable(

38+

params.filePath,

39+

`#!/usr/bin/env node

40+

import fs from "node:fs/promises";

41+42+

const logPath = ${JSON.stringify(params.logPath)};

43+

const delayMs = ${params.delayMs ?? 0};

44+

const hang = ${params.hang === true};

45+46+

let buffer = "";

47+

let pendingTimer;

48+

let keepAlive;

49+

function log(line) {

50+

void fs.appendFile(logPath, line + "\\n", "utf8").catch(() => {});

51+

}

52+

function send(message) {

53+

process.stdout.write(JSON.stringify(message) + "\\n");

54+

}

55+

function handle(message) {

56+

if (!message || typeof message !== "object") {

57+

return;

58+

}

59+

log("recv " + String(message.method ?? "unknown"));

60+

if (message.method === "initialize") {

61+

send({

62+

jsonrpc: "2.0",

63+

id: message.id,

64+

result: {

65+

protocolVersion: message.params?.protocolVersion ?? "2025-03-26",

66+

capabilities: { tools: {} },

67+

serverInfo: { name: "test-list-tools", version: "1.0.0" },

68+

},

69+

});

70+

return;

71+

}

72+

if (message.method === "notifications/initialized") {

73+

return;

74+

}

75+

if (message.method === "tools/list") {

76+

if (hang) {

77+

log("hang tools/list");

78+

keepAlive = setInterval(() => {}, 1000);

79+

return;

80+

}

81+

log("delay tools/list " + delayMs);

82+

pendingTimer = setTimeout(() => {

83+

send({

84+

jsonrpc: "2.0",

85+

id: message.id,

86+

result: {

87+

tools: [

88+

{

89+

name: "slow_tool",

90+

description: "Returned after a slow catalog response.",

91+

inputSchema: { type: "object", properties: {} },

92+

},

93+

],

94+

},

95+

});

96+

}, delayMs);

97+

}

98+

}

99+

process.stdin.setEncoding("utf8");

100+

function shutdown() {

101+

if (pendingTimer) {

102+

clearTimeout(pendingTimer);

103+

}

104+

if (keepAlive) {

105+

clearInterval(keepAlive);

106+

}

107+

process.exit(0);

108+

}

109+

process.stdin.on("data", (chunk) => {

110+

buffer += chunk;

111+

while (true) {

112+

const newline = buffer.indexOf("\\n");

113+

if (newline < 0) {

114+

return;

115+

}

116+

const line = buffer.slice(0, newline).replace(/\\r$/, "");

117+

buffer = buffer.slice(newline + 1);

118+

if (line.trim()) {

119+

handle(JSON.parse(line));

120+

}

121+

}

122+

});

123+

process.stdin.on("end", shutdown);

124+

process.on("SIGTERM", shutdown);

125+

process.on("SIGINT", shutdown);`,

126+

);

127+

}

128+129+

async function waitForFileText(

130+

filePath: string,

131+

expectedText: string,

132+

timeoutMs: number,

133+

): Promise<void> {

134+

const deadline = Date.now() + timeoutMs;

135+

let lastText = "";

136+

while (Date.now() < deadline) {

137+

try {

138+

lastText = await fs.readFile(filePath, "utf8");

139+

if (lastText.includes(expectedText)) {

140+

return;

141+

}

142+

} catch {

143+

// The server may not have written the log file yet.

144+

}

145+

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

146+

}

147+

throw new Error(

148+

`Timed out waiting for ${expectedText} in ${filePath}; saw ${JSON.stringify(lastText)}`,

149+

);

150+

}

2415125152

function makeRuntime(

26153

tools: Array<{ toolName: string; description: string }>,

@@ -181,6 +308,95 @@ describe("session MCP runtime", () => {

181308

expect(activeLeases).toBe(0);

182309

});

183310311+

it("keeps MCP tools/list responses that exceed the connection timeout but finish within the internal catalog timeout", async () => {

312+

const tempDir = await fs.mkdtemp(path.join(os.tmpdir(), "bundle-mcp-slow-listtools-"));

313+

const serverPath = path.join(tempDir, "slow-list-tools.mjs");

314+

const logPath = path.join(tempDir, "server.log");

315+

await writeListToolsMcpServer({

316+

filePath: serverPath,

317+

logPath,

318+

delayMs: 750,

319+

});

320+321+

const runtime = await getOrCreateSessionMcpRuntime({

322+

sessionId: "session-slow-listtools-server-timeout",

323+

sessionKey: "agent:test:session-slow-listtools-server-timeout",

324+

workspaceDir: "/workspace",

325+

cfg: {

326+

mcp: {

327+

servers: {

328+

slowListTools: {

329+

command: process.execPath,

330+

args: [serverPath],

331+

connectionTimeoutMs: 500,

332+

},

333+

},

334+

},

335+

},

336+

});

337+338+

try {

339+

const catalog = await runtime.getCatalog();

340+341+

expect(catalog.tools.map((tool) => tool.toolName)).toEqual(["slow_tool"]);

342+

expect(catalog.servers.slowListTools).toMatchObject({

343+

serverName: "slowListTools",

344+

toolCount: 1,

345+

});

346+

await expect(fs.readFile(logPath, "utf8")).resolves.toContain("delay tools/list 750");

347+

} finally {

348+

await runtime.dispose();

349+

await fs.rm(tempDir, { recursive: true, force: true });

350+

}

351+

});

352+353+

it("times out default-config hung bundle MCP tools/list using the internal catalog timeout", async () => {

354+

const tempDir = await fs.mkdtemp(path.join(os.tmpdir(), "bundle-mcp-listtools-timeout-"));

355+

const serverPath = path.join(tempDir, "hanging-list-tools.mjs");

356+

const logPath = path.join(tempDir, "server.log");

357+

await writeListToolsMcpServer({ filePath: serverPath, logPath, hang: true });

358+359+

const runtime = await getOrCreateSessionMcpRuntime({

360+

sessionId: "session-listtools-server-timeout",

361+

sessionKey: "agent:test:session-listtools-server-timeout",

362+

workspaceDir: "/workspace",

363+

cfg: {

364+

mcp: {

365+

servers: {

366+

hangingListTools: {

367+

command: process.execPath,

368+

args: [serverPath],

369+

},

370+

},

371+

},

372+

},

373+

});

374+

const catalogResult = runtime.getCatalog().then(

375+

(catalog) => ({ status: "resolved" as const, catalog }),

376+

(error: unknown) => ({ status: "rejected" as const, error }),

377+

);

378+379+

try {

380+

await waitForFileText(logPath, "recv tools/list", LIST_TOOLS_SERVER_LOG_TIMEOUT_MS);

381+

const result = await Promise.race([

382+

catalogResult,

383+

new Promise<{ status: "pending" }>((resolve) => {

384+

setTimeout(() => resolve({ status: "pending" }), LIST_TOOLS_TEST_DEADLINE_MS);

385+

}),

386+

]);

387+388+

expect(result.status).toBe("resolved");

389+

if (result.status === "resolved") {

390+

expect(result.catalog.tools).toEqual([]);

391+

expect(result.catalog.servers).toEqual({});

392+

}

393+

} finally {

394+

await runtime.dispose();

395+

await Promise.race([catalogResult, new Promise((resolve) => setTimeout(resolve, 1000))]);

396+

await fs.rm(tempDir, { recursive: true, force: true });

397+

}

398+

});

399+184400

it("reuses repeated materialization and recreates after explicit disposal", async () => {

185401

const created: SessionMcpRuntime[] = [];

186402

const disposed: string[] = [];