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

推荐订阅源

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(qa-lab): report scenario pack coverage · openclaw/openclaw@dcd98bf fix(plugins): drop stale tlon tool contract · openclaw/openclaw@d70dc4b fix(installer): prefer tar for portable Node extraction · openclaw/openclaw@a54a881 fix(codex): deliver native subagent completions feat: add context-engine host capability requirements (#84994) · openclaw/openclaw@cff5244 fix(release): keep shrinkwrap pinned to pnpm lock · openclaw/openclaw@9d24fde fix: surface plan updates as status notices · openclaw/openclaw@dc04503 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): bound diagnostic buffers · openclaw/openclaw@bdcaac0 fix(diagnostics): surface async queue drops chore(diagnostics): refresh plugin sdk baseline · openclaw/openclaw@ab684f5 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): dedupe shared-directory watchers across agent workspaces… · openclaw/openclaw@3e94290 fix(skills): document watcher edge cases, add teardown/rebuild tests,… · openclaw/openclaw@19ff77e fix(skills): type watcher mock calls in dedupe regression tests · openclaw/openclaw@bb73f0a 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: include plugin shrinkwraps in dependency reports · openclaw/openclaw@82f69a2 test: update shrinkwrap packaging expectations · openclaw/openclaw@a1b05aa fix: honor shrinkwrap when bundling plugin deps fix: keep bundled plugin peers nested · openclaw/openclaw@86faf65 fix: opt acpx out of bundled runtime deps · openclaw/openclaw@9914e25 chore: refresh shrinkwrap metadata test: refresh shrinkwrap after rebase · openclaw/openclaw@8b0537c fix: opt codex out of bundled runtime deps · openclaw/openclaw@fcecbd8 chore: refresh shrinkwrap for Testbox npm · openclaw/openclaw@b2dc449 fix: honor overrides in npm shrinkwrap generation · openclaw/openclaw@0d28040 fix: make bundled plugin packages portable chore: add shrinkwrap to plugin npm packages · openclaw/openclaw@b6c8807 fix: cover plugin package locks in dependency review · openclaw/openclaw@bfa5b39 fix: publish explicit plugin bundled dependencies · openclaw/openclaw@976da39 chore: harden npm shrinkwrap release path feat: bundle plugin npm dependencies · openclaw/openclaw@de022bb 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 fix: break plugin metadata snapshot cycle · openclaw/openclaw@4ee8a2a test: align exec event routing proof (#83743) (thanks @Kaspre) · openclaw/openclaw@7b48956 test node exec event wake metadata · openclaw/openclaw@37207c6 fix: preserve route-bound direct thread events · openclaw/openclaw@0d8c9ca fix heartbeat event routing for main-scoped DMs fix: route direct thread event wakes to main DMs · openclaw/openclaw@0acfb7b 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
fix(acp): honor terminal turn results · openclaw/openclaw@635b947
steipete · 2026-05-17 · via Recent Commits to openclaw:main

@@ -5,6 +5,7 @@ import {

55

type AcpRuntime,

66

type AcpRuntimeCapabilities,

77

type AcpRuntimeDoctorReport,

8+

type AcpRuntimeEvent,

89

type AcpRuntimeStatus,

910

} from "openclaw/plugin-sdk/acp-runtime-backend";

1011

import type { OpenClawPluginService, OpenClawPluginServiceContext } from "openclaw/plugin-sdk/core";

@@ -23,6 +24,9 @@ type AcpxRuntimeLike = AcpRuntime & {

2324

doctor?(): Promise<AcpRuntimeDoctorReport>;

2425

isHealthy(): boolean;

2526

};

27+

type AcpRuntimeTurnInput = Parameters<AcpRuntime["runTurn"]>[0];

28+

type AcpRuntimeTurn = ReturnType<NonNullable<AcpRuntime["startTurn"]>>;

29+

type AcpRuntimeTurnResult = Awaited<AcpRuntimeTurn["result"]>;

26302731

type DeferredServiceState = {

2832

ctx: OpenClawPluginServiceContext | null;

@@ -43,6 +47,157 @@ function shouldRunStartupProbe(env: NodeJS.ProcessEnv = process.env): boolean {

4347

return env[ENABLE_STARTUP_PROBE_ENV] !== "0" && env[SKIP_RUNTIME_PROBE_ENV] !== "1";

4448

}

454950+

function createDeferredResult<T>() {

51+

let resolve!: (value: T) => void;

52+

let reject!: (error: unknown) => void;

53+

const promise = new Promise<T>((resolvePromise, rejectPromise) => {

54+

resolve = resolvePromise;

55+

reject = rejectPromise;

56+

});

57+

return { promise, resolve, reject };

58+

}

59+60+

class LegacyRunTurnEventQueue {

61+

private readonly items: AcpRuntimeEvent[] = [];

62+

private readonly waits: Array<{

63+

resolve: (value: AcpRuntimeEvent | null) => void;

64+

reject: (error: unknown) => void;

65+

}> = [];

66+

private closed = false;

67+

private error: unknown;

68+69+

push(item: AcpRuntimeEvent): void {

70+

if (this.closed) {

71+

return;

72+

}

73+

const waiter = this.waits.shift();

74+

if (waiter) {

75+

waiter.resolve(item);

76+

return;

77+

}

78+

this.items.push(item);

79+

}

80+81+

clear(): void {

82+

this.items.length = 0;

83+

}

84+85+

close(): void {

86+

if (this.closed) {

87+

return;

88+

}

89+

this.closed = true;

90+

for (const waiter of this.waits.splice(0)) {

91+

waiter.resolve(null);

92+

}

93+

}

94+95+

fail(error: unknown): void {

96+

if (this.closed) {

97+

return;

98+

}

99+

this.error = error;

100+

this.closed = true;

101+

for (const waiter of this.waits.splice(0)) {

102+

waiter.reject(error);

103+

}

104+

}

105+106+

private async next(): Promise<AcpRuntimeEvent | null> {

107+

const item = this.items.shift();

108+

if (item) {

109+

return item;

110+

}

111+

if (this.error) {

112+

throw this.error;

113+

}

114+

if (this.closed) {

115+

return null;

116+

}

117+

return await new Promise<AcpRuntimeEvent | null>((resolve, reject) => {

118+

this.waits.push({ resolve, reject });

119+

});

120+

}

121+122+

async *iterate(): AsyncIterable<AcpRuntimeEvent> {

123+

for (;;) {

124+

const item = await this.next();

125+

if (!item) {

126+

return;

127+

}

128+

yield item;

129+

}

130+

}

131+

}

132+133+

function legacyRunTurnAsStartTurn(runtime: AcpRuntime, input: AcpRuntimeTurnInput): AcpRuntimeTurn {

134+

const result = createDeferredResult<AcpRuntimeTurnResult>();

135+

result.promise.catch(() => {});

136+

const queue = new LegacyRunTurnEventQueue();

137+

let resultSettled = false;

138+

const settleResult = (next: AcpRuntimeTurnResult) => {

139+

if (resultSettled) {

140+

return;

141+

}

142+

resultSettled = true;

143+

result.resolve(next);

144+

};

145+

void (async () => {

146+

try {

147+

for await (const event of runtime.runTurn(input)) {

148+

if (event.type === "done") {

149+

settleResult({

150+

status: "completed",

151+

...(event.stopReason ? { stopReason: event.stopReason } : {}),

152+

});

153+

continue;

154+

}

155+

if (event.type === "error") {

156+

settleResult({

157+

status: "failed",

158+

error: {

159+

message: event.message,

160+

...(event.code ? { code: event.code } : {}),

161+

...(event.detailCode ? { detailCode: event.detailCode } : {}),

162+

...(event.retryable === undefined ? {} : { retryable: event.retryable }),

163+

},

164+

});

165+

continue;

166+

}

167+

queue.push(event);

168+

}

169+

settleResult({

170+

status: "failed",

171+

error: {

172+

code: "ACP_TURN_FAILED",

173+

message: "ACP turn ended without a terminal done event.",

174+

},

175+

});

176+

} catch (error) {

177+

result.reject(error);

178+

queue.fail(error);

179+

return;

180+

}

181+

queue.close();

182+

})();

183+

return {

184+

requestId: input.requestId,

185+

events: queue.iterate(),

186+

result: result.promise,

187+

async cancel(inputArgs) {

188+

await runtime.cancel({ handle: input.handle, reason: inputArgs?.reason });

189+

},

190+

async closeStream() {

191+

queue.clear();

192+

queue.close();

193+

},

194+

};

195+

}

196+197+

function startRuntimeTurn(runtime: AcpRuntime, input: AcpRuntimeTurnInput): AcpRuntimeTurn {

198+

return runtime.startTurn?.(input) ?? legacyRunTurnAsStartTurn(runtime, input);

199+

}

200+46201

async function startRealService(state: DeferredServiceState): Promise<AcpxRuntimeLike> {

47202

if (state.realRuntime) {

48203

return state.realRuntime;

@@ -70,6 +225,26 @@ function createDeferredRuntime(state: DeferredServiceState): AcpxRuntimeLike {

70225

async ensureSession(input) {

71226

return await (await startRealService(state)).ensureSession(input);

72227

},

228+

startTurn(input) {

229+

const turnPromise = startRealService(state).then((runtime) =>

230+

startRuntimeTurn(runtime, input),

231+

);

232+

return {

233+

requestId: input.requestId,

234+

events: {

235+

async *[Symbol.asyncIterator]() {

236+

yield* (await turnPromise).events;

237+

},

238+

},

239+

result: turnPromise.then((turn) => turn.result),

240+

cancel(inputArgs) {

241+

return turnPromise.then((turn) => turn.cancel(inputArgs));

242+

},

243+

closeStream(inputArgs) {

244+

return turnPromise.then((turn) => turn.closeStream(inputArgs));

245+

},

246+

};

247+

},

73248

async *runTurn(input) {

74249

yield* (await startRealService(state)).runTurn(input);

75250

},