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

推荐订阅源

Recent Announcements
Recent Announcements
让小产品的独立变现更简单 - ezindie.com
让小产品的独立变现更简单 - ezindie.com
O
OpenAI News
D
Docker
freeCodeCamp Programming Tutorials: Python, JavaScript, Git & More
N
Netflix TechBlog - Medium
人人都是产品经理
人人都是产品经理
Y
Y Combinator Blog
M
MIT News - Artificial intelligence
奇客Solidot–传递最新科技情报
奇客Solidot–传递最新科技情报
博客园 - 司徒正美
C
CXSECURITY Database RSS Feed - CXSecurity.com
阮一峰的网络日志
阮一峰的网络日志
K
Kaspersky official blog
Security Latest
Security Latest
T
Tailwind CSS Blog
cs.AI updates on arXiv.org
cs.AI updates on arXiv.org
V
Vulnerabilities – Threatpost
W
WeLiveSecurity
N
News and Events Feed by Topic
aimingoo的专栏
aimingoo的专栏
美团技术团队
OSCHINA 社区最新新闻
OSCHINA 社区最新新闻
Google DeepMind News
Google DeepMind News
CTFtime.org: upcoming CTF events
CTFtime.org: upcoming CTF events
C
Cyber Attacks, Cyber Crime and Cyber Security
Cyber Security Advisories - MS-ISAC
Cyber Security Advisories - MS-ISAC
B
Blog
T
The Blog of Author Tim Ferriss
Google DeepMind News
Google DeepMind News
Help Net Security
Help Net Security
爱范儿
爱范儿
宝玉的分享
宝玉的分享
腾讯CDC
H
Heimdal Security Blog
Webroot Blog
Webroot Blog
AI
AI
WordPress大学
WordPress大学
Recorded Future
Recorded Future
SecWiki News
SecWiki News
cs.CV updates on arXiv.org
cs.CV updates on arXiv.org
Security Archives - TechRepublic
Security Archives - TechRepublic
Google Online Security Blog
Google Online Security Blog
C
Check Point Blog
TaoSecurity Blog
TaoSecurity Blog
Cisco Talos Blog
Cisco Talos Blog
The Cloudflare Blog
www.infosecurity-magazine.com
www.infosecurity-magazine.com
博客园 - Franky
云风的 BLOG
云风的 BLOG

Recent Commits to openclaw:main

test: merge chat side-result checks · openclaw/openclaw@ddd2c2a test: merge cron history checks · openclaw/openclaw@f7eb746 test: merge responsive navigation shell checks · openclaw/openclaw@c2e4b47 docs(changelog): add codex oauth fixes · openclaw/openclaw@628e6cd test: merge navigation routing cases · openclaw/openclaw@5d8cecb Tests: mock channel registry bundled fallback · openclaw/openclaw@2b08233 Secrets: avoid broad web search discovery for single plugin config · openclaw/openclaw@a464f59 test: merge config view browser checks · openclaw/openclaw@20cf511 fix(status): align oauth health with runtime · openclaw/openclaw@eed7116 feat: add macOS screen snapshots for monitor preview (#67954) thanks … · openclaw/openclaw@f377db1 fix: report shared auth scopes in hello-ok (#67810) thanks @BunsDev · openclaw/openclaw@0b6c39b Auto-reply: avoid eager bundled route fallback · openclaw/openclaw@3ea1bf4 Tests: narrow session binding contract setup · openclaw/openclaw@54e4e16 fix(macOS): enable undo/redo in webchat composer text input (#34962) · openclaw/openclaw@00951dc Tests: speed up channel setup promotion · openclaw/openclaw@82b529a Docs: refresh agent instructions · openclaw/openclaw@5775fe2 fix(auth): serialize OAuth refresh across agents to fix #26322 (#67876) · openclaw/openclaw@8e79080 test: allow ollama public surface boundary test · openclaw/openclaw@7d4f1a6 Docs: add test performance guardrails · openclaw/openclaw@89706d3 Tests: restore context-engine usage proof · openclaw/openclaw@e4c4f95 Tests: slim context engine runtime coverage · openclaw/openclaw@74c198f ci: retry failed custom checkouts · openclaw/openclaw@0ee5baf test: trim duplicate provider auth onboarding cases · openclaw/openclaw@1ffc02e matrix: fix sessions_spawn --thread subagent session spawning (#67643) · openclaw/openclaw@1ce2596 test: reduce auth choice fixture churn · openclaw/openclaw@857b9cd test: mock health status config boundaries · openclaw/openclaw@9d5ab4a test: mock onboard config io boundary · openclaw/openclaw@299694d test: mock legacy state plugin boundaries · openclaw/openclaw@2713089 test: mock channel install boundaries · openclaw/openclaw@b945248 test: mock doctor preview channel boundaries · openclaw/openclaw@b1a3ad4 test: trim doctor command hotspots · openclaw/openclaw@c66f16a test: isolate agent auth and spawn hotspots · openclaw/openclaw@9285935 test: stabilize MCP startup disposal race · openclaw/openclaw@dd9d2eb test: merge browser contract server suites · openclaw/openclaw@5817a76 test: narrow ollama provider discovery setup · openclaw/openclaw@a0d9598 build: declare qa-lab aimock runtime dependency · openclaw/openclaw@24431e5 test: speed up safe-bins exec harness · openclaw/openclaw@ee856ab test: preserve tool helpers in embedded runner mocks · openclaw/openclaw@acd86a0 refactor: move memory embeddings into provider plugins · openclaw/openclaw@77e6e4c test: reuse system-run temp fixtures · openclaw/openclaw@7e9ff0f test: trim hotspot wait overhead · openclaw/openclaw@12a59b0 Check: avoid duplicate boundary prep · openclaw/openclaw@baf11b8 test: reduce hotspot fixture overhead · openclaw/openclaw@3a59edd feat(ui): overhaul settings and slash command UX (#67819) thanks @Bun… · openclaw/openclaw@2cfb660 QA Matrix: exit cleanly on failure · openclaw/openclaw@42805d2 QA Matrix: isolate scenario coverage · openclaw/openclaw@7e659e1 Matrix: refresh crypto bootstrap state · openclaw/openclaw@94081d8 QA Lab: add provider registry · openclaw/openclaw@bb7e982 Matrix: add plugin changelog · openclaw/openclaw@4acab55 test: trim more hotspot overhead · openclaw/openclaw@f485311 test: trim remaining hotspot tests · openclaw/openclaw@6ba8626 test: narrow hotspot mocks · openclaw/openclaw@dbc8179 test: isolate gemini embedding request helpers · openclaw/openclaw@cd330f5 test: trim memory and mcp hotspots · openclaw/openclaw@fd48dfa test: slim provider registry mocks · openclaw/openclaw@2e08c77 test: harden Parallels update smoke · openclaw/openclaw@1a98090 feat: default Anthropic to Opus 4.7 · openclaw/openclaw@628b454 fix: harden node-host shell payload mutability checks · openclaw/openclaw@75c551e fix: land node-host approval binding for native binaries (#66731) (th… · openclaw/openclaw@29919bb CI: add daily schedule to CodeQL workflow (#67645) · openclaw/openclaw@69d25f5 fix(gateway): capture config hash after plugin auto-enable to prevent… · openclaw/openclaw@8c11210 fix: repair sanitized replay tool results before send (#67620) (thank… · openclaw/openclaw@c3c7a99 fix: restrict HTML timeout short-circuit to transient statuses · openclaw/openclaw@de129a6 fix: keep TUI watchdog bound to active run (#67401) (thanks @xantorres) · openclaw/openclaw@3525273 Gateway/skills: dedupe skills prefix-match + drop dead fallback on log · openclaw/openclaw@d7f489f Extensions/lmstudio: back off inference preload after consecutive fai… · openclaw/openclaw@b555214 TUI/streaming: add watchdog that resets the activity indicator after … · openclaw/openclaw@f44ab20 Agents/tool-loop: enable unknown-tool stream guard by default · openclaw/openclaw@36ed367 Gateway/skills: invalidate session skills snapshot on config write · openclaw/openclaw@b23d59a fix: classify HTML provider error pages correctly (#67642) (thanks @s… · openclaw/openclaw@e588e90 fix(skills): remove unused model-usage import (#67641) · openclaw/openclaw@55f05df docs(changelog): credit codex fix superseded PRs · openclaw/openclaw@e485f24 fix(openai-codex): normalize stale transport metadata in resolution a… · openclaw/openclaw@90801ba CI: pin Docker-related GitHub Actions (#67632) · openclaw/openclaw@f697b01 Android: modernize WebView and discovery API usage (#67627) · openclaw/openclaw@44a6e50 fix(deps): bump hono to 4.12.14 and @hono/node-server to 1.19.14 (GHS… · openclaw/openclaw@fbccc18 fix(deps): bump dompurify to 3.4.0 (#67614) · openclaw/openclaw@2c2dc00 CI: add explicit permissions to all workflow jobs (fixes code-scannin… · openclaw/openclaw@01b7516 fix: register bundled TTS providers and route overrides correctly (#6… · openclaw/openclaw@6ea3cdd fix: align host tilde paths with OS home (#62804) (thanks @stainlu) · openclaw/openclaw@ecfaf64 fix: flush creds queue before reconnect socket open (#67464) (thanks … · openclaw/openclaw@405c63f fix: strip standalone <function> tool call tags from visible text (#6… · openclaw/openclaw@78df859 fix(agents): preserve cli session metadata before transcript persist … · openclaw/openclaw@898fd04 docs(changelog): move cli transcript entry · openclaw/openclaw@c1817c6 fix(agents): normalize cli transcript api field · openclaw/openclaw@3a3fae0 docs(changelog): note cli transcript persistence · openclaw/openclaw@6c343f1 fix(agents): persist cli transcript turns · openclaw/openclaw@b8ef507 fix(msteams): harden security-sensitive flows (#65841) · openclaw/openclaw@c56b56e [Dashboard] Fix exec approval modal overflow for long command content… · openclaw/openclaw@053c5b0 Docs: remove QA changelog entry · openclaw/openclaw@7fd5771 QA: fix private runtime source loading (#67428) · openclaw/openclaw@d5933af docs(gateway): correct protocol.md schema path, hello-ok example, aut… · openclaw/openclaw@489404d CI: pin Node 22 runners to 22.18.0 · openclaw/openclaw@4ffa621 models.authStatus: normalize provider ids + tighten env-backed escape… · openclaw/openclaw@f2fdb9d Update CHANGELOG.md · openclaw/openclaw@7694a92 test(parallels): clean up npm update guard jobs · openclaw/openclaw@045ea7b Plugins: prefer scanDir override paths · openclaw/openclaw@b2974da fix(dreaming): default storage.mode to "separate" so phase blocks sto… · openclaw/openclaw@8c392f0 fix(memory-core): skip dreaming transcript ingestion via session stor… · openclaw/openclaw@a1b01f0 fix: dedupe replayed exec.finished node events (#67281) · openclaw/openclaw@5dcf526
feat: Add /models add hot-reload model registration (#70211) · openclaw/openclaw@f328c21
Takhoffman · 2026-04-23 · via Recent Commits to openclaw:main
Original file line numberDiff line numberDiff line change

@@ -6,6 +6,7 @@ Docs: https://docs.openclaw.ai

66
77

### Changes

88
9+

- Models/commands: add `/models add <provider> <modelId>` so you can register a model from chat and use it without restarting the gateway; keep `/models` as a simple provider browser while adding clearer add guidance and copy-friendly command examples. (#70211) Thanks @Takhoffman.

910

- Pi/models: update the bundled pi packages to `0.68.1` and let the OpenCode Go catalog come from pi instead of plugin-maintained model aliases, adding the refreshed `opencode-go/kimi-k2.6`, Qwen, GLM, MiMo, and MiniMax entries.

1011

- CLI/doctor plugins: lazy-load doctor plugin paths and prefer installed plugin `dist/*` runtime entries over source-adjacent JavaScript fallbacks, reducing the measured `doctor --non-interactive` runtime by about 74% while keeping cold doctor startup on built plugin artifacts. (#69840) Thanks @gumadeiras.

1112

- WhatsApp/groups+direct: forward per-group and per-direct `systemPrompt` config into inbound context `GroupSystemPrompt` so configured per-chat behavioral instructions are injected on every turn. Supports `"*"` wildcard fallback and account-scoped overrides under `channels.whatsapp.accounts.<id>.{groups,direct}`; account maps fully replace root maps (no deep merge), matching the existing `requireMention` pattern. Closes #7011. (#59553) Thanks @Bluetegu.

Original file line numberDiff line numberDiff line change

@@ -307,7 +307,7 @@ By default, components are single use. Set `components.reusable=true` to allow b

307307
308308

To restrict who can click a button, set `allowedUsers` on that button (Discord user IDs, tags, or `*`). When configured, unmatched users receive an ephemeral denial.

309309
310-

The `/model` and `/models` slash commands open an interactive model picker with provider and model dropdowns plus a Submit step. The picker reply is ephemeral and only the invoking user can use it.

310+

The `/model` and `/models` slash commands open an interactive model picker with provider and model dropdowns plus a Submit step. `/models add` also supports adding a new provider/model entry from chat, and newly added models show up without restarting the gateway. The picker reply is ephemeral and only the invoking user can use it.

311311
312312

File attachments:

313313
Original file line numberDiff line numberDiff line change

@@ -361,8 +361,8 @@ Surface different features that extend the above defaults.

361361

},

362362

{

363363

"command": "/models",

364-

"description": "List providers or models for a provider",

365-

"usage_hint": "[provider] [page] [limit=<n>|size=<n>|all]"

364+

"description": "List providers/models or add a model",

365+

"usage_hint": "[provider] [page] [limit=<n>|size=<n>|all] | add <provider> <modelId>"

366366

},

367367

{

368368

"command": "/help",

Original file line numberDiff line numberDiff line change

@@ -114,6 +114,9 @@ Notes:

114114
115115

- `/model` (and `/model list`) is a compact, numbered picker (model family + available providers).

116116

- On Discord, `/model` and `/models` open an interactive picker with provider and model dropdowns plus a Submit step.

117+

- `/models add` lets you add a provider/model entry from chat without editing config manually.

118+

- `/models add <provider> <modelId>` is the fastest path; bare `/models add` starts a provider-first guided flow where supported.

119+

- After `/models add`, the new model becomes available in `/models` and `/model` without restarting the gateway.

117120

- `/model <#>` selects from that picker.

118121

- `/model` persists the new session selection immediately.

119122

- If the agent is idle, the next run uses the new model right away.

@@ -132,6 +135,14 @@ Notes:

132135
133136

Full command behavior/config: [Slash commands](/tools/slash-commands).

134137
138+

Examples:

139+
140+

```text

141+

/models add

142+

/models add ollama glm-5.1:cloud

143+

/models add lmstudio qwen/qwen3.5-9b

144+

```

145+
135146

## CLI commands

136147
137148

```bash

Original file line numberDiff line numberDiff line change

@@ -68,7 +68,7 @@ import {

6868

withResolvedTelegramForumFlag,

6969

} from "./bot/helpers.js";

7070

import type { TelegramContext, TelegramGetChat } from "./bot/types.js";

71-

import { buildCommandsPaginationKeyboard } from "./command-ui.js";

71+

import { buildCommandsPaginationKeyboard, buildTelegramModelsMenuButtons } from "./command-ui.js";

7272

import {

7373

resolveTelegramConversationBaseSessionKey,

7474

resolveTelegramConversationRoute,

@@ -1509,7 +1509,7 @@ export const registerTelegramHandlers = ({

15091509

id: p,

15101510

count: byProvider.get(p)?.size ?? 0,

15111511

}));

1512-

const buttons = buildProviderKeyboard(providerInfos);

1512+

const buttons = buildTelegramModelsMenuButtons({ providers: providerInfos });

15131513

try {

15141514

await editMessageWithButtons("Select a provider:", buttons);

15151515

} catch (err) {

@@ -1527,7 +1527,7 @@ export const registerTelegramHandlers = ({

15271527

id: p,

15281528

count: byProvider.get(p)?.size ?? 0,

15291529

}));

1530-

const buttons = buildProviderKeyboard(providerInfos);

1530+

const buttons = buildTelegramModelsMenuButtons({ providers: providerInfos });

15311531

try {

15321532

await editMessageWithButtons(

15331533

`Unknown provider: ${provider}\n\nSelect a provider:`,

@@ -1580,7 +1580,7 @@ export const registerTelegramHandlers = ({

15801580

id: p,

15811581

count: byProvider.get(p)?.size ?? 0,

15821582

}));

1583-

const buttons = buildProviderKeyboard(providerInfos);

1583+

const buttons = buildTelegramModelsMenuButtons({ providers: providerInfos });

15841584

try {

15851585

await editMessageWithButtons(

15861586

`Could not resolve model "${selection.model}".\n\nSelect a provider:`,

Original file line numberDiff line numberDiff line change

@@ -3274,6 +3274,16 @@ describe("createTelegramBot", () => {

32743274

expect(buildModelsProviderDataMock).toHaveBeenCalledTimes(2);

32753275

expect(editMessageTextSpy).toHaveBeenCalledTimes(1);

32763276

expect(editMessageTextSpy.mock.calls[0]?.[2]).toContain("Select a provider:");

3277+

expect(

3278+

(

3279+

editMessageTextSpy.mock.calls[0]?.[3] as {

3280+

reply_markup?: { inline_keyboard?: unknown[][] };

3281+

}

3282+

)?.reply_markup?.inline_keyboard?.[0]?.[0],

3283+

).toEqual({

3284+

text: "Add model",

3285+

callback_data: "/models add",

3286+

});

32773287

});

32783288
32793289

it("retries command pagination callbacks after a bubbled edit failure", async () => {

@@ -3654,6 +3664,16 @@ describe("createTelegramBot", () => {

36543664
36553665

expect(editMessageTextSpy).toHaveBeenCalledTimes(2);

36563666

expect(editMessageTextSpy.mock.calls.at(-1)?.[2]).toContain("Select a provider:");

3667+

expect(

3668+

(

3669+

editMessageTextSpy.mock.calls.at(-1)?.[3] as {

3670+

reply_markup?: { inline_keyboard?: unknown[][] };

3671+

}

3672+

)?.reply_markup?.inline_keyboard?.[0]?.[0],

3673+

).toEqual({

3674+

text: "Add model",

3675+

callback_data: "/models add",

3676+

});

36573677

});

36583678
36593679

it("retries model selection callbacks after a bubbled session-store failure", async () => {

Original file line numberDiff line numberDiff line change

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

99

toPluginMessageContext,

1010

toPluginMessageSentEvent,

1111

} from "openclaw/plugin-sdk/hook-runtime";

12+

import type { ReplyPayloadDelivery } from "openclaw/plugin-sdk/interactive-runtime";

1213

import { buildOutboundMediaLoadOptions } from "openclaw/plugin-sdk/media-runtime";

1314

import { isGifMedia, kindFromMime } from "openclaw/plugin-sdk/media-runtime";

1415

import {

@@ -487,7 +488,7 @@ async function deliverMediaReply(params: {

487488

}

488489
489490

async function maybePinFirstDeliveredMessage(params: {

490-

pin: NonNullable<ReplyPayload["delivery"]>["pin"] | undefined;

491+

pin: ReplyPayloadDelivery["pin"];

491492

bot: Bot;

492493

chatId: string;

493494

runtime: RuntimeEnv;

Original file line numberDiff line numberDiff line change

@@ -9,6 +9,26 @@ import {

99
1010

export { buildCommandsPaginationKeyboard };

1111
12+

export function buildTelegramModelsMenuButtons(params: { providers: ProviderInfo[] }) {

13+

return [

14+

[{ text: "Add model", callback_data: "/models add" }],

15+

...buildProviderKeyboard(params.providers),

16+

];

17+

}

18+
19+

export function buildTelegramModelsMenuChannelData(params: {

20+

providers: ProviderInfo[];

21+

}): ReplyPayload["channelData"] | null {

22+

if (params.providers.length === 0) {

23+

return null;

24+

}

25+

return {

26+

telegram: {

27+

buttons: buildTelegramModelsMenuButtons(params),

28+

},

29+

};

30+

}

31+
1232

export function buildTelegramCommandsListChannelData(params: {

1333

currentPage: number;

1434

totalPages: number;

@@ -41,6 +61,25 @@ export function buildTelegramModelsProviderChannelData(params: {

4161

};

4262

}

4363
64+

export function buildTelegramModelsAddProviderChannelData(params: {

65+

providers: Array<{ id: string }>;

66+

}): ReplyPayload["channelData"] | null {

67+

if (params.providers.length === 0) {

68+

return null;

69+

}

70+

const buttons = params.providers.map((provider) => [

71+

{

72+

text: provider.id,

73+

callback_data: `/models add ${provider.id}`,

74+

},

75+

]);

76+

return {

77+

telegram: {

78+

buttons,

79+

},

80+

};

81+

}

82+
4483

export function buildTelegramModelsListChannelData(params: {

4584

provider: string;

4685

models: readonly string[];

Original file line numberDiff line numberDiff line change

@@ -28,6 +28,42 @@ function resolveAccount(cfg: OpenClawConfig, accountId: string): ResolvedTelegra

2828

}

2929
3030

describe("createTelegramPluginBase config duplicate token guard", () => {

31+

it("wires the top-level models menu adapter into the production plugin", () => {

32+

const channelData = telegramPluginBase.commands?.buildModelsMenuChannelData?.({

33+

providers: [

34+

{ id: "anthropic", count: 2 },

35+

{ id: "openai", count: 3 },

36+

],

37+

});

38+
39+

expect(channelData).toEqual({

40+

telegram: {

41+

buttons: [

42+

[{ text: "Add model", callback_data: "/models add" }],

43+

[

44+

{ text: "anthropic (2)", callback_data: "mdl_list_anthropic_1" },

45+

{ text: "openai (3)", callback_data: "mdl_list_openai_1" },

46+

],

47+

],

48+

},

49+

});

50+

});

51+
52+

it("wires the guided add-provider adapter into the production plugin", () => {

53+

const channelData = telegramPluginBase.commands?.buildModelsAddProviderChannelData?.({

54+

providers: [{ id: "ollama" }, { id: "lmstudio" }],

55+

});

56+
57+

expect(channelData).toEqual({

58+

telegram: {

59+

buttons: [

60+

[{ text: "ollama", callback_data: "/models add ollama" }],

61+

[{ text: "lmstudio", callback_data: "/models add lmstudio" }],

62+

],

63+

},

64+

});

65+

});

66+
3167

it("marks secondary account as not configured when token is shared", async () => {

3268

const cfg = createCfg();

3369

const alertsAccount = resolveAccount(cfg, "alerts");

Original file line numberDiff line numberDiff line change

@@ -19,7 +19,9 @@ import {

1919

import {

2020

buildTelegramCommandsListChannelData,

2121

buildTelegramModelBrowseChannelData,

22+

buildTelegramModelsAddProviderChannelData,

2223

buildTelegramModelsListChannelData,

24+

buildTelegramModelsMenuChannelData,

2325

buildTelegramModelsProviderChannelData,

2426

} from "./command-ui.js";

2527

import { TelegramChannelConfigSchema } from "./config-schema.js";

@@ -148,7 +150,9 @@ export function createTelegramPluginBase(params: {

148150

nativeCommandsAutoEnabled: true,

149151

nativeSkillsAutoEnabled: true,

150152

buildCommandsListChannelData: buildTelegramCommandsListChannelData,

153+

buildModelsMenuChannelData: buildTelegramModelsMenuChannelData,

151154

buildModelsProviderChannelData: buildTelegramModelsProviderChannelData,

155+

buildModelsAddProviderChannelData: buildTelegramModelsAddProviderChannelData,

152156

buildModelsListChannelData: buildTelegramModelsListChannelData,

153157

buildModelBrowseChannelData: buildTelegramModelBrowseChannelData,

154158

},