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

推荐订阅源

N
News and Events Feed by Topic
Malwarebytes
Malwarebytes
Threat Intelligence Blog | Flashpoint
Threat Intelligence Blog | Flashpoint
C
Cybersecurity and Infrastructure Security Agency CISA
F
Future of Privacy Forum
C
Cisco Blogs
T
The Exploit Database - CXSecurity.com
A
Arctic Wolf
S
Securelist
K
Kaspersky official blog
S
Schneier on Security
T
ThreatConnect
T
Tenable Blog
Spread Privacy
Spread Privacy
T
True Tiger Recordings
AWS News Blog
AWS News Blog
F
Fox-IT International blog
量子位
T
Threatpost
V
Vulnerabilities – Threatpost
C
CERT Recently Published Vulnerability Notes
Cisco Talos Blog
Cisco Talos Blog
GbyAI
GbyAI
宝玉的分享
宝玉的分享
腾讯CDC
G
Google Developers Blog
aimingoo的专栏
aimingoo的专栏
Cyberwarzone
Cyberwarzone
有赞技术团队
有赞技术团队
S
SegmentFault 最新的问题
OSCHINA 社区最新新闻
OSCHINA 社区最新新闻
V
Visual Studio Blog
U
Unit 42
雷峰网
雷峰网
cs.CV updates on arXiv.org
cs.CV updates on arXiv.org
Simon Willison's Weblog
Simon Willison's Weblog
O
OpenAI News
freeCodeCamp Programming Tutorials: Python, JavaScript, Git & More
The GitHub Blog
The GitHub Blog
The Register - Security
The Register - Security
MyScale Blog
MyScale Blog
小众软件
小众软件
A
About on SuperTechFans
Last Week in AI
Last Week in AI
Y
Y Combinator Blog
博客园 - 三生石上(FineUI控件)
美团技术团队
Google Online Security Blog
Google Online Security Blog
P
Proofpoint News Feed
MongoDB | Blog
MongoDB | Blog

Recent Commits to openclaw:main

fix(scripts): harden Windows native opus install · openclaw/openclaw@43252c8 fix(agents): match runtime policy entries when session provider is em… fix(scripts): harden Windows generated formatting · openclaw/openclaw@0a98559 fix(mcp): bound tools/list during catalog discovery (#85063) · openclaw/openclaw@07f500a fix(test): focus plugin binding Docker smoke · openclaw/openclaw@dfa1a51 test(e2e): fail release memory indexing errors test(daemon): fail launchd integration bootstrap errors · openclaw/openclaw@af07769 feat(imessage): support thumb approval reactions (#85952) · openclaw/openclaw@5c7980f fix(crabbox): default macos aws runs on demand fix(scripts): preserve test passthrough args · openclaw/openclaw@e4332f7 fix(e2e): harden Windows plugin assertions fix(test): mount upgrade survivor helper · openclaw/openclaw@5f03154 fix(android): prevent stale chat during session switches fix(android): keep permission setup action visible · openclaw/openclaw@94bc18a style(android): sharpen voice mode surfaces · openclaw/openclaw@c452510 fix(android): hide internal chat content blocks · openclaw/openclaw@d86ed21 style(android): refine list surface spacing · openclaw/openclaw@955909c feat(android): add pair new gateway action · openclaw/openclaw@cc5eb97 Advance iMessage catchup cursor after live handling (#85475) · openclaw/openclaw@102555c fix(scripts): ignore forwarded arg separator · openclaw/openclaw@79ee70c fix(test): fail empty gateway startup samples · openclaw/openclaw@5a8ce6a fix(e2e): harden Windows kitchen sink assertions · openclaw/openclaw@87a2eba fix(e2e): harden Telegram credential paths on Windows · openclaw/openclaw@c643370 fix(android): align setup pairing scopes fix(android): complete qr setup operator handoff · openclaw/openclaw@be9bb77 fix(test): copy cleanup smoke prepare hook · openclaw/openclaw@dbc08f6 fix(secrets): allow hash in exec SecretRef ids (#86072) · openclaw/openclaw@675158c fix(media): use static image compression metadata · openclaw/openclaw@694d45e fix(release): verify large plugin npm packs fix(test): require kitchen sink diagnostic canaries · openclaw/openclaw@7e51f83 fix(scripts): harden Windows upgrade survivor recipe · openclaw/openclaw@483d7be fix(installer): count verify progress stage fix: Refine PR template for review state (#86054) fix(test): repair split agent shard runs · openclaw/openclaw@125d82c fix(codex): harden Windows protocol formatting · openclaw/openclaw@ce48e4c fix(openrouter): use endpoint context limits (#86041) · openclaw/openclaw@dd01a2e test(qa): remove brittle capability flip setup turn fix(telegram): migrate legacy cache sidecars · openclaw/openclaw@eb9b882 fix(telegram): migrate account topic cache sidecars · openclaw/openclaw@5cfb12f fix(scripts): harden Windows ZAI fallback repro · openclaw/openclaw@5be62e7 style(android): sharpen v2 screen rhythm · openclaw/openclaw@400d90a test(qa): extend capability flip setup budget · openclaw/openclaw@c91c3c6 fix(android): simplify gateway status copy fix(android): route offline voice to gateway setup fix(scripts): harden Windows control UI i18n commands · openclaw/openclaw@581c8a6 fix(android): stop operator chat subscription · openclaw/openclaw@5c15859 fix(test): fail missing kitchen sink rss samples test(qa): widen capability flip restart budget · openclaw/openclaw@c7d4e9e fix(android): smooth gateway pairing recovery · openclaw/openclaw@60e6ccd fix(test): suppress rolldown timing noise · openclaw/openclaw@6d9b388 style(android): fix talk mode ktlint formatting · openclaw/openclaw@01b284c fix(telegram): store topic cache in plugin state fix(telegram): store bot info cache in plugin state · openclaw/openclaw@2ed5296 fix(test): sync sparse AWS Crabbox runs from full checkout · openclaw/openclaw@0f82c81 fix(release): harden Windows cross-os command shims · openclaw/openclaw@7154767 fix(test): harden Docker resource ceilings test(telegram): keep startup limiter coverage focused test(telegram): isolate startup probe limiter timing · openclaw/openclaw@04d86e0 test(release): harden plugin prerelease checks · openclaw/openclaw@578e73f fix(telegram): serialize topic dispatch replies (#85709) · openclaw/openclaw@62b51a6 test(release): stabilize plugin prerelease checks · openclaw/openclaw@3679151 fix(test): fail live gateway false greens · openclaw/openclaw@295339d fix(test): build startup artifacts for smoke scripts · openclaw/openclaw@3838e45 test(telegram): wait for startup probe slots · openclaw/openclaw@0a8af67 test(codex): match sandbox exec-server yolo policy · openclaw/openclaw@783290f fix(test): fail missing explicit test targets · openclaw/openclaw@9ff4d36 test(codex): avoid full sandbox exec-server turn run · openclaw/openclaw@558c1bc fix(ci): keep Crabbox pnpm hydration shims writable · openclaw/openclaw@bca1ac0 fix(release): harden Windows release-check npm probes · openclaw/openclaw@75ac11a fix(docker): parse peer-suffixed lockfile packages · openclaw/openclaw@cf46f2e fix(docker): seed lockfile packages before prune · openclaw/openclaw@f799da0 fix(docker): seed lockfile snapshot tarballs before prune · openclaw/openclaw@2cd93f1 test(codex): type thread start mock params · openclaw/openclaw@a4ef3a2 test(codex): avoid full sandbox run in thread-start test · openclaw/openclaw@11bf642 fix(plugins): harden Windows npm package staging · openclaw/openclaw@abdd8a4 test(codex): complete sandbox turn inline · openclaw/openclaw@c14a0c6 fix(release): harden Windows npm shim verification · openclaw/openclaw@a56f452 test(release): type metadata snapshot mock params · openclaw/openclaw@f878959 test(release): finish plugin metadata prerelease sync fix(update): avoid broad tag fetches for dev updates (#84737) · openclaw/openclaw@501f2cb Fix iMessage slash command acknowledgements (#82642) · openclaw/openclaw@4d15020 test(release): align prerelease contracts · openclaw/openclaw@02f53e6 test(release): align plugin prerelease checks fix(docker): copy prepare hook before install · openclaw/openclaw@0ba6b23 fix: share signed thinking replay policy fix(memory): strip invalid thinking signatures for signed-thinking pr… · openclaw/openclaw@41329c0 fix: preserve signed thinking tool ids · openclaw/openclaw@906476a fix(scripts): harden Windows install checks · openclaw/openclaw@d21abb8 fix(release): keep private QA markers out of bundled alias code · openclaw/openclaw@b972ac1 fix(discord): harden realtime voice wake joins · openclaw/openclaw@fdfcb07 test(parallels): harden release VM smoke isolation · openclaw/openclaw@3839b48 fix: keep blank agent allowlists fail closed (#85849) · openclaw/openclaw@0f83c93 fix: harden session allowlist glob matching (#85849) (thanks @SebTardif) · openclaw/openclaw@88aa713 fix(security): replace regex wildcard matching with linear-time glob … fix(telegram): normalize legacy action targets · openclaw/openclaw@32631eb fix: notify chat when main session recovery fails · openclaw/openclaw@cf61b87 docs(changelog): note restart recovery notice · openclaw/openclaw@ae9308b fix(minimax): normalize OAuth token expiry to absolute millisecond ti… fix(memory-wiki): show vault totals in palace summary (#85824) Isolate boot-md startup sessions (#85919) · openclaw/openclaw@8deb1ef
test(e2e): select installable bundled plugins · openclaw/openclaw@0acc3e3
vincentkoc · 2026-05-25 · via Recent Commits to openclaw:main

@@ -1,53 +1,93 @@

1+

import { spawnSync } from "node:child_process";

12

import fs from "node:fs";

23

import path from "node:path";

3445

const readJson = (file) => JSON.parse(fs.readFileSync(file, "utf8"));

566-

function loadManifestEntries() {

7-

const explicit = (process.env.OPENCLAW_BUNDLED_PLUGIN_SWEEP_IDS || "")

8-

.split(/[,\s]+/u)

9-

.map((entry) => entry.trim())

10-

.filter(Boolean);

11-

const extensionRoot = path.join(process.cwd(), "dist", "extensions");

12-

const manifestEntries = fs

13-

.readdirSync(extensionRoot, { withFileTypes: true })

14-

.filter((entry) => entry.isDirectory())

15-

.map((entry) => {

16-

const manifestPath = path.join(extensionRoot, entry.name, "openclaw.plugin.json");

17-

if (!fs.existsSync(manifestPath)) {

7+

function resolveOpenClawEntry() {

8+

if (process.env.OPENCLAW_ENTRY) {

9+

return process.env.OPENCLAW_ENTRY;

10+

}

11+

for (const entry of ["dist/index.mjs", "dist/index.js"]) {

12+

if (fs.existsSync(entry)) {

13+

return entry;

14+

}

15+

}

16+

throw new Error("Missing OPENCLAW_ENTRY and dist/index.(m)js");

17+

}

18+19+

function readPluginsList() {

20+

const entry = resolveOpenClawEntry();

21+

const result = spawnSync(process.execPath, [entry, "plugins", "list", "--json"], {

22+

cwd: process.cwd(),

23+

encoding: "utf8",

24+

env: process.env,

25+

});

26+

if (result.status !== 0) {

27+

throw new Error(

28+

`Unable to list packaged bundled plugins: ${result.stderr || result.stdout || `exit ${result.status}`}`,

29+

);

30+

}

31+

const payload = JSON.parse(result.stdout);

32+

return Array.isArray(payload.plugins) ? payload.plugins : [];

33+

}

34+35+

function pluginRequiresConfig(pluginDir) {

36+

const manifestPath = path.join(pluginDir, "openclaw.plugin.json");

37+

if (!fs.existsSync(manifestPath)) {

38+

throw new Error(`missing bundled plugin manifest: ${manifestPath}`);

39+

}

40+

const manifest = readJson(manifestPath);

41+

const required = manifest.configSchema?.required;

42+

return Array.isArray(required) && required.some((value) => typeof value === "string");

43+

}

44+45+

async function loadPackagedBundledEntries() {

46+

return readPluginsList()

47+

.filter((plugin) => plugin?.origin === "bundled")

48+

.map((plugin) => {

49+

const id = typeof plugin.id === "string" ? plugin.id.trim() : "";

50+

const rootDir = typeof plugin.rootDir === "string" ? plugin.rootDir.trim() : "";

51+

const source = typeof plugin.source === "string" ? plugin.source.trim() : "";

52+

const pluginDir = rootDir || (source ? path.dirname(source) : "");

53+

if (!id || !pluginDir) {

1854

return null;

1955

}

20-

const manifest = readJson(manifestPath);

21-

const id = typeof manifest.id === "string" ? manifest.id.trim() : "";

22-

if (!id) {

23-

throw new Error(`Bundled plugin manifest is missing id: ${manifestPath}`);

24-

}

25-

const required = manifest.configSchema?.required;

2656

return {

2757

id,

28-

dir: entry.name,

29-

requiresConfig:

30-

Array.isArray(required) && required.some((value) => typeof value === "string"),

58+

dir: path.basename(pluginDir),

59+

rootDir: pluginDir,

60+

requiresConfig: pluginRequiresConfig(pluginDir),

3161

};

3262

})

3363

.filter(Boolean)

3464

.toSorted((a, b) => a.id.localeCompare(b.id));

65+

}

66+67+

async function loadManifestEntries() {

68+

const explicit = (process.env.OPENCLAW_BUNDLED_PLUGIN_SWEEP_IDS || "")

69+

.split(/[,\s]+/u)

70+

.map((entry) => entry.trim())

71+

.filter(Boolean);

72+

const manifestEntries = await loadPackagedBundledEntries();

35733674

if (explicit.length === 0) {

3775

return manifestEntries;

3876

}

39-

return explicit.map(

40-

(lookup) =>

41-

manifestEntries.find((entry) => entry.id === lookup || entry.dir === lookup) || {

42-

id: lookup,

43-

dir: lookup,

44-

requiresConfig: false,

45-

},

46-

);

77+

const available = manifestEntries.map((entry) => entry.id).join(", ");

78+

return explicit.map((lookup) => {

79+

const found = manifestEntries.find((entry) => entry.id === lookup || entry.dir === lookup);

80+

if (!found) {

81+

throw new Error(

82+

`OPENCLAW_BUNDLED_PLUGIN_SWEEP_IDS entry is not an installable bundled plugin in this package: ${lookup}. Available: ${available}`,

83+

);

84+

}

85+

return found;

86+

});

4787

}

488849-

function selectedManifestEntries() {

50-

const allEntries = loadManifestEntries();

89+

async function selectedManifestEntries() {

90+

const allEntries = await loadManifestEntries();

5191

const total = Number.parseInt(process.env.OPENCLAW_BUNDLED_PLUGIN_SWEEP_TOTAL || "1", 10);

5292

const index = Number.parseInt(process.env.OPENCLAW_BUNDLED_PLUGIN_SWEEP_INDEX || "0", 10);

5393

if (!Number.isInteger(total) || total < 1) {

@@ -85,15 +125,23 @@ function assertInstalled(pluginId, pluginDir, requiresConfig) {

85125

}

86126

if (

87127

typeof record.sourcePath !== "string" ||

88-

!record.sourcePath.includes(`/dist/extensions/${pluginDir}`)

128+

![`/dist/extensions/${pluginDir}`, `/dist-runtime/extensions/${pluginDir}`].some((fragment) =>

129+

record.sourcePath.includes(fragment),

130+

)

89131

) {

90132

throw new Error(`unexpected bundled source path for ${pluginId}: ${record.sourcePath}`);

91133

}

92134

if (record.installPath !== record.sourcePath) {

93135

throw new Error(`bundled install path should equal source path for ${pluginId}`);

94136

}

95137

const paths = config.plugins?.load?.paths || [];

96-

if (paths.some((entry) => String(entry).includes(`/dist/extensions/${pluginDir}`))) {

138+

if (

139+

paths.some((entry) =>

140+

[`/dist/extensions/${pluginDir}`, `/dist-runtime/extensions/${pluginDir}`].some(

141+

(fragment) => String(entry).includes(fragment),

142+

),

143+

)

144+

) {

97145

throw new Error(`config load paths should not include bundled install path for ${pluginId}`);

98146

}

99147

if (requiresConfig && config.plugins?.entries?.[pluginId]?.enabled === true) {

@@ -123,7 +171,13 @@ function assertUninstalled(pluginId, pluginDir) {

123171

throw new Error(`install record still present after uninstall for ${pluginId}`);

124172

}

125173

const paths = config.plugins?.load?.paths || [];

126-

if (paths.some((entry) => String(entry).includes(`/dist/extensions/${pluginDir}`))) {

174+

if (

175+

paths.some((entry) =>

176+

[`/dist/extensions/${pluginDir}`, `/dist-runtime/extensions/${pluginDir}`].some(

177+

(fragment) => String(entry).includes(fragment),

178+

),

179+

)

180+

) {

127181

throw new Error(`load path still present after uninstall for ${pluginId}`);

128182

}

129183

if (config.plugins?.entries?.[pluginId]) {

@@ -145,8 +199,8 @@ function assertUninstalled(pluginId, pluginDir) {

145199146200

const [command, pluginId, pluginDir, requiresConfig] = process.argv.slice(2);

147201

if (command === "select") {

148-

for (const entry of selectedManifestEntries()) {

149-

console.log(`${entry.id}\t${entry.dir}\t${entry.requiresConfig ? "1" : "0"}`);

202+

for (const entry of await selectedManifestEntries()) {

203+

console.log(`${entry.id}\t${entry.dir}\t${entry.requiresConfig ? "1" : "0"}\t${entry.rootDir}`);

150204

}

151205

} else if (command === "assert-installed") {

152206

assertInstalled(pluginId, pluginDir, requiresConfig === "1");