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

推荐订阅源

Google DeepMind News
Google DeepMind News
F
Fortinet All Blogs
阮一峰的网络日志
阮一峰的网络日志
Apple Machine Learning Research
Apple Machine Learning Research
爱范儿
爱范儿
WordPress大学
WordPress大学
让小产品的独立变现更简单 - ezindie.com
让小产品的独立变现更简单 - ezindie.com
J
Java Code Geeks
罗磊的独立博客
S
SegmentFault 最新的问题
V
V2EX
V
Visual Studio Blog
钛媒体:引领未来商业与生活新知
钛媒体:引领未来商业与生活新知
美团技术团队
博客园 - 三生石上(FineUI控件)
Stack Overflow Blog
Stack Overflow Blog
Y
Y Combinator Blog
MyScale Blog
MyScale Blog
D
Docker
Google DeepMind News
Google DeepMind News
Blog — PlanetScale
Blog — PlanetScale
M
Microsoft Research Blog - Microsoft Research
Martin Fowler
Martin Fowler
S
Secure Thoughts
B
Blog
cs.CL updates on arXiv.org
cs.CL updates on arXiv.org
www.infosecurity-magazine.com
www.infosecurity-magazine.com
Recent Announcements
Recent Announcements
MongoDB | Blog
MongoDB | Blog
C
Cisco Blogs
C
CERT Recently Published Vulnerability Notes
T
True Tiger Recordings
GbyAI
GbyAI
P
Proofpoint News Feed
P
Privacy International News Feed
Jina AI
Jina AI
The Cloudflare Blog
I
Intezer
AWS News Blog
AWS News Blog
Hacker News - Newest:
Hacker News - Newest: "LLM"
S
Security Archives - TechRepublic
NISL@THU
NISL@THU
The Register - Security
The Register - Security
Recent Commits to openclaw:main
Recent Commits to openclaw:main
P
Palo Alto Networks Blog
S
Schneier on Security
L
LINUX DO - 热门话题
C
CXSECURITY Database RSS Feed - CXSecurity.com
Security Latest
Security Latest
C
Cybersecurity and Infrastructure Security Agency CISA

Recent Commits to openclaw:main

test(google): narrow web search fake timers · openclaw/openclaw@fe7d13c fix(installer): extract portable Node with ZipFile · openclaw/openclaw@ffa6cd8 fix(gateway): defer provider auth prewarm after startup (#85369) · openclaw/openclaw@69255f8 fix(talk): stabilize realtime voice consults · openclaw/openclaw@683ad75 test(qa): tolerate slow gateway rpc startup · openclaw/openclaw@29118a0 fix(diagnostics): surface async queue drops chore(diagnostics): refresh plugin sdk baseline · openclaw/openclaw@ab684f5 fix(diagnostics): bound diagnostic buffers · openclaw/openclaw@bdcaac0 fix(installer): copy portable Node into place · openclaw/openclaw@c21ca88 fix(cli): recover replaced device approvals (#85342) · openclaw/openclaw@6ea907c test(release): align prerelease validation · openclaw/openclaw@0def3e2 fix(installer): install portable Node directory atomically · openclaw/openclaw@2890b1a fix(runtime-llm): avoid duplicate provider prefix in allowlist diagno… · openclaw/openclaw@937a756 fix(gateway): include openclaw bin in service PATH (#84475) · openclaw/openclaw@66d1d13 fix(gateway): handle concurrent launchd bootstrap restart race (#84722) · openclaw/openclaw@ba86716 feat: support pi and opencode autoreview engines · openclaw/openclaw@31a189d ci(package): gate acceptance on package integrity · openclaw/openclaw@5275929 ci(release): bypass pnpm for tsdown package build · openclaw/openclaw@fea89cd ci(release): harden docker package build · openclaw/openclaw@7b1fbe1 test(release): align prerelease validation baselines · openclaw/openclaw@04ebdc6 fix(codex): skip native web search transcript mirroring (#85346) · openclaw/openclaw@c3531fc fix(gateway): harden launchd reload handoff race recovery (#84641) · openclaw/openclaw@fc7a531 fix: honor per-model provider transport overrides (#80488) fix(skills): type watcher mock calls in dedupe regression tests · openclaw/openclaw@bb73f0a fix(skills): dedupe shared-directory watchers across agent workspaces… · openclaw/openclaw@3e94290 fix(skills): document watcher edge cases, add teardown/rebuild tests,… · openclaw/openclaw@19ff77e fix(infra): allow macos browser open over ssh env (#85340) · openclaw/openclaw@47d66fe fix(update): preserve package service state during cutover (#83026) · openclaw/openclaw@a15797a fix(gateway): broadcast agent-run error payloads (#85355) · openclaw/openclaw@07e61fc test(e2e): avoid synthetic channel config in plugin smoke fix(cli): suppress systemd hints for live gateway (#85336) · openclaw/openclaw@a00c583 fix(cli): honor agent for model auth logout (#85326) · openclaw/openclaw@fc47c1f fix(gateway): eager-load lifecycle runtime to survive in-place upgrad… · openclaw/openclaw@4a91385 fix(doctor): point codex asset warning at migrate plan (#85324) fix(update): harden managed handoff cwd (#83875) · openclaw/openclaw@1bafc23 docs(release): prepare 2026.5.21 notes ci(crabbox): harden docker hydration refactor(crabbox): parse provider list from binary help instead of ha… test(plugins): add kitchen sink rpc docker lane · openclaw/openclaw@6f6da5f test(plugins): keep rpc source walk on source call gateway test(plugins): run kitchen sink rpc lane without tsx test(qa-lab): add bus tool trace scenario · openclaw/openclaw@2b39613 fix(cron): classify network retry errors (#85344) fix(installer): bootstrap portable Windows Node · openclaw/openclaw@3551e98 fix(ui): move chat session search into picker (#85303) · openclaw/openclaw@1fdc73a fix: opt codex out of bundled runtime deps · openclaw/openclaw@fcecbd8 chore: refresh shrinkwrap metadata fix: keep bundled plugin peers nested · openclaw/openclaw@86faf65 test: update shrinkwrap packaging expectations · openclaw/openclaw@a1b05aa feat: bundle plugin npm dependencies · openclaw/openclaw@de022bb chore: refresh shrinkwrap for Testbox npm · openclaw/openclaw@b2dc449 fix: honor overrides in npm shrinkwrap generation · openclaw/openclaw@0d28040 chore: harden npm shrinkwrap release path fix: cover plugin package locks in dependency review · openclaw/openclaw@bfa5b39 fix: make bundled plugin packages portable fix: publish explicit plugin bundled dependencies · openclaw/openclaw@976da39 chore: add shrinkwrap to plugin npm packages · openclaw/openclaw@b6c8807 fix: opt acpx out of bundled runtime deps · openclaw/openclaw@9914e25 fix: include plugin shrinkwraps in dependency reports · openclaw/openclaw@82f69a2 fix: honor shrinkwrap when bundling plugin deps test: refresh shrinkwrap after rebase · openclaw/openclaw@8b0537c fix: limit subagent bootstrap defaults · openclaw/openclaw@56308a7 feat: update autoreview engine coverage · openclaw/openclaw@ab1fedb fix(messages): strip unsupported citation markers (#85204) (thanks @n… · openclaw/openclaw@0a95e53 test(qa-lab): report live transport coverage lanes · openclaw/openclaw@fda0baf fix(gateway): close child ACP sessions on parent reset/delete · openclaw/openclaw@136c927 fix: preserve Google Gemini 3 cron thinking (#85300) docs(skills): exclude SDK boundary bug sweeps · openclaw/openclaw@85e468d feat(plugin-sdk): add generic channel poll sender (#85299) · openclaw/openclaw@c9a0f03 fix(agents): preserve OpenAI reasoning token usage · openclaw/openclaw@0ddf51c test(e2e): harden plugin smoke cleanup fix(plugins): resolve native plugin sdk aliases (#85298) · openclaw/openclaw@6b1c868 fix(update): keep service logs out of json output · openclaw/openclaw@03f61cd fix(agent): retry transient gateway handshake closes · openclaw/openclaw@ff79299 fix(codex): keep interrupted turns visible-answer eligible (#84494) · openclaw/openclaw@8523e09 test(agents): narrow bundle mcp e2e setup · openclaw/openclaw@6bd430e test: add mocked Control UI E2E tests and playwright for local verifi… fix(code-mode): return structured worker error codes · openclaw/openclaw@edab653 fix: land code-mode structured worker errors (#83444) (thanks @Kaspre) · openclaw/openclaw@70dd315 test node exec event wake metadata · openclaw/openclaw@37207c6 fix heartbeat event routing for main-scoped DMs test: align exec event routing proof (#83743) (thanks @Kaspre) · openclaw/openclaw@7b48956 fix: route direct thread event wakes to main DMs · openclaw/openclaw@0acfb7b fix: break plugin metadata snapshot cycle · openclaw/openclaw@4ee8a2a fix: preserve route-bound direct thread events · openclaw/openclaw@0d8c9ca test(plugins): retry bundled smoke health probes test(gateway): bind auth-free websocket harness to loopback · openclaw/openclaw@2b1c01f test(plugins): keep npm peer prune mock directory-safe · openclaw/openclaw@a12e302 chore(ui): refresh fa control ui locale fix(ci): allow release update restarts · openclaw/openclaw@b859654 chore(ui): refresh nl control ui locale · openclaw/openclaw@cc6d222 chore(ui): refresh vi control ui locale · openclaw/openclaw@b59ab5b chore(ui): refresh th control ui locale · openclaw/openclaw@f483f59 chore(ui): refresh id control ui locale · openclaw/openclaw@c222ef0 chore(ui): refresh pl control ui locale · openclaw/openclaw@0050b8e chore(ui): refresh uk control ui locale · openclaw/openclaw@6b4aec9 chore(ui): refresh tr control ui locale · openclaw/openclaw@940a950 chore(ui): refresh ar control ui locale · openclaw/openclaw@d11c2e4 chore(ui): refresh it control ui locale · openclaw/openclaw@c99a29d chore(ui): refresh fr control ui locale · openclaw/openclaw@a7ba47c
fix(gateway): isolate hot reload channel failures · openclaw/openclaw@93db190
steipete · 2026-05-17 · via Recent Commits to openclaw:main

@@ -226,6 +226,159 @@ describe("gateway restart deferral preflight", () => {

226226

});

227227

});

228228229+

describe("gateway channel hot reload handlers", () => {

230+

function createChannelReloadPlan(channels: ChannelKind[]): GatewayReloadPlan {

231+

return {

232+

changedPaths: channels.map((channel) => `channels.${channel}.enabled`),

233+

restartGateway: false,

234+

restartReasons: [],

235+

hotReasons: ["channels"],

236+

reloadHooks: false,

237+

restartGmailWatcher: false,

238+

restartCron: false,

239+

restartHeartbeat: false,

240+

restartHealthMonitor: false,

241+

reloadPlugins: false,

242+

restartChannels: new Set(channels),

243+

disposeMcpRuntimes: false,

244+

noopPaths: [],

245+

};

246+

}

247+248+

async function withChannelReloadsEnabled(run: () => Promise<void>) {

249+

const previousSkipChannels = process.env.OPENCLAW_SKIP_CHANNELS;

250+

const previousSkipProviders = process.env.OPENCLAW_SKIP_PROVIDERS;

251+

delete process.env.OPENCLAW_SKIP_CHANNELS;

252+

delete process.env.OPENCLAW_SKIP_PROVIDERS;

253+

try {

254+

await run();

255+

} finally {

256+

if (previousSkipChannels === undefined) {

257+

delete process.env.OPENCLAW_SKIP_CHANNELS;

258+

} else {

259+

process.env.OPENCLAW_SKIP_CHANNELS = previousSkipChannels;

260+

}

261+

if (previousSkipProviders === undefined) {

262+

delete process.env.OPENCLAW_SKIP_PROVIDERS;

263+

} else {

264+

process.env.OPENCLAW_SKIP_PROVIDERS = previousSkipProviders;

265+

}

266+

}

267+

}

268+269+

it("continues restarting later channels after a hot-reload stop failure", async () => {

270+

const events: string[] = [];

271+

const setState = vi.fn();

272+

const logChannels = { info: vi.fn(), error: vi.fn() };

273+

const stopChannel = vi.fn(async (channel: ChannelKind) => {

274+

events.push(`stop:${channel}`);

275+

if (channel === "telegram") {

276+

throw new Error("stop failed");

277+

}

278+

});

279+

const startChannel = vi.fn(async (channel: ChannelKind) => {

280+

events.push(`start:${channel}`);

281+

});

282+

const { applyHotReload } = createGatewayReloadHandlers({

283+

deps: {} as never,

284+

broadcast: vi.fn(),

285+

getState: () => ({

286+

hooksConfig: {} as never,

287+

hookClientIpConfig: {} as never,

288+

heartbeatRunner: { stop: vi.fn(), updateConfig: vi.fn() } as never,

289+

cronState: {

290+

cron: { start: vi.fn(async () => {}), stop: vi.fn() },

291+

storePath: "/tmp/cron.json",

292+

cronEnabled: false,

293+

} as never,

294+

channelHealthMonitor: null,

295+

}),

296+

setState,

297+

startChannel,

298+

stopChannel,

299+

reloadPlugins: vi.fn(

300+

async (): Promise<GatewayPluginReloadResult> => ({

301+

restartChannels: new Set(),

302+

activeChannels: new Set(),

303+

}),

304+

),

305+

logHooks: { info: vi.fn(), warn: vi.fn(), error: vi.fn() },

306+

logChannels,

307+

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

308+

logReload: { info: vi.fn(), warn: vi.fn() },

309+

createHealthMonitor: () => null,

310+

});

311+312+

await withChannelReloadsEnabled(async () => {

313+

await expect(

314+

applyHotReload(createChannelReloadPlan(["telegram", "discord"]), {}),

315+

).rejects.toThrow("failed to restart channels during hot reload: telegram");

316+

});

317+318+

expect(events).toEqual(["stop:telegram", "stop:discord", "start:discord"]);

319+

expect(logChannels.error).toHaveBeenCalledWith(

320+

"failed to restart telegram channel during hot reload: stop failed",

321+

);

322+

expect(setState).not.toHaveBeenCalled();

323+

});

324+325+

it("continues restarting later channels after a hot-reload start failure", async () => {

326+

const events: string[] = [];

327+

const setState = vi.fn();

328+

const logChannels = { info: vi.fn(), error: vi.fn() };

329+

const stopChannel = vi.fn(async (channel: ChannelKind) => {

330+

events.push(`stop:${channel}`);

331+

});

332+

const startChannel = vi.fn(async (channel: ChannelKind) => {

333+

events.push(`start:${channel}`);

334+

if (channel === "telegram") {

335+

throw new Error("start failed");

336+

}

337+

});

338+

const { applyHotReload } = createGatewayReloadHandlers({

339+

deps: {} as never,

340+

broadcast: vi.fn(),

341+

getState: () => ({

342+

hooksConfig: {} as never,

343+

hookClientIpConfig: {} as never,

344+

heartbeatRunner: { stop: vi.fn(), updateConfig: vi.fn() } as never,

345+

cronState: {

346+

cron: { start: vi.fn(async () => {}), stop: vi.fn() },

347+

storePath: "/tmp/cron.json",

348+

cronEnabled: false,

349+

} as never,

350+

channelHealthMonitor: null,

351+

}),

352+

setState,

353+

startChannel,

354+

stopChannel,

355+

reloadPlugins: vi.fn(

356+

async (): Promise<GatewayPluginReloadResult> => ({

357+

restartChannels: new Set(),

358+

activeChannels: new Set(),

359+

}),

360+

),

361+

logHooks: { info: vi.fn(), warn: vi.fn(), error: vi.fn() },

362+

logChannels,

363+

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

364+

logReload: { info: vi.fn(), warn: vi.fn() },

365+

createHealthMonitor: () => null,

366+

});

367+368+

await withChannelReloadsEnabled(async () => {

369+

await expect(

370+

applyHotReload(createChannelReloadPlan(["telegram", "discord"]), {}),

371+

).rejects.toThrow("failed to restart channels during hot reload: telegram");

372+

});

373+374+

expect(events).toEqual(["stop:telegram", "start:telegram", "stop:discord", "start:discord"]);

375+

expect(logChannels.error).toHaveBeenCalledWith(

376+

"failed to restart telegram channel during hot reload: start failed",

377+

);

378+

expect(setState).not.toHaveBeenCalled();

379+

});

380+

});

381+229382

describe("gateway Gmail hot reload handlers", () => {

230383

function createGmailReloadPlan(): GatewayReloadPlan {

231384

return {

@@ -586,6 +739,115 @@ describe("gateway Gmail hot reload handlers", () => {

586739

});

587740588741

describe("gateway plugin hot reload handlers", () => {

742+

it("rolls back stopped channels when plugin pre-replace stop fails", async () => {

743+

const previousSkipChannels = process.env.OPENCLAW_SKIP_CHANNELS;

744+

const previousSkipProviders = process.env.OPENCLAW_SKIP_PROVIDERS;

745+

delete process.env.OPENCLAW_SKIP_CHANNELS;

746+

delete process.env.OPENCLAW_SKIP_PROVIDERS;

747+

const cron = { start: vi.fn(async () => {}), stop: vi.fn() };

748+

const heartbeatRunner = {

749+

stop: vi.fn(),

750+

updateConfig: vi.fn(),

751+

};

752+

const setState = vi.fn();

753+

const logChannels = { info: vi.fn(), error: vi.fn() };

754+

const events: string[] = [];

755+

const startChannel = vi.fn(async (channel: ChannelKind) => {

756+

events.push(`start:${channel}`);

757+

});

758+

const stopChannel = vi.fn(async (channel: ChannelKind) => {

759+

events.push(`stop:${channel}`);

760+

if (channel === "discord") {

761+

throw new Error("stop failed");

762+

}

763+

});

764+

const reloadPlugins = vi.fn(

765+

async (params: {

766+

beforeReplace: (channels: ReadonlySet<ChannelKind>) => Promise<void>;

767+

}): Promise<GatewayPluginReloadResult> => {

768+

events.push("reload:start");

769+

await params.beforeReplace(new Set(["telegram", "discord"]));

770+

events.push("registry:replace");

771+

return {

772+

restartChannels: new Set(),

773+

activeChannels: new Set(),

774+

};

775+

},

776+

);

777+

const { applyHotReload } = createGatewayReloadHandlers({

778+

deps: {} as never,

779+

broadcast: vi.fn(),

780+

getState: () => ({

781+

hooksConfig: {} as never,

782+

hookClientIpConfig: {} as never,

783+

heartbeatRunner: heartbeatRunner as never,

784+

cronState: { cron, storePath: "/tmp/cron.json", cronEnabled: false } as never,

785+

channelHealthMonitor: null,

786+

}),

787+

setState,

788+

startChannel,

789+

stopChannel,

790+

reloadPlugins,

791+

logHooks: { info: vi.fn(), warn: vi.fn(), error: vi.fn() },

792+

logChannels,

793+

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

794+

logReload: { info: vi.fn(), warn: vi.fn() },

795+

createHealthMonitor: () => null,

796+

});

797+798+

try {

799+

await expect(

800+

applyHotReload(

801+

{

802+

changedPaths: ["plugins.enabled"],

803+

restartGateway: false,

804+

restartReasons: [],

805+

hotReasons: ["plugins.enabled"],

806+

reloadHooks: false,

807+

restartGmailWatcher: false,

808+

restartCron: false,

809+

restartHeartbeat: false,

810+

restartHealthMonitor: false,

811+

reloadPlugins: true,

812+

restartChannels: new Set(),

813+

disposeMcpRuntimes: false,

814+

noopPaths: [],

815+

},

816+

{

817+

plugins: {

818+

enabled: false,

819+

},

820+

},

821+

),

822+

).rejects.toThrow("failed to stop channels before plugin reload: discord");

823+

} finally {

824+

if (previousSkipChannels === undefined) {

825+

delete process.env.OPENCLAW_SKIP_CHANNELS;

826+

} else {

827+

process.env.OPENCLAW_SKIP_CHANNELS = previousSkipChannels;

828+

}

829+

if (previousSkipProviders === undefined) {

830+

delete process.env.OPENCLAW_SKIP_PROVIDERS;

831+

} else {

832+

process.env.OPENCLAW_SKIP_PROVIDERS = previousSkipProviders;

833+

}

834+

}

835+836+

expect(events).toEqual([

837+

"reload:start",

838+

"stop:telegram",

839+

"stop:discord",

840+

"start:telegram",

841+

"start:discord",

842+

]);

843+

expect(logChannels.error).toHaveBeenCalledWith(

844+

"failed to stop discord channel before plugin reload: stop failed",

845+

);

846+

expect(startChannel).toHaveBeenCalledWith("telegram");

847+

expect(startChannel).toHaveBeenCalledWith("discord");

848+

expect(setState).not.toHaveBeenCalled();

849+

});

850+589851

it("stops removed channel plugins from broad activation before swapping plugin runtime", async () => {

590852

const previousSkipChannels = process.env.OPENCLAW_SKIP_CHANNELS;

591853

const previousSkipProviders = process.env.OPENCLAW_SKIP_PROVIDERS;