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

推荐订阅源

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 chore(diagnostics): refresh plugin sdk baseline · openclaw/openclaw@ab684f5 fix(diagnostics): bound diagnostic buffers · openclaw/openclaw@bdcaac0 fix(diagnostics): surface async queue drops 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 test(release): align prerelease validation baselines · openclaw/openclaw@04ebdc6 ci(release): harden docker package build · openclaw/openclaw@7b1fbe1 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): dedupe shared-directory watchers across agent workspaces… · openclaw/openclaw@3e94290 fix(skills): type watcher mock calls in dedupe regression tests · openclaw/openclaw@bb73f0a 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): run kitchen sink rpc lane without tsx test(plugins): keep rpc source walk on source call gateway 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: include plugin shrinkwraps in dependency reports · openclaw/openclaw@82f69a2 chore: add shrinkwrap to plugin npm packages · openclaw/openclaw@b6c8807 feat: bundle plugin npm dependencies · openclaw/openclaw@de022bb fix: keep bundled plugin peers nested · openclaw/openclaw@86faf65 chore: refresh shrinkwrap metadata test: refresh shrinkwrap after rebase · openclaw/openclaw@8b0537c test: update shrinkwrap packaging expectations · openclaw/openclaw@a1b05aa chore: refresh shrinkwrap for Testbox npm · openclaw/openclaw@b2dc449 fix: honor overrides in npm shrinkwrap generation · openclaw/openclaw@0d28040 fix: cover plugin package locks in dependency review · openclaw/openclaw@bfa5b39 fix: make bundled plugin packages portable fix: opt codex out of bundled runtime deps · openclaw/openclaw@fcecbd8 fix: publish explicit plugin bundled dependencies · openclaw/openclaw@976da39 fix: honor shrinkwrap when bundling plugin deps chore: harden npm shrinkwrap release path fix: opt acpx out of bundled runtime deps · openclaw/openclaw@9914e25 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: land code-mode structured worker errors (#83444) (thanks @Kaspre) · openclaw/openclaw@70dd315 fix(code-mode): return structured worker error codes · openclaw/openclaw@edab653 test node exec event wake metadata · openclaw/openclaw@37207c6 fix: break plugin metadata snapshot cycle · openclaw/openclaw@4ee8a2a 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: 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): drain replies during restart close · openclaw/openclaw@5aac793
vincentkoc · 2026-05-17 · via Recent Commits to openclaw:main

@@ -1,4 +1,5 @@

11

import { describe, expect, it, vi } from "vitest";

2+

import type { GatewayServer } from "../../gateway/server.impl.js";

23

import type { GatewayBonjourBeacon } from "../../infra/bonjour-discovery.js";

34

import { pickBeaconHost, pickGatewayPort } from "./discover.js";

45

@@ -249,13 +250,33 @@ function createRuntimeWithExitSignal(exitCallOrder?: string[]) {

249250

return { runtime, exited };

250251

}

251252252-

type GatewayCloseFn = (...args: unknown[]) => Promise<void>;

253+

type GatewayCloseFn = GatewayServer["close"];

253254

type LoopRuntime = {

254255

log: (...args: unknown[]) => void;

255256

error: (...args: unknown[]) => void;

256257

exit: (code: number) => void;

257258

};

258259260+

function createCloseMock() {

261+

return vi.fn<GatewayCloseFn>(async (_opts) => {});

262+

}

263+264+

function expectRestartCloseCall(

265+

close: ReturnType<typeof createCloseMock>,

266+

maxDrainTimeoutMs: number,

267+

) {

268+

expect(close).toHaveBeenCalledWith(

269+

expect.objectContaining({

270+

reason: "gateway restarting",

271+

restartExpectedMs: 1500,

272+

drainTimeoutMs: expect.any(Number),

273+

}),

274+

);

275+

const closeArgs = close.mock.calls[0]?.[0];

276+

expect(closeArgs?.drainTimeoutMs).toBeLessThanOrEqual(maxDrainTimeoutMs);

277+

expect(closeArgs?.drainTimeoutMs).toBeGreaterThanOrEqual(0);

278+

}

279+259280

function createSignaledStart(close: GatewayCloseFn) {

260281

let resolveStarted: (() => void) | null = null;

261282

const started = new Promise<void>((resolve) => {

@@ -304,7 +325,7 @@ async function waitForLoopCondition(predicate: () => boolean, message: string) {

304325

}

305326306327

async function createSignaledLoopHarness(exitCallOrder?: string[]) {

307-

const close = vi.fn(async () => {});

328+

const close = createCloseMock();

308329

const { start, started } = createSignaledStart(close);

309330

const { runtime, exited } = createRuntimeWithExitSignal(exitCallOrder);

310331

const { loopPromise } = await runLoopWithStart({ start, runtime });

@@ -361,8 +382,8 @@ describe("runGatewayLoop", () => {

361382

getActiveTaskCount.mockReturnValueOnce(1).mockReturnValue(0);

362383363384

await withIsolatedSignals(async ({ captureSignal }) => {

364-

const closeFirst = vi.fn(async () => {});

365-

const closeSecond = vi.fn(async () => {});

385+

const closeFirst = createCloseMock();

386+

const closeSecond = createCloseMock();

366387

const { runtime, exited } = createRuntimeWithExitSignal();

367388

let resolveSecond: (() => void) | null = null;

368389

const startedSecond = new Promise<void>((resolve) => {

@@ -391,10 +412,7 @@ describe("runGatewayLoop", () => {

391412

expect(consumeGatewayRestartIntentPayloadSync).toHaveBeenCalledOnce();

392413

expect(markGatewayDraining).toHaveBeenCalledOnce();

393414

expect(waitForActiveTasks).toHaveBeenCalledWith(90_000);

394-

expect(closeFirst).toHaveBeenCalledWith({

395-

reason: "gateway restarting",

396-

restartExpectedMs: 1500,

397-

});

415+

expectRestartCloseCall(closeFirst, 90_000);

398416

await startedSecond;

399417

expect(start).toHaveBeenCalledTimes(2);

400418

await new Promise<void>((resolve) => setImmediate(resolve));

@@ -430,6 +448,27 @@ describe("runGatewayLoop", () => {

430448

});

431449

});

432450451+

it("caps reply drain time for unbounded SIGTERM restarts", async () => {

452+

vi.clearAllMocks();

453+

consumeGatewayRestartIntentPayloadSync.mockReturnValueOnce({ waitMs: 0 });

454+455+

await withIsolatedSignals(async ({ captureSignal }) => {

456+

const { close, start, exited } = await createSignaledLoopHarness();

457+

const sigterm = captureSignal("SIGTERM");

458+

const sigint = captureSignal("SIGINT");

459+460+

sigterm();

461+

await new Promise<void>((resolve) => setImmediate(resolve));

462+

await new Promise<void>((resolve) => setImmediate(resolve));

463+464+

expectRestartCloseCall(close, 15_000);

465+

expect(start).toHaveBeenCalledTimes(2);

466+467+

sigint();

468+

await expect(exited).resolves.toBe(0);

469+

});

470+

});

471+433472

it("aborts active embedded runs after a short restart drain grace", async () => {

434473

vi.clearAllMocks();

435474

consumeGatewayRestartIntentPayloadSync.mockReturnValueOnce({});

@@ -473,10 +512,7 @@ describe("runGatewayLoop", () => {

473512

expect(gatewayLog.warn).toHaveBeenCalledWith(

474513

"failed to mark interrupted main sessions for restart recovery: Error: store read-only",

475514

);

476-

expect(close).toHaveBeenCalledWith({

477-

reason: "gateway restarting",

478-

restartExpectedMs: 1500,

479-

});

515+

expectRestartCloseCall(close, 90_000);

480516

expect(start).toHaveBeenCalledTimes(2);

481517482518

sigint();

@@ -567,12 +603,12 @@ describe("runGatewayLoop", () => {

567603

waitForActiveEmbeddedRuns.mockResolvedValueOnce({ drained: true });

568604569605

type StartServer = () => Promise<{

570-

close: (opts: { reason: string; restartExpectedMs: number | null }) => Promise<void>;

606+

close: GatewayCloseFn;

571607

}>;

572608573-

const closeFirst = vi.fn(async () => {});

574-

const closeSecond = vi.fn(async () => {});

575-

const closeThird = vi.fn(async () => {});

609+

const closeFirst = createCloseMock();

610+

const closeSecond = createCloseMock();

611+

const closeThird = createCloseMock();

576612

const { runtime, exited } = createRuntimeWithExitSignal();

577613578614

const start = vi.fn<StartServer>();

@@ -639,10 +675,7 @@ describe("runGatewayLoop", () => {

639675

});

640676

expect(markGatewayDraining).toHaveBeenCalledTimes(1);

641677

expect(gatewayLog.warn).toHaveBeenCalledWith(DRAIN_TIMEOUT_LOG);

642-

expect(closeFirst).toHaveBeenCalledWith({

643-

reason: "gateway restarting",

644-

restartExpectedMs: 1500,

645-

});

678+

expectRestartCloseCall(closeFirst, 1_234);

646679

expect(markGatewaySigusr1RestartHandled).toHaveBeenCalledTimes(1);

647680

expect(resetAllLanes).toHaveBeenCalledTimes(1);

648681

expect(resetGatewayRestartStateForInProcessRestart).toHaveBeenCalledTimes(1);

@@ -652,10 +685,7 @@ describe("runGatewayLoop", () => {

652685653686

await startedThird;

654687

await new Promise<void>((resolve) => setImmediate(resolve));

655-

expect(closeSecond).toHaveBeenCalledWith({

656-

reason: "gateway restarting",

657-

restartExpectedMs: 1500,

658-

});

688+

expectRestartCloseCall(closeSecond, 1_234);

659689

expect(markGatewaySigusr1RestartHandled).toHaveBeenCalledTimes(2);

660690

expect(markGatewayDraining).toHaveBeenCalledTimes(2);

661691

expect(resetAllLanes).toHaveBeenCalledTimes(2);

@@ -681,8 +711,8 @@ describe("runGatewayLoop", () => {

681711

});

682712683713

await withIsolatedSignals(async ({ captureSignal }) => {

684-

const closeFirst = vi.fn(async () => {});

685-

const closeSecond = vi.fn(async () => {});

714+

const closeFirst = createCloseMock();

715+

const closeSecond = createCloseMock();

686716

const { runtime, exited } = createRuntimeWithExitSignal();

687717

let releaseFirstStart!: () => void;

688718

const firstStartMayReturn = new Promise<void>((resolve) => {

@@ -729,10 +759,7 @@ describe("runGatewayLoop", () => {

729759

"expected queued SIGUSR1 to trigger the second gateway start",

730760

);

731761

await startedSecond;

732-

expect(closeFirst).toHaveBeenCalledWith({

733-

reason: "gateway restarting",

734-

restartExpectedMs: 1500,

735-

});

762+

expectRestartCloseCall(closeFirst, 90_000);

736763

expect(markGatewaySigusr1RestartHandled).toHaveBeenCalledTimes(1);

737764

expect(markGatewayDraining).toHaveBeenCalledTimes(1);

738765

expect(resetAllLanes).toHaveBeenCalledTimes(1);

@@ -869,8 +896,8 @@ describe("runGatewayLoop", () => {

869896

});

870897871898

await withIsolatedSignals(async ({ captureSignal }) => {

872-

const closeFirst = vi.fn(async () => {});

873-

const closeThird = vi.fn(async () => {});

899+

const closeFirst = createCloseMock();

900+

const closeThird = createCloseMock();

874901

const { runtime, exited } = createRuntimeWithExitSignal();

875902

let sigusr1: (() => void) | null = null;

876903

let resolveThirdStart: (() => void) | null = null;

@@ -909,10 +936,7 @@ describe("runGatewayLoop", () => {

909936

"expected queued SIGUSR1 to advance past failed restart startup",

910937

);

911938

await startedThird;

912-

expect(closeFirst).toHaveBeenCalledWith({

913-

reason: "gateway restarting",

914-

restartExpectedMs: 1500,

915-

});

939+

expectRestartCloseCall(closeFirst, 90_000);

916940

expect(markGatewaySigusr1RestartHandled).toHaveBeenCalledTimes(2);

917941

expect(markGatewayDraining).toHaveBeenCalledTimes(2);

918942

expect(resetAllLanes).toHaveBeenCalledTimes(2);

@@ -938,8 +962,8 @@ describe("runGatewayLoop", () => {

938962

});

939963940964

await withIsolatedSignals(async ({ captureSignal }) => {

941-

const closeFirst = vi.fn(async () => {});

942-

const closeThird = vi.fn(async () => {});

965+

const closeFirst = createCloseMock();

966+

const closeThird = createCloseMock();

943967

const { runtime, exited } = createRuntimeWithExitSignal();

944968

let resolveThirdStart: (() => void) | null = null;

945969

const startedThird = new Promise<void>((resolve) => {

@@ -980,10 +1004,7 @@ describe("runGatewayLoop", () => {

9801004

"expected post-failure SIGUSR1 to retry gateway startup",

9811005

);

9821006

await startedThird;

983-

expect(closeFirst).toHaveBeenCalledWith({

984-

reason: "gateway restarting",

985-

restartExpectedMs: 1500,

986-

});

1007+

expectRestartCloseCall(closeFirst, 90_000);

9871008

expect(markGatewaySigusr1RestartHandled).toHaveBeenCalledTimes(2);

9881009

expect(markGatewayDraining).toHaveBeenCalledTimes(2);

9891010

expect(resetAllLanes).toHaveBeenCalledTimes(2);