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

推荐订阅源

H
Help Net Security
T
ThreatConnect
SecWiki News
SecWiki News
F
Future of Privacy Forum
AWS News Blog
AWS News Blog
C
Cisco Blogs
A
Arctic Wolf
Vercel News
Vercel News
The GitHub Blog
The GitHub Blog
Scott Helme
Scott Helme
V
V2EX
博客园 - 叶小钗
阮一峰的网络日志
阮一峰的网络日志
K
Kaspersky official blog
G
Google Developers Blog
freeCodeCamp Programming Tutorials: Python, JavaScript, Git & More
P
Privacy International News Feed
C
Cyber Attacks, Cyber Crime and Cyber Security
N
News | PayPal Newsroom
Schneier on Security
Schneier on Security
NISL@THU
NISL@THU
Microsoft Azure Blog
Microsoft Azure Blog
量子位
The Hacker News
The Hacker News
Stack Overflow Blog
Stack Overflow Blog
Security Latest
Security Latest
M
Microsoft Research Blog - Microsoft Research
Google Online Security Blog
Google Online Security Blog
博客园_首页
C
CXSECURITY Database RSS Feed - CXSecurity.com
I
InfoQ
Google DeepMind News
Google DeepMind News
Y
Y Combinator Blog
The Cloudflare Blog
Microsoft Security Blog
Microsoft Security Blog
Martin Fowler
Martin Fowler
Cisco Talos Blog
Cisco Talos Blog
钛媒体:引领未来商业与生活新知
钛媒体:引领未来商业与生活新知
T
Troy Hunt's Blog
F
Fox-IT International blog
S
Security @ Cisco Blogs
博客园 - 司徒正美
cs.CV updates on arXiv.org
cs.CV updates on arXiv.org
C
Comments on: Blog
Threat Intelligence Blog | Flashpoint
Threat Intelligence Blog | Flashpoint
L
LINUX DO - 最新话题
GbyAI
GbyAI
Project Zero
Project Zero
腾讯CDC
T
Tailwind CSS Blog

Recent Commits to openclaw:main

refactor: share realtime output activity tracking (#86661) fix(memory-wiki): bound compile page reads (#86660) test(qa-matrix): use larger media coverage jpeg fix(cli): reject unknown command help roots (#81083) (thanks @YB0y) · openclaw/openclaw@bec7d56 test: improve test profiling helpers · openclaw/openclaw@68ab48b perf: reduce fuzzy matching allocations · openclaw/openclaw@ec7ad3b perf: reduce runtime cache churn · openclaw/openclaw@1531fe2 refactor: reuse forced consult coordinator in discord voice (#86656) · openclaw/openclaw@0164fd5 fix(scripts): drain codex-cli metadata stdout (#84239) (thanks @Iftek… fix(test): avoid discord voice tts activation tax · openclaw/openclaw@75ac0b5 fix(codex): disable native thread personality (#85891) (thanks @lastg… · openclaw/openclaw@0f35ec2 Refactor realtime voice turn context tracking (#86650) fix(discord): surface silent reply-delivery skips and remove runtime.… · openclaw/openclaw@3a48366 test(discord): cover deliver-lambda abort-skip path via processDiscor… · openclaw/openclaw@48adcb1 docs: update changelog for landed bug fixes · openclaw/openclaw@75c6cf2 test(qa-matrix): use valid media coverage jpeg · openclaw/openclaw@0f54221 fix(gmail-watcher): strip listeners from old process after settleProc… · openclaw/openclaw@0a38932 fix(gmail-watcher): prevent TDZ in settleProcess and guard exit handl… · openclaw/openclaw@94968c8 fix(hooks): stop existing Gmail watcher before re-entry to prevent leaks fix(codex): honor yolo app-server approval policy · openclaw/openclaw@7b30291 fix(usage-cost): invalidate durable cache on missing-cost semantics c… · openclaw/openclaw@9c79a0f fix(usage-cost): preserve transport-recorded positive cost for unpric… · openclaw/openclaw@6e85869 fix(usage-cost): only flag catalog-default zeros, preserve operator-c… · openclaw/openclaw@1670249 fix: treat zero-rate usage cost as unknown · openclaw/openclaw@116c600 fix(usage-cost): surface unpriced-model spend as missingCostEntries i… · openclaw/openclaw@1cc0a96 fix(irc): normalize channel route ids · openclaw/openclaw@9cb1e47 test(irc): cover transient channel join · openclaw/openclaw@c4c80ce fix(irc): store inbound channel routes as channel:#name and join befo… · openclaw/openclaw@63dee51 fix(test): harden macos onboarding e2e · openclaw/openclaw@cd96542 fix(agents): strip markdown code spans from IDENTITY.md values and la… · openclaw/openclaw@55c9a6b perf: reduce runtime cache churn · openclaw/openclaw@5b6d03e fix(pi-runner): flush blocks after compaction retry (#85288) (thanks … · openclaw/openclaw@0d4575a fix(gateway): abort stale agent runs on restart · openclaw/openclaw@a122d80 fix(ui): harden control e2e browser setup · openclaw/openclaw@4424daf fix(telegram): keep overlapping DM replies deliverable (#85361) (than… · openclaw/openclaw@0f67dfd fix(openai): route compaction through Codex auth provider (#86408) · openclaw/openclaw@f4cfa01 refactor: share realtime forced consult coordination · openclaw/openclaw@5dccba7 test(qa-lab): add runtime confidence reports build: refresh dependency pins (#86628) · openclaw/openclaw@cda7c30 test: port release validation stabilizers · openclaw/openclaw@9f7485e fix(cron): stop forcing message tool for delivery · openclaw/openclaw@c51fa0d fix(google): omit request config with cached content test: stabilize release validation test harnesses fix(test): bound kitchen sink command output · openclaw/openclaw@f1197ed fix(discord): stabilize realtime wake-name feedback test(config): guard legacy agentRuntime regression perf: precompute audio resample kernels fix(codex): allow env api-key app-server bootstrap · openclaw/openclaw@009b18c refactor: reuse shared coercion helpers (#86419) · openclaw/openclaw@77d9ac3 fix(cron): preserve runtime snapshot for isolated delivery · openclaw/openclaw@a98660e fix(test): model active assistant failover attempts · openclaw/openclaw@c55bee5 docs: update changelog for bug sweep landings test: fix mock signatures for tsgo · openclaw/openclaw@aa05c5c docs: document fail-closed behavior for rejected modelPatterns · openclaw/openclaw@36f269d fix(security): guard plugin modelPatterns with compileSafeRegex · openclaw/openclaw@117e082 docs(manifest): note safe-regex validation for modelPatterns · openclaw/openclaw@e7c7ee4 style: use bracket notation for __openclaw to satisfy no-underscore-d… · openclaw/openclaw@9a6c161 fix(security): escape field names in transcript regex extraction · openclaw/openclaw@fe8d99d test: tighten oversized metadata assertion to check exact id in __ope… · openclaw/openclaw@aff8e64 fix(logging): exit on stdout/stderr EPIPE instead of spinning · openclaw/openclaw@2aa5f17 fix(logging): preserve failure exit on EPIPE · openclaw/openclaw@623a60a fix(logging): keep string failure codes on EPIPE · openclaw/openclaw@78a1e7d fix(scripts): docs-spellcheck.sh fails on bash 3.2 with set -u · openclaw/openclaw@fef57f9 fix(docs): keep spellcheck bash 3.2-compatible · openclaw/openclaw@778fa87 fix(test): assert e2e agent reply payloads · openclaw/openclaw@74f3a1e test(gateway): pin live gateway models to pi runtime · openclaw/openclaw@c88f660 perf: speed up local TUI startup · openclaw/openclaw@a0023fb refactor: share realtime voice activation helpers (#86615) · openclaw/openclaw@d0ab0d9 fix(feishu): render native presentation buttons (#86588) · openclaw/openclaw@170e0aa fix(test): narrow plugin gauntlet prebuild · openclaw/openclaw@423f7d2 fix: route Discord gateway metadata through proxy (#86601) · openclaw/openclaw@5b6d409 fix: tighten Discord voice wake matching (#86595) · openclaw/openclaw@f00a912 refactor(logging): share diagnostic message lifecycle · openclaw/openclaw@baab4cf fix(cron): restore suspended lanes to default concurrency · openclaw/openclaw@e844d1d fix(auth): emit one-shot doctor-pointer warning for Keychain-only leg… · openclaw/openclaw@a61d530 fix(codex): recover stale preflight bindings (#86602) · openclaw/openclaw@9b9d897 fix(cron): preserve unsupported payload rows on writes · openclaw/openclaw@c916906 fix(cron): canonicalize preserved row ids · openclaw/openclaw@985bc93 test(cron): pin sequential duration regression · openclaw/openclaw@8351556 docs: update changelog for cron preservation (#86415) · openclaw/openclaw@bdc6b32 build: bump qs to patched release · openclaw/openclaw@9330b76 fix(status): prefer active OAuth for runtime aliases chore(acpx): bump bundled acpx to 0.10.0 · openclaw/openclaw@407cf8e docs: make changelog release-owned · openclaw/openclaw@c0f2d89 fix(google): stop appending preview to flash lite · openclaw/openclaw@915c820 docs: update changelog for bug sweep landings · openclaw/openclaw@cd7994f fix(crabbox): detect timed macos js commands · openclaw/openclaw@44bb0be fix(mantis): release telegram user leases on startup failure · openclaw/openclaw@cf27567 fix(agents): keep cron media completions run-scoped · openclaw/openclaw@f5d2db2 fix(agents): deliver stale cron media completions · openclaw/openclaw@f01b2a8 fix(agents): notify stale cron media failures · openclaw/openclaw@baf469f guide workspace-only scratch paths fix(cron): gate lifecycle diagnostic events behind isDiagnosticsEnabled · openclaw/openclaw@4853222 fix(cron): emit message.queued/processed for isolated-agent turns fix(cron): address review — drop unsupported taskLabel, pair with ses… · openclaw/openclaw@804a31e fix(cron): report rotated session in final diagnostics · openclaw/openclaw@207a5a2 fix: hydrate current turn image attachments · openclaw/openclaw@b5ada80 fix(gateway): ignore inherited launchd env for respawn · openclaw/openclaw@177ebdc fix(test): preserve undici exports in discord proxy tests · openclaw/openclaw@b0c8a4d fix: raise default cron concurrency · openclaw/openclaw@bc12e04
perf: reduce runtime metadata hotpath churn · openclaw/openclaw@1d21224
steipete · 2026-05-26 · via Recent Commits to openclaw:main

@@ -124,56 +124,49 @@ function pickMemoRelevantEnv(env: NodeJS.ProcessEnv): Record<string, string> {

124124

);

125125

}

126126127-

function cloneOwnerMaps(owners: PluginMetadataSnapshotOwnerMaps): PluginMetadataSnapshotOwnerMaps {

128-

return {

129-

channels: new Map(owners.channels),

130-

channelConfigs: new Map(owners.channelConfigs),

131-

providers: new Map(owners.providers),

132-

modelCatalogProviders: new Map(owners.modelCatalogProviders),

133-

cliBackends: new Map(owners.cliBackends),

134-

setupProviders: new Map(owners.setupProviders),

135-

commandAliases: new Map(owners.commandAliases),

136-

contracts: new Map(owners.contracts),

137-

};

127+

function throwReadonlyPluginMetadataMutation(): never {

128+

throw new TypeError("Plugin metadata snapshots are immutable");

138129

}

139130140-

function cloneSnapshotValue<T>(value: T): T {

141-

return value && typeof value === "object" ? structuredClone(value) : value;

131+

function freezeSnapshotValue<T>(value: T, seen = new WeakSet<object>()): T {

132+

if (!value || typeof value !== "object") {

133+

return value;

134+

}

135+

if (seen.has(value)) {

136+

return value;

137+

}

138+

seen.add(value);

139+

if (value instanceof Map) {

140+

for (const [key, entry] of value) {

141+

freezeSnapshotValue(key, seen);

142+

freezeSnapshotValue(entry, seen);

143+

}

144+

Object.defineProperties(value, {

145+

clear: { value: throwReadonlyPluginMetadataMutation },

146+

delete: { value: throwReadonlyPluginMetadataMutation },

147+

set: { value: throwReadonlyPluginMetadataMutation },

148+

});

149+

return Object.freeze(value);

150+

}

151+

if (value instanceof Set) {

152+

for (const entry of value) {

153+

freezeSnapshotValue(entry, seen);

154+

}

155+

Object.defineProperties(value, {

156+

add: { value: throwReadonlyPluginMetadataMutation },

157+

clear: { value: throwReadonlyPluginMetadataMutation },

158+

delete: { value: throwReadonlyPluginMetadataMutation },

159+

});

160+

return Object.freeze(value);

161+

}

162+

for (const entry of Object.values(value)) {

163+

freezeSnapshotValue(entry, seen);

164+

}

165+

return Object.freeze(value);

142166

}

143167144-

function clonePluginManifestRecord(plugin: PluginManifestRecord): PluginManifestRecord {

145-

return cloneSnapshotValue(plugin);

146-

}

147-148-

function clonePluginMetadataSnapshot(snapshot: PluginMetadataSnapshot): PluginMetadataSnapshot {

149-

const plugins = snapshot.plugins.map(clonePluginManifestRecord);

150-

const pluginsById = new Map(plugins.map((plugin) => [plugin.id, plugin]));

151-

const diagnostics = snapshot.diagnostics.map(cloneSnapshotValue);

152-

return {

153-

...snapshot,

154-

index: {

155-

...snapshot.index,

156-

installRecords: cloneSnapshotValue(snapshot.index.installRecords ?? {}),

157-

plugins: snapshot.index.plugins.map(cloneSnapshotValue),

158-

diagnostics: snapshot.index.diagnostics.map(cloneSnapshotValue),

159-

},

160-

registryDiagnostics: snapshot.registryDiagnostics.map(cloneSnapshotValue),

161-

manifestRegistry: {

162-

...snapshot.manifestRegistry,

163-

plugins,

164-

diagnostics,

165-

},

166-

plugins,

167-

diagnostics,

168-

byPluginId: new Map(

169-

[...snapshot.byPluginId.entries()].map(([pluginId, plugin]) => [

170-

pluginId,

171-

pluginsById.get(plugin.id) ?? clonePluginManifestRecord(plugin),

172-

]),

173-

),

174-

owners: cloneOwnerMaps(snapshot.owners),

175-

metrics: { ...snapshot.metrics },

176-

};

168+

function freezePluginMetadataSnapshot(snapshot: PluginMetadataSnapshot): PluginMetadataSnapshot {

169+

return freezeSnapshotValue(snapshot);

177170

}

178171179172

function resolvePersistedRegistryFastMemoFingerprint(params: {

@@ -366,6 +359,10 @@ function indexesMatch(

366359

);

367360

}

368361362+

function cloneSnapshotInput<T>(value: T): T {

363+

return value && typeof value === "object" ? structuredClone(value) : value;

364+

}

365+369366

function normalizeInstalledPluginIndex(index: InstalledPluginIndex): InstalledPluginIndex {

370367

return {

371368

version: index.version ?? 1,

@@ -374,9 +371,9 @@ function normalizeInstalledPluginIndex(index: InstalledPluginIndex): InstalledPl

374371

migrationVersion: index.migrationVersion ?? 1,

375372

policyHash: index.policyHash ?? "",

376373

generatedAtMs: index.generatedAtMs ?? 0,

377-

installRecords: index.installRecords ?? {},

378-

plugins: index.plugins ?? [],

379-

diagnostics: index.diagnostics ?? [],

374+

installRecords: cloneSnapshotInput(index.installRecords ?? {}),

375+

plugins: (index.plugins ?? []).map(cloneSnapshotInput),

376+

diagnostics: (index.diagnostics ?? []).map(cloneSnapshotInput),

380377

...(index.warning ? { warning: index.warning } : {}),

381378

...(index.refreshReason ? { refreshReason: index.refreshReason } : {}),

382379

} as InstalledPluginIndex;

@@ -511,20 +508,16 @@ export function loadPluginMetadataSnapshot(

511508

const memoKey = computePluginMetadataSnapshotMemoKey({ params, registryState });

512509

const memo = findPluginMetadataSnapshotMemo(memoKey);

513510

if (memo?.key === memoKey) {

514-

return measureDiagnosticsTimelineSpanSync(

515-

"plugins.metadata.scan",

516-

() => clonePluginMetadataSnapshot(memo.snapshot),

517-

{

518-

phase: activeTimelineSpan?.phase ?? "startup",

519-

config: params.config,

520-

env: params.env,

521-

attributes: {

522-

cacheHit: true,

523-

hasWorkspaceDir: params.workspaceDir !== undefined,

524-

hasInstalledIndex: params.index !== undefined,

525-

},

511+

return measureDiagnosticsTimelineSpanSync("plugins.metadata.scan", () => memo.snapshot, {

512+

phase: activeTimelineSpan?.phase ?? "startup",

513+

config: params.config,

514+

env: params.env,

515+

attributes: {

516+

cacheHit: true,

517+

hasWorkspaceDir: params.workspaceDir !== undefined,

518+

hasInstalledIndex: params.index !== undefined,

526519

},

527-

);

520+

});

528521

}

529522530523

const result = measureDiagnosticsTimelineSpanSync(

@@ -540,12 +533,13 @@ export function loadPluginMetadataSnapshot(

540533

},

541534

},

542535

);

536+

const snapshot = freezePluginMetadataSnapshot(result.snapshot);

543537

if (canMemoizePluginMetadataSnapshotResult(result)) {

544538

const cachedRegistryState =

545539

result.registrySource === "derived"

546540

? resolvePersistedRegistryMemoState({

547541

env,

548-

index: result.snapshot.index,

542+

index: snapshot.index,

549543

...(params.stateDir ? { stateDir: resolveUserPath(params.stateDir, env) } : {}),

550544

...(params.preferPersisted !== undefined

551545

? { preferPersisted: params.preferPersisted }

@@ -555,10 +549,10 @@ export function loadPluginMetadataSnapshot(

555549

rememberPluginMetadataSnapshotMemo({

556550

key: computePluginMetadataSnapshotMemoKey({ params, registryState: cachedRegistryState }),

557551

registryState: cachedRegistryState,

558-

snapshot: clonePluginMetadataSnapshot(result.snapshot),

552+

snapshot,

559553

});

560554

}

561-

return result.snapshot;

555+

return snapshot;

562556

}

563557564558

function canMemoizePluginMetadataSnapshotResult(result: {