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

推荐订阅源

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(cli): route node status hints to stdout (#85780) · openclaw/openclaw@749692e fix(oc-path): support deep config edits (#86060) 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 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: repair anchorless iMessage watch payloads · openclaw/openclaw@f37fbc9
omarshahine · 2026-05-25 · via Recent Commits to openclaw:main

@@ -0,0 +1,155 @@

1+

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

2+

import { isIMessageAnchorless, repairIMessageConversationAnchor } from "./conversation-repair.js";

3+

import type { IMessagePayload } from "./types.js";

4+5+

function anchorlessMessage(overrides: Partial<IMessagePayload> = {}): IMessagePayload {

6+

return {

7+

id: 9500,

8+

guid: "ANCHORLESS-GUID-1",

9+

chat_id: 0,

10+

sender: "+15550001111",

11+

is_from_me: false,

12+

text: "https://example.com",

13+

chat_guid: "",

14+

chat_identifier: "",

15+

chat_name: "",

16+

participants: null,

17+

is_group: false,

18+

...overrides,

19+

};

20+

}

21+22+

function mockClient(chats: Array<{ id: number; messages: Record<string, unknown>[] }>) {

23+

const request = vi.fn(async (method: string, params?: Record<string, unknown>) => {

24+

if (method === "chats.list") {

25+

return { chats: chats.map((chat) => ({ id: chat.id })) };

26+

}

27+

if (method === "messages.history") {

28+

return {

29+

messages: chats.find((chat) => chat.id === params?.chat_id)?.messages ?? [],

30+

};

31+

}

32+

throw new Error(`unexpected method ${method}`);

33+

});

34+

return { request };

35+

}

36+37+

describe("isIMessageAnchorless", () => {

38+

it("detects explicit broken conversation anchors", () => {

39+

expect(isIMessageAnchorless(anchorlessMessage())).toBe(true);

40+

expect(isIMessageAnchorless(anchorlessMessage({ chat_guid: undefined }))).toBe(true);

41+

expect(isIMessageAnchorless(anchorlessMessage({ chat_identifier: undefined }))).toBe(true);

42+

expect(

43+

isIMessageAnchorless(

44+

anchorlessMessage({ chat_id: undefined, chat_guid: "", chat_identifier: "" }),

45+

),

46+

).toBe(true);

47+

});

48+49+

it("does not classify sender-only direct messages as anchorless", () => {

50+

expect(

51+

isIMessageAnchorless({

52+

guid: "DM-GUID",

53+

sender: "+15550001111",

54+

is_from_me: false,

55+

text: "hello",

56+

}),

57+

).toBe(false);

58+

});

59+60+

it("does not classify messages with any usable conversation anchor", () => {

61+

expect(isIMessageAnchorless(anchorlessMessage({ chat_id: 349 }))).toBe(false);

62+

expect(isIMessageAnchorless(anchorlessMessage({ chat_guid: "iMessage;+;chat349" }))).toBe(

63+

false,

64+

);

65+

expect(isIMessageAnchorless(anchorlessMessage({ chat_identifier: "chat349" }))).toBe(false);

66+

});

67+

});

68+69+

describe("repairIMessageConversationAnchor", () => {

70+

it("passes through non-anchorless messages without recovery RPCs", async () => {

71+

const message = anchorlessMessage({ chat_id: 349, is_group: true });

72+

const client = mockClient([]);

73+74+

await expect(

75+

repairIMessageConversationAnchor({ client: client as never, message }),

76+

).resolves.toBe(message);

77+

expect(client.request).not.toHaveBeenCalled();

78+

});

79+80+

it("recovers the conversation from recent history by GUID", async () => {

81+

const message = anchorlessMessage();

82+

const client = mockClient([

83+

{ id: 100, messages: [{ guid: "OTHER-GUID", chat_id: 100, is_group: true }] },

84+

{

85+

id: 349,

86+

messages: [

87+

{

88+

guid: "ANCHORLESS-GUID-1",

89+

chat_id: 349,

90+

chat_guid: "iMessage;+;chat349",

91+

chat_identifier: "chat349",

92+

chat_name: "Project group",

93+

participants: ["+15550001111", "+15550002222"],

94+

is_group: true,

95+

},

96+

],

97+

},

98+

]);

99+100+

const repaired = await repairIMessageConversationAnchor({

101+

client: client as never,

102+

message,

103+

});

104+105+

expect(repaired).toMatchObject({

106+

chat_id: 349,

107+

chat_guid: "iMessage;+;chat349",

108+

chat_identifier: "chat349",

109+

chat_name: "Project group",

110+

participants: ["+15550001111", "+15550002222"],

111+

is_group: true,

112+

});

113+

});

114+115+

it("drops fail-closed when the GUID cannot be matched", async () => {

116+

const runtime = { error: vi.fn() };

117+

const client = mockClient([{ id: 349, messages: [{ guid: "OTHER-GUID", chat_id: 349 }] }]);

118+119+

await expect(

120+

repairIMessageConversationAnchor({

121+

client: client as never,

122+

message: anchorlessMessage(),

123+

runtime,

124+

}),

125+

).resolves.toBeNull();

126+

expect(runtime.error.mock.calls.at(-1)?.[0]).toContain("no recent chat matched");

127+

});

128+129+

it("drops fail-closed when history finds the GUID but no usable anchor", async () => {

130+

const runtime = { error: vi.fn() };

131+

const client = mockClient([

132+

{

133+

id: 349,

134+

messages: [

135+

{

136+

guid: "ANCHORLESS-GUID-1",

137+

chat_id: 0,

138+

chat_guid: "",

139+

chat_identifier: "",

140+

is_group: false,

141+

},

142+

],

143+

},

144+

]);

145+146+

await expect(

147+

repairIMessageConversationAnchor({

148+

client: client as never,

149+

message: anchorlessMessage(),

150+

runtime,

151+

}),

152+

).resolves.toBeNull();

153+

expect(runtime.error.mock.calls.at(-1)?.[0]).toContain("no usable conversation anchor");

154+

});

155+

});