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

推荐订阅源

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: suppress async media incomplete-turn errors (#85933) · openclaw/openclaw@35dcd42 migrate auth credentials · openclaw/openclaw@f036bac fix migrate auth lint · openclaw/openclaw@50e6cb0 fix migrate auth opt-out precedence · openclaw/openclaw@2016a51 honor migrate auth opt-out in plan address migrate auth review comments · openclaw/openclaw@0a98c2d fix ci blockers for migrate auth docs: add migrate auth changelog (#85667) fix(scripts): avoid duplicate install smoke ui build · openclaw/openclaw@b1b2841 fix(telegram): preserve inbound text entities (#83873) · openclaw/openclaw@b552919 chore: ignore Python bytecode caches · openclaw/openclaw@b6b2755 fix: make autoreview progress visible · openclaw/openclaw@236edb2 ci(release): fix plugin prerelease extension batch invocation test(telegram): provide topic cache store in message context harness · openclaw/openclaw@ff1fde1 test(agents): complete provider runtime test mocks · openclaw/openclaw@be8cd12 test(telegram): type topic cache harness store · openclaw/openclaw@84ab206 test(agents): sync provider runtime mocks · openclaw/openclaw@a289dd9 refactor: keep plain text tool-call promotion private (#86374) · openclaw/openclaw@c3ab2de fix(discord): suppress self-reply prompt echoes (#86238) docs: clarify config migration policy · openclaw/openclaw@c44367f fix(perf): fail startup bench on early gateway exit · openclaw/openclaw@a8fc28c fix: prevent plain text tool call leaks (#86222) · openclaw/openclaw@cd62780 fix: handle npm min-release-age in installers · openclaw/openclaw@316d97c fix(scripts): include ui:build in build-all full and ciArtifacts prof… · openclaw/openclaw@6704d0a fix(e2e): sample Windows kitchen sink gateway RSS · openclaw/openclaw@73189e3 fix(cron): respect isolated target and error on missing remove id (#8… · openclaw/openclaw@6709f4e fix(pi-embedded-runner): propagate trigger-derived priority to the gl… · openclaw/openclaw@0580f57 fix(cli): suppress self-update version warnings · openclaw/openclaw@e2bd20f fix: preserve webchat source reply details · openclaw/openclaw@aa50c51 docs: replace OpenClaw docs skill and add plugin permissions guide · openclaw/openclaw@0dabb70 fix(codex): preserve source reply mode for active runs (#86325) · openclaw/openclaw@b962110 fix: make compaction reinjection opt-in · openclaw/openclaw@ab910f8 fix codex usage-limit recovery copy (#86305) · openclaw/openclaw@c3c8a65 feat(ui): add ephemeral Activity tab · openclaw/openclaw@3dd0e8e fix(tests): harden native macos plugin proof · openclaw/openclaw@a5d5604 fix(commitments): serialize load-modify-save with in-process queue + … · openclaw/openclaw@d3c293d Fail Codex compaction at the Codex boundary (#85958) · openclaw/openclaw@dd47e47 fix(docker): restore config parent ownership · openclaw/openclaw@908b894 docs: clarify config default review policy (#86329) · openclaw/openclaw@3a03dd5 docs: clean changelog script entries · openclaw/openclaw@0eead19 fix(scripts): budget restart benchmark timeouts · openclaw/openclaw@5bd5509 fix: align ui vitest config assertion · openclaw/openclaw@730fd19 fix: route unit ui vitest targets narrowly · openclaw/openclaw@777402e fix: route explicit ui vitest targets narrowly · openclaw/openclaw@56a383c fix(android): harden play media permission removal fix(webchat): stabilize live transcript run state · openclaw/openclaw@119a01c fix(scripts): fail restart benchmark regressions · openclaw/openclaw@95d1b39 fix(openai): scope external codex auth to realtime fix(openai): prefer codex auth for GPT realtime · openclaw/openclaw@48c4f57 fix(openai): discover codex cli auth for provider checks · openclaw/openclaw@4656275 fix(android): keep talk mode on realtime relay · openclaw/openclaw@70614f8 test(android): add gateway connect adb probe · openclaw/openclaw@d7aa1f3 fix(android): stabilize realtime talk connection state · openclaw/openclaw@ffb02a5 test(android): add voice mode adb e2e harness · openclaw/openclaw@e52a3b3 fix(ci): stabilize deadcode and catalog checks · openclaw/openclaw@3db1508 fix(scripts): prebuild gateway cpu bench · openclaw/openclaw@ca70015 fix(e2e): harden bundled lifecycle probe on Windows · openclaw/openclaw@4798264 test(e2e): sample kitchen sink rpc peak rss · openclaw/openclaw@60c0f24 fix(scripts): remove stale deadcode allowlist entries · openclaw/openclaw@ea3bb92 fix(telegram): route polling diagnostics away from errors · openclaw/openclaw@b5c1199 fix(plugins): support linked source checkouts on Windows · openclaw/openclaw@793e300 fix(gateway): back off session tool mirrors under pressure (#84846) · openclaw/openclaw@42bdc94 fix(config): skip shell env fallback on Windows (#85739) · openclaw/openclaw@06bf302 fix(gateway): avoid duplicate session message broadcasts · openclaw/openclaw@1459044 fix: repair anchorless iMessage watch payloads · openclaw/openclaw@f37fbc9 fix(cli): route node status hints to stdout (#85780) · openclaw/openclaw@749692e fix(oc-path): support deep config edits (#86060) · openclaw/openclaw@3a72a30 fix(config): quiet benign metadata anomaly output · openclaw/openclaw@f3f4f29 fix(test): fail multi-node update regressions · openclaw/openclaw@732cf54 fix(google-vertex): support production ADC modes (#83971) · openclaw/openclaw@f09b4eb test(e2e): expose corrupt plugin deps smoke · openclaw/openclaw@fa3ff4d fix(codex): log app-server approval promotion trigger · openclaw/openclaw@d9af23f test(e2e): harden multi-node update smoke Clean up browser MCP subprocess tree (#85832) · openclaw/openclaw@8dc6b4d fix(agents): log warnings instead of swallowing subagent errors (#82943) · openclaw/openclaw@907bc03 fix(compaction): preserve partial summary on mid-chain chunk failure … · openclaw/openclaw@f0061dd fix(config): do not suppress recovery retry after failed backup resto… · openclaw/openclaw@5d174a5 chore: release 2026.5.25 fix(installer): support alpine cli installs · openclaw/openclaw@f68ed72 test(agents): keep runtime-plan provider mock current fix(scripts): launch env package scripts on Windows · openclaw/openclaw@4d4ce9e fix(agents): cache fallback provider resolution · openclaw/openclaw@3c8d101 fix(test): make import timing scripts Windows-safe · openclaw/openclaw@8ae9977 fix(telegram): transient Telegram pairing prompts (#85555) · openclaw/openclaw@8209426 fix(test): make max Vitest scripts Windows-safe · openclaw/openclaw@b681d5d fix(doctor): migrate Feishu account bot names (#86081) · openclaw/openclaw@9e8cc7e fix(scripts): prefilter conflict marker scans docs: add ClawSweeper review policy to AGENTS (#86197) · openclaw/openclaw@242e876 fix(installer): avoid before with npm release-age configs (#85491) · openclaw/openclaw@4742db6 fix(e2e): retry Windows kitchen sink probes · openclaw/openclaw@3e275a5 fix(installer): install node with apk on alpine fix(installer): detect musl linux shells · openclaw/openclaw@acfed37 perf(plugins,gateway): thread metadata snapshot + discovery through h… · openclaw/openclaw@8ccb11c fix(ui): split control ui runtime chunks · openclaw/openclaw@8bf4f7d refactor(config): extract GoogleChat schema into zod-schema.providers… · openclaw/openclaw@fe34141 fix(update): suppress internal handoff version warnings · openclaw/openclaw@6cc8244 test(e2e): select installable bundled plugins · openclaw/openclaw@0acc3e3 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 migrate supported auth imports · openclaw/openclaw@44bb2be
fuller-stack · 2026-05-25 · via Recent Commits to openclaw:main

@@ -1,3 +1,4 @@

1+

import { createHash } from "node:crypto";

12

import { loadAuthProfileStoreWithoutExternalProfiles } from "openclaw/plugin-sdk/agent-runtime";

23

import {

34

createMigrationItem,

@@ -38,8 +39,12 @@ const HERMES_AUTH_DISPLAY_NAME = "Hermes import";

38393940

type HermesCodexAuthCandidate = {

4041

access: string;

42+

accountId?: string;

4143

refresh: string;

44+

sourceKind: "hermes-auth-json" | "opencode-auth-json";

45+

sourceCredentialIndex?: number;

4246

sourceLabel: string;

47+

sourcePath: string;

4348

updatedAt?: number;

4449

};

4550

@@ -86,7 +91,7 @@ function decodeJwtPayload(token: string): Record<string, unknown> | undefined {

8691

}

8792

}

889389-

function resolveCodexIdentity(access: string): CodexIdentity {

94+

function resolveCodexIdentity(access: string, accountId?: string): CodexIdentity {

9095

const payload = decodeJwtPayload(access);

9196

const auth = isRecord(payload?.["https://api.openai.com/auth"])

9297

? payload["https://api.openai.com/auth"]

@@ -95,11 +100,11 @@ function resolveCodexIdentity(access: string): CodexIdentity {

95100

? payload["https://api.openai.com/profile"]

96101

: {};

97102

const email = readString(profile.email);

98-

const accountId = readString(auth.chatgpt_account_id);

103+

const resolvedAccountId = accountId ?? readString(auth.chatgpt_account_id);

99104

const chatgptPlanType = readString(auth.chatgpt_plan_type);

100105

if (email) {

101106

return {

102-

...(accountId ? { accountId } : {}),

107+

...(resolvedAccountId ? { accountId: resolvedAccountId } : {}),

103108

...(chatgptPlanType ? { chatgptPlanType } : {}),

104109

email,

105110

profileName: email,

@@ -109,9 +114,10 @@ function resolveCodexIdentity(access: string): CodexIdentity {

109114

readString(auth.chatgpt_account_user_id) ??

110115

readString(auth.chatgpt_user_id) ??

111116

readString(auth.user_id) ??

112-

readString(payload?.sub);

117+

readString(payload?.sub) ??

118+

resolvedAccountId;

113119

return {

114-

...(accountId ? { accountId } : {}),

120+

...(resolvedAccountId ? { accountId: resolvedAccountId } : {}),

115121

...(chatgptPlanType ? { chatgptPlanType } : {}),

116122

...(stableSubject

117123

? { profileName: `id-${Buffer.from(stableSubject).toString("base64url")}` }

@@ -131,7 +137,24 @@ function resolveAccessTokenExpiry(access: string): number | undefined {

131137

return undefined;

132138

}

133139134-

function readProviderTokens(auth: Record<string, unknown>): HermesCodexAuthCandidate | undefined {

140+

function sourceCredentialFingerprint(candidate: HermesCodexAuthCandidate): string {

141+

const hash = createHash("sha256");

142+

for (const part of [

143+

candidate.sourceKind,

144+

candidate.accountId ?? "",

145+

candidate.access,

146+

candidate.refresh,

147+

]) {

148+

hash.update(part);

149+

hash.update("\0");

150+

}

151+

return hash.digest("hex");

152+

}

153+154+

function readProviderTokens(

155+

auth: Record<string, unknown>,

156+

sourcePath: string,

157+

): HermesCodexAuthCandidate | undefined {

135158

const providers = isRecord(auth.providers) ? auth.providers : {};

136159

const provider = isRecord(providers[OPENAI_CODEX_PROVIDER_ID])

137160

? providers[OPENAI_CODEX_PROVIDER_ID]

@@ -145,12 +168,17 @@ function readProviderTokens(auth: Record<string, unknown>): HermesCodexAuthCandi

145168

return {

146169

access,

147170

refresh,

171+

sourceKind: "hermes-auth-json",

148172

sourceLabel: "Hermes active OpenAI Codex provider",

173+

sourcePath,

149174

updatedAt: readTimestamp(provider?.last_refresh),

150175

};

151176

}

152177153-

function readPoolTokens(auth: Record<string, unknown>): HermesCodexAuthCandidate[] {

178+

function readPoolTokens(

179+

auth: Record<string, unknown>,

180+

sourcePath: string,

181+

): HermesCodexAuthCandidate[] {

154182

const pool = isRecord(auth.credential_pool) ? auth.credential_pool : {};

155183

const entries = Array.isArray(pool[OPENAI_CODEX_PROVIDER_ID])

156184

? pool[OPENAI_CODEX_PROVIDER_ID]

@@ -169,7 +197,9 @@ function readPoolTokens(auth: Record<string, unknown>): HermesCodexAuthCandidate

169197

candidates.push({

170198

access,

171199

refresh,

200+

sourceKind: "hermes-auth-json",

172201

sourceLabel: label,

202+

sourcePath,

173203

updatedAt: readTimestamp(entry.last_refresh) ?? readTimestamp(entry.last_status_at),

174204

});

175205

}

@@ -180,7 +210,7 @@ async function readHermesCodexAuthCandidates(

180210

authPath: string | undefined,

181211

): Promise<HermesCodexAuthCandidate[]> {

182212

const raw = await readText(authPath);

183-

if (!raw) {

213+

if (!raw || !authPath) {

184214

return [];

185215

}

186216

let parsed: unknown;

@@ -192,9 +222,49 @@ async function readHermesCodexAuthCandidates(

192222

if (!isRecord(parsed)) {

193223

return [];

194224

}

195-

return [readProviderTokens(parsed), ...readPoolTokens(parsed)]

225+

const candidates = [readProviderTokens(parsed, authPath), ...readPoolTokens(parsed, authPath)]

196226

.filter((candidate): candidate is HermesCodexAuthCandidate => candidate !== undefined)

197227

.toSorted((left, right) => (right.updatedAt ?? 0) - (left.updatedAt ?? 0));

228+

candidates.forEach((candidate, index) => {

229+

candidate.sourceCredentialIndex = index;

230+

});

231+

return candidates;

232+

}

233+234+

async function readOpenCodeOpenAICandidates(

235+

authPath: string | undefined,

236+

): Promise<HermesCodexAuthCandidate[]> {

237+

const raw = await readText(authPath);

238+

if (!raw || !authPath) {

239+

return [];

240+

}

241+

let parsed: unknown;

242+

try {

243+

parsed = JSON.parse(raw);

244+

} catch {

245+

return [];

246+

}

247+

if (!isRecord(parsed)) {

248+

return [];

249+

}

250+

const openai = isRecord(parsed.openai) ? parsed.openai : undefined;

251+

const access = readString(openai?.access);

252+

const accountId = readString(openai?.accountId);

253+

const refresh = readString(openai?.refresh);

254+

if (!access || !refresh) {

255+

return [];

256+

}

257+

return [

258+

{

259+

access,

260+

...(accountId ? { accountId } : {}),

261+

refresh,

262+

sourceKind: "opencode-auth-json",

263+

sourceCredentialIndex: 0,

264+

sourceLabel: "OpenCode OpenAI OAuth credential",

265+

sourcePath: authPath,

266+

},

267+

];

198268

}

199269200270

function credentialExtra(identity: CodexIdentity): Record<string, unknown> | undefined {

@@ -219,7 +289,7 @@ function buildAuthResult(

219289

candidate: HermesCodexAuthCandidate,

220290

fallbackProfileName = "hermes-import",

221291

): ProviderAuthResult {

222-

const identity = resolveCodexIdentity(candidate.access);

292+

const identity = resolveCodexIdentity(candidate.access, candidate.accountId);

223293

return buildOauthProviderAuthResult({

224294

providerId: OPENAI_CODEX_PROVIDER_ID,

225295

defaultModel: OPENAI_CODEX_DEFAULT_MODEL,

@@ -243,10 +313,13 @@ function authProfileDedupeKey(profile: HermesCodexAuthProfile): string {

243313

return `${profile.credential.provider}:profile:${profile.sourceProfileId}`;

244314

}

245315246-

async function readHermesCodexAuthProfiles(

247-

authPath: string | undefined,

316+

async function readCodexAuthProfilesFromSource(

317+

source: HermesSource,

248318

): Promise<HermesCodexAuthProfile[]> {

249-

const candidates = await readHermesCodexAuthCandidates(authPath);

319+

const candidates = [

320+

...(await readHermesCodexAuthCandidates(source.authPath)),

321+

...(await readOpenCodeOpenAICandidates(source.opencodeAuthPath)),

322+

].toSorted((left, right) => (right.updatedAt ?? 0) - (left.updatedAt ?? 0));

250323

const profiles: HermesCodexAuthProfile[] = [];

251324

const seen = new Set<string>();

252325

for (const [index, candidate] of candidates.entries()) {

@@ -273,6 +346,24 @@ async function readHermesCodexAuthProfiles(

273346

return profiles;

274347

}

275348349+

async function readCodexAuthProfilesFromPath(params: {

350+

sourcePath: string | undefined;

351+

sourceKind: unknown;

352+

}): Promise<HermesCodexAuthProfile[]> {

353+

if (params.sourceKind === "opencode-auth-json") {

354+

return await readCodexAuthProfilesFromSource({

355+

root: "",

356+

archivePaths: [],

357+

...(params.sourcePath ? { opencodeAuthPath: params.sourcePath } : {}),

358+

});

359+

}

360+

return await readCodexAuthProfilesFromSource({

361+

root: "",

362+

archivePaths: [],

363+

...(params.sourcePath ? { authPath: params.sourcePath } : {}),

364+

});

365+

}

366+276367

function findMatchingProfile(

277368

store: AuthProfileStore,

278369

credential: OAuthCredential,

@@ -305,12 +396,47 @@ function oauthAuthProfileConfig(

305396

};

306397

}

307398399+

function matchesSourceCredentialFingerprint(

400+

profile: HermesCodexAuthProfile,

401+

fingerprint: string,

402+

): boolean {

403+

return sourceCredentialFingerprint(profile.candidate) === fingerprint;

404+

}

405+406+

function findPlannedAuthProfile(params: {

407+

profiles: HermesCodexAuthProfile[];

408+

sourceProfileId: string;

409+

sourceCredentialIndex?: number;

410+

sourceCredentialFingerprint?: string;

411+

}): HermesCodexAuthProfile | undefined {

412+

const bySourceProfileId = params.profiles.find(

413+

(entry) => entry.sourceProfileId === params.sourceProfileId,

414+

);

415+

const fingerprint = params.sourceCredentialFingerprint;

416+

if (!fingerprint) {

417+

return bySourceProfileId;

418+

}

419+

if (bySourceProfileId && matchesSourceCredentialFingerprint(bySourceProfileId, fingerprint)) {

420+

return bySourceProfileId;

421+

}

422+

const byIndex =

423+

params.sourceCredentialIndex === undefined

424+

? undefined

425+

: params.profiles.find(

426+

(entry) => entry.candidate.sourceCredentialIndex === params.sourceCredentialIndex,

427+

);

428+

if (byIndex && matchesSourceCredentialFingerprint(byIndex, fingerprint)) {

429+

return byIndex;

430+

}

431+

return params.profiles.find((entry) => matchesSourceCredentialFingerprint(entry, fingerprint));

432+

}

433+308434

export async function buildAuthItems(params: {

309435

ctx: MigrationProviderContext;

310436

source: HermesSource;

311437

targets: PlannedTargets;

312438

}): Promise<MigrationItem[]> {

313-

const profiles = await readHermesCodexAuthProfiles(params.source.authPath);

439+

const profiles = await readCodexAuthProfilesFromSource(params.source);

314440

if (profiles.length === 0) {

315441

return [];

316442

}

@@ -335,7 +461,7 @@ export async function buildAuthItems(params: {

335461

id: itemId,

336462

kind: "auth",

337463

action: skipped ? "skip" : "create",

338-

source: params.source.authPath,

464+

source: profile.candidate.sourcePath,

339465

target: `${params.targets.agentDir}/auth-profiles.json#${profileId}`,

340466

status: skipped ? "skipped" : conflict ? "conflict" : "planned",

341467

sensitive: true,

@@ -350,8 +476,12 @@ export async function buildAuthItems(params: {

350476

details: {

351477

provider: OPENAI_CODEX_PROVIDER_ID,

352478

profileId,

479+

...(typeof profile.candidate.sourceCredentialIndex === "number"

480+

? { sourceCredentialIndex: profile.candidate.sourceCredentialIndex }

481+

: {}),

482+

sourceCredentialFingerprint: sourceCredentialFingerprint(profile.candidate),

353483

sourceProfileId: profile.sourceProfileId,

354-

sourceKind: "hermes-auth-json",

484+

sourceKind: profile.candidate.sourceKind,

355485

sourceLabel: profile.candidate.sourceLabel,

356486

},

357487

});

@@ -370,12 +500,27 @@ export async function applyAuthItem(

370500

const profileId = typeof item.details?.profileId === "string" ? item.details.profileId : "";

371501

const sourceProfileId =

372502

typeof item.details?.sourceProfileId === "string" ? item.details.sourceProfileId : profileId;

503+

const sourceCredentialIndex =

504+

typeof item.details?.sourceCredentialIndex === "number"

505+

? item.details.sourceCredentialIndex

506+

: undefined;

507+

const sourceCredentialFingerprint =

508+

typeof item.details?.sourceCredentialFingerprint === "string"

509+

? item.details.sourceCredentialFingerprint

510+

: undefined;

373511

if (!source || !profileId) {

374512

return markMigrationItemError(item, HERMES_REASON_MISSING_SECRET_METADATA);

375513

}

376-

const profile = (await readHermesCodexAuthProfiles(source)).find(

377-

(entry) => entry.sourceProfileId === sourceProfileId,

378-

);

514+

const profiles = await readCodexAuthProfilesFromPath({

515+

sourcePath: source,

516+

sourceKind: item.details?.sourceKind,

517+

});

518+

const profile = findPlannedAuthProfile({

519+

profiles,

520+

sourceProfileId,

521+

...(sourceCredentialIndex === undefined ? {} : { sourceCredentialIndex }),

522+

...(sourceCredentialFingerprint ? { sourceCredentialFingerprint } : {}),

523+

});

379524

if (!profile) {

380525

return markMigrationItemSkipped(item, HERMES_REASON_SECRET_NO_LONGER_PRESENT);

381526

}