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

推荐订阅源

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(auto-reply): suppress repeated silent tokens (#86848) · openclaw/openclaw@8ec2b2d fix(commands): preserve async skill commands · openclaw/openclaw@1313e15 fix(docker): bound telegram npm installs · openclaw/openclaw@130464e fix(mac): use corepack pnpm for app packaging · openclaw/openclaw@728b61a fix: mark ios watch app as watchkit app · openclaw/openclaw@1600bcd test: fix bundled install mock typing · openclaw/openclaw@669bfdd docs: explain bundled plugin npm override · openclaw/openclaw@40fa750 fix: keep bundled OpenClaw plugins image-owned · openclaw/openclaw@771675e fix: preserve whatsapp inbound batch order · openclaw/openclaw@84a33c7 perf: cache npm globalconfig lookups · openclaw/openclaw@3f524a6 fix(daemon): ignore recursive Windows gateway wrapper · openclaw/openclaw@126a336 fix(docker): bound live setup commands · openclaw/openclaw@eb15c44 fix: restore ios build stability · openclaw/openclaw@1daef79 feat(plugin-sdk): add reaction approval helpers (#86735) · openclaw/openclaw@7d6b7f4 test(auto-reply): type manifest catalog harness mock · openclaw/openclaw@4f83cd6 fix(docker): bound live docker runs · openclaw/openclaw@96307ca test(auto-reply): mock manifest model catalog in trigger harness · openclaw/openclaw@989d449 fix(crabbox): scope env-wrapped macOS bootstrap · openclaw/openclaw@2f7bfdb fix(channel): handle plugin channel markdown fallback · openclaw/openclaw@6158742 fix(docker): require bounded e2e docker commands · openclaw/openclaw@3736d7b fix(codex): share native hook relay registry (#73950) · openclaw/openclaw@6729dea fix(release): stabilize plugin prerelease tests · openclaw/openclaw@5a684c4 fix(diagnostics): flush OTel trace batches · openclaw/openclaw@c4b9f54 fix(memory): reject invalid CLI numeric options · openclaw/openclaw@d569e41 fix(codex): bound app-server timeout fallout · openclaw/openclaw@5a7d5c6 fix(agents): keep model browse normalization bounded · openclaw/openclaw@9fc71e9 fix: stabilize media-related tests · openclaw/openclaw@a818556 fix(ci): preserve docker pull retry failures · openclaw/openclaw@be2213e fix(build): stabilize shrinkwrap generation · openclaw/openclaw@538b537 fix(ui): ignore stale running session rows · openclaw/openclaw@1705189 ci: support windows node download fallback · openclaw/openclaw@bb48fcf test(agents): pin native anthropic replay policy · openclaw/openclaw@acd3ce0 fix(status): surface systemd gateway hygiene (#86976) · openclaw/openclaw@0a085bf fix(ui): show failed tool results as errors (#85786) ci: allow Windows Node 22 patch range · openclaw/openclaw@ce4db4f ci: enforce Node 22 floor in setup helper · openclaw/openclaw@1d972af Fix status JSON plugin scan (#87001) · openclaw/openclaw@f3e6158 fix(telegram): preserve command slots for aliases (#85270) · openclaw/openclaw@77505da fix(agents): handle deferred maintenance drain · openclaw/openclaw@94fb547 test: keep legacy tool-result error proof ci: fix post-merge Rastermill checks · openclaw/openclaw@b546998 fix(agents): mark repaired legacy tool results errored · openclaw/openclaw@8523d32 docs(changelog): note rastermill exif fix docs(changelog): regroup 2026.5.26 release notes · openclaw/openclaw@a6973ab fix: keep EXIF normalization best-effort (#86923) · openclaw/openclaw@acb942f build: update rastermill pin · openclaw/openclaw@7d4d751 build: use rastermill 0.3.0 · openclaw/openclaw@cee8c87 ci: normalize Windows toolcache paths · openclaw/openclaw@03ae999 refactor: use unified rastermill encode API · openclaw/openclaw@a3325c9 ci: satisfy opengrep git add guard · openclaw/openclaw@16d06aa refactor: delegate image processing to Rastermill · openclaw/openclaw@50b98a1 build: update rastermill dependency · openclaw/openclaw@e6edcca fix: infer realtime smoke dev server type · openclaw/openclaw@4e84229 refactor: delegate image limits to Rastermill · openclaw/openclaw@4f728f8 fix(agents): repair legacy tool results before replay · openclaw/openclaw@4e45b11 fix(config): narrow profiled tool section doctor repair (#87030) · openclaw/openclaw@3c16648 test: fix current suite drift · openclaw/openclaw@80655fe fix(lock): require owner identity proof before stale removal · openclaw/openclaw@daa7b1d fix(deps): pin shrinkwrap patch drift to pnpm lock · openclaw/openclaw@d8a14e7 revert: 60bec8c duplicate tool display guard · openclaw/openclaw@e09f89d fix(e2e): bound docker package preparation · openclaw/openclaw@38edae7 fix(cli): add Windows stack-size respawn (#87031) · openclaw/openclaw@5e8f498 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
fix(gateway): reject RPCs from invalidated device-token clients durin… · openclaw/openclaw@1e1cf14
davidangular · 2026-05-27 · via Recent Commits to openclaw:main

@@ -1,6 +1,77 @@

11

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

22

import type { GatewayServerLiveState } from "./server-live-state.js";

3-

import { createGatewayRequestContext } from "./server-request-context.js";

3+

import {

4+

createGatewayRequestContext,

5+

type GatewayRequestContextParams,

6+

} from "./server-request-context.js";

7+8+

function makeContextParams(

9+

overrides: Partial<GatewayRequestContextParams> = {},

10+

): GatewayRequestContextParams {

11+

const runtimeState: Pick<GatewayServerLiveState, "cronState"> = {

12+

cronState: {

13+

cron: { start: vi.fn(), stop: vi.fn() } as never,

14+

storePath: "/tmp/cron",

15+

cronEnabled: true,

16+

},

17+

};

18+

return {

19+

deps: {} as never,

20+

runtimeState,

21+

getRuntimeConfig: vi.fn(() => ({}) as never),

22+

execApprovalManager: undefined,

23+

pluginApprovalManager: undefined,

24+

loadGatewayModelCatalog: vi.fn(async () => []),

25+

getHealthCache: vi.fn(() => null),

26+

refreshHealthSnapshot: vi.fn(async () => ({}) as never),

27+

logHealth: { error: vi.fn() },

28+

logGateway: { warn: vi.fn(), info: vi.fn(), error: vi.fn() } as never,

29+

incrementPresenceVersion: vi.fn(() => 1),

30+

getHealthVersion: vi.fn(() => 1),

31+

broadcast: vi.fn(),

32+

broadcastToConnIds: vi.fn(),

33+

nodeSendToSession: vi.fn(),

34+

nodeSendToAllSubscribed: vi.fn(),

35+

nodeSubscribe: vi.fn(),

36+

nodeUnsubscribe: vi.fn(),

37+

nodeUnsubscribeAll: vi.fn(),

38+

hasConnectedTalkNode: vi.fn(() => false),

39+

clients: new Set(),

40+

enforceSharedGatewayAuthGenerationForConfigWrite: vi.fn(),

41+

nodeRegistry: {} as never,

42+

agentRunSeq: new Map(),

43+

chatAbortControllers: new Map(),

44+

chatAbortedRuns: new Map(),

45+

chatRunBuffers: new Map(),

46+

chatDeltaSentAt: new Map(),

47+

chatDeltaLastBroadcastLen: new Map(),

48+

chatDeltaLastBroadcastText: new Map(),

49+

agentDeltaSentAt: new Map(),

50+

bufferedAgentEvents: new Map(),

51+

addChatRun: vi.fn(),

52+

removeChatRun: vi.fn(),

53+

subscribeSessionEvents: vi.fn(),

54+

unsubscribeSessionEvents: vi.fn(),

55+

subscribeSessionMessageEvents: vi.fn(),

56+

unsubscribeSessionMessageEvents: vi.fn(),

57+

unsubscribeAllSessionEvents: vi.fn(),

58+

getSessionEventSubscriberConnIds: vi.fn(() => new Set<string>()),

59+

registerToolEventRecipient: vi.fn(),

60+

dedupe: new Map(),

61+

wizardSessions: new Map(),

62+

findRunningWizard: vi.fn(() => null),

63+

purgeWizardSession: vi.fn(),

64+

getRuntimeSnapshot: vi.fn(() => ({}) as never),

65+

startChannel: vi.fn(async () => undefined),

66+

stopChannel: vi.fn(async () => undefined),

67+

markChannelLoggedOut: vi.fn(),

68+

wizardRunner: vi.fn(async () => undefined),

69+

broadcastVoiceWakeChanged: vi.fn(),

70+

broadcastVoiceWakeRoutingChanged: vi.fn(),

71+

unavailableGatewayMethods: new Set(),

72+

...overrides,

73+

};

74+

}

475576

describe("createGatewayRequestContext", () => {

677

it("reads cron state live from runtime state", () => {

@@ -82,4 +153,66 @@ describe("createGatewayRequestContext", () => {

82153

expect(context.cron).toBe(cronB);

83154

expect(context.cronStorePath).toBe("/tmp/cron-b");

84155

});

156+157+

it("invalidateClientsForDevice sets the flag on matching clients without closing the socket", () => {

158+

const target = {

159+

connId: "conn-target",

160+

connect: { device: { id: "device-1" }, role: "primary" },

161+

socket: { close: vi.fn() },

162+

};

163+

const unrelated = {

164+

connId: "conn-unrelated",

165+

connect: { device: { id: "device-2" }, role: "primary" },

166+

socket: { close: vi.fn() },

167+

};

168+

const clients = new Set([target, unrelated]) as never;

169+170+

const context = createGatewayRequestContext(makeContextParams({ clients }));

171+

context.invalidateClientsForDevice?.("device-1", { reason: "device-token-rotated" });

172+173+

expect((target as { invalidated?: boolean }).invalidated).toBe(true);

174+

expect((target as { invalidatedReason?: string }).invalidatedReason).toBe(

175+

"device-token-rotated",

176+

);

177+

expect(target.socket.close).not.toHaveBeenCalled();

178+179+

expect((unrelated as { invalidated?: boolean }).invalidated).toBeUndefined();

180+

expect(unrelated.socket.close).not.toHaveBeenCalled();

181+

});

182+183+

it("disconnectClientsForDevice also marks the invalidated flag before closing", () => {

184+

const target = {

185+

connId: "conn-target",

186+

connect: { device: { id: "device-1" }, role: "primary" },

187+

socket: { close: vi.fn() },

188+

};

189+

const clients = new Set([target]) as never;

190+191+

const context = createGatewayRequestContext(makeContextParams({ clients }));

192+

context.disconnectClientsForDevice?.("device-1");

193+194+

expect((target as { invalidated?: boolean }).invalidated).toBe(true);

195+

expect((target as { invalidatedReason?: string }).invalidatedReason).toBe("device-removed");

196+

expect(target.socket.close).toHaveBeenCalledWith(4001, "device removed");

197+

});

198+199+

it("invalidateClientsForDevice filters by role when provided", () => {

200+

const primary = {

201+

connId: "conn-primary",

202+

connect: { device: { id: "device-1" }, role: "primary" },

203+

socket: { close: vi.fn() },

204+

};

205+

const secondary = {

206+

connId: "conn-secondary",

207+

connect: { device: { id: "device-1" }, role: "secondary" },

208+

socket: { close: vi.fn() },

209+

};

210+

const clients = new Set([primary, secondary]) as never;

211+212+

const context = createGatewayRequestContext(makeContextParams({ clients }));

213+

context.invalidateClientsForDevice?.("device-1", { role: "primary" });

214+215+

expect((primary as { invalidated?: boolean }).invalidated).toBe(true);

216+

expect((secondary as { invalidated?: boolean }).invalidated).toBeUndefined();

217+

});

85218

});