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

推荐订阅源

让小产品的独立变现更简单 - ezindie.com
让小产品的独立变现更简单 - ezindie.com
人人都是产品经理
人人都是产品经理
Cisco Talos Blog
Cisco Talos Blog
钛媒体:引领未来商业与生活新知
钛媒体:引领未来商业与生活新知
V
V2EX
博客园 - 三生石上(FineUI控件)
Martin Fowler
Martin Fowler
WordPress大学
WordPress大学
D
Docker
S
SegmentFault 最新的问题
博客园 - 聂微东
美团技术团队
Apple Machine Learning Research
Apple Machine Learning Research
月光博客
月光博客
奇客Solidot–传递最新科技情报
奇客Solidot–传递最新科技情报
Last Week in AI
Last Week in AI
M
MIT News - Artificial intelligence
F
Fortinet All Blogs
freeCodeCamp Programming Tutorials: Python, JavaScript, Git & More
The GitHub Blog
The GitHub Blog
GbyAI
GbyAI
L
LangChain Blog
Vercel News
Vercel News
博客园 - 叶小钗
MongoDB | Blog
MongoDB | Blog
Stack Overflow Blog
Stack Overflow Blog
H
Help Net Security
OSCHINA 社区最新新闻
OSCHINA 社区最新新闻
The Cloudflare Blog
Engineering at Meta
Engineering at Meta
T
Threat Research - Cisco Blogs
T
Threatpost
Scott Helme
Scott Helme
T
Tailwind CSS Blog
Latest news
Latest news
Stack Overflow Blog
Stack Overflow Blog
Blog — PlanetScale
Blog — PlanetScale
The Register - Security
The Register - Security
罗磊的独立博客
P
Proofpoint News Feed
腾讯CDC
S
Schneier on Security
雷峰网
雷峰网
A
About on SuperTechFans
T
Tenable Blog
F
Full Disclosure
Cyberwarzone
Cyberwarzone
博客园_首页
有赞技术团队
有赞技术团队
K
Kaspersky official blog

Recent Commits to openclaw:main

fix: skip browser cleanup when browser is disabled · openclaw/openclaw@4ac90a5 perf: skip unnecessary setup auth fallback · openclaw/openclaw@39e9873 ci(release): checkout approval helper · openclaw/openclaw@427df01 ci(release): allow direct publish recovery · openclaw/openclaw@50b7a2f test(scripts): expose kitchen sink command RSS · openclaw/openclaw@b93ed3f refactor(msteams): persist conversation and poll stores in sqlite · openclaw/openclaw@a2b2c4a fix(agents): dedupe subagent browser session cleanup · openclaw/openclaw@a9a86f7 fix(build): avoid stale agent-core dts warnings (#87915) · openclaw/openclaw@371a8ab Move cron persistence to SQLite (#88285) · openclaw/openclaw@005da57 fix(ui): keep selected chat model visible after session switch · openclaw/openclaw@d11e82a fix(llm): repair invalid streaming unicode escapes ci: update Blacksmith Testbox actions refactor: move model catalog normalization into core package fix(codex): keep app-server continuation turns alive · openclaw/openclaw@961691d test(agents): wait for MCP method-not-found log · openclaw/openclaw@2780f54 fix(scripts): quiet minimal runtime asset copies · openclaw/openclaw@37058ad ci: skip codeql network shard for test-only changes · openclaw/openclaw@37c6e2d fix: remove redundant unknown union ci: keep harness changes on fast checks (#88429) · openclaw/openclaw@e24a9c5 chore: remove inert skill workshop package · openclaw/openclaw@d9c0d09 fix(auth): bound oauth mirror expiry · openclaw/openclaw@0c7ab41 fix(export-html): guard msg.content and result.content filter/iterati… · openclaw/openclaw@5811693 fix(agents): bound auth health expiry · openclaw/openclaw@445ff22 ci: stabilize changed checks · openclaw/openclaw@602364f fix(agents): bound discovery auth expiry · openclaw/openclaw@c73e8ee fix: harden skill workshop proposal results · openclaw/openclaw@7d19f89 fix: preserve trusted policy checks for skill workshop · openclaw/openclaw@908fc35 fix: refresh skill workshop generated surfaces · openclaw/openclaw@77c6bee refactor: move skill research capture logic · openclaw/openclaw@6eb6730 fix: serialize skill proposal lifecycle mutations · openclaw/openclaw@41044a2 fix: keep autonomous skill capture opt-in · openclaw/openclaw@43e4b9d fix: allow concise skill update descriptions · openclaw/openclaw@28290a4 fix: preserve auto-captured skill updates · openclaw/openclaw@0b49710 fix: scan skill proposal prompt content · openclaw/openclaw@131e662 fix: refresh skill workshop CI expectations · openclaw/openclaw@515d4ff chore: keep skill workshop package manifest inert · openclaw/openclaw@dcc329a fix: align skill proposal revise validation · openclaw/openclaw@7051bf1 fix: approve final skill workshop tool params · openclaw/openclaw@3cd368e fix: serialize skill proposal creation limits · openclaw/openclaw@d6d1cc2 fix: bound skill workshop descriptions · openclaw/openclaw@3a9e7df refactor: rename skill workshop agent tool · openclaw/openclaw@2383cfd fix: enforce skill workshop proposal bounds · openclaw/openclaw@e4905ce fix(plugin-sdk): bound oauth result expiry fix(agents): harden message dts and block timestamps fix(outbound): bound delivery recovery deadline fix(shared): bound epoch expiry helpers fix(media): bound provider operation deadlines · openclaw/openclaw@06e0fd3 fix(agents): bound run drain deadlines · openclaw/openclaw@51cceaf fix(github-copilot): bound device code expiry feat: improve MCP operability fix(models): bound pasted token expiry · openclaw/openclaw@9cb9851 fix(plugins): bound scheduled turn delays fix(skill-creator): sort .skill entries deterministically · openclaw/openclaw@878e433 fix(qqbot): bound reminder schedule time fix(memory): bound qmd embed backoff · openclaw/openclaw@caac973 fix(discord): bound timeout member expiry · openclaw/openclaw@6399b6a fix(qqbot): skip token cache on invalid clock · openclaw/openclaw@472606d fix(infra): bound device bootstrap expiry · openclaw/openclaw@1774965 fix(cron): bound relative at timestamps · openclaw/openclaw@e0248fc fix(crestodian): bound rescue approval expiry · openclaw/openclaw@6a753ad fix(agents): bound codex cli fallback expiry · openclaw/openclaw@53812bd fix(gateway): forward stop sequences across providers · openclaw/openclaw@fe3c3ac feat: expand workboard orchestration metadata (#88408) · openclaw/openclaw@5435b45 fix(discord): bound rest rate-limit deadlines · openclaw/openclaw@abc26b0 fix(agents): show exec target node in tool display · openclaw/openclaw@64533ba fix(telegram): bound transport cooldown expiry · openclaw/openclaw@7d4bf8f test(release): harden live release checks · openclaw/openclaw@bdb0fde fix(anthropic): bound setup token expiry · openclaw/openclaw@926a165 fix(bedrock): bound mantle runtime token expiry · openclaw/openclaw@70b6fdd fix(agents): bound sqlite cache expiry · openclaw/openclaw@9ad7f5b fix(agents): bound google prompt cache expiry · openclaw/openclaw@1ee751d fix(agents): bound auth profile block expiry · openclaw/openclaw@30e3ca0 fix(telegram): bound error cooldown expiry · openclaw/openclaw@1f6c1ea fix(discord): bound component registry expiry · openclaw/openclaw@8654353 fix(discord): carry reply typing feedback through queue · openclaw/openclaw@6f20f29 fix(discord): omit undefined component registry fields fix(gateway): bound node pending work expiry fix(gateway): explain ignored restart signal · openclaw/openclaw@bc77f7a fix(media): dedupe duplicate inbound media path urls fix(gateway): bound plugin node capability expiry fix(install): show npm install progress without gum · openclaw/openclaw@1c9851e fix(commitments): bound terminal failure cooldown expiry · openclaw/openclaw@a4f6240 test(tasks): cover task domain view mappers (#86755) · openclaw/openclaw@8d3fe21 fix(imessage): bound private api negative cache expiry refactor(matrix): move ephemeral state to plugin sqlite (#88387) fix(imessage): bound approval reaction poll expiry fix(agents): bound exec followup handoff expiry · openclaw/openclaw@cbad1b6 fix(sandbox): bound novnc observer token expiry · openclaw/openclaw@f4cd5e4 test(release): wait for live probe cleanup · openclaw/openclaw@0e7773d fix(agents): bound exec approval request expiry · openclaw/openclaw@d8e7734 fix(google): bound realtime browser session expiry · openclaw/openclaw@da7fb64 fix(msteams): bound delegated token expiry · openclaw/openclaw@3fffb34 fix(workboard): bound claim expiry checks · openclaw/openclaw@0dd67e2 fix(browser): bound armed dialog expiry · openclaw/openclaw@4df27b9 fix(commands): bound private approval route expiry · openclaw/openclaw@e708a87 fix(agents): hide sessions_send alias normalization · openclaw/openclaw@2dacc6d fix(plugin-state): bound ttl expiry writes · openclaw/openclaw@9660e42 fix(skills): bound upload expiry checks · openclaw/openclaw@522da25 fix(exec): bound approval pending expiry · openclaw/openclaw@d44621b test(release): skip unavailable anthropic live models
refactor(agents): bind subagent threads in core (#88416) · openclaw/openclaw@3fc0df9
steipete · 2026-05-31 · via Recent Commits to openclaw:main
Original file line numberDiff line numberDiff line change

@@ -141,7 +141,9 @@ observation-only.

141141
142142

**Subagents**

143143
144-

- `subagent_spawning` / `subagent_delivery_target` / `subagent_spawned` / `subagent_ended` - coordinate subagent routing and completion delivery

144+

- `subagent_spawned` / `subagent_ended` - observe subagent launch and completion.

145+

- `subagent_delivery_target` - compatibility hook for completion delivery when no core session binding can project a route.

146+

- `subagent_spawning` - deprecated compatibility hook. Core now prepares `thread: true` subagent bindings through channel session-binding adapters before `subagent_spawned` fires.

145147

- `subagent_spawned` includes `resolvedModel` and `resolvedProvider` when OpenClaw has resolved the child session's native model before launch.

146148
147149

**Lifecycle**

@@ -464,6 +466,10 @@ before the next major release:

464466

- **`before_agent_start`** remains for compatibility. New plugins should use

465467

`before_model_resolve` and `before_prompt_build` instead of the combined

466468

phase.

469+

- **`subagent_spawning`** remains for compatibility with older plugins, but

470+

new plugins should not return thread routing from it. Core prepares

471+

`thread: true` subagent bindings through channel session-binding adapters

472+

before `subagent_spawned` fires.

467473

- **`deactivate`** remains as a deprecated cleanup compatibility alias until

468474

after 2026-08-16. New plugins should use `gateway_stop`.

469475

- **`onResolution` in `before_tool_call`** now uses the typed

Original file line numberDiff line numberDiff line change

@@ -792,6 +792,35 @@ canonical replacement.

792792
793793

</Accordion>

794794
795+

<Accordion title="subagent_spawning hook → core thread binding">

796+

**Old**: `api.on("subagent_spawning", handler)` returning

797+

`threadBindingReady` or `deliveryOrigin`.

798+
799+

**New**: let core prepare `thread: true` subagent bindings through the

800+

channel session-binding adapter. Use `api.on("subagent_spawned", handler)`

801+

only for post-launch observation.

802+
803+

```typescript

804+

// Before

805+

api.on("subagent_spawning", async () => ({

806+

status: "ok",

807+

threadBindingReady: true,

808+

deliveryOrigin: { channel: "discord", to: "channel:123", threadId: "456" },

809+

}));

810+
811+

// After

812+

api.on("subagent_spawned", async (event) => {

813+

await observeSubagentLaunch(event);

814+

});

815+

```

816+
817+

`subagent_spawning`, `PluginHookSubagentSpawningEvent`,

818+

`PluginHookSubagentSpawningResult`, and

819+

`SubagentLifecycleHookRunner.runSubagentSpawning(...)` remain only as

820+

deprecated compatibility surfaces while external plugins migrate.

821+
822+

</Accordion>

823+
795824

<Accordion title="Provider discovery types → provider catalog types">

796825

Four discovery type aliases are now thin wrappers over the

797826

catalog-era types:

Original file line numberDiff line numberDiff line change

@@ -291,14 +291,12 @@ same sub-agent session.

291291
292292

### Thread supporting channels

293293
294-

**Discord** is currently the only supported channel. It supports

295-

persistent thread-bound subagent sessions (`sessions_spawn` with

296-

`thread: true`), manual thread controls (`/focus`, `/unfocus`, `/agents`,

297-

`/session idle`, `/session max-age`), and adapter keys

298-

`channels.discord.threadBindings.enabled`,

299-

`channels.discord.threadBindings.idleHours`,

300-

`channels.discord.threadBindings.maxAgeHours`, and

301-

`channels.discord.threadBindings.spawnSessions`.

294+

Any channel with a session-binding adapter can support persistent

295+

thread-bound subagent sessions (`sessions_spawn` with `thread: true`).

296+

Bundled adapters currently include Discord threads, Matrix threads,

297+

Telegram forum topics, and current-conversation bindings for Feishu.

298+

Use the per-channel `threadBindings` config keys for enablement,

299+

timeouts, and `spawnSessions`.

302300
303301

### Quick flow

304302
Original file line numberDiff line numberDiff line change

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

44

} from "openclaw/plugin-sdk/channel-test-helpers";

55

import type { OpenClawPluginApi } from "openclaw/plugin-sdk/core";

66

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

7+

import { handleDiscordSubagentSpawning } from "./subagent-hooks.js";

78
89

type ThreadBindingRecord = {

910

accountId: string;

@@ -85,7 +86,10 @@ function registerHandlersForTest(

8586

) {

8687

return registerHookHandlersForTest<OpenClawPluginApi>({

8788

config,

88-

register: registerDiscordSubagentHooks,

89+

register: (api) => {

90+

registerDiscordSubagentHooks(api);

91+

api.on("subagent_spawning", (event) => handleDiscordSubagentSpawning(api, event));

92+

},

8993

});

9094

}

9195
Original file line numberDiff line numberDiff line change

@@ -12,10 +12,6 @@ function loadDiscordSubagentHooksModule() {

1212

// Subagent hooks live behind a dedicated barrel so the bundled entry can

1313

// register one stable hook wiring path while keeping the handler module lazy.

1414

export function registerDiscordSubagentHooks(api: OpenClawPluginApi): void {

15-

api.on("subagent_spawning", async (event) => {

16-

const { handleDiscordSubagentSpawning } = await loadDiscordSubagentHooksModule();

17-

return await handleDiscordSubagentSpawning(api, event);

18-

});

1915

api.on("subagent_ended", async (event) => {

2016

const { handleDiscordSubagentEnded } = await loadDiscordSubagentHooksModule();

2117

handleDiscordSubagentEnded(event);

Original file line numberDiff line numberDiff line change

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

55

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

66

import type { ClawdbotConfig, OpenClawPluginApi } from "../runtime-api.js";

77

import { registerFeishuSubagentHooks } from "../subagent-hooks-api.js";

8+

import { handleFeishuSubagentSpawning } from "./subagent-hooks.js";

89

import {

910

createFeishuThreadBindingManager,

1011

testing as threadBindingTesting,

@@ -18,7 +19,10 @@ const baseConfig: ClawdbotConfig = {

1819

function registerHandlersForTest(config: Record<string, unknown> = baseConfig) {

1920

return registerHookHandlersForTest<OpenClawPluginApi>({

2021

config,

21-

register: registerFeishuSubagentHooks,

22+

register: (api) => {

23+

registerFeishuSubagentHooks(api);

24+

api.on("subagent_spawning", (event, ctx) => handleFeishuSubagentSpawning(event, ctx));

25+

},

2226

});

2327

}

2428
Original file line numberDiff line numberDiff line change

@@ -10,10 +10,6 @@ function loadFeishuSubagentHooksModule() {

1010

}

1111
1212

export function registerFeishuSubagentHooks(api: OpenClawPluginApi): void {

13-

api.on("subagent_spawning", async (event, ctx) => {

14-

const { handleFeishuSubagentSpawning } = await loadFeishuSubagentHooksModule();

15-

return await handleFeishuSubagentSpawning(event, ctx);

16-

});

1713

api.on("subagent_delivery_target", async (event) => {

1814

const { handleFeishuSubagentDeliveryTarget } = await loadFeishuSubagentHooksModule();

1915

return handleFeishuSubagentDeliveryTarget(event);

Original file line numberDiff line numberDiff line change

@@ -135,17 +135,14 @@ describe("matrix plugin", () => {

135135
136136

expect(runtimeMocks.ensureMatrixCryptoRuntime).not.toHaveBeenCalled();

137137

expect(on.mock.calls.map(([hookName]) => hookName)).toEqual([

138-

"subagent_spawning",

139138

"subagent_ended",

140139

"subagent_delivery_target",

141140

]);

142141

const handlers = Object.fromEntries(on.mock.calls);

143-

await expect(handlers.subagent_spawning({ id: "spawn" })).resolves.toBe("spawned");

144142

await expect(handlers.subagent_ended({ id: "ended" })).resolves.toBeUndefined();

145143

await expect(handlers.subagent_delivery_target({ id: "target" })).resolves.toBe(

146144

"delivery-target",

147145

);

148-

expect(runtimeMocks.handleMatrixSubagentSpawning).toHaveBeenCalledWith(api, { id: "spawn" });

149146

expect(runtimeMocks.handleMatrixSubagentEnded).toHaveBeenCalledWith({ id: "ended" });

150147

expect(runtimeMocks.handleMatrixSubagentDeliveryTarget).toHaveBeenCalledWith({ id: "target" });

151148

});

Original file line numberDiff line numberDiff line change

@@ -55,7 +55,10 @@ const fakeApi = { config: {} } as never;

5555

function registerHandlersForTest(config: Record<string, unknown> = {}) {

5656

return registerHookHandlersForTest<MatrixEntryPluginApi>({

5757

config,

58-

register: registerMatrixSubagentHooks,

58+

register: (api) => {

59+

registerMatrixSubagentHooks(api);

60+

api.on("subagent_spawning", (event) => handleMatrixSubagentSpawning(api, event));

61+

},

5962

});

6063

}

6164
Original file line numberDiff line numberDiff line change

@@ -10,10 +10,6 @@ function loadMatrixSubagentHooksModule() {

1010

}

1111
1212

export function registerMatrixSubagentHooks(api: OpenClawPluginApi): void {

13-

api.on("subagent_spawning", async (event) => {

14-

const { handleMatrixSubagentSpawning } = await loadMatrixSubagentHooksModule();

15-

return await handleMatrixSubagentSpawning(api, event);

16-

});

1713

api.on("subagent_ended", async (event) => {

1814

const { handleMatrixSubagentEnded } = await loadMatrixSubagentHooksModule();

1915

await handleMatrixSubagentEnded(event);