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

推荐订阅源

Stack Overflow Blog
Stack Overflow Blog
酷 壳 – CoolShell
酷 壳 – CoolShell
P
Proofpoint News Feed
Apple Machine Learning Research
Apple Machine Learning Research
T
Tailwind CSS Blog
罗磊的独立博客
F
Future of Privacy Forum
The Register - Security
The Register - Security
MyScale Blog
MyScale Blog
P
Privacy & Cybersecurity Law Blog
V
Visual Studio Blog
T
Tenable Blog
F
Fortinet All Blogs
D
Docker
V
Vulnerabilities – Threatpost
Cyberwarzone
Cyberwarzone
A
Arctic Wolf
T
Threat Research - Cisco Blogs
I
Intezer
T
Tor Project blog
大猫的无限游戏
大猫的无限游戏
MongoDB | Blog
MongoDB | Blog
博客园 - 司徒正美
AWS News Blog
AWS News Blog
freeCodeCamp Programming Tutorials: Python, JavaScript, Git & More
G
GRAHAM CLULEY
T
Threatpost
美团技术团队
K
Kaspersky official blog
F
Fox-IT International blog
Hugging Face - Blog
Hugging Face - Blog
Vercel News
Vercel News
P
Palo Alto Networks Blog
Google DeepMind News
Google DeepMind News
T
The Blog of Author Tim Ferriss
S
Schneier on Security
腾讯CDC
Cisco Talos Blog
Cisco Talos Blog
C
Check Point Blog
博客园 - 叶小钗
I
InfoQ
Threat Intelligence Blog | Flashpoint
Threat Intelligence Blog | Flashpoint
Blog — PlanetScale
Blog — PlanetScale
F
Full Disclosure
T
True Tiger Recordings
CTFtime.org: upcoming CTF events
CTFtime.org: upcoming CTF events
E
Exploit-DB.com RSS Feed
L
LINUX DO - 热门话题
J
Java Code Geeks
C
CERT Recently Published Vulnerability Notes

Recent Commits to openclaw:main

test(qa): annotate live transport RTT measurements fix(qqbot): derive outbound watchdog from configured timeouts (#85267… fix(test): clean kitchen sink rpc temp state · openclaw/openclaw@6f695c1 fix: quiet missing daily memory reads perf: speed up agent transcript lookup · openclaw/openclaw@e7ad116 fix: guard QMD session stem fallback (#86482) · openclaw/openclaw@2e3b59b Guard OpenAI chat payload turns (#86497) · openclaw/openclaw@489e415 fix(gateway): keep session tool mirrors under pressure · openclaw/openclaw@459e89a docs: route github creation through agent transcript test(tools): add unmocked image custom-provider auth regression (#85733) · openclaw/openclaw@f0bfb3f refactor(plugin-sdk): rename plain text tool-call compat wrapper docs(skills): defer private release locators · openclaw/openclaw@23d38e4 Replace Sharp image backend with Photon (#86437) · openclaw/openclaw@b9f975b fix(agents): release embedded-attempt session lock on every exit path… · openclaw/openclaw@32ddfc2 fix: accept OpenClaw voice wake confusions (#86507) fix(crabbox): bootstrap macos js toolchain chore: add agent transcript skill · openclaw/openclaw@d63e8d4 fix(gateway): dedupe session tool fanout · openclaw/openclaw@89a21db fix: Hook ingress token unlocks password-mode gateway auth (#86453) · openclaw/openclaw@d51f268 fix #86077: keep fallback errors candidate scoped (#86134) · openclaw/openclaw@d6b7fe8 fix(diagnostics): reclaim wedged session lanes with a stale leaked ac… · openclaw/openclaw@6f76d9f fix: derive plugin media trust from metadata (#86410) · openclaw/openclaw@e761eb8 fix(media-understanding): normalize HEIC before image descriptions (#… · openclaw/openclaw@75c7236 fix: accept leading fuzzy Discord voice wake names (#86484) · openclaw/openclaw@8fe4f34 feat: promote provider tool call stream wrapper (#86489) fix(test): dedupe kitchen sink command assertions test: derive deprecated sdk usage guard (#86403) fix(qa): extend memory fallback Windows budget fix(ui): move control ui chunking helper out of runtime source · openclaw/openclaw@968c87d fix: quiet retained lost task noise (#86475) fix(build): keep control ui chunking out of deadcode · openclaw/openclaw@dc26069 fix: rotate realtime voice sessions on max duration · openclaw/openclaw@dc2c4aa fix(test): stream bundled plugin sweep logs · openclaw/openclaw@fc3cd49 docs: add bugfix changelog credits · openclaw/openclaw@2e7e4bc fix(models): show oauth marker auth status (#86378) · openclaw/openclaw@a6df39d fix: seed cron task progress summaries (#86313) · openclaw/openclaw@92afd8b fix(update): exclude prerelease tags from stable git channel (#86260) · openclaw/openclaw@28f169b fix(doctor): warn and continue when cron job store is unreadable (#86… fix(gateway): clear runtime config snapshot before in-process restart… · openclaw/openclaw@90caa3b fix(scripts): restore sparse crabbox changed gates · openclaw/openclaw@d270879 fix(build): support Windows UI builds · openclaw/openclaw@0bb9b42 Fix local embedding worker safety (#85348) · openclaw/openclaw@7ff29a9 fix(ui): scope chat session picker to active agent (#85965) · openclaw/openclaw@70c7d6f [codex] improve iOS realtime talk mode (#86355) · openclaw/openclaw@9ca52ce fix(scripts): dedupe docker lane resources · openclaw/openclaw@5e94469 docs: add code size guidance · openclaw/openclaw@9a60fcf fix(test): avoid source gateway import in rpc walk · openclaw/openclaw@e9b8a6e docs: add bugfix changelog entries · openclaw/openclaw@f950132 Fix heartbeat response loop guard (#86324) (#86357) · openclaw/openclaw@e2c174e fix(memory-core): filter REM dreaming candidates to light-staged entr… · openclaw/openclaw@8b42771 fix(telegram): propagate forum topic names into agent context (#86299) fix(slack): keep downloaded files out of reply media (#86318) · openclaw/openclaw@2fcd481 fix(cron): accept plus durations for one-shot jobs (#86341) · openclaw/openclaw@9239f94 fix(plugins): clear metadata memo at lifecycle boundaries · openclaw/openclaw@e7c696a chore(skills): normalize release skill routing · openclaw/openclaw@4737e19 docs(release): require early performance regression check · openclaw/openclaw@0336938 fix(qa): capture Windows gateway metrics · openclaw/openclaw@9afbfc1 feat(qa): add coverage scenario matching · openclaw/openclaw@a1fe86a fix(perf): avoid duplicate docker package ui build build: enable modern TypeScript module syntax · openclaw/openclaw@bbc1772 ci: include performance evidence in release validation fix(providers): stream ordinary tool-like prose promptly fix(perf): harden gateway restart bench exits · openclaw/openclaw@82bbcf6 fix(gateway): gate talk secret bootstrap handoff (#85690) · openclaw/openclaw@c791e42 fix: suppress async media incomplete-turn errors (#85933) · openclaw/openclaw@35dcd42 migrate auth credentials · openclaw/openclaw@f036bac fix migrate auth lint · openclaw/openclaw@50e6cb0 fix migrate supported auth imports · openclaw/openclaw@44bb2be fix migrate auth opt-out precedence · openclaw/openclaw@2016a51 honor migrate auth opt-out in plan · openclaw/openclaw@17edec7 address migrate auth review comments · openclaw/openclaw@0a98c2d fix ci blockers for migrate auth docs: add migrate auth changelog (#85667) · openclaw/openclaw@f7fcbdb fix(scripts): avoid duplicate install smoke ui build · openclaw/openclaw@b1b2841 fix(telegram): preserve inbound text entities (#83873) · openclaw/openclaw@b552919 chore: ignore Python bytecode caches · openclaw/openclaw@b6b2755 fix: make autoreview progress visible · openclaw/openclaw@236edb2 ci(release): fix plugin prerelease extension batch invocation test(telegram): provide topic cache store in message context harness · openclaw/openclaw@ff1fde1 test(agents): complete provider runtime test mocks · openclaw/openclaw@be8cd12 test(telegram): type topic cache harness store · openclaw/openclaw@84ab206 test(agents): sync provider runtime mocks · openclaw/openclaw@a289dd9 refactor: keep plain text tool-call promotion private (#86374) · openclaw/openclaw@c3ab2de fix(discord): suppress self-reply prompt echoes (#86238) docs: clarify config migration policy · openclaw/openclaw@c44367f fix(perf): fail startup bench on early gateway exit · openclaw/openclaw@a8fc28c fix: prevent plain text tool call leaks (#86222) · openclaw/openclaw@cd62780 fix: handle npm min-release-age in installers · openclaw/openclaw@316d97c fix(scripts): include ui:build in build-all full and ciArtifacts prof… · openclaw/openclaw@6704d0a fix(e2e): sample Windows kitchen sink gateway RSS · openclaw/openclaw@73189e3 fix(cron): respect isolated target and error on missing remove id (#8… · openclaw/openclaw@6709f4e fix(pi-embedded-runner): propagate trigger-derived priority to the gl… · openclaw/openclaw@0580f57 fix(cli): suppress self-update version warnings · openclaw/openclaw@e2bd20f fix: preserve webchat source reply details · openclaw/openclaw@aa50c51 docs: replace OpenClaw docs skill and add plugin permissions guide · openclaw/openclaw@0dabb70 fix(codex): preserve source reply mode for active runs (#86325) · openclaw/openclaw@b962110 fix: make compaction reinjection opt-in · openclaw/openclaw@ab910f8 fix codex usage-limit recovery copy (#86305) · openclaw/openclaw@c3c8a65 feat(ui): add ephemeral Activity tab · openclaw/openclaw@3dd0e8e fix(tests): harden native macos plugin proof · openclaw/openclaw@a5d5604
fix: tighten empty plugin registry reuse · openclaw/openclaw@026cfb6
steipete · 2026-05-25 · via Recent Commits to openclaw:main

@@ -7,6 +7,8 @@ const mocks = vi.hoisted(() => ({

77

vi.fn<typeof import("../loader.js").resolveCompatibleRuntimePluginRegistry>(),

88

resolveRuntimePluginRegistry: vi.fn<typeof import("../loader.js").resolveRuntimePluginRegistry>(),

99

getActivePluginRegistry: vi.fn<typeof import("../runtime.js").getActivePluginRegistry>(),

10+

getActivePluginRegistryWorkspaceDir:

11+

vi.fn<typeof import("../runtime.js").getActivePluginRegistryWorkspaceDir>(),

1012

resolveConfiguredChannelPluginIds:

1113

vi.fn<typeof import("../channel-plugin-ids.js").resolveConfiguredChannelPluginIds>(),

1214

resolveDiscoverableScopedChannelPluginIds:

@@ -76,7 +78,9 @@ vi.mock("../runtime.js", () => ({

7678

getActivePluginHttpRouteRegistry: () => null,

7779

getActivePluginRegistry: (...args: Parameters<typeof mocks.getActivePluginRegistry>) =>

7880

mocks.getActivePluginRegistry(...args),

79-

getActivePluginRegistryWorkspaceDir: () => undefined,

81+

getActivePluginRegistryWorkspaceDir: (

82+

...args: Parameters<typeof mocks.getActivePluginRegistryWorkspaceDir>

83+

) => mocks.getActivePluginRegistryWorkspaceDir(...args),

8084

}));

81858286

vi.mock("../channel-plugin-ids.js", () => ({

@@ -119,6 +123,7 @@ describe("ensurePluginRegistryLoaded", () => {

119123

mocks.resolveCompatibleRuntimePluginRegistry.mockReset();

120124

mocks.resolveRuntimePluginRegistry.mockReset();

121125

mocks.getActivePluginRegistry.mockReset();

126+

mocks.getActivePluginRegistryWorkspaceDir.mockReset();

122127

mocks.resolveConfiguredChannelPluginIds.mockReset();

123128

mocks.resolveDiscoverableScopedChannelPluginIds.mockReset();

124129

mocks.resolveChannelPluginIds.mockReset();

@@ -129,6 +134,7 @@ describe("ensurePluginRegistryLoaded", () => {

129134

resetPluginRegistryLoadedForTests();

130135131136

mocks.getActivePluginRegistry.mockReturnValue(null);

137+

mocks.getActivePluginRegistryWorkspaceDir.mockReturnValue(undefined);

132138

mocks.resolveCompatibleRuntimePluginRegistry.mockReturnValue(undefined);

133139

mocks.loadOpenClawPlugins.mockReturnValue(createEmptyPluginRegistry());

134140

mocks.resolveRuntimePluginRegistry.mockImplementation(

@@ -333,6 +339,33 @@ describe("ensurePluginRegistryLoaded", () => {

333339

expect(load.workspaceDir).toBe("/resolved-workspace");

334340

});

335341342+

it("does not reuse non-empty all-scope registries without loader compatibility", () => {

343+

mocks.resolveEffectivePluginIds.mockReturnValue(["demo"]);

344+345+

ensurePluginRegistryLoaded({

346+

scope: "all",

347+

config: { plugins: { allow: ["demo"] } } as never,

348+

});

349+

const activeRegistry = createEmptyPluginRegistry();

350+

activeRegistry.plugins.push({

351+

id: "demo",

352+

source: "/tmp/demo.js",

353+

origin: "workspace",

354+

enabled: true,

355+

status: "loaded",

356+

} as never);

357+

mocks.getActivePluginRegistry.mockReturnValue(activeRegistry);

358+

mocks.getActivePluginRegistryWorkspaceDir.mockReturnValue("/resolved-workspace");

359+

mocks.loadOpenClawPlugins.mockClear();

360+361+

ensurePluginRegistryLoaded({

362+

scope: "all",

363+

config: { plugins: { allow: ["demo"], entries: { demo: { value: "changed" } } } } as never,

364+

});

365+366+

expect(loadOptions().onlyPluginIds).toEqual(["demo"]);

367+

});

368+336369

it("preserves empty all-scope loads instead of widening to all discovered plugins", () => {

337370

mocks.resolveEffectivePluginIds.mockReturnValue([]);

338371

@@ -344,6 +377,149 @@ describe("ensurePluginRegistryLoaded", () => {

344377

expect(loadOptions().onlyPluginIds).toEqual([]);

345378

});

346379380+

it("reuses an active empty registry for repeated empty all-scope loads", () => {

381+

mocks.resolveEffectivePluginIds.mockReturnValue([]);

382+383+

ensurePluginRegistryLoaded({

384+

scope: "all",

385+

config: { plugins: { enabled: true } } as never,

386+

});

387+

const emptyRegistry = createEmptyPluginRegistry();

388+

mocks.getActivePluginRegistry.mockReturnValue(emptyRegistry);

389+

mocks.getActivePluginRegistryWorkspaceDir.mockReturnValue("/resolved-workspace");

390+

mocks.loadOpenClawPlugins.mockClear();

391+392+

ensurePluginRegistryLoaded({

393+

scope: "all",

394+

config: { plugins: { enabled: true } } as never,

395+

});

396+397+

expect(mocks.loadOpenClawPlugins).not.toHaveBeenCalled();

398+

});

399+400+

it("does not reuse an empty active registry from another workspace", () => {

401+

mocks.resolveEffectivePluginIds.mockReturnValue([]);

402+403+

ensurePluginRegistryLoaded({

404+

scope: "all",

405+

config: { plugins: { enabled: true } } as never,

406+

});

407+

const emptyRegistry = createEmptyPluginRegistry();

408+

mocks.getActivePluginRegistry.mockReturnValue(emptyRegistry);

409+

mocks.getActivePluginRegistryWorkspaceDir.mockReturnValue("/other-workspace");

410+

mocks.loadOpenClawPlugins.mockClear();

411+412+

ensurePluginRegistryLoaded({

413+

scope: "all",

414+

config: { plugins: { enabled: true } } as never,

415+

});

416+417+

expect(loadOptions().onlyPluginIds).toEqual([]);

418+

});

419+420+

it("does not reuse a non-empty active registry for empty all-scope loads", () => {

421+

mocks.resolveEffectivePluginIds.mockReturnValue([]);

422+423+

ensurePluginRegistryLoaded({

424+

scope: "all",

425+

config: { plugins: { enabled: true } } as never,

426+

});

427+

const staleRegistry = createEmptyPluginRegistry();

428+

staleRegistry.plugins.push({

429+

id: "stale",

430+

source: "/tmp/stale.js",

431+

origin: "workspace",

432+

enabled: true,

433+

status: "loaded",

434+

} as never);

435+

mocks.getActivePluginRegistry.mockReturnValue(staleRegistry);

436+

mocks.loadOpenClawPlugins.mockClear();

437+438+

ensurePluginRegistryLoaded({

439+

scope: "all",

440+

config: { plugins: { enabled: true } } as never,

441+

});

442+443+

expect(loadOptions().onlyPluginIds).toEqual([]);

444+

});

445+446+

it("does not reuse a disabled-record registry for empty all-scope loads", () => {

447+

mocks.resolveEffectivePluginIds.mockReturnValue([]);

448+449+

ensurePluginRegistryLoaded({

450+

scope: "all",

451+

config: { plugins: { enabled: true } } as never,

452+

});

453+

const disabledRegistry = createEmptyPluginRegistry();

454+

disabledRegistry.plugins.push({

455+

id: "disabled",

456+

source: "/tmp/disabled.js",

457+

origin: "workspace",

458+

enabled: false,

459+

status: "disabled",

460+

} as never);

461+

mocks.getActivePluginRegistry.mockReturnValue(disabledRegistry);

462+

mocks.getActivePluginRegistryWorkspaceDir.mockReturnValue("/resolved-workspace");

463+

mocks.loadOpenClawPlugins.mockClear();

464+465+

ensurePluginRegistryLoaded({

466+

scope: "all",

467+

config: { plugins: { enabled: true } } as never,

468+

});

469+470+

expect(loadOptions().onlyPluginIds).toEqual([]);

471+

});

472+473+

it("does not reuse a failed diagnostic registry for explicit plugin scopes", () => {

474+

const failedRegistry = createEmptyPluginRegistry();

475+

failedRegistry.plugins.push({

476+

id: "failed",

477+

source: "/tmp/failed.js",

478+

origin: "workspace",

479+

enabled: true,

480+

status: "error",

481+

} as never);

482+

failedRegistry.diagnostics.push({

483+

level: "error",

484+

pluginId: "failed",

485+

message: "failed to load",

486+

} as never);

487+

mocks.getActivePluginRegistry.mockReturnValue(failedRegistry);

488+

mocks.getActivePluginRegistryWorkspaceDir.mockReturnValue("/resolved-workspace");

489+490+

ensurePluginRegistryLoaded({

491+

scope: "all",

492+

config: { plugins: { enabled: true } } as never,

493+

onlyPluginIds: ["failed"],

494+

});

495+496+

expect(loadOptions().onlyPluginIds).toEqual(["failed"]);

497+

});

498+499+

it("does not reuse a setup-only registry for explicit plugin scopes", () => {

500+

const setupRegistry = createEmptyPluginRegistry();

501+

setupRegistry.plugins.push({

502+

id: "setup-only",

503+

source: "/tmp/setup-only.js",

504+

origin: "workspace",

505+

enabled: false,

506+

status: "disabled",

507+

} as never);

508+

setupRegistry.channelSetups.push({

509+

pluginId: "setup-only",

510+

} as never);

511+

mocks.getActivePluginRegistry.mockReturnValue(setupRegistry);

512+

mocks.getActivePluginRegistryWorkspaceDir.mockReturnValue("/resolved-workspace");

513+514+

ensurePluginRegistryLoaded({

515+

scope: "all",

516+

config: { plugins: { enabled: true } } as never,

517+

onlyPluginIds: ["setup-only"],

518+

});

519+520+

expect(loadOptions().onlyPluginIds).toEqual(["setup-only"]);

521+

});

522+347523

it("reuses a compatible active registry instead of forcing a broad reload", () => {

348524

const activeRegistry = createEmptyPluginRegistry();

349525

activeRegistry.plugins.push({