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

推荐订阅源

Cisco Talos Blog
Cisco Talos Blog
S
Securelist
C
Cisco Blogs
D
DataBreaches.Net
钛媒体:引领未来商业与生活新知
钛媒体:引领未来商业与生活新知
V
Vulnerabilities – Threatpost
Latest news
Latest news
T
The Exploit Database - CXSecurity.com
小众软件
小众软件
S
SegmentFault 最新的问题
罗磊的独立博客
I
Intezer
雷峰网
雷峰网
T
Threatpost
博客园 - 叶小钗
阮一峰的网络日志
阮一峰的网络日志
A
About on SuperTechFans
AWS News Blog
AWS News Blog
A
Arctic Wolf
P
Privacy International News Feed
The Register - Security
The Register - Security
Vercel News
Vercel News
L
LangChain Blog
S
Schneier on Security
D
Docker
J
Java Code Geeks
L
LINUX DO - 热门话题
cs.CL updates on arXiv.org
cs.CL updates on arXiv.org
M
MIT News - Artificial intelligence
Spread Privacy
Spread Privacy
MyScale Blog
MyScale Blog
量子位
Cyber Security Advisories - MS-ISAC
Cyber Security Advisories - MS-ISAC
K
Kaspersky official blog
C
CERT Recently Published Vulnerability Notes
Know Your Adversary
Know Your Adversary
D
Darknet – Hacking Tools, Hacker News & Cyber Security
Recorded Future
Recorded Future
C
Cyber Attacks, Cyber Crime and Cyber Security
Scott Helme
Scott Helme
Security Latest
Security Latest
人人都是产品经理
人人都是产品经理
T
Threat Research - Cisco Blogs
Cyberwarzone
Cyberwarzone
F
Full Disclosure
Threat Intelligence Blog | Flashpoint
Threat Intelligence Blog | Flashpoint
Jina AI
Jina AI
NISL@THU
NISL@THU
P
Proofpoint News Feed
T
The Blog of Author Tim Ferriss

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
refactor: share sessions list changed test helpers · openclaw/openclaw@9d9a614
vincentkoc · 2026-06-02 · via Recent Commits to openclaw:main

@@ -15,6 +15,8 @@ const { createSessionStoreDir, openClient } = setupGatewaySessionsTestHarness();

1515

type MockCalls = {

1616

mock: { calls: unknown[][] };

1717

};

18+

type SessionStoreEntryOptions = Parameters<typeof sessionStoreEntry>[1];

19+

type MutationMethod = "sessions.patch" | "sessions.compact";

18201921

function isRecord(value: unknown): value is Record<string, unknown> {

2022

return typeof value === "object" && value !== null;

@@ -116,20 +118,31 @@ async function invokeSessionsList({

116118

return { request, respond };

117119

}

118120119-

async function invokeSessionsPatch(params: Record<string, unknown>) {

121+

async function invokeSessionMutation({

122+

method,

123+

params,

124+

context = {},

125+

subscribedConnIds = new Set(["conn-1"]),

126+

}: {

127+

method: MutationMethod;

128+

params: Record<string, unknown>;

129+

context?: Record<string, unknown>;

130+

subscribedConnIds?: Set<string>;

131+

}) {

120132

const broadcastToConnIds = vi.fn();

121133

const respond = vi.fn();

122134

const sessionsHandlers = await getSessionsHandlers();

123135

const { getRuntimeConfig } = await getGatewayConfigModule();

124-

await sessionsHandlers["sessions.patch"]({

136+

await sessionsHandlers[method]({

125137

req: {} as never,

126138

params,

127139

respond,

128140

context: {

129141

broadcastToConnIds,

130-

getSessionEventSubscriberConnIds: () => new Set(["conn-1"]),

142+

getSessionEventSubscriberConnIds: () => subscribedConnIds,

131143

loadGatewayModelCatalog: async () => ({ providers: [] }),

132144

getRuntimeConfig,

145+

...context,

133146

} as never,

134147

client: null,

135148

isWebchatConnect: () => false,

@@ -140,6 +153,31 @@ async function invokeSessionsPatch(params: Record<string, unknown>) {

140153

};

141154

}

142155156+

async function invokeSessionsPatch(params: Record<string, unknown>) {

157+

return invokeSessionMutation({ method: "sessions.patch", params });

158+

}

159+160+

async function writeMainSessionStore(options?: SessionStoreEntryOptions) {

161+

await createSessionStoreDir();

162+

await writeSessionStore({

163+

entries: {

164+

main: sessionStoreEntry("sess-main", options),

165+

},

166+

});

167+

}

168+169+

function expectMainPatchBroadcast(

170+

result: Awaited<ReturnType<typeof invokeSessionsPatch>>,

171+

expected: Record<string, unknown>,

172+

) {

173+

expectFields(result.responsePayload, { ok: true, key: "agent:main:main" });

174+

expectChangedBroadcast(result.broadcastToConnIds, {

175+

sessionKey: "agent:main:main",

176+

reason: "patch",

177+

...expected,

178+

});

179+

}

180+143181

async function setupGlobalAgentSessionStores({

144182

writePrimeStore = false,

145183

withTranscripts = false,

@@ -257,25 +295,33 @@ async function invokeSessionsCompact({

257295

params: Record<string, unknown>;

258296

subscribedConnIds?: Set<string>;

259297

}) {

260-

const broadcastToConnIds = vi.fn();

261-

const respond = vi.fn();

262-

const sessionsHandlers = await getSessionsHandlers();

263-

await sessionsHandlers["sessions.compact"]({

264-

req: {} as never,

298+

return invokeSessionMutation({

299+

method: "sessions.compact",

265300

params,

266-

respond,

267301

context: {

268-

broadcastToConnIds,

269-

getSessionEventSubscriberConnIds: () => subscribedConnIds,

270302

getRuntimeConfig,

271-

} as never,

272-

client: null,

273-

isWebchatConnect: () => false,

303+

},

304+

subscribedConnIds,

274305

});

275-

return {

276-

broadcastToConnIds,

277-

responsePayload: expectRespondPayload(respond),

278-

};

306+

}

307+308+

async function expectListedSessionActiveRun(

309+

requestId: string,

310+

run: Record<string, unknown>,

311+

expected: boolean,

312+

) {

313+

await writeMainSessionStore();

314+315+

const { respond } = await invokeSessionsList({

316+

requestId,

317+

context: {

318+

chatAbortControllers: new Map([["run-1", { sessionKey: "agent:main:main", ...run }]]),

319+

},

320+

});

321+322+

const payload = expectRespondPayload(respond);

323+

const session = findSession(payload, "agent:main:main");

324+

expect(session.hasActiveRun).toBe(expected);

279325

}

280326281327

test("sessions.list keeps bulk rows lightweight and uses persisted model fields", async () => {

@@ -369,17 +415,12 @@ test("sessions.list keeps bulk rows lightweight and uses persisted model fields"

369415

});

370416371417

test("sessions.list uses the gateway model catalog for effective thinking defaults", async () => {

372-

await createSessionStoreDir();

373418

testState.agentConfig = {

374419

model: { primary: "test-provider/reasoner" },

375420

};

376-

await writeSessionStore({

377-

entries: {

378-

main: sessionStoreEntry("sess-main", {

379-

modelProvider: "test-provider",

380-

model: "reasoner",

381-

}),

382-

},

421+

await writeMainSessionStore({

422+

modelProvider: "test-provider",

423+

model: "reasoner",

383424

});

384425385426

const { respond } = await invokeSessionsList({

@@ -407,67 +448,23 @@ test("sessions.list uses the gateway model catalog for effective thinking defaul

407448

});

408449409450

test("sessions.list marks sessions with active abortable runs", async () => {

410-

await createSessionStoreDir();

411-

await writeSessionStore({

412-

entries: {

413-

main: sessionStoreEntry("sess-main"),

414-

},

415-

});

416-417-

const { respond } = await invokeSessionsList({

418-

requestId: "req-sessions-list-active-run",

419-

context: {

420-

chatAbortControllers: new Map([["run-1", { sessionKey: "agent:main:main" }]]),

421-

},

422-

});

423-424-

const payload = expectRespondPayload(respond);

425-

const session = findSession(payload, "agent:main:main");

426-

expect(session.hasActiveRun).toBe(true);

451+

await expectListedSessionActiveRun("req-sessions-list-active-run", {}, true);

427452

});

428453429454

test("sessions.list ignores terminal abortable runs kept for retry guards", async () => {

430-

await createSessionStoreDir();

431-

await writeSessionStore({

432-

entries: {

433-

main: sessionStoreEntry("sess-main"),

434-

},

435-

});

436-437-

const { respond } = await invokeSessionsList({

438-

requestId: "req-sessions-list-terminal-run",

439-

context: {

440-

chatAbortControllers: new Map([

441-

["run-1", { sessionKey: "agent:main:main", projectSessionActive: false }],

442-

]),

443-

},

444-

});

445-446-

const payload = expectRespondPayload(respond);

447-

const session = findSession(payload, "agent:main:main");

448-

expect(session.hasActiveRun).toBe(false);

455+

await expectListedSessionActiveRun(

456+

"req-sessions-list-terminal-run",

457+

{ projectSessionActive: false },

458+

false,

459+

);

449460

});

450461451462

test("sessions.list ignores hidden internal abortable runs", async () => {

452-

await createSessionStoreDir();

453-

await writeSessionStore({

454-

entries: {

455-

main: sessionStoreEntry("sess-main"),

456-

},

457-

});

458-459-

const { respond } = await invokeSessionsList({

460-

requestId: "req-sessions-list-hidden-run",

461-

context: {

462-

chatAbortControllers: new Map([

463-

["run-1", { sessionKey: "agent:main:main", controlUiVisible: false }],

464-

]),

465-

},

466-

});

467-468-

const payload = expectRespondPayload(respond);

469-

const session = findSession(payload, "agent:main:main");

470-

expect(session.hasActiveRun).toBe(false);

463+

await expectListedSessionActiveRun(

464+

"req-sessions-list-hidden-run",

465+

{ controlUiVisible: false },

466+

false,

467+

);

471468

});

472469473470

test("sessions.list yields before responding during bulk transcript hydration", async () => {

@@ -518,12 +515,7 @@ test("sessions.list yields before responding during bulk transcript hydration",

518515

});

519516520517

test("sessions.list does not block on slow model catalog discovery", async () => {

521-

await createSessionStoreDir();

522-

await writeSessionStore({

523-

entries: {

524-

main: sessionStoreEntry("sess-main"),

525-

},

526-

});

518+

await writeMainSessionStore();

527519528520

vi.useFakeTimers();

529521

try {

@@ -586,15 +578,12 @@ test("sessions.changed mutation events include live usage metadata", async () =>

586578

},

587579

});

588580589-

const { broadcastToConnIds, responsePayload } = await invokeSessionsPatch({

581+

const result = await invokeSessionsPatch({

590582

key: "main",

591583

label: "Renamed",

592584

});

593585594-

expectFields(responsePayload, { ok: true, key: "agent:main:main" });

595-

expectChangedBroadcast(broadcastToConnIds, {

596-

sessionKey: "agent:main:main",

597-

reason: "patch",

586+

expectMainPatchBroadcast(result, {

598587

totalTokens: 6_643,

599588

totalTokensFresh: true,

600589

contextTokens: 123_456,

@@ -605,59 +594,36 @@ test("sessions.changed mutation events include live usage metadata", async () =>

605594

});

606595607596

test("sessions.changed mutation events include live session setting metadata", async () => {

608-

await createSessionStoreDir();

609-

await writeSessionStore({

610-

entries: {

611-

main: sessionStoreEntry("sess-main", {

612-

verboseLevel: "on",

613-

responseUsage: "full",

614-

fastMode: true,

615-

lastChannel: "telegram",

616-

lastTo: "-100123",

617-

lastAccountId: "acct-1",

618-

lastThreadId: 42,

619-

}),

620-

},

621-

});

622-623-

const { broadcastToConnIds, responsePayload } = await invokeSessionsPatch({

624-

key: "main",

625-

verboseLevel: "on",

626-

});

627-628-

expectFields(responsePayload, { ok: true, key: "agent:main:main" });

629-

expectChangedBroadcast(broadcastToConnIds, {

630-

sessionKey: "agent:main:main",

631-

reason: "patch",

597+

const sessionSettings = {

632598

verboseLevel: "on",

633599

responseUsage: "full",

634600

fastMode: true,

635601

lastChannel: "telegram",

636602

lastTo: "-100123",

637603

lastAccountId: "acct-1",

638604

lastThreadId: 42,

605+

} satisfies SessionStoreEntryOptions;

606+

await writeMainSessionStore(sessionSettings);

607+608+

const result = await invokeSessionsPatch({

609+

key: "main",

610+

verboseLevel: "on",

639611

});

612+613+

expectMainPatchBroadcast(result, sessionSettings);

640614

});

641615642616

test("sessions.changed mutation events include sendPolicy metadata", async () => {

643-

await createSessionStoreDir();

644-

await writeSessionStore({

645-

entries: {

646-

main: sessionStoreEntry("sess-main", {

647-

sendPolicy: "deny",

648-

}),

649-

},

617+

await writeMainSessionStore({

618+

sendPolicy: "deny",

650619

});

651620652-

const { broadcastToConnIds, responsePayload } = await invokeSessionsPatch({

621+

const result = await invokeSessionsPatch({

653622

key: "main",

654623

sendPolicy: "deny",

655624

});

656625657-

expectFields(responsePayload, { ok: true, key: "agent:main:main" });

658-

expectChangedBroadcast(broadcastToConnIds, {

659-

sessionKey: "agent:main:main",

660-

reason: "patch",

626+

expectMainPatchBroadcast(result, {

661627

sendPolicy: "deny",

662628

});

663629

});