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

推荐订阅源

O
OpenAI News
I
InfoQ
云风的 BLOG
云风的 BLOG
博客园 - 【当耐特】
D
DataBreaches.Net
H
Help Net Security
爱范儿
爱范儿
F
Fortinet All Blogs
CTFtime.org: upcoming CTF events
CTFtime.org: upcoming CTF events
N
Netflix TechBlog - Medium
WordPress大学
WordPress大学
GbyAI
GbyAI
宝玉的分享
宝玉的分享
Martin Fowler
Martin Fowler
博客园_首页
C
Check Point Blog
OSCHINA 社区最新新闻
OSCHINA 社区最新新闻
G
Google Developers Blog
Apple Machine Learning Research
Apple Machine Learning Research
小众软件
小众软件
M
MIT News - Artificial intelligence
Recent Announcements
Recent Announcements
P
Proofpoint News Feed
L
LangChain Blog
阮一峰的网络日志
阮一峰的网络日志
V
V2EX
MyScale Blog
MyScale Blog
Recorded Future
Recorded Future
B
Blog
J
Java Code Geeks
T
The Blog of Author Tim Ferriss
Jina AI
Jina AI
博客园 - Franky
B
Blog RSS Feed
The GitHub Blog
The GitHub Blog
量子位
博客园 - 叶小钗
Hugging Face - Blog
Hugging Face - Blog
Cyberwarzone
Cyberwarzone
Google Online Security Blog
Google Online Security Blog
SecWiki News
SecWiki News
V
Vulnerabilities – Threatpost
AWS News Blog
AWS News Blog
Cisco Talos Blog
Cisco Talos Blog
G
GRAHAM CLULEY
T
Tor Project blog
腾讯CDC
美团技术团队
Application and Cybersecurity Blog
Application and Cybersecurity Blog
N
News and Events Feed by Topic

Recent Commits to openclaw:main

test: merge chat side-result checks · openclaw/openclaw@ddd2c2a test: merge cron history checks · openclaw/openclaw@f7eb746 test: merge responsive navigation shell checks · openclaw/openclaw@c2e4b47 docs(changelog): add codex oauth fixes · openclaw/openclaw@628e6cd test: merge navigation routing cases · openclaw/openclaw@5d8cecb Tests: mock channel registry bundled fallback · openclaw/openclaw@2b08233 Secrets: avoid broad web search discovery for single plugin config · openclaw/openclaw@a464f59 test: merge config view browser checks · openclaw/openclaw@20cf511 fix(status): align oauth health with runtime · openclaw/openclaw@eed7116 feat: add macOS screen snapshots for monitor preview (#67954) thanks … · openclaw/openclaw@f377db1 fix: report shared auth scopes in hello-ok (#67810) thanks @BunsDev · openclaw/openclaw@0b6c39b Auto-reply: avoid eager bundled route fallback · openclaw/openclaw@3ea1bf4 Tests: narrow session binding contract setup · openclaw/openclaw@54e4e16 fix(macOS): enable undo/redo in webchat composer text input (#34962) · openclaw/openclaw@00951dc Tests: speed up channel setup promotion · openclaw/openclaw@82b529a Docs: refresh agent instructions · openclaw/openclaw@5775fe2 fix(auth): serialize OAuth refresh across agents to fix #26322 (#67876) · openclaw/openclaw@8e79080 test: allow ollama public surface boundary test · openclaw/openclaw@7d4f1a6 Docs: add test performance guardrails · openclaw/openclaw@89706d3 Tests: restore context-engine usage proof · openclaw/openclaw@e4c4f95 Tests: slim context engine runtime coverage · openclaw/openclaw@74c198f ci: retry failed custom checkouts · openclaw/openclaw@0ee5baf test: trim duplicate provider auth onboarding cases · openclaw/openclaw@1ffc02e matrix: fix sessions_spawn --thread subagent session spawning (#67643) · openclaw/openclaw@1ce2596 test: reduce auth choice fixture churn · openclaw/openclaw@857b9cd test: mock health status config boundaries · openclaw/openclaw@9d5ab4a test: mock onboard config io boundary · openclaw/openclaw@299694d test: mock legacy state plugin boundaries · openclaw/openclaw@2713089 test: mock channel install boundaries · openclaw/openclaw@b945248 test: mock doctor preview channel boundaries · openclaw/openclaw@b1a3ad4 test: trim doctor command hotspots · openclaw/openclaw@c66f16a test: isolate agent auth and spawn hotspots · openclaw/openclaw@9285935 test: stabilize MCP startup disposal race · openclaw/openclaw@dd9d2eb test: merge browser contract server suites · openclaw/openclaw@5817a76 test: narrow ollama provider discovery setup · openclaw/openclaw@a0d9598 build: declare qa-lab aimock runtime dependency · openclaw/openclaw@24431e5 test: speed up safe-bins exec harness · openclaw/openclaw@ee856ab test: preserve tool helpers in embedded runner mocks · openclaw/openclaw@acd86a0 refactor: move memory embeddings into provider plugins · openclaw/openclaw@77e6e4c test: reuse system-run temp fixtures · openclaw/openclaw@7e9ff0f test: trim hotspot wait overhead · openclaw/openclaw@12a59b0 Check: avoid duplicate boundary prep · openclaw/openclaw@baf11b8 test: reduce hotspot fixture overhead · openclaw/openclaw@3a59edd feat(ui): overhaul settings and slash command UX (#67819) thanks @Bun… · openclaw/openclaw@2cfb660 QA Matrix: exit cleanly on failure · openclaw/openclaw@42805d2 QA Matrix: isolate scenario coverage · openclaw/openclaw@7e659e1 Matrix: refresh crypto bootstrap state · openclaw/openclaw@94081d8 QA Lab: add provider registry · openclaw/openclaw@bb7e982 Matrix: add plugin changelog · openclaw/openclaw@4acab55 test: trim more hotspot overhead · openclaw/openclaw@f485311 test: trim remaining hotspot tests · openclaw/openclaw@6ba8626 test: narrow hotspot mocks · openclaw/openclaw@dbc8179 test: isolate gemini embedding request helpers · openclaw/openclaw@cd330f5 test: trim memory and mcp hotspots · openclaw/openclaw@fd48dfa test: slim provider registry mocks · openclaw/openclaw@2e08c77 test: harden Parallels update smoke · openclaw/openclaw@1a98090 feat: default Anthropic to Opus 4.7 · openclaw/openclaw@628b454 fix: harden node-host shell payload mutability checks · openclaw/openclaw@75c551e fix: land node-host approval binding for native binaries (#66731) (th… · openclaw/openclaw@29919bb CI: add daily schedule to CodeQL workflow (#67645) · openclaw/openclaw@69d25f5 fix(gateway): capture config hash after plugin auto-enable to prevent… · openclaw/openclaw@8c11210 fix: repair sanitized replay tool results before send (#67620) (thank… · openclaw/openclaw@c3c7a99 fix: restrict HTML timeout short-circuit to transient statuses · openclaw/openclaw@de129a6 fix: keep TUI watchdog bound to active run (#67401) (thanks @xantorres) · openclaw/openclaw@3525273 Gateway/skills: dedupe skills prefix-match + drop dead fallback on log · openclaw/openclaw@d7f489f Extensions/lmstudio: back off inference preload after consecutive fai… · openclaw/openclaw@b555214 TUI/streaming: add watchdog that resets the activity indicator after … · openclaw/openclaw@f44ab20 Agents/tool-loop: enable unknown-tool stream guard by default · openclaw/openclaw@36ed367 Gateway/skills: invalidate session skills snapshot on config write · openclaw/openclaw@b23d59a fix: classify HTML provider error pages correctly (#67642) (thanks @s… · openclaw/openclaw@e588e90 fix(skills): remove unused model-usage import (#67641) · openclaw/openclaw@55f05df docs(changelog): credit codex fix superseded PRs · openclaw/openclaw@e485f24 fix(openai-codex): normalize stale transport metadata in resolution a… · openclaw/openclaw@90801ba CI: pin Docker-related GitHub Actions (#67632) · openclaw/openclaw@f697b01 Android: modernize WebView and discovery API usage (#67627) · openclaw/openclaw@44a6e50 fix(deps): bump hono to 4.12.14 and @hono/node-server to 1.19.14 (GHS… · openclaw/openclaw@fbccc18 fix(deps): bump dompurify to 3.4.0 (#67614) · openclaw/openclaw@2c2dc00 CI: add explicit permissions to all workflow jobs (fixes code-scannin… · openclaw/openclaw@01b7516 fix: register bundled TTS providers and route overrides correctly (#6… · openclaw/openclaw@6ea3cdd fix: align host tilde paths with OS home (#62804) (thanks @stainlu) · openclaw/openclaw@ecfaf64 fix: flush creds queue before reconnect socket open (#67464) (thanks … · openclaw/openclaw@405c63f fix: strip standalone <function> tool call tags from visible text (#6… · openclaw/openclaw@78df859 fix(agents): preserve cli session metadata before transcript persist … · openclaw/openclaw@898fd04 docs(changelog): move cli transcript entry · openclaw/openclaw@c1817c6 fix(agents): normalize cli transcript api field · openclaw/openclaw@3a3fae0 docs(changelog): note cli transcript persistence · openclaw/openclaw@6c343f1 fix(agents): persist cli transcript turns · openclaw/openclaw@b8ef507 fix(msteams): harden security-sensitive flows (#65841) · openclaw/openclaw@c56b56e [Dashboard] Fix exec approval modal overflow for long command content… · openclaw/openclaw@053c5b0 Docs: remove QA changelog entry · openclaw/openclaw@7fd5771 QA: fix private runtime source loading (#67428) · openclaw/openclaw@d5933af docs(gateway): correct protocol.md schema path, hello-ok example, aut… · openclaw/openclaw@489404d CI: pin Node 22 runners to 22.18.0 · openclaw/openclaw@4ffa621 models.authStatus: normalize provider ids + tighten env-backed escape… · openclaw/openclaw@f2fdb9d Update CHANGELOG.md · openclaw/openclaw@7694a92 test(parallels): clean up npm update guard jobs · openclaw/openclaw@045ea7b Plugins: prefer scanDir override paths · openclaw/openclaw@b2974da fix(dreaming): default storage.mode to "separate" so phase blocks sto… · openclaw/openclaw@8c392f0 fix(memory-core): skip dreaming transcript ingestion via session stor… · openclaw/openclaw@a1b01f0 fix: dedupe replayed exec.finished node events (#67281) · openclaw/openclaw@5dcf526
fix(telegram): render progress drafts as rich previews · openclaw/openclaw@663fabb
obviyus · 2026-06-15 · via Recent Commits to openclaw:main

@@ -1812,9 +1812,9 @@ describe("dispatchTelegramMessage draft streaming", () => {

18121812

telegramCfg: { streaming: { mode: "partial" } },

18131813

});

181418141815-

expect(mockCallArg(answerDraftStream.update)).toContain("Exec");

1816-

expect(answerDraftStream.update).toHaveBeenNthCalledWith(2, "Done ");

1817-

expect(answerDraftStream.update).toHaveBeenNthCalledWith(3, "Done answer");

1815+

expect(mockCallArg(answerDraftStream.updatePreview).text).toContain("Exec");

1816+

expect(answerDraftStream.update).toHaveBeenNthCalledWith(1, "Done ");

1817+

expect(answerDraftStream.update).toHaveBeenNthCalledWith(2, "Done answer");

18181818

expect(answerDraftStream.update).toHaveBeenLastCalledWith("Done answer.");

18191819

expect(deliverReplies).not.toHaveBeenCalled();

18201820

});

@@ -2219,15 +2219,14 @@ describe("dispatchTelegramMessage draft streaming", () => {

2219221922202220

expect(answerDraftStream.update).toHaveBeenNthCalledWith(1, "Site A shows X.");

22212221

expect(answerDraftStream.update).toHaveBeenNthCalledWith(2, "Site A shows X.");

2222-

expect(answerDraftStream.update).toHaveBeenNthCalledWith(

2223-

3,

2224-

expect.stringMatching(/`🛠 Exec`$/),

2222+

expect(answerDraftStream.updatePreview).toHaveBeenCalledWith(

2223+

expect.objectContaining({ text: expect.stringMatching(/`🛠 Exec`$/) }),

22252224

);

2226-

expect(answerDraftStream.update).toHaveBeenNthCalledWith(4, "Final answer");

2225+

expect(answerDraftStream.update).toHaveBeenNthCalledWith(3, "Final answer");

22272226

expect(answerDraftStream.clear).toHaveBeenCalledTimes(1);

22282227

expect(answerDraftStream.forceNewMessage).toHaveBeenCalledTimes(2);

22292228

const progressResetOrder = answerDraftStream.forceNewMessage.mock.invocationCallOrder[0];

2230-

const progressUpdateOrder = answerDraftStream.update.mock.invocationCallOrder[2];

2229+

const progressUpdateOrder = answerDraftStream.updatePreview.mock.invocationCallOrder[0];

22312230

expect(progressResetOrder).toBeLessThan(progressUpdateOrder);

22322231

expect(deliverReplies).not.toHaveBeenCalled();

22332232

});

@@ -2247,12 +2246,11 @@ describe("dispatchTelegramMessage draft streaming", () => {

22472246

await dispatchWithContext({ context: createContext() });

2248224722492248

expect(answerDraftStream.update).toHaveBeenNthCalledWith(1, "Site A shows X.");

2250-

expect(answerDraftStream.update).toHaveBeenNthCalledWith(

2251-

2,

2252-

expect.stringMatching(/`🛠 Exec`$/),

2249+

expect(answerDraftStream.updatePreview).toHaveBeenCalledWith(

2250+

expect.objectContaining({ text: expect.stringMatching(/`🛠 Exec`$/) }),

22532251

);

2254-

expect(answerDraftStream.update).toHaveBeenNthCalledWith(3, "Site B shows Y.");

2255-

expect(answerDraftStream.update).toHaveBeenNthCalledWith(4, "Final answer");

2252+

expect(answerDraftStream.update).toHaveBeenNthCalledWith(2, "Site B shows Y.");

2253+

expect(answerDraftStream.update).toHaveBeenNthCalledWith(3, "Final answer");

22562254

expect(answerDraftStream.forceNewMessage).toHaveBeenCalledTimes(2);

22572255

expect(answerDraftStream.clear).toHaveBeenCalledTimes(1);

22582256

expect(deliverReplies).not.toHaveBeenCalled();

@@ -2290,16 +2288,15 @@ describe("dispatchTelegramMessage draft streaming", () => {

2290228822912289

await dispatchWithContext({ context: createContext() });

229222902293-

expect(answerDraftStream.update).toHaveBeenNthCalledWith(

2294-

1,

2295-

expect.stringMatching(/`🛠 Exec`$/),

2291+

expect(answerDraftStream.updatePreview).toHaveBeenCalledWith(

2292+

expect.objectContaining({ text: expect.stringMatching(/`🛠 Exec`$/) }),

22962293

);

2297-

expect(answerDraftStream.update).toHaveBeenNthCalledWith(2, "Branch is up to date");

2294+

expect(answerDraftStream.update).toHaveBeenNthCalledWith(1, "Branch is up to date");

22982295

expect(answerDraftStream.forceNewMessage).toHaveBeenCalledTimes(1);

22992296

expect(answerDraftStream.clear).toHaveBeenCalledTimes(1);

23002297

const clearOrder = answerDraftStream.clear.mock.invocationCallOrder[0];

23012298

const rotationOrder = answerDraftStream.forceNewMessage.mock.invocationCallOrder[0];

2302-

const finalUpdateOrder = answerDraftStream.update.mock.invocationCallOrder[1];

2299+

const finalUpdateOrder = answerDraftStream.update.mock.invocationCallOrder[0];

23032300

expect(clearOrder).toBeLessThan(rotationOrder);

23042301

expect(rotationOrder).toBeLessThan(finalUpdateOrder);

23052302

});

@@ -2317,16 +2314,15 @@ describe("dispatchTelegramMessage draft streaming", () => {

2317231423182315

await dispatchWithContext({ context: createContext() });

231923162320-

expect(answerDraftStream.update).toHaveBeenNthCalledWith(

2321-

1,

2322-

expect.stringMatching(/`🛠 Exec`$/),

2317+

expect(answerDraftStream.updatePreview).toHaveBeenCalledWith(

2318+

expect.objectContaining({ text: expect.stringMatching(/`🛠 Exec`$/) }),

23232319

);

2324-

expect(answerDraftStream.update).toHaveBeenNthCalledWith(2, "Branch is up to date");

2320+

expect(answerDraftStream.update).toHaveBeenNthCalledWith(1, "Branch is up to date");

23252321

expect(answerDraftStream.forceNewMessage).toHaveBeenCalledTimes(1);

23262322

expect(answerDraftStream.clear).toHaveBeenCalledTimes(1);

23272323

const clearOrder = answerDraftStream.clear.mock.invocationCallOrder[0];

23282324

const rotationOrder = answerDraftStream.forceNewMessage.mock.invocationCallOrder[0];

2329-

const finalUpdateOrder = answerDraftStream.update.mock.invocationCallOrder[1];

2325+

const finalUpdateOrder = answerDraftStream.update.mock.invocationCallOrder[0];

23302326

expect(clearOrder).toBeLessThan(rotationOrder);

23312327

expect(rotationOrder).toBeLessThan(finalUpdateOrder);

23322328

});

@@ -2373,9 +2369,13 @@ describe("dispatchTelegramMessage draft streaming", () => {

23732369

telegramCfg: { streaming: { mode: "progress" } },

23742370

});

237523712376-

expect(answerDraftStream.update).toHaveBeenCalledWith(

2377-

"Cracking\n\n`🛠️ Exec`\n\n`🛠️ git rev-parse --abbrev-ref HEAD`",

2378-

);

2372+

expect(answerDraftStream.updatePreview).toHaveBeenCalledWith({

2373+

text: "Cracking\n\n`🛠️ Exec`\n`🛠️ git rev-parse --abbrev-ref HEAD`",

2374+

richMessage: {

2375+

html: "<b>Cracking</b><br><b>🛠️ Exec</b><br><b>🛠️ Exec</b> <code>git rev-parse --abbrev-ref HEAD</code>",

2376+

skip_entity_detection: true,

2377+

},

2378+

});

23792379

expect(answerDraftStream.update).not.toHaveBeenCalledWith("Branch is up to date");

23802380

expect(answerDraftStream.forceNewMessage).toHaveBeenCalledTimes(1);

23812381

expect(answerDraftStream.clear).toHaveBeenCalledTimes(1);

@@ -2400,12 +2400,18 @@ describe("dispatchTelegramMessage draft streaming", () => {

24002400

telegramCfg: { streaming: { mode: "progress" } },

24012401

});

240224022403-

expect(answerDraftStream.update).toHaveBeenCalledTimes(2);

2404-

expect(answerDraftStream.update).toHaveBeenNthCalledWith(1, "Cracking\n\n`🛠️ Exec`");

2405-

expect(answerDraftStream.update).toHaveBeenNthCalledWith(2, trailingFinalStatusText);

2403+

expect(answerDraftStream.updatePreview).toHaveBeenCalledWith({

2404+

text: "Cracking\n\n`🛠️ Exec`",

2405+

richMessage: {

2406+

html: "<b>Cracking</b><br><b>🛠️ Exec</b>",

2407+

skip_entity_detection: true,

2408+

},

2409+

});

2410+

expect(answerDraftStream.update).toHaveBeenCalledTimes(1);

2411+

expect(answerDraftStream.update).toHaveBeenNthCalledWith(1, trailingFinalStatusText);

24062412

expect(answerDraftStream.forceNewMessage).toHaveBeenCalledTimes(2);

24072413

expect(answerDraftStream.forceNewMessage.mock.invocationCallOrder[1]).toBeLessThan(

2408-

answerDraftStream.update.mock.invocationCallOrder[1],

2414+

answerDraftStream.update.mock.invocationCallOrder[0],

24092415

);

24102416

expectDeliveredReply(0, { text: "Branch is up to date" });

24112417

});

@@ -2430,11 +2436,13 @@ describe("dispatchTelegramMessage draft streaming", () => {

24302436

telegramCfg: { streaming: { mode: "progress", progress: { label: "Shelling" } } },

24312437

});

243224382433-

expect(answerDraftStream.update).not.toHaveBeenCalledWith(

2439+

expect(answerDraftStream.updatePreview).not.toHaveBeenCalledWith(

24342440

expect.stringContaining("stdout line one"),

24352441

);

2436-

expect(answerDraftStream.update).toHaveBeenLastCalledWith(

2437-

"Shelling\n\n`🛠️ Exec`\n\n`🔎 Web Search: docs lookup`",

2442+

expect(answerDraftStream.updatePreview).toHaveBeenLastCalledWith(

2443+

expect.objectContaining({

2444+

text: "Shelling\n\n`🛠️ Exec`\n`🔎 Web Search: docs lookup`",

2445+

}),

24382446

);

24392447

expect(deliverReplies).not.toHaveBeenCalled();

24402448

});

@@ -2456,8 +2464,10 @@ describe("dispatchTelegramMessage draft streaming", () => {

24562464

telegramCfg: { streaming: { mode: "progress", progress: { label: "Shelling" } } },

24572465

});

245824662459-

expect(answerDraftStream.update).toHaveBeenCalledTimes(1);

2460-

expect(answerDraftStream.update).toHaveBeenCalledWith("Shelling\n\n`🛠️ Exec`");

2467+

expect(answerDraftStream.updatePreview).toHaveBeenCalledTimes(1);

2468+

expect(answerDraftStream.updatePreview).toHaveBeenCalledWith(

2469+

expect.objectContaining({ text: "Shelling\n\n`🛠️ Exec`" }),

2470+

);

24612471

expectDeliveredReply(0, { text: "Branch is up to date" });

24622472

});

24632473

@@ -2484,8 +2494,10 @@ describe("dispatchTelegramMessage draft streaming", () => {

24842494

telegramCfg: { streaming: { mode: "progress", progress: { label: "Shelling" } } },

24852495

});

248624962487-

expect(answerDraftStream.update).toHaveBeenCalledTimes(1);

2488-

expect(answerDraftStream.update).toHaveBeenCalledWith("Shelling\n\n`🛠️ Exec`");

2497+

expect(answerDraftStream.updatePreview).toHaveBeenCalledTimes(1);

2498+

expect(answerDraftStream.updatePreview).toHaveBeenCalledWith(

2499+

expect.objectContaining({ text: "Shelling\n\n`🛠️ Exec`" }),

2500+

);

24892501

expectDeliveredReply(0, { text: "Branch is up to date" });

24902502

});

24912503

@@ -2516,8 +2528,10 @@ describe("dispatchTelegramMessage draft streaming", () => {

25162528

telegramCfg: { streaming: { mode: "progress", progress: { label: "Shelling" } } },

25172529

});

251825302519-

expect(answerDraftStream.update).toHaveBeenCalledTimes(1);

2520-

expect(answerDraftStream.update).toHaveBeenCalledWith("Shelling\n\n`🛠️ Exec`");

2531+

expect(answerDraftStream.updatePreview).toHaveBeenCalledTimes(1);

2532+

expect(answerDraftStream.updatePreview).toHaveBeenCalledWith(

2533+

expect.objectContaining({ text: "Shelling\n\n`🛠️ Exec`" }),

2534+

);

25212535

expectDeliveredReply(0, { text: "Branch is up to date" });

25222536

});

25232537

@@ -2648,7 +2662,13 @@ describe("dispatchTelegramMessage draft streaming", () => {

26482662

telegramCfg: { streaming: { mode: "progress", progress: { label: "Shelling" } } },

26492663

});

265026642651-

expect(draftStream.update).toHaveBeenCalledWith("Shelling\n\n`🛠️ Exec`");

2665+

expect(draftStream.updatePreview).toHaveBeenCalledWith({

2666+

text: "Shelling\n\n`🛠️ Exec`",

2667+

richMessage: {

2668+

html: "<b>Shelling</b><br><b>🛠️ Exec</b>",

2669+

skip_entity_detection: true,

2670+

},

2671+

});

26522672

expect(draftStream.flush).toHaveBeenCalled();

26532673

});

26542674

@@ -2670,7 +2690,13 @@ describe("dispatchTelegramMessage draft streaming", () => {

26702690

});

2671269126722692

expect(createTelegramDraftStream).toHaveBeenCalledTimes(1);

2673-

expect(draftStream.update).toHaveBeenCalledWith("Shelling\n\n`🛠️ Exec`\n\n• _Checking files_");

2693+

expect(draftStream.updatePreview).toHaveBeenCalledWith({

2694+

text: "Shelling\n\n`🛠️ Exec`\n• _Checking files_",

2695+

richMessage: {

2696+

html: "<b>Shelling</b><br><b>🛠️ Exec</b><br><i>Checking files</i>",

2697+

skip_entity_detection: true,

2698+

},

2699+

});

26742700

});

2675270126762702

it("renders configured Telegram commentary progress from preamble item events", async () => {

@@ -2697,7 +2723,13 @@ describe("dispatchTelegramMessage draft streaming", () => {

26972723

},

26982724

});

269927252700-

expect(draftStream.update).toHaveBeenCalledWith("Shelling\n\n_Checking recent context_");

2726+

expect(draftStream.updatePreview).toHaveBeenCalledWith({

2727+

text: "Shelling\n\n_Checking recent context_",

2728+

richMessage: {

2729+

html: "<b>Shelling</b><br><i>Checking recent context</i>",

2730+

skip_entity_detection: true,

2731+

},

2732+

});

27012733

});

2702273427032735

it("suppresses Telegram preamble progress when commentary is disabled", async () => {

@@ -2724,7 +2756,9 @@ describe("dispatchTelegramMessage draft streaming", () => {

27242756

},

27252757

});

272627582727-

expect(draftStream.update).not.toHaveBeenCalledWith(expect.stringContaining("Checking recent"));

2759+

expect(draftStream.updatePreview).not.toHaveBeenCalledWith(

2760+

expect.objectContaining({ text: expect.stringContaining("Checking recent") }),

2761+

);

27282762

});

2729276327302764

it("keeps the progress draft label when tool progress lines are hidden", async () => {

@@ -2748,7 +2782,10 @@ describe("dispatchTelegramMessage draft streaming", () => {

27482782

},

27492783

});

275027842751-

expect(draftStream.update).toHaveBeenCalledWith("Shelling");

2785+

expect(draftStream.updatePreview).toHaveBeenCalledWith({

2786+

text: "Shelling",

2787+

richMessage: { html: "<b>Shelling</b>", skip_entity_detection: true },

2788+

});

27522789

expect(draftStream.flush).toHaveBeenCalled();

27532790

});

27542791

@@ -2777,10 +2814,20 @@ describe("dispatchTelegramMessage draft streaming", () => {

27772814

},

27782815

});

277928162780-

await vi.waitFor(() => expect(draftStream.update).toHaveBeenCalledWith("Working"));

2781-

expect(draftStream.update).not.toHaveBeenCalledWith("Working.");

2782-

expect(draftStream.update).not.toHaveBeenCalledWith("Working..");

2783-

expect(draftStream.update).not.toHaveBeenCalledWith("Working...");

2817+

await vi.waitFor(() =>

2818+

expect(draftStream.updatePreview).toHaveBeenCalledWith(

2819+

expect.objectContaining({ text: "Working" }),

2820+

),

2821+

);

2822+

expect(draftStream.updatePreview).not.toHaveBeenCalledWith(

2823+

expect.objectContaining({ text: "Working." }),

2824+

);

2825+

expect(draftStream.updatePreview).not.toHaveBeenCalledWith(

2826+

expect.objectContaining({ text: "Working.." }),

2827+

);

2828+

expect(draftStream.updatePreview).not.toHaveBeenCalledWith(

2829+

expect.objectContaining({ text: "Working..." }),

2830+

);

27842831

finishRun?.();

27852832

await run;

27862833

});

@@ -2800,10 +2847,10 @@ describe("dispatchTelegramMessage draft streaming", () => {

28002847

const pendingToolStart = replyOptions?.onToolStart?.({ name: "exec", phase: "start" });

28012848

await Promise.resolve();

28022849

await Promise.resolve();

2803-

const updateBeforeStatusReaction = draftStream.update.mock.calls.at(-1)?.[0];

2850+

const updateBeforeStatusReaction = draftStream.updatePreview.mock.calls.at(-1)?.[0]?.text;

28042851

releaseSetTool?.();

28052852

await pendingToolStart;

2806-

expect(updateBeforeStatusReaction).toMatch(/^Shelling\n`🛠 Exec`$/);

2853+

expect(updateBeforeStatusReaction).toBe("Shelling\n\n`🛠️ Exec`");

28072854

return { queuedFinal: false };

28082855

});

28092856

@@ -2839,9 +2886,13 @@ describe("dispatchTelegramMessage draft streaming", () => {

28392886

telegramCfg: { streaming: { mode: "progress", progress: { label: "Shelling" } } },

28402887

});

284128882842-

expect(draftStream.update).toHaveBeenCalledWith(

2843-

"Shelling\n\n`🔎 Web Search: docs lookup`\n\n• `tests passed`",

2844-

);

2889+

expect(draftStream.updatePreview).toHaveBeenCalledWith({

2890+

text: "Shelling\n\n`🔎 Web Search: docs lookup`\n• `tests passed`",

2891+

richMessage: {

2892+

html: "<b>Shelling</b><br><b>🔎 Web Search</b> <code>docs lookup</code><br><b>Update</b> <code>tests passed</code>",

2893+

skip_entity_detection: true,

2894+

},

2895+

});

28452896

expect(draftStream.forceNewMessage).toHaveBeenCalledTimes(1);

28462897

expect(draftStream.materialize).not.toHaveBeenCalled();

28472898

expect(draftStream.clear).toHaveBeenCalledTimes(1);

@@ -3984,7 +4035,7 @@ describe("dispatchTelegramMessage draft streaming", () => {

39844035

telegramCfg: { streaming: { mode: "partial" } },

39854036

});

398640373987-

expect(mockCallArg(answerDraftStream.update)).toContain("Exec");

4038+

expect(mockCallArg(answerDraftStream.updatePreview).text).toContain("Exec");

39884039

expect(answerDraftStream.update).toHaveBeenLastCalledWith("block after progress");

39894040

});

39904041