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

推荐订阅源

N
News and Events Feed by Topic
Malwarebytes
Malwarebytes
Threat Intelligence Blog | Flashpoint
Threat Intelligence Blog | Flashpoint
C
Cybersecurity and Infrastructure Security Agency CISA
F
Future of Privacy Forum
C
Cisco Blogs
T
The Exploit Database - CXSecurity.com
A
Arctic Wolf
S
Securelist
K
Kaspersky official blog
S
Schneier on Security
T
ThreatConnect
T
Tenable Blog
Spread Privacy
Spread Privacy
T
True Tiger Recordings
AWS News Blog
AWS News Blog
F
Fox-IT International blog
量子位
T
Threatpost
V
Vulnerabilities – Threatpost
C
CERT Recently Published Vulnerability Notes
Cisco Talos Blog
Cisco Talos Blog
GbyAI
GbyAI
宝玉的分享
宝玉的分享
腾讯CDC
G
Google Developers Blog
aimingoo的专栏
aimingoo的专栏
Cyberwarzone
Cyberwarzone
有赞技术团队
有赞技术团队
S
SegmentFault 最新的问题
OSCHINA 社区最新新闻
OSCHINA 社区最新新闻
V
Visual Studio Blog
U
Unit 42
雷峰网
雷峰网
cs.CV updates on arXiv.org
cs.CV updates on arXiv.org
Simon Willison's Weblog
Simon Willison's Weblog
O
OpenAI News
freeCodeCamp Programming Tutorials: Python, JavaScript, Git & More
The GitHub Blog
The GitHub Blog
The Register - Security
The Register - Security
MyScale Blog
MyScale Blog
小众软件
小众软件
A
About on SuperTechFans
Last Week in AI
Last Week in AI
Y
Y Combinator Blog
博客园 - 三生石上(FineUI控件)
美团技术团队
Google Online Security Blog
Google Online Security Blog
P
Proofpoint News Feed
MongoDB | Blog
MongoDB | Blog

Recent Commits to openclaw:main

fix(telegram): route polling diagnostics away from errors fix(plugins): support linked source checkouts on Windows fix(gateway): back off session tool mirrors under pressure (#84846) fix(config): skip shell env fallback on Windows (#85739) fix: repair anchorless iMessage watch payloads · openclaw/openclaw@f37fbc9 fix(cli): route node status hints to stdout (#85780) · openclaw/openclaw@749692e fix(oc-path): support deep config edits (#86060) · openclaw/openclaw@3a72a30 fix(config): quiet benign metadata anomaly output · openclaw/openclaw@f3f4f29 fix(test): fail multi-node update regressions · openclaw/openclaw@732cf54 fix(google-vertex): support production ADC modes (#83971) · openclaw/openclaw@f09b4eb test(e2e): expose corrupt plugin deps smoke · openclaw/openclaw@fa3ff4d fix(codex): log app-server approval promotion trigger · openclaw/openclaw@d9af23f test(e2e): harden multi-node update smoke Clean up browser MCP subprocess tree (#85832) · openclaw/openclaw@8dc6b4d fix(agents): log warnings instead of swallowing subagent errors (#82943) · openclaw/openclaw@907bc03 fix(compaction): preserve partial summary on mid-chain chunk failure … · openclaw/openclaw@f0061dd fix(config): do not suppress recovery retry after failed backup resto… · openclaw/openclaw@5d174a5 chore: release 2026.5.25 fix(installer): support alpine cli installs · openclaw/openclaw@f68ed72 test(agents): keep runtime-plan provider mock current fix(scripts): launch env package scripts on Windows · openclaw/openclaw@4d4ce9e fix(agents): cache fallback provider resolution · openclaw/openclaw@3c8d101 fix(test): make import timing scripts Windows-safe · openclaw/openclaw@8ae9977 fix(telegram): transient Telegram pairing prompts (#85555) · openclaw/openclaw@8209426 fix(test): make max Vitest scripts Windows-safe · openclaw/openclaw@b681d5d fix(doctor): migrate Feishu account bot names (#86081) · openclaw/openclaw@9e8cc7e fix(scripts): prefilter conflict marker scans docs: add ClawSweeper review policy to AGENTS (#86197) · openclaw/openclaw@242e876 fix(installer): avoid before with npm release-age configs (#85491) · openclaw/openclaw@4742db6 fix(e2e): retry Windows kitchen sink probes · openclaw/openclaw@3e275a5 fix(installer): install node with apk on alpine fix(installer): detect musl linux shells · openclaw/openclaw@acfed37 perf(plugins,gateway): thread metadata snapshot + discovery through h… · openclaw/openclaw@8ccb11c fix(ui): split control ui runtime chunks · openclaw/openclaw@8bf4f7d refactor(config): extract GoogleChat schema into zod-schema.providers… · openclaw/openclaw@fe34141 fix(update): suppress internal handoff version warnings · openclaw/openclaw@6cc8244 test(e2e): select installable bundled plugins · openclaw/openclaw@0acc3e3 fix(scripts): harden Windows native opus install · openclaw/openclaw@43252c8 fix(agents): match runtime policy entries when session provider is em… fix(scripts): harden Windows generated formatting · openclaw/openclaw@0a98559 fix(mcp): bound tools/list during catalog discovery (#85063) · openclaw/openclaw@07f500a fix(test): focus plugin binding Docker smoke · openclaw/openclaw@dfa1a51 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
fix(gateway): avoid duplicate session message broadcasts · openclaw/openclaw@1459044
galiniliev · 2026-05-25 · via Recent Commits to openclaw:main

@@ -93,7 +93,7 @@ function waitForSessionsChangedMessagePhase(

9393

);

9494

}

959596-

async function emitTranscriptUpdateAndCollectEvents(params: {

96+

async function emitTranscriptUpdateAndCollectMessageEvent(params: {

9797

ws: Awaited<ReturnType<Awaited<ReturnType<typeof createGatewaySuiteHarness>>["openWs"]>>;

9898

sessionKey: string;

9999

sessionFile: string;

@@ -102,7 +102,6 @@ async function emitTranscriptUpdateAndCollectEvents(params: {

102102

messageSeq?: number;

103103

}) {

104104

const messageEventPromise = waitForSessionMessageEvent(params.ws, params.sessionKey);

105-

const changedEventPromise = waitForSessionsChangedMessagePhase(params.ws, params.sessionKey);

106105107106

emitSessionTranscriptUpdate({

108107

sessionFile: params.sessionFile,

@@ -112,11 +111,8 @@ async function emitTranscriptUpdateAndCollectEvents(params: {

112111

...(typeof params.messageSeq === "number" ? { messageSeq: params.messageSeq } : {}),

113112

});

114113115-

const [messageEvent, changedEvent] = await Promise.all([

116-

messageEventPromise,

117-

changedEventPromise,

118-

]);

119-

return { messageEvent, changedEvent };

114+

const messageEvent = await messageEventPromise;

115+

return { messageEvent };

120116

}

121117122118

async function expectNoMessageWithin(params: {

@@ -317,7 +313,7 @@ describe("session.message websocket events", () => {

317313

);

318314319315

await withOperatorSessionSubscriber(async (ws) => {

320-

const { messageEvent } = await emitTranscriptUpdateAndCollectEvents({

316+

const { messageEvent } = await emitTranscriptUpdateAndCollectMessageEvent({

321317

ws,

322318

sessionKey: "agent:main:main",

323319

sessionFile: transcriptPath,

@@ -441,7 +437,57 @@ describe("session.message websocket events", () => {

441437

});

442438

});

443439444-

test("includes live usage metadata on session.message and sessions.changed transcript events", async () => {

440+

test("does not duplicate displayable transcript updates with sessions.changed", async () => {

441+

const storePath = await createSessionStoreFile();

442+

await writeSessionStore({

443+

entries: {

444+

main: {

445+

sessionId: "sess-main",

446+

updatedAt: Date.now(),

447+

},

448+

},

449+

storePath,

450+

});

451+452+

await withOperatorSessionSubscriber(async (ws) => {

453+

const messageEventPromise = waitForSessionMessageEvent(ws, "agent:main:main");

454+

await expectNoMessageWithin({

455+

action: () => {

456+

emitSessionTranscriptUpdate({

457+

sessionFile: path.join(path.dirname(storePath), "sess-main.jsonl"),

458+

sessionKey: "agent:main:main",

459+

message: {

460+

role: "assistant",

461+

content: [{ type: "text", text: "single frame" }],

462+

timestamp: Date.now(),

463+

},

464+

messageId: "msg-single-frame",

465+

messageSeq: 1,

466+

});

467+

},

468+

watch: (timeoutMs) =>

469+

onceMessage(

470+

ws,

471+

(message) =>

472+

message.type === "event" &&

473+

message.event === "sessions.changed" &&

474+

(message.payload as { phase?: string; sessionKey?: string } | undefined)?.phase ===

475+

"message" &&

476+

(message.payload as { sessionKey?: string } | undefined)?.sessionKey ===

477+

"agent:main:main",

478+

timeoutMs,

479+

),

480+

});

481+

const messageEvent = await messageEventPromise;

482+

expectRecordFields(messageEvent.payload, {

483+

sessionKey: "agent:main:main",

484+

messageId: "msg-single-frame",

485+

messageSeq: 1,

486+

});

487+

});

488+

});

489+490+

test("includes live usage metadata on session.message transcript events", async () => {

445491

const storePath = await createSessionStoreFile();

446492

await writeSessionStore({

447493

entries: {

@@ -482,7 +528,7 @@ describe("session.message websocket events", () => {

482528

);

483529484530

await withOperatorSessionSubscriber(async (ws) => {

485-

const { messageEvent, changedEvent } = await emitTranscriptUpdateAndCollectEvents({

531+

const { messageEvent } = await emitTranscriptUpdateAndCollectMessageEvent({

486532

ws,

487533

sessionKey: "agent:main:main",

488534

sessionFile: transcriptPath,

@@ -500,18 +546,6 @@ describe("session.message websocket events", () => {

500546

modelProvider: "openai",

501547

model: "gpt-5.4",

502548

});

503-

expectRecordFields(changedEvent.payload, {

504-

sessionKey: "agent:main:main",

505-

phase: "message",

506-

messageId: "msg-usage",

507-

messageSeq: 1,

508-

totalTokens: 2_400,

509-

totalTokensFresh: true,

510-

contextTokens: 123_456,

511-

estimatedCostUsd: 0.0042,

512-

modelProvider: "openai",

513-

model: "gpt-5.4",

514-

});

515549

});

516550

});

517551

@@ -528,7 +562,7 @@ describe("session.message websocket events", () => {

528562

});

529563530564

await withOperatorSessionSubscriber(async (ws) => {

531-

const { messageEvent, changedEvent } = await emitTranscriptUpdateAndCollectEvents({

565+

const { messageEvent } = await emitTranscriptUpdateAndCollectMessageEvent({

532566

ws,

533567

sessionKey: "agent:main:main",

534568

sessionFile: path.join(path.dirname(storePath), "missing-transcript.jsonl"),

@@ -540,25 +574,18 @@ describe("session.message websocket events", () => {

540574

messageId: "msg-carried-seq",

541575

messageSeq: 7,

542576

});

543-544577

expectRecordFields(messageEvent.payload, {

545578

sessionKey: "agent:main:main",

546579

messageId: "msg-carried-seq",

547580

messageSeq: 7,

548581

});

549-

expectRecordFields(changedEvent.payload, {

550-

sessionKey: "agent:main:main",

551-

phase: "message",

552-

messageId: "msg-carried-seq",

553-

messageSeq: 7,

554-

});

555582

const payload = requireRecord(messageEvent.payload, "session.message payload");

556583

const message = requireRecord(payload.message, "session.message payload message");

557584

expect((message["__openclaw"] as { seq?: unknown } | undefined)?.seq).toBe(7);

558585

});

559586

});

560587561-

test("includes spawnedBy metadata on session.message and sessions.changed transcript events", async () => {

588+

test("includes spawnedBy metadata on session.message transcript events", async () => {

562589

const storePath = await createSessionStoreFile();

563590

const transcriptPath = path.join(path.dirname(storePath), "sess-child.jsonl");

564591

await writeSessionStore({

@@ -605,16 +632,6 @@ describe("session.message websocket events", () => {

605632

(message.payload as { sessionKey?: string } | undefined)?.sessionKey ===

606633

"agent:main:child",

607634

);

608-

const changedEventPromise = onceMessage(

609-

ws,

610-

(message) =>

611-

message.type === "event" &&

612-

message.event === "sessions.changed" &&

613-

(message.payload as { phase?: string; sessionKey?: string } | undefined)?.phase ===

614-

"message" &&

615-

(message.payload as { sessionKey?: string } | undefined)?.sessionKey ===

616-

"agent:main:child",

617-

);

618635619636

emitSessionTranscriptUpdate({

620637

sessionFile: transcriptPath,

@@ -623,10 +640,7 @@ describe("session.message websocket events", () => {

623640

messageId: "msg-spawn",

624641

});

625642626-

const [messageEvent, changedEvent] = await Promise.all([

627-

messageEventPromise,

628-

changedEventPromise,

629-

]);

643+

const messageEvent = await messageEventPromise;

630644

expectRecordFields(messageEvent.payload, {

631645

sessionKey: "agent:main:child",

632646

spawnedBy: "agent:main:main",

@@ -637,23 +651,12 @@ describe("session.message websocket events", () => {

637651

subagentControlScope: "children",

638652

parentSessionKey: "agent:main:main",

639653

});

640-

expectRecordFields(changedEvent.payload, {

641-

sessionKey: "agent:main:child",

642-

phase: "message",

643-

spawnedBy: "agent:main:main",

644-

spawnedWorkspaceDir: "/tmp/subagent-workspace",

645-

forkedFromParent: true,

646-

spawnDepth: 2,

647-

subagentRole: "orchestrator",

648-

subagentControlScope: "children",

649-

parentSessionKey: "agent:main:main",

650-

});

651654

} finally {

652655

ws.close();

653656

}

654657

});

655658656-

test("includes route thread metadata on session.message and sessions.changed transcript events", async () => {

659+

test("includes route thread metadata on session.message transcript events", async () => {

657660

const storePath = await createSessionStoreFile();

658661

const transcriptPath = path.join(path.dirname(storePath), "sess-thread.jsonl");

659662

await writeSessionStore({

@@ -685,7 +688,7 @@ describe("session.message websocket events", () => {

685688

);

686689687690

await withOperatorSessionSubscriber(async (ws) => {

688-

const { messageEvent, changedEvent } = await emitTranscriptUpdateAndCollectEvents({

691+

const { messageEvent } = await emitTranscriptUpdateAndCollectMessageEvent({

689692

ws,

690693

sessionKey: "agent:main:main",

691694

sessionFile: transcriptPath,

@@ -699,14 +702,6 @@ describe("session.message websocket events", () => {

699702

lastAccountId: "acct-1",

700703

lastThreadId: 42,

701704

});

702-

expectRecordFields(changedEvent.payload, {

703-

sessionKey: "agent:main:main",

704-

phase: "message",

705-

lastChannel: "telegram",

706-

lastTo: "-100123",

707-

lastAccountId: "acct-1",

708-

lastThreadId: 42,

709-

});

710705

});

711706

});

712707