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

推荐订阅源

Martin Fowler
Martin Fowler
cs.CV updates on arXiv.org
cs.CV updates on arXiv.org
T
Threat Research - Cisco Blogs
Threat Intelligence Blog | Flashpoint
Threat Intelligence Blog | Flashpoint
C
Cyber Attacks, Cyber Crime and Cyber Security
cs.CL updates on arXiv.org
cs.CL updates on arXiv.org
T
Troy Hunt's Blog
V
V2EX - 技术
Hacker News - Newest:
Hacker News - Newest: "LLM"
H
Heimdal Security Blog
T
Tor Project blog
IT之家
IT之家
Project Zero
Project Zero
GbyAI
GbyAI
Security Latest
Security Latest
S
Security Archives - TechRepublic
人人都是产品经理
人人都是产品经理
大猫的无限游戏
大猫的无限游戏
Spread Privacy
Spread Privacy
S
Security Affairs
A
Arctic Wolf
C
Cybersecurity and Infrastructure Security Agency CISA
I
Intezer
P
Palo Alto Networks Blog
宝玉的分享
宝玉的分享
Google DeepMind News
Google DeepMind News
T
Threatpost
I
InfoQ
F
Full Disclosure
Blog — PlanetScale
Blog — PlanetScale
Last Week in AI
Last Week in AI
Cisco Talos Blog
Cisco Talos Blog
N
Netflix TechBlog - Medium
MyScale Blog
MyScale Blog
H
Help Net Security
S
Securelist
Y
Y Combinator Blog
月光博客
月光博客
博客园_首页
Engineering at Meta
Engineering at Meta
酷 壳 – CoolShell
酷 壳 – CoolShell
J
Java Code Geeks
CTFtime.org: upcoming CTF events
CTFtime.org: upcoming CTF events
A
About on SuperTechFans
K
Kaspersky official blog
Microsoft Azure Blog
Microsoft Azure Blog
Vercel News
Vercel News
阮一峰的网络日志
阮一峰的网络日志
T
The Exploit Database - CXSecurity.com
B
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) fix(gateway): capture config hash after plugin auto-enable to prevent… · openclaw/openclaw@8c11210 fix: repair sanitized replay tool results before send (#67620) (thank… fix: restrict HTML timeout short-circuit to transient statuses fix: keep TUI watchdog bound to active run (#67401) (thanks @xantorres) Gateway/skills: dedupe skills prefix-match + drop dead fallback on log Extensions/lmstudio: back off inference preload after consecutive fai… TUI/streaming: add watchdog that resets the activity indicator after … Agents/tool-loop: enable unknown-tool stream guard by default · openclaw/openclaw@36ed367 Gateway/skills: invalidate session skills snapshot on config write fix: classify HTML provider error pages correctly (#67642) (thanks @s… 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… fix(deps): bump dompurify to 3.4.0 (#67614) CI: add explicit permissions to all workflow jobs (fixes code-scannin… fix: register bundled TTS providers and route overrides correctly (#6… fix: align host tilde paths with OS home (#62804) (thanks @stainlu) 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
fix(diffs): refresh live tool config · openclaw/openclaw@44965bf
vincentkoc · 2026-04-23 · via Recent Commits to openclaw:main

@@ -192,6 +192,138 @@ describe("PlaywrightDiffScreenshotter", () => {

192192

});

193193194194

describe("diffs plugin registration", () => {

195+

it("uses live runtime tool config through the registered tool factory", async () => {

196+

type RegisteredTool = {

197+

execute?: (toolCallId: string, params: Record<string, unknown>) => Promise<unknown>;

198+

};

199+

type HttpRouteHandler = (

200+

req: IncomingMessage,

201+

res: ServerResponse,

202+

) => boolean | Promise<boolean>;

203+

type RegisteredHttpRouteParams = Parameters<OpenClawPluginApi["registerHttpRoute"]>[0];

204+205+

let registeredToolFactory:

206+

| ((ctx: OpenClawPluginToolContext) => RegisteredTool | RegisteredTool[] | null | undefined)

207+

| undefined;

208+

let registeredHttpRouteHandler: HttpRouteHandler | undefined;

209+

let configFile: OpenClawConfig = {

210+

gateway: {

211+

port: 18789,

212+

bind: "loopback",

213+

},

214+

plugins: {

215+

entries: {

216+

diffs: {

217+

config: {

218+

viewerBaseUrl: "https://startup.example.com/openclaw",

219+

defaults: {

220+

mode: "view",

221+

theme: "light",

222+

background: false,

223+

layout: "split",

224+

showLineNumbers: false,

225+

diffIndicators: "classic",

226+

lineSpacing: 2,

227+

},

228+

},

229+

},

230+

},

231+

},

232+

} as OpenClawConfig;

233+234+

const api = createTestPluginApi({

235+

id: "diffs",

236+

name: "Diffs",

237+

description: "Diffs",

238+

source: "test",

239+

config: {

240+

gateway: {

241+

port: 18789,

242+

bind: "loopback",

243+

},

244+

},

245+

pluginConfig: {

246+

viewerBaseUrl: "https://startup.example.com/openclaw",

247+

defaults: {

248+

mode: "view",

249+

theme: "light",

250+

background: false,

251+

layout: "split",

252+

showLineNumbers: false,

253+

diffIndicators: "classic",

254+

lineSpacing: 2,

255+

},

256+

},

257+

runtime: {

258+

config: {

259+

loadConfig: () => configFile,

260+

},

261+

} as never,

262+

registerTool(tool: Parameters<OpenClawPluginApi["registerTool"]>[0]) {

263+

registeredToolFactory = typeof tool === "function" ? tool : () => tool;

264+

},

265+

registerHttpRoute(params: RegisteredHttpRouteParams) {

266+

registeredHttpRouteHandler = params.handler as HttpRouteHandler;

267+

},

268+

on: vi.fn(),

269+

});

270+271+

registerDiffsPlugin(api as unknown as OpenClawPluginApi);

272+273+

configFile = {

274+

...configFile,

275+

plugins: {

276+

entries: {

277+

diffs: {

278+

config: {

279+

viewerBaseUrl: "https://live.example.com/gateway",

280+

defaults: {

281+

mode: "view",

282+

theme: "dark",

283+

background: true,

284+

layout: "unified",

285+

showLineNumbers: true,

286+

diffIndicators: "bars",

287+

lineSpacing: 1.6,

288+

},

289+

},

290+

},

291+

},

292+

},

293+

} as OpenClawConfig;

294+295+

const registeredTool = registeredToolFactory?.({

296+

agentId: "main",

297+

sessionId: "session-456",

298+

messageChannel: "discord",

299+

agentAccountId: "default",

300+

}) as RegisteredTool | undefined;

301+

const result = await registeredTool?.execute?.("tool-1", {

302+

before: "one\n",

303+

after: "two\n",

304+

});

305+

const details = (result as { details?: Record<string, unknown> } | undefined)?.details;

306+

const viewerPath = String(details?.viewerPath);

307+

const res = createMockServerResponse();

308+

const handled = await registeredHttpRouteHandler?.(

309+

localReq({

310+

method: "GET",

311+

url: viewerPath,

312+

}),

313+

res,

314+

);

315+316+

expect(handled).toBe(true);

317+

expect(String(details?.viewerUrl)).toContain("https://live.example.com/gateway");

318+

expect(res.statusCode).toBe(200);

319+

expect(String(res.body)).toContain('body data-theme="dark"');

320+

expect(String(res.body)).toContain('"backgroundEnabled":true');

321+

expect(String(res.body)).toContain('"diffStyle":"unified"');

322+

expect(String(res.body)).toContain('"disableLineNumbers":false');

323+

expect(String(res.body)).toContain('"diffIndicators":"bars"');

324+

expect(String(res.body)).toContain("--diffs-line-height: 24px;");

325+

});

326+195327

it("uses live runtime viewer-access config through the registered HTTP handler", async () => {

196328

type RegisteredTool = {

197329

execute?: (toolCallId: string, params: Record<string, unknown>) => Promise<unknown>;

@@ -299,12 +431,6 @@ describe("diffs plugin registration", () => {

299431300432

expect(handled).toBe(true);

301433

expect(res.statusCode).toBe(200);

302-

expect(String(res.body)).toContain('body data-theme="light"');

303-

expect(String(res.body)).toContain('"backgroundEnabled":false');

304-

expect(String(res.body)).toContain('"diffStyle":"split"');

305-

expect(String(res.body)).toContain('"disableLineNumbers":true');

306-

expect(String(res.body)).toContain('"diffIndicators":"classic"');

307-

expect(String(res.body)).toContain("--diffs-line-height: 30px;");

308434

expect((result as { details?: Record<string, unknown> } | undefined)?.details?.context).toEqual(

309435

{

310436

agentId: "main",

@@ -344,6 +470,107 @@ describe("diffs plugin registration", () => {

344470

expect(proxiedHandled).toBe(true);

345471

expect(proxiedRes.statusCode).toBe(404);

346472

});

473+474+

it("fails closed for remote viewer access when the live diffs plugin entry is removed", async () => {

475+

type RegisteredTool = {

476+

execute?: (toolCallId: string, params: Record<string, unknown>) => Promise<unknown>;

477+

};

478+

type HttpRouteHandler = (

479+

req: IncomingMessage,

480+

res: ServerResponse,

481+

) => boolean | Promise<boolean>;

482+

type RegisteredHttpRouteParams = Parameters<OpenClawPluginApi["registerHttpRoute"]>[0];

483+484+

let registeredToolFactory:

485+

| ((ctx: OpenClawPluginToolContext) => RegisteredTool | RegisteredTool[] | null | undefined)

486+

| undefined;

487+

let registeredHttpRouteHandler: HttpRouteHandler | undefined;

488+

let configFile: OpenClawConfig = {

489+

gateway: {

490+

port: 18789,

491+

bind: "loopback",

492+

},

493+

plugins: {

494+

entries: {

495+

diffs: {

496+

config: {

497+

security: {

498+

allowRemoteViewer: true,

499+

},

500+

},

501+

},

502+

},

503+

},

504+

} as OpenClawConfig;

505+506+

const api = createTestPluginApi({

507+

id: "diffs",

508+

name: "Diffs",

509+

description: "Diffs",

510+

source: "test",

511+

config: {

512+

gateway: {

513+

port: 18789,

514+

bind: "loopback",

515+

},

516+

},

517+

pluginConfig: {

518+

security: {

519+

allowRemoteViewer: true,

520+

},

521+

},

522+

runtime: {

523+

config: {

524+

loadConfig: () => configFile,

525+

},

526+

} as never,

527+

registerTool(tool: Parameters<OpenClawPluginApi["registerTool"]>[0]) {

528+

registeredToolFactory = typeof tool === "function" ? tool : () => tool;

529+

},

530+

registerHttpRoute(params: RegisteredHttpRouteParams) {

531+

registeredHttpRouteHandler = params.handler as HttpRouteHandler;

532+

},

533+

on: vi.fn(),

534+

});

535+536+

registerDiffsPlugin(api as unknown as OpenClawPluginApi);

537+538+

const registeredTool = registeredToolFactory?.({

539+

agentId: "main",

540+

sessionId: "session-789",

541+

messageChannel: "discord",

542+

agentAccountId: "default",

543+

}) as RegisteredTool | undefined;

544+

const result = await registeredTool?.execute?.("tool-1", {

545+

before: "one\n",

546+

after: "two\n",

547+

});

548+

const viewerPath = String(

549+

(result as { details?: Record<string, unknown> } | undefined)?.details?.viewerPath,

550+

);

551+552+

configFile = {

553+

...configFile,

554+

plugins: {

555+

entries: {},

556+

},

557+

} as OpenClawConfig;

558+559+

const proxiedRes = createMockServerResponse();

560+

const proxiedHandled = await registeredHttpRouteHandler?.(

561+

localReq({

562+

method: "GET",

563+

url: viewerPath,

564+

headers: {

565+

"x-forwarded-for": "203.0.113.10",

566+

},

567+

}),

568+

proxiedRes,

569+

);

570+571+

expect(proxiedHandled).toBe(true);

572+

expect(proxiedRes.statusCode).toBe(404);

573+

});

347574

});

348575349576

function createConfig(): OpenClawConfig {