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

推荐订阅源

F
Full Disclosure
Recorded Future
Recorded Future
T
Tenable Blog
S
Securelist
C
CERT Recently Published Vulnerability Notes
T
Threatpost
S
Schneier on Security
A
Arctic Wolf
The Hacker News
The Hacker News
C
CXSECURITY Database RSS Feed - CXSecurity.com
Know Your Adversary
Know Your Adversary
P
Privacy International News Feed
Threat Intelligence Blog | Flashpoint
Threat Intelligence Blog | Flashpoint
The Register - Security
The Register - Security
Cisco Talos Blog
Cisco Talos Blog
AWS News Blog
AWS News Blog
K
Kaspersky official blog
T
True Tiger Recordings
T
Threat Research - Cisco Blogs
V
Vulnerabilities – Threatpost
P
Palo Alto Networks Blog
T
The Exploit Database - CXSecurity.com
小众软件
小众软件
B
Blog
Cyber Security Advisories - MS-ISAC
Cyber Security Advisories - MS-ISAC
Microsoft Azure Blog
Microsoft Azure Blog
Cyberwarzone
Cyberwarzone
C
Cybersecurity and Infrastructure Security Agency CISA
T
Tor Project blog
Spread Privacy
Spread Privacy
Malwarebytes
Malwarebytes
P
Proofpoint News Feed
F
Fox-IT International blog
F
Fortinet All Blogs
P
Privacy & Cybersecurity Law Blog
G
GRAHAM CLULEY
量子位
Latest news
Latest news
OSCHINA 社区最新新闻
OSCHINA 社区最新新闻
博客园 - 叶小钗
Project Zero
Project Zero
T
Tailwind CSS Blog
N
Netflix TechBlog - Medium
Martin Fowler
Martin Fowler
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
I
Intezer
博客园_首页
腾讯CDC
H
Hackread – Cybersecurity News, Data Breaches, AI and More
D
Darknet – Hacking Tools, Hacker News & Cyber Security

Recent Commits to openclaw:main

fix(lock): require owner identity proof before stale removal fix(deps): pin shrinkwrap patch drift to pnpm lock fix(e2e): bound docker package preparation fix(cli): add Windows stack-size respawn (#87031) fix(agents): preserve sessions_spawn transcript payloads (#82203) · openclaw/openclaw@ef86d8c fix(agents): guard duplicate tool display metadata (#87025) · openclaw/openclaw@60bec8c ci(release): port 2026.5.25 release gate fixes · openclaw/openclaw@f7e2d9b chore: update tool display snapshot · openclaw/openclaw@ad71c42 fix(web-search): keep runtime legacy merge out of validation (#86818) · openclaw/openclaw@4a85cd7 fix(cli): default logs to local timestamps (#85387) · openclaw/openclaw@3127808 fix(agents): dedupe transcripts tool display config · openclaw/openclaw@8788ae1 fix(updater): exclude prerelease tags from stable git channel (#86559) · openclaw/openclaw@e070519 fix(agents): memoize session lock owner args · openclaw/openclaw@c430fcd fix: dedupe transcripts tool display metadata · openclaw/openclaw@0f49bbb fix(cli): validate timeout and banner TTY state · openclaw/openclaw@abb85cc fix(codex): project newer history on app-server resume (#86677) fix(codex): keep turn timeouts inside Codex (#86476) fix(auto-reply): stage sandboxed workspace media · openclaw/openclaw@f22c3a5 fix(e2e): support plain telegram install timeouts fix(mac): harden restart and dSYM packaging · openclaw/openclaw@639e7ff fix(exec): avoid default approval store writes (#86964) · openclaw/openclaw@4d65936 fix(agents): restore current guard checks (#86934) · openclaw/openclaw@9b1b6d0 docs(changelog): prepare 2026.5.26 notes · openclaw/openclaw@983b338 fix(commands): keep slash handling off reply startup · openclaw/openclaw@c2b56de fix(reply): defer context compaction safely · openclaw/openclaw@ed3ae0d fix(telegram): refine typing and progress drafts · openclaw/openclaw@0afccc6 fix(codex): gate profiler timing and startup setup · openclaw/openclaw@21c25bb fix(agents): avoid runtime model hydration on hot paths · openclaw/openclaw@7951cc0 fix(reply): reduce visible reply delivery latency · openclaw/openclaw@699c047 docs(changelog): note reply latency fixes · openclaw/openclaw@29a1dc2 fix(e2e): support plain timeout wrappers fix(channels): preserve direct native progress callbacks · openclaw/openclaw@e750041 fix: tighten parser edge cases (#86999) · openclaw/openclaw@174cd49 fix(e2e): clean stale docker lane containers · openclaw/openclaw@3968288 fix(e2e): bound docker lifecycle hangs · openclaw/openclaw@71cb607 fix(gateway): bound live agent model probes fix(e2e): bound plugin binding docker smoke · openclaw/openclaw@b36fa1d fix(e2e): preserve docker run failure status feat(discord): bucket large model picker menus fix(telegram): derive DM topics from bot capability · openclaw/openclaw@aa117ec fix: improve discord voice playback and wake replies fix(e2e): kill timed kitchen rpc command groups · openclaw/openclaw@23aeb58 ci: use supported codex mini live target · openclaw/openclaw@b56ddcc fix(ci): kill wedged bun smoke commands fix(e2e): bound corrupt plugin update runs · openclaw/openclaw@2b63eb2 ci: require codex profiles for live probes · openclaw/openclaw@6930538 docs: clarify inline comment guidance docs: update changelog for landed sweep fixes · openclaw/openclaw@8c575bd fix(agents): disclose scoped session list results (#86944) · openclaw/openclaw@598aad4 fix(telegram): treat ENETDOWN as transient network failure (#86762) · openclaw/openclaw@1fd8de8 fix(mac): harden package script safety · openclaw/openclaw@564e0bb fix(ci): kill wedged checkout fetches · openclaw/openclaw@c867ecb ci: restore codex replay live probe contract · openclaw/openclaw@9fd8158 fix(codex): preserve oversized native thread reuse · openclaw/openclaw@7a14741 fix(scripts): detect timed changed gates · openclaw/openclaw@a5eee8f ci: stop waiting for nonexistent capability restart wake · openclaw/openclaw@3c6fd49 fix(e2e): route plugin update through timeout helper · openclaw/openclaw@e8f584e fix: respect root options in startup guards (#86927) · openclaw/openclaw@7e6837b fix: tighten CLI utility failure handling (#86918) · openclaw/openclaw@0ec2928 fix: preserve config and hook contracts (#86911) fix: tighten small runtime parsing guards (#86909) fix(ci): kill timed tui pty test runs · openclaw/openclaw@081e295 fix(logging): preserve env placeholders during redaction · openclaw/openclaw@6c18c21 fix(memory-core): avoid per-file watcher FD fan-out for memory direct… fix: use current config sdk contract in feishu doctor · openclaw/openclaw@5535eef Validate wide-area DNS zone domains [AI] (#84136) · openclaw/openclaw@84b9704 ci: stabilize release live QA gates · openclaw/openclaw@27359ec ci: harden live release gates · openclaw/openclaw@cf21c8a ci: normalize Windows toolcache Node paths perf: cache read-only channel resolution chore: remove stale codex test conversion · openclaw/openclaw@fdb7848 fix(test): explain missing vitest dependency · openclaw/openclaw@373b3bf fix(e2e): kill timed docker scenario runners · openclaw/openclaw@d5bf325 fix: add transcripts tool display metadata · openclaw/openclaw@645cbf6 docs: update changelog for landed fixes · openclaw/openclaw@12b81d8 fix(agents): route btw through embedded stream resolver (#86312) fix(telegram): treat targeted bot commands as mentions (#86553) · openclaw/openclaw@c7821bd fix(e2e): route doctor switch commands through timeout helper · openclaw/openclaw@9ced76a fix: ignore other codex thread completions · openclaw/openclaw@ead847f fix(feishu): repair stale channel state · openclaw/openclaw@b7c461a fix: remove stale image provider assertions · openclaw/openclaw@0973a7e fix: accept trailing fuzzy voice wake questions · openclaw/openclaw@d001d35 fix(podman): bound setup image builds · openclaw/openclaw@d6fcb56 fix(podman): kill timed container launches · openclaw/openclaw@6118f3f fix(scripts): preserve native pnpm exec paths · openclaw/openclaw@fb853de fix(codex): bridge cli api-key auth into app-server fix: update Discord voice to libopus-wasm 0.1.0 · openclaw/openclaw@0f605ee fix(e2e): kill timed docker helper commands · openclaw/openclaw@e89afa6 fix(e2e): kill timed live docker runs · openclaw/openclaw@dc0d4c2 fix(ci): kill timed website installer docker steps · openclaw/openclaw@d54c906 fix(scripts): trim macOS node bootstrap · openclaw/openclaw@4ff5a61 fix(ci): kill timed install smoke docker steps · openclaw/openclaw@cf6f9ad fix(setup): kill timed image pulls when supported fix(test): default Vitest stall watchdog fix(ci): bound crabbox hydrate downloads · openclaw/openclaw@d1c8f09 fix(control-ui): guard stale overview usage refresh fix(qa): require genai otel model spans (#86920) refactor: move transcripts into core fix(message-tool): hydrate structured reply attachments fix(e2e): kill timed npm install process groups
revert: 60bec8c duplicate tool display guard · openclaw/openclaw@e09f89d
joshavant · 2026-05-27 · via Recent Commits to openclaw:main

@@ -1,12 +1,10 @@

11

import fs from "node:fs";

22

import path from "node:path";

33

import { fileURLToPath } from "node:url";

4-

import ts from "typescript";

54

import { TOOL_DISPLAY_CONFIG, type ToolDisplayConfig } from "../src/agents/tool-display-config.js";

6576

const scriptDir = path.dirname(fileURLToPath(import.meta.url));

87

const repoRoot = path.resolve(scriptDir, "..");

9-

const configPath = path.join(repoRoot, "src/agents/tool-display-config.ts");

108

const outputPath = path.join(

119

repoRoot,

1210

"apps/shared/OpenClawKit/Sources/OpenClawKit/Resources/tool-display.json",

@@ -18,63 +16,49 @@ const toolSources = [

1816

path.join(repoRoot, "src/auto-reply/reply/acp-projector.ts"),

1917

];

201821-

type DuplicateToolKey = {

22-

name: string;

23-

lines: number[];

24-

};

19+

const args = new Set(process.argv.slice(2));

20+

const shouldCheck = args.has("--check");

21+

const shouldWrite = args.has("--write");

252226-

export function main(argv = process.argv.slice(2)): number {

27-

const args = new Set(argv);

28-

const shouldCheck = args.has("--check");

29-

const shouldWrite = args.has("--write");

30-31-

if (!shouldCheck && !shouldWrite) {

32-

console.error("Usage: node --import tsx scripts/tool-display.ts --check|--write");

33-

return 1;

34-

}

35-36-

const duplicateErrors = collectToolDisplayDuplicateErrors({ includeSnapshot: shouldCheck });

37-

if (duplicateErrors.length > 0) {

38-

console.error(duplicateErrors.join("\n"));

39-

return 1;

40-

}

41-42-

const expected = serializeToolDisplayConfig();

43-

ensureCoreToolCoverage();

23+

if (!shouldCheck && !shouldWrite) {

24+

console.error("Usage: node --import tsx scripts/tool-display.ts --check|--write");

25+

process.exit(1);

26+

}

442745-

if (shouldWrite) {

46-

fs.mkdirSync(path.dirname(outputPath), { recursive: true });

47-

fs.writeFileSync(outputPath, expected);

48-

process.stdout.write(`wrote ${path.relative(repoRoot, outputPath)}\n`);

49-

return 0;

50-

}

28+

const expected = serializeToolDisplayConfig();

29+

ensureCoreToolCoverage();

513052-

if (!fs.existsSync(path.dirname(outputPath))) {

53-

process.stdout.write(

54-

`skip tool-display snapshot check; missing ${path.relative(repoRoot, path.dirname(outputPath))}\n`,

55-

);

56-

return 0;

57-

}

31+

if (shouldWrite) {

32+

fs.mkdirSync(path.dirname(outputPath), { recursive: true });

33+

fs.writeFileSync(outputPath, expected);

34+

process.stdout.write(`wrote ${path.relative(repoRoot, outputPath)}\n`);

35+

process.exit(0);

36+

}

583759-

if (!fs.existsSync(outputPath)) {

60-

console.error(

61-

`missing generated snapshot: ${path.relative(repoRoot, outputPath)}\nrun: pnpm tool-display:write`,

62-

);

63-

return 1;

64-

}

38+

if (!fs.existsSync(path.dirname(outputPath))) {

39+

process.stdout.write(

40+

`skip tool-display snapshot check; missing ${path.relative(repoRoot, path.dirname(outputPath))}\n`,

41+

);

42+

process.exit(0);

43+

}

654466-

const actual = fs.readFileSync(outputPath, "utf8");

67-

if (actual !== expected) {

68-

console.error(

69-

`tool-display snapshot is stale: ${path.relative(repoRoot, outputPath)}\nrun: pnpm tool-display:write`,

70-

);

71-

return 1;

72-

}

45+

if (!fs.existsSync(outputPath)) {

46+

console.error(

47+

`missing generated snapshot: ${path.relative(repoRoot, outputPath)}\nrun: pnpm tool-display:write`,

48+

);

49+

process.exit(1);

50+

}

735174-

process.stdout.write("tool-display snapshot is up to date\n");

75-

return 0;

52+

const actual = fs.readFileSync(outputPath, "utf8");

53+

if (actual !== expected) {

54+

console.error(

55+

`tool-display snapshot is stale: ${path.relative(repoRoot, outputPath)}\nrun: pnpm tool-display:write`,

56+

);

57+

process.exit(1);

7658

}

775960+

process.stdout.write("tool-display snapshot is up to date\n");

61+7862

function ensureCoreToolCoverage() {

7963

const toolNames = new Set<string>();

8064

for (const sourcePath of toolSources) {

@@ -108,131 +92,3 @@ function collectToolNamesFromFile(sourcePath: string, names: Set<string>) {

10892

function serializeToolDisplayConfig(config: ToolDisplayConfig = TOOL_DISPLAY_CONFIG): string {

10993

return `${JSON.stringify(config, null, 2)}\n`;

11094

}

111-112-

function collectToolDisplayDuplicateErrors(options: { includeSnapshot: boolean }): string[] {

113-

const duplicateErrors: string[] = [];

114-

const configSource = fs.readFileSync(configPath, "utf8");

115-

const configDuplicates = collectToolDisplayConfigDuplicateKeys(configSource, configPath);

116-

if (configDuplicates.length > 0) {

117-

duplicateErrors.push(

118-

formatDuplicateToolKeyError(path.relative(repoRoot, configPath), configDuplicates),

119-

);

120-

}

121-122-

if (options.includeSnapshot && fs.existsSync(outputPath)) {

123-

const snapshotSource = fs.readFileSync(outputPath, "utf8");

124-

const snapshotDuplicates = collectToolDisplaySnapshotDuplicateKeys(snapshotSource, outputPath);

125-

if (snapshotDuplicates.length > 0) {

126-

duplicateErrors.push(

127-

formatDuplicateToolKeyError(path.relative(repoRoot, outputPath), snapshotDuplicates),

128-

);

129-

}

130-

}

131-

return duplicateErrors;

132-

}

133-134-

export function collectToolDisplayConfigDuplicateKeys(

135-

source: string,

136-

sourcePath = "src/agents/tool-display-config.ts",

137-

): DuplicateToolKey[] {

138-

const sourceFile = ts.createSourceFile(sourcePath, source, ts.ScriptTarget.Latest, true);

139-

let toolsObject: ts.ObjectLiteralExpression | undefined;

140-

visitToolDisplayConfig(sourceFile, (configObject) => {

141-

toolsObject = findObjectProperty(configObject, "tools");

142-

});

143-

return toolsObject ? collectDuplicatePropertyKeys(toolsObject, sourceFile) : [];

144-

}

145-146-

export function collectToolDisplaySnapshotDuplicateKeys(

147-

source: string,

148-

sourcePath = "tool-display.json",

149-

): DuplicateToolKey[] {

150-

const sourceFile = ts.parseJsonText(sourcePath, source);

151-

const statement = sourceFile.statements[0];

152-

if (!statement || !ts.isExpressionStatement(statement)) {

153-

return [];

154-

}

155-

const root = statement.expression;

156-

if (!ts.isObjectLiteralExpression(root)) {

157-

return [];

158-

}

159-

const toolsObject = findObjectProperty(root, "tools");

160-

return toolsObject ? collectDuplicatePropertyKeys(toolsObject, sourceFile) : [];

161-

}

162-163-

export function formatDuplicateToolKeyError(

164-

relativePath: string,

165-

duplicates: DuplicateToolKey[],

166-

): string {

167-

const formatted = duplicates

168-

.map((duplicate) => `${duplicate.name} at lines ${duplicate.lines.join(", ")}`)

169-

.join("; ");

170-

return `tool-display metadata has duplicate tool ids in ${relativePath}: ${formatted}`;

171-

}

172-173-

function visitToolDisplayConfig(

174-

node: ts.Node,

175-

onConfig: (configObject: ts.ObjectLiteralExpression) => void,

176-

) {

177-

if (

178-

ts.isVariableDeclaration(node) &&

179-

ts.isIdentifier(node.name) &&

180-

node.name.text === "TOOL_DISPLAY_CONFIG" &&

181-

node.initializer &&

182-

ts.isObjectLiteralExpression(node.initializer)

183-

) {

184-

onConfig(node.initializer);

185-

return;

186-

}

187-

ts.forEachChild(node, (child) => visitToolDisplayConfig(child, onConfig));

188-

}

189-190-

function findObjectProperty(

191-

object: ts.ObjectLiteralExpression,

192-

propertyName: string,

193-

): ts.ObjectLiteralExpression | undefined {

194-

for (const property of object.properties) {

195-

if (

196-

ts.isPropertyAssignment(property) &&

197-

getPropertyNameText(property.name) === propertyName &&

198-

ts.isObjectLiteralExpression(property.initializer)

199-

) {

200-

return property.initializer;

201-

}

202-

}

203-

return undefined;

204-

}

205-206-

function collectDuplicatePropertyKeys(

207-

object: ts.ObjectLiteralExpression,

208-

sourceFile: ts.SourceFile,

209-

): DuplicateToolKey[] {

210-

const keyLines = new Map<string, number[]>();

211-

for (const property of object.properties) {

212-

if (!ts.isPropertyAssignment(property)) {

213-

continue;

214-

}

215-

const name = getPropertyNameText(property.name);

216-

if (!name) {

217-

continue;

218-

}

219-

const line =

220-

sourceFile.getLineAndCharacterOfPosition(property.name.getStart(sourceFile)).line + 1;

221-

keyLines.set(name, [...(keyLines.get(name) ?? []), line]);

222-

}

223-

return [...keyLines.entries()]

224-

.filter(([, lines]) => lines.length > 1)

225-

.map(([name, lines]) => ({ name, lines }))

226-

.toSorted((left, right) => left.name.localeCompare(right.name));

227-

}

228-229-

function getPropertyNameText(name: ts.PropertyName): string | undefined {

230-

if (ts.isIdentifier(name) || ts.isStringLiteral(name) || ts.isNumericLiteral(name)) {

231-

return name.text;

232-

}

233-

return undefined;

234-

}

235-236-

if (process.argv[1] && path.resolve(process.argv[1]) === fileURLToPath(import.meta.url)) {

237-

process.exitCode = main();

238-

}