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

推荐订阅源

大猫的无限游戏
大猫的无限游戏
T
Threatpost
Cyber Security Advisories - MS-ISAC
Cyber Security Advisories - MS-ISAC
博客园 - 聂微东
让小产品的独立变现更简单 - ezindie.com
让小产品的独立变现更简单 - ezindie.com
阮一峰的网络日志
阮一峰的网络日志
Jina AI
Jina AI
Google DeepMind News
Google DeepMind News
freeCodeCamp Programming Tutorials: Python, JavaScript, Git & More
Martin Fowler
Martin Fowler
GbyAI
GbyAI
N
News and Events Feed by Topic
Cloudbric
Cloudbric
博客园 - 【当耐特】
Google Online Security Blog
Google Online Security Blog
TaoSecurity Blog
TaoSecurity Blog
S
Secure Thoughts
Microsoft Security Blog
Microsoft Security Blog
Application and Cybersecurity Blog
Application and Cybersecurity Blog
Forbes - Security
Forbes - Security
E
Exploit-DB.com RSS Feed
Hacker News - Newest:
Hacker News - Newest: "LLM"
L
LangChain Blog
T
Tailwind CSS Blog
Attack and Defense Labs
Attack and Defense Labs
人人都是产品经理
人人都是产品经理
V
Vulnerabilities – Threatpost
Security Latest
Security Latest
Engineering at Meta
Engineering at Meta
T
Tenable Blog
P
Privacy & Cybersecurity Law Blog
L
LINUX DO - 热门话题
S
SegmentFault 最新的问题
酷 壳 – CoolShell
酷 壳 – CoolShell
Latest news
Latest news
AI
AI
Microsoft Azure Blog
Microsoft Azure Blog
Apple Machine Learning Research
Apple Machine Learning Research
S
Security @ Cisco Blogs
H
Help Net Security
Webroot Blog
Webroot Blog
A
About on SuperTechFans
Y
Y Combinator Blog
博客园 - Franky
V
V2EX - 技术
腾讯CDC
N
News | PayPal Newsroom
Recent Commits to openclaw:main
Recent Commits to openclaw:main
P
Palo Alto Networks Blog
C
Cyber Attacks, Cyber Crime and Cyber Security

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(sessions): guard unsafe transcript serialization cache · openclaw/openclaw@568f2d5
vincentkoc · 2026-06-16 · via Recent Commits to openclaw:main

@@ -245,6 +245,54 @@ describe("SessionManager.open", () => {

245245

}

246246

});

247247248+

it("publishes owned snapshots when a safe append pushes the transcript over the cache limit", async () => {

249+

const dir = await makeTempDir();

250+

const sessionFile = path.join(dir, "large-session.jsonl");

251+

const maxCachedSessionBytes = 32 * 1024 * 1024;

252+

const headerLine = JSON.stringify(buildSessionHeader(dir));

253+

const largeEntryBase = {

254+

type: "message",

255+

id: "assistant-1",

256+

parentId: null,

257+

timestamp: "2026-06-04T00:00:01.000Z",

258+

message: buildAssistantMessage(""),

259+

};

260+

const initialTranscriptWithContent = (content: string) =>

261+

`${headerLine}\n${JSON.stringify({

262+

...largeEntryBase,

263+

message: buildAssistantMessage(content),

264+

})}\n`;

265+

let filler = "x".repeat(

266+

maxCachedSessionBytes - Buffer.byteLength(initialTranscriptWithContent(""), "utf8") - 16,

267+

);

268+

while (

269+

Buffer.byteLength(initialTranscriptWithContent(filler), "utf8") >

270+

maxCachedSessionBytes - 16

271+

) {

272+

filler = filler.slice(0, -1024);

273+

}

274+

await fs.writeFile(sessionFile, initialTranscriptWithContent(filler), "utf8");

275+276+

const sessionManager = SessionManager.open(sessionFile, dir, dir);

277+

const publishSessionFileSnapshot = vi.fn(() => true);

278+

await withOwnedSessionTranscriptWrites(

279+

{

280+

sessionFile,

281+

canAdvanceSessionEntryCache: () => true,

282+

publishSessionFileSnapshot,

283+

withSessionWriteLock: async (run) => await run(),

284+

},

285+

async () => {

286+

sessionManager.appendMessage(buildAssistantMessage("small append"));

287+

},

288+

);

289+290+

expect(Buffer.byteLength(await fs.readFile(sessionFile, "utf8"), "utf8")).toBeGreaterThan(

291+

maxCachedSessionBytes,

292+

);

293+

expect(publishSessionFileSnapshot).toHaveBeenCalledTimes(1);

294+

});

295+248296

it("invalidates warm entries after an append outside the owned write context", async () => {

249297

const dir = await makeTempDir();

250298

const sessionFile = path.join(dir, "session.jsonl");

@@ -370,7 +418,7 @@ describe("SessionManager.open", () => {

370418

expect(warmEntry).toMatchObject({ data: { value: "first" } });

371419

});

372420373-

it("validates the transcript prefix after extension-owned entries are serialized", async () => {

421+

it("validates the transcript prefix after entries with custom serializers are serialized", async () => {

374422

const appenders: Array<{

375423

name: string;

376424

append: (manager: SessionManager, value: unknown) => void;

@@ -402,55 +450,144 @@ describe("SessionManager.open", () => {

402450

append: (manager, value) =>

403451

manager.branchWithSummary("assistant-1", "summary", { value }, true),

404452

},

453+

{

454+

name: "tool_result_details",

455+

append: (manager, value) =>

456+

manager.appendMessage({

457+

role: "toolResult",

458+

toolCallId: "call-1",

459+

toolName: "test",

460+

content: [{ type: "text", text: "ok" }],

461+

details: { value },

462+

isError: false,

463+

timestamp: Date.now(),

464+

} as Parameters<SessionManager["appendMessage"]>[0]),

465+

},

405466

];

406467407-

for (const { name, append } of appenders) {

408-

const dir = await makeTempDir();

409-

const sessionFile = path.join(dir, `${name}.jsonl`);

410-

const originalEntry = {

411-

type: "message",

412-

id: "assistant-1",

413-

parentId: null,

414-

timestamp: "2026-06-04T00:00:01.000Z",

415-

message: buildAssistantMessage("message 1"),

416-

};

417-

const replacementEntry = {

418-

...originalEntry,

419-

message: buildAssistantMessage("changed 1"),

468+

const serializerCases: Array<{

469+

name: string;

470+

createValue: (rewriteTranscript: () => void) => {

471+

value: unknown;

472+

cleanup?: () => void;

420473

};

421-

const headerLine = JSON.stringify(buildSessionHeader(dir));

422-

await fs.writeFile(sessionFile, `${headerLine}\n${JSON.stringify(originalEntry)}\n`, "utf8");

423-424-

const sessionManager = SessionManager.open(sessionFile, dir, dir);

425-

const publishSessionFileSnapshot = vi.fn(() => true);

426-

await withOwnedSessionTranscriptWrites(

427-

{

428-

sessionFile,

429-

canAdvanceSessionEntryCache: () => true,

430-

publishSessionFileSnapshot,

431-

withSessionWriteLock: async (run) => await run(),

432-

},

433-

async () => {

434-

append(sessionManager, {

474+

}> = [

475+

{

476+

name: "own_to_json",

477+

createValue: (rewriteTranscript) => ({

478+

value: {

435479

toJSON() {

436-

writeFileSync(

437-

sessionFile,

438-

`${headerLine}\n${JSON.stringify(replacementEntry)}\n`,

439-

"utf8",

440-

);

480+

rewriteTranscript();

441481

return "persisted";

442482

},

483+

},

484+

}),

485+

},

486+

{

487+

name: "non_enumerable_array_index",

488+

createValue: (rewriteTranscript) => {

489+

const array = ["placeholder"];

490+

Object.defineProperty(array, "0", {

491+

configurable: true,

492+

enumerable: false,

493+

value: {

494+

toJSON() {

495+

rewriteTranscript();

496+

return "persisted";

497+

},

498+

},

443499

});

500+

return { value: array };

444501

},

445-

);

502+

},

503+

{

504+

name: "bigint_to_json",

505+

createValue: (rewriteTranscript) => {

506+

const originalBigIntToJson = Object.getOwnPropertyDescriptor(BigInt.prototype, "toJSON");

507+

// eslint-disable-next-line no-extend-native -- JSON.stringify invokes BigInt.prototype.toJSON when present.

508+

Object.defineProperty(BigInt.prototype, "toJSON", {

509+

configurable: true,

510+

value() {

511+

rewriteTranscript();

512+

return "persisted";

513+

},

514+

});

515+

return {

516+

value: 1n,

517+

cleanup: () => {

518+

if (originalBigIntToJson) {

519+

// eslint-disable-next-line no-extend-native -- Restore the serializer installed for this case.

520+

Object.defineProperty(BigInt.prototype, "toJSON", originalBigIntToJson);

521+

} else {

522+

delete (BigInt.prototype as { toJSON?: unknown }).toJSON;

523+

}

524+

},

525+

};

526+

},

527+

},

528+

];

446529447-

expect(

448-

SessionManager.open(sessionFile, dir, dir)

449-

.getEntries()

450-

.filter((entry) => entry.type === "message")

451-

.map((entry) => readMessageContent(entry)),

452-

).toEqual(["changed 1"]);

453-

expect(publishSessionFileSnapshot).toHaveBeenCalledTimes(1);

530+

for (const { name, append } of appenders) {

531+

for (const serializerCase of serializerCases) {

532+

const dir = await makeTempDir();

533+

const sessionFile = path.join(dir, `${name}-${serializerCase.name}.jsonl`);

534+

const originalEntry = {

535+

type: "message",

536+

id: "assistant-1",

537+

parentId: null,

538+

timestamp: "2026-06-04T00:00:01.000Z",

539+

message: buildAssistantMessage("message 1"),

540+

};

541+

const replacementEntry = {

542+

...originalEntry,

543+

message: buildAssistantMessage("changed 1"),

544+

};

545+

const headerLine = JSON.stringify(buildSessionHeader(dir));

546+

await fs.writeFile(

547+

sessionFile,

548+

`${headerLine}\n${JSON.stringify(originalEntry)}\n`,

549+

"utf8",

550+

);

551+552+

const sessionManager = SessionManager.open(sessionFile, dir, dir);

553+

let cacheAdvanceChecks = 0;

554+

const publishSessionFileSnapshot = vi.fn(() => true);

555+

const { value, cleanup } = serializerCase.createValue(() => {

556+

writeFileSync(

557+

sessionFile,

558+

`${headerLine}\n${JSON.stringify(replacementEntry)}\n`,

559+

"utf8",

560+

);

561+

});

562+563+

try {

564+

await withOwnedSessionTranscriptWrites(

565+

{

566+

sessionFile,

567+

canAdvanceSessionEntryCache: () => {

568+

cacheAdvanceChecks += 1;

569+

return true;

570+

},

571+

publishSessionFileSnapshot,

572+

withSessionWriteLock: async (run) => await run(),

573+

},

574+

async () => {

575+

append(sessionManager, value);

576+

},

577+

);

578+

} finally {

579+

cleanup?.();

580+

}

581+582+

expect(

583+

SessionManager.open(sessionFile, dir, dir)

584+

.getEntries()

585+

.filter((entry) => entry.type === "message")

586+

.map((entry) => readMessageContent(entry)),

587+

).toEqual(name === "tool_result_details" ? ["changed 1", "ok"] : ["changed 1"]);

588+

expect(cacheAdvanceChecks, `${name}/${serializerCase.name}`).toBe(0);

589+

expect(publishSessionFileSnapshot, `${name}/${serializerCase.name}`).not.toHaveBeenCalled();

590+

}

454591

}

455592

});

456593