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

推荐订阅源

C
Cybersecurity and Infrastructure Security Agency CISA
cs.CL updates on arXiv.org
cs.CL updates on arXiv.org
D
Darknet – Hacking Tools, Hacker News & Cyber Security
Know Your Adversary
Know Your Adversary
Malwarebytes
Malwarebytes
K
Kaspersky official blog
The Register - Security
The Register - Security
N
News and Events Feed by Topic
H
Hacker News: Front Page
T
The Exploit Database - CXSecurity.com
T
Tor Project blog
S
Secure Thoughts
Stack Overflow Blog
Stack Overflow Blog
Stack Overflow Blog
Stack Overflow Blog
Recent Announcements
Recent Announcements
Vercel News
Vercel News
Threat Intelligence Blog | Flashpoint
Threat Intelligence Blog | Flashpoint
L
LINUX DO - 热门话题
T
ThreatConnect
量子位
Apple Machine Learning Research
Apple Machine Learning Research
Application and Cybersecurity Blog
Application and Cybersecurity Blog
S
Security Archives - TechRepublic
Recent Commits to openclaw:main
Recent Commits to openclaw:main
雷峰网
雷峰网
F
Fortinet All Blogs
Y
Y Combinator Blog
Last Week in AI
Last Week in AI
月光博客
月光博客
P
Proofpoint News Feed
C
Cyber Attacks, Cyber Crime and Cyber Security
AWS News Blog
AWS News Blog
T
Tailwind CSS Blog
cs.CV updates on arXiv.org
cs.CV updates on arXiv.org
罗磊的独立博客
P
Privacy & Cybersecurity Law Blog
U
Unit 42
L
LINUX DO - 最新话题
M
MIT News - Artificial intelligence
OSCHINA 社区最新新闻
OSCHINA 社区最新新闻
Cyberwarzone
Cyberwarzone
V
Vulnerabilities – Threatpost
F
Fox-IT International blog
MongoDB | Blog
MongoDB | Blog
Google Online Security Blog
Google Online Security Blog
博客园 - 司徒正美
C
CXSECURITY Database RSS Feed - CXSecurity.com
Engineering at Meta
Engineering at Meta
C
Check Point Blog
李成银的技术随笔

Recent Commits to openclaw:main

fix(installer): tolerate WSL UNC launch cwd docs: absorb docs sweep fix: cancel stale provider auth prewarms (#85503) fix(security): escape entry.id in HTML export to prevent attribute XS… · openclaw/openclaw@7bc4a33 fix: guard openai-completions tool payload with supportsTools compat … · openclaw/openclaw@76a025c fix(slack): surface auth.test failure + normalize explicit-bot mentio… · openclaw/openclaw@995a020 chore(release): bump version to 2026.5.22 fix(ui): keep chat session search inline (#85490) · openclaw/openclaw@2601453 Policy: add secret and auth conformance checks (#81974) · openclaw/openclaw@c85feac fix(delivery): log failDelivery errors instead of silently swallowing… · openclaw/openclaw@f75789f fix(cli): validate debug proxy numeric options (#84260) · openclaw/openclaw@5c866a1 fix(daemon): use exit code instead of localized text for schtasks fal… · openclaw/openclaw@501e74d fix(auto-reply): enforce word boundary in slash command prefix match … · openclaw/openclaw@5c614de fix(message-tool): normalize send body aliases (#84102) docs: absorb maintainer docs sweep · openclaw/openclaw@e0fda55 fix(gateway): coalesce provider auth rewarms docs(voyage): clarify API key setup (#81803) · openclaw/openclaw@57178b1 docs(config): quote bracket config paths (#83058) · openclaw/openclaw@88f50e8 docs: link Copilot model availability (#76252) · openclaw/openclaw@14b2b8a test(telegram): await watchdog registration event · openclaw/openclaw@9fae5f7 ci: run binding command escape in release checks · openclaw/openclaw@4b63502 fix(tui): dismiss watchdog notice when response actually arrives (#77… · openclaw/openclaw@b741ddb test: add docker proof for plugin binding command escape · openclaw/openclaw@d756e1c test(telegram): wait for polling watchdog deterministically · openclaw/openclaw@7c9127c ci: skip pnpm auto repair in Crabbox shell · openclaw/openclaw@0241a6e fix(gateway): add .catch() to SIGTERM/SIGUSR1 signal handlers (#83131) fix: release cron runtime state after isolated runs (#85053) · openclaw/openclaw@247e536 fix(cron): suppress fatal error completion announce (#83724) · openclaw/openclaw@0c7220f fix(exec): parse nested approval metadata in async followups (#72268) · openclaw/openclaw@34c441c ci: share Crabbox hydrate pnpm store · openclaw/openclaw@7552634 chore(release): refresh plugin SDK baseline · openclaw/openclaw@736e7de ci(release): harden node setup before pnpm cache · openclaw/openclaw@a26aba6 ci(release): pass node pin to pnpm setup ci: use stable pnpm wrapper for Crabbox hydrate · openclaw/openclaw@b00d306 docs: refine maintainer docs sweep docs(memory): add guidance for action-sensitive memories (#82788) · openclaw/openclaw@bd04b1e docs(feishu): add dynamicAgentCreation and per-user isolation docs (#… docs(secrets): clarify agent-readable plaintext boundary (#84574) · openclaw/openclaw@ce5dcb0 docs(channels): document ackReactionScope for Slack & Telegram (DM go… · openclaw/openclaw@bbbed26 build(pnpm): use packageManager as pnpm source · openclaw/openclaw@a0702e1 ci: export Crabbox hydrate pnpm layout · openclaw/openclaw@f6840ac fix(gateway): preserve fresh agent session state · openclaw/openclaw@6f41653 fix(gateway): attribute agent wait timeouts ci: keep Crabbox hydrate runs reusable · openclaw/openclaw@489ea84 Speed up /models browse replies (#84735) · openclaw/openclaw@936dfaa docs: update changelog for plugin binding command escape (#85188) · openclaw/openclaw@9fc5346 Let binding commands escape plugin routes · openclaw/openclaw@af12082 ci: fix Crabbox hydrate pnpm modules dir · openclaw/openclaw@c9b17c5 Restore Control UI gateway token pairing [AI] (#85459) · openclaw/openclaw@10cb0a5 fix(docker): accept single-object pnpm list output · openclaw/openclaw@5e97045 fix: apply docs sweep updates · openclaw/openclaw@59aef2f fix(update): roll back failed git updates · openclaw/openclaw@769fd0b test(docker): expect prod store seed command fix(agents): bound embedded compaction write locks · openclaw/openclaw@46de078 fix(update): repair managed npm plugin peers · openclaw/openclaw@571f364 fix(update): repair managed npm plugin peers (#83794) (thanks @fuller… · openclaw/openclaw@de8a82a fix(telegram): honor table mode in outbound chunks (#85455) · openclaw/openclaw@7fc691a fix(docker): precreate owned named volume targets (#85454) · openclaw/openclaw@d8b9736 fix(ui): strip ANSI from displayed gateway logs (#85453) · openclaw/openclaw@664611c fix(skills): accept macos os requirement on darwin (#85451) fix(gateway): preserve message-tool replies in chat history test: track Docker prod store seed command · openclaw/openclaw@9a816f4 fix: satisfy prod store package list lint · openclaw/openclaw@d5247d0 test(plugins): clear lookup metadata memo fix(docker): seed prod store before offline prune · openclaw/openclaw@6788aa1 fix(memory): expand home paths in extra memory paths (#85449) · openclaw/openclaw@48bf037 docs: add security FAQ guidance chore(deps): refresh npm shrinkwraps docs: clarify OpenAI HTTP client guidance docs: remove stale showcase intro videos · openclaw/openclaw@00d3dca fix(gateway): point model override error to config docs docs: document secrets provider plan fields docs: clarify media directive formatting · openclaw/openclaw@c876fec docs: align memory search cache default refactor(ios): centralize setup auth parsing test(release): wait for config reload log proof refactor(ios): consolidate manual auth override inputs · openclaw/openclaw@d93c597 fix(ui): hide thinking options for non-reasoning models (#85406) · openclaw/openclaw@bb4d88e fix(ui): attach pasted data image text (#85392) · openclaw/openclaw@a03a8d9 fix(gateway): preserve OpenAI usage aliases in chat history (#85383) · openclaw/openclaw@d9c6c5f feat(ios): add realtime talk relay mode · openclaw/openclaw@e730e9b fix(browser): hint WSL portproxy CDP empty replies (#85379) · openclaw/openclaw@933f01c fix(installer): persist portable Git on Windows · openclaw/openclaw@5b90a48 fix(opencode-go): strip Kimi reasoning replay fields (#85377) · openclaw/openclaw@d22bcfc fix(build): normalize cache paths on Windows (#85437) · openclaw/openclaw@81d22c8 fix(update): detect nested macOS gateway ancestry (#85391) · openclaw/openclaw@adc6adc fix(docker): seed offline prune store in runtime stage · openclaw/openclaw@faf2a6c fix(ci): stabilize npm shrinkwrap metadata · openclaw/openclaw@21bedd3 fix(codex): route node exec through OpenClaw tools · openclaw/openclaw@5cc0dbc test(installer): track portable node root helper · openclaw/openclaw@9364b21 fix(ui): sync talk transcript translations fix(ui): localize talk transcript labels · openclaw/openclaw@8fc48af fix(release): stabilize config restart QA · openclaw/openclaw@cc91ff0 fix(installer): extract portable Node directly test(qa-lab): report scenario pack coverage · openclaw/openclaw@dcd98bf fix(plugins): drop stale tlon tool contract · openclaw/openclaw@d70dc4b fix(installer): prefer tar for portable Node extraction · openclaw/openclaw@a54a881 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
Add TUI PTY integration coverage (#85485) · openclaw/openclaw@0a50cbd
RomneyDa · 2026-05-23 · via Recent Commits to openclaw:main

@@ -0,0 +1,294 @@

1+

import { spawn } from "node:child_process";

2+

import { mkdir, readFile, writeFile } from "node:fs/promises";

3+

import { createRequire } from "node:module";

4+

import path from "node:path";

5+6+

type Options = {

7+

altScreen: boolean;

8+

mirrorPath: string;

9+

mode: "fake" | "local" | "all";

10+

vitestArgs: string[];

11+

};

12+13+

const DEFAULT_MIRROR_PATH = path.join(process.cwd(), ".artifacts", "tui-pty-mirror", "latest.ansi");

14+

const require = createRequire(import.meta.url);

15+

const MODE_TEST_FILES = {

16+

fake: ["src/tui/tui-pty-harness.e2e.test.ts"],

17+

local: ["src/tui/tui-pty-local.e2e.test.ts"],

18+

all: ["src/tui/tui-pty-harness.e2e.test.ts", "src/tui/tui-pty-local.e2e.test.ts"],

19+

} as const;

20+

const MIRROR_TERMINAL_QUERIES = ["\x1b[?u", "\x1b[16t"];

21+

const DEFAULT_PTY_COLS = 100;

22+

const DEFAULT_PTY_ROWS = 30;

23+24+

function readOption(args: string[], name: string): string | undefined {

25+

const idx = args.indexOf(name);

26+

if (idx < 0) {

27+

return undefined;

28+

}

29+

return args[idx + 1]?.trim() || undefined;

30+

}

31+32+

function readMode(args: string[]): Options["mode"] {

33+

const mode = readOption(args, "--mode") ?? "fake";

34+

if (mode === "fake" || mode === "local" || mode === "all") {

35+

return mode;

36+

}

37+

throw new Error(`--mode must be fake, local, or all; got ${JSON.stringify(mode)}`);

38+

}

39+40+

function parseOptions(args = process.argv.slice(2)): Options {

41+

const separator = args.indexOf("--");

42+

const ownArgs = separator >= 0 ? args.slice(0, separator) : args;

43+

const vitestArgs = separator >= 0 ? args.slice(separator + 1) : [];

44+

const mirrorPath =

45+

readOption(ownArgs, "--mirror-path") !== undefined

46+

? path.resolve(readOption(ownArgs, "--mirror-path") ?? "")

47+

: DEFAULT_MIRROR_PATH;

48+

return {

49+

altScreen: !ownArgs.includes("--no-alt-screen"),

50+

mirrorPath,

51+

mode: readMode(ownArgs),

52+

vitestArgs,

53+

};

54+

}

55+56+

function delay(ms: number): Promise<void> {

57+

return new Promise((resolve) => setTimeout(resolve, ms));

58+

}

59+60+

function shouldUseAltScreen(options: Options) {

61+

return options.altScreen && process.stdout.isTTY;

62+

}

63+64+

function resolveVitestCliEntry(): string {

65+

const vitestPackageJson = require.resolve("vitest/package.json");

66+

return path.join(path.dirname(vitestPackageJson), "vitest.mjs");

67+

}

68+69+

function currentTerminalDimension(value: number | undefined, fallback: number): string {

70+

return String(value && value > 0 ? value : fallback);

71+

}

72+73+

async function createMirrorFile(mirrorPath: string): Promise<void> {

74+

await mkdir(path.dirname(mirrorPath), { recursive: true });

75+

await writeFile(mirrorPath, "", "utf8");

76+

}

77+78+

async function readNewMirrorData(mirrorPath: string, offset: number) {

79+

const data = await readFile(mirrorPath);

80+

const nextOffset = data.byteLength;

81+

if (nextOffset < offset) {

82+

return { chunk: data, offset: nextOffset };

83+

}

84+

if (nextOffset === offset) {

85+

return { chunk: Buffer.alloc(0), offset };

86+

}

87+

return { chunk: data.subarray(offset), offset: nextOffset };

88+

}

89+90+

async function main(): Promise<void> {

91+

const options = parseOptions();

92+

const useAltScreen = shouldUseAltScreen(options);

93+

await createMirrorFile(options.mirrorPath);

94+95+

const child = spawn(

96+

process.execPath,

97+

[

98+

"--no-maglev",

99+

resolveVitestCliEntry(),

100+

"run",

101+

"--config",

102+

"test/vitest/vitest.tui-pty.config.ts",

103+

...MODE_TEST_FILES[options.mode],

104+

"--reporter=dot",

105+

...options.vitestArgs,

106+

],

107+

{

108+

cwd: process.cwd(),

109+

env: {

110+

...process.env,

111+

OPENCLAW_TUI_PTY_MIRROR_PATH: options.mirrorPath,

112+

OPENCLAW_TUI_PTY_INCLUDE_LOCAL: options.mode === "fake" ? "0" : "1",

113+

OPENCLAW_TUI_PTY_COLS: currentTerminalDimension(process.stdout.columns, DEFAULT_PTY_COLS),

114+

OPENCLAW_TUI_PTY_ROWS: currentTerminalDimension(process.stdout.rows, DEFAULT_PTY_ROWS),

115+

OPENCLAW_TUI_PTY_TYPE_CHUNK_SIZE: process.env.OPENCLAW_TUI_PTY_TYPE_CHUNK_SIZE ?? "4",

116+

OPENCLAW_TUI_PTY_TYPE_DELAY_MS: process.env.OPENCLAW_TUI_PTY_TYPE_DELAY_MS ?? "25",

117+

},

118+

stdio: ["ignore", "pipe", "pipe"],

119+

},

120+

);

121+122+

let childStdout = "";

123+

let childStderr = "";

124+

let restored = false;

125+

let mirrorOffset = 0;

126+

let mirrorFilterPending = "";

127+

let sawMirrorOutput = false;

128+

const startedAt = Date.now();

129+130+

const filterMirrorTerminalQueries = (chunk: Buffer) => {

131+

const input = mirrorFilterPending + chunk.toString("utf8");

132+

let output = "";

133+

mirrorFilterPending = "";

134+

for (let idx = 0; idx < input.length; idx += 1) {

135+

const rest = input.slice(idx);

136+

const fullMatch = MIRROR_TERMINAL_QUERIES.find((query) => rest.startsWith(query));

137+

if (fullMatch) {

138+

idx += fullMatch.length - 1;

139+

continue;

140+

}

141+

const partialMatch = MIRROR_TERMINAL_QUERIES.find((query) => query.startsWith(rest));

142+

if (partialMatch) {

143+

mirrorFilterPending = rest;

144+

break;

145+

}

146+

output += input[idx];

147+

}

148+

return output;

149+

};

150+151+

const writeMirrorChunk = (chunk: Buffer) => {

152+

const filteredChunk = filterMirrorTerminalQueries(chunk);

153+

if (filteredChunk.length === 0) {

154+

return;

155+

}

156+

if (!sawMirrorOutput && useAltScreen) {

157+

process.stdout.write("\x1b[2J\x1b[H");

158+

}

159+

sawMirrorOutput = true;

160+

process.stdout.write(filteredChunk);

161+

};

162+163+

const restoreScreen = () => {

164+

if (restored) {

165+

return;

166+

}

167+

restored = true;

168+

if (useAltScreen) {

169+

process.stdout.write("\x1b[?1049l");

170+

}

171+

};

172+173+

const stopChild = () => {

174+

child.kill("SIGINT");

175+

setTimeout(() => child.kill("SIGTERM"), 500).unref();

176+

};

177+178+

const ignoredInput = (chunk: Buffer) => {

179+

if (chunk.includes(0x03)) {

180+

stopChild();

181+

}

182+

};

183+

const hadRawMode = process.stdin.isTTY && process.stdin.isRaw;

184+

if (useAltScreen && process.stdin.isTTY) {

185+

process.stdin.setRawMode(true);

186+

process.stdin.resume();

187+

process.stdin.on("data", ignoredInput);

188+

}

189+190+

const restoreInput = () => {

191+

if (!process.stdin.isTTY) {

192+

return;

193+

}

194+

process.stdin.off("data", ignoredInput);

195+

process.stdin.setRawMode(hadRawMode);

196+

if (!hadRawMode) {

197+

process.stdin.pause();

198+

}

199+

};

200+201+

const drainParentInput = async () => {

202+

if (!useAltScreen || !process.stdin.isTTY) {

203+

return;

204+

}

205+

await delay(100);

206+

};

207+208+

const renderWaitingStatus = () => {

209+

if (!useAltScreen || sawMirrorOutput) {

210+

return;

211+

}

212+

const elapsedSeconds = Math.floor((Date.now() - startedAt) / 1000);

213+

process.stdout.write(

214+

[

215+

"\x1b[2J\x1b[H",

216+

"openclaw TUI PTY tests",

217+

"",

218+

`Mode: ${options.mode}`,

219+

`Waiting for the first TUI frame... ${elapsedSeconds}s`,

220+

`Mirror: ${options.mirrorPath}`,

221+

"",

222+

"Vitest output is buffered and will print after the mirrored TUI run exits.",

223+

].join("\n"),

224+

);

225+

};

226+227+

if (useAltScreen) {

228+

process.stdout.write("\x1b[?1049h\x1b[?25l");

229+

renderWaitingStatus();

230+

}

231+232+

child.stdout?.on("data", (chunk: Buffer) => {

233+

childStdout += chunk.toString("utf8");

234+

});

235+

child.stderr?.on("data", (chunk: Buffer) => {

236+

childStderr += chunk.toString("utf8");

237+

});

238+239+

let childExit: { code: number | null; signal: NodeJS.Signals | null } | null = null;

240+

child.on("exit", (code, signal) => {

241+

childExit = { code, signal };

242+

});

243+244+

process.once("SIGINT", stopChild);

245+246+

try {

247+

for (;;) {

248+

if (childExit) {

249+

break;

250+

}

251+

const result = await readNewMirrorData(options.mirrorPath, mirrorOffset);

252+

mirrorOffset = result.offset;

253+

if (result.chunk.byteLength > 0) {

254+

writeMirrorChunk(result.chunk);

255+

} else {

256+

renderWaitingStatus();

257+

}

258+

await delay(sawMirrorOutput ? 25 : 250);

259+

}

260+261+

const result = await readNewMirrorData(options.mirrorPath, mirrorOffset);

262+

if (result.chunk.byteLength > 0) {

263+

writeMirrorChunk(result.chunk);

264+

}

265+

} finally {

266+

await drainParentInput();

267+

restoreInput();

268+

if (useAltScreen) {

269+

process.stdout.write("\x1b[?2026l\x1b[?2004l\x1b[>4;0m\x1b[?25h");

270+

}

271+

restoreScreen();

272+

}

273+274+

if (childStdout) {

275+

process.stdout.write(childStdout);

276+

}

277+

if (childStderr) {

278+

process.stderr.write(childStderr);

279+

}

280+281+

if (childExit.signal) {

282+

throw new Error(`TUI PTY tests exited with signal ${childExit.signal}`);

283+

}

284+

if (childExit.code !== 0) {

285+

process.exitCode = childExit.code ?? 1;

286+

}

287+

}

288+289+

main().catch((error) => {

290+

process.stderr.write(

291+

`${error instanceof Error ? error.stack || error.message : String(error)}\n`,

292+

);

293+

process.exit(1);

294+

});