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

推荐订阅源

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

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(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
fix(google-vertex): support production ADC modes (#83971) · openclaw/openclaw@f09b4eb
damianFelixP · 2026-05-25 · via Recent Commits to openclaw:main

@@ -4,21 +4,40 @@ import path from "node:path";

44

import type { Model } from "@earendil-works/pi-ai";

55

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

667-

const { buildGuardedModelFetchMock, guardedFetchMock } = vi.hoisted(() => ({

8-

buildGuardedModelFetchMock: vi.fn(),

9-

guardedFetchMock: vi.fn(),

10-

}));

7+

const {

8+

buildGuardedModelFetchMock,

9+

guardedFetchMock,

10+

googleAuthGetAccessTokenMock,

11+

googleAuthMock,

12+

} = vi.hoisted(() => {

13+

const googleAuthGetAccessTokenMock = vi.fn();

14+

return {

15+

buildGuardedModelFetchMock: vi.fn(),

16+

guardedFetchMock: vi.fn(),

17+

googleAuthGetAccessTokenMock,

18+

googleAuthMock: vi.fn(function GoogleAuthMock() {

19+

return {

20+

getAccessToken: googleAuthGetAccessTokenMock,

21+

};

22+

}),

23+

};

24+

});

11251226

vi.mock("openclaw/plugin-sdk/provider-transport-runtime", async (importOriginal) => ({

1327

...(await importOriginal()),

1428

buildGuardedModelFetch: buildGuardedModelFetchMock,

1529

}));

163031+

vi.mock("google-auth-library", () => ({

32+

GoogleAuth: googleAuthMock,

33+

}));

34+1735

let buildGoogleGenerativeAiParams: typeof import("./transport-stream.js").buildGoogleGenerativeAiParams;

1836

let buildGoogleGemini3FirstResponseRetryParams: typeof import("./transport-stream.js").buildGoogleGemini3FirstResponseRetryParams;

1937

let createGoogleGenerativeAiTransportStreamFn: typeof import("./transport-stream.js").createGoogleGenerativeAiTransportStreamFn;

2038

let createGoogleVertexTransportStreamFn: typeof import("./transport-stream.js").createGoogleVertexTransportStreamFn;

2139

let hasGoogleVertexAuthorizedUserAdcSync: typeof import("./vertex-adc.js").hasGoogleVertexAuthorizedUserAdcSync;

40+

let resolveGoogleVertexAuthorizedUserHeaders: typeof import("./vertex-adc.js").resolveGoogleVertexAuthorizedUserHeaders;

2241

let resetGoogleVertexAuthorizedUserTokenCacheForTest: typeof import("./vertex-adc.js").resetGoogleVertexAuthorizedUserTokenCacheForTest;

23422443

const MODEL_PROVIDER_REQUEST_TRANSPORT_SYMBOL = Symbol.for(

@@ -254,13 +273,18 @@ describe("google transport stream", () => {

254273

createGoogleGenerativeAiTransportStreamFn,

255274

createGoogleVertexTransportStreamFn,

256275

} = await import("./transport-stream.js"));

257-

({ hasGoogleVertexAuthorizedUserAdcSync, resetGoogleVertexAuthorizedUserTokenCacheForTest } =

258-

await import("./vertex-adc.js"));

276+

({

277+

hasGoogleVertexAuthorizedUserAdcSync,

278+

resolveGoogleVertexAuthorizedUserHeaders,

279+

resetGoogleVertexAuthorizedUserTokenCacheForTest,

280+

} = await import("./vertex-adc.js"));

259281

});

260282261283

beforeEach(() => {

262284

buildGuardedModelFetchMock.mockReset();

263285

guardedFetchMock.mockReset();

286+

googleAuthGetAccessTokenMock.mockReset();

287+

googleAuthMock.mockClear();

264288

buildGuardedModelFetchMock.mockReturnValue(guardedFetchMock);

265289

resetGoogleVertexAuthorizedUserTokenCacheForTest();

266290

});

@@ -271,6 +295,7 @@ describe("google transport stream", () => {

271295272296

afterAll(() => {

273297

vi.doUnmock("openclaw/plugin-sdk/provider-transport-runtime");

298+

vi.doUnmock("google-auth-library");

274299

vi.resetModules();

275300

});

276301

@@ -695,6 +720,89 @@ describe("google transport stream", () => {

695720

});

696721

});

697722723+

it("detects supported Vertex ADC sources synchronously", async () => {

724+

const tempDir = await mkdtemp(path.join(os.tmpdir(), "openclaw-google-vertex-adc-detect-"));

725+

for (const type of ["authorized_user", "external_account", "service_account"]) {

726+

const credentialsPath = path.join(tempDir, `${type}.json`);

727+

await writeFile(credentialsPath, JSON.stringify({ type }), "utf8");

728+729+

expect(

730+

hasGoogleVertexAuthorizedUserAdcSync({

731+

GOOGLE_APPLICATION_CREDENTIALS: credentialsPath,

732+

}),

733+

).toBe(true);

734+

}

735+736+

expect(

737+

hasGoogleVertexAuthorizedUserAdcSync({

738+

HOME: path.join(tempDir, "empty-home"),

739+

KUBERNETES_SERVICE_HOST: "10.0.0.1",

740+

}),

741+

).toBe(false);

742+

});

743+744+

it("resolves non-file Vertex ADC through google-auth-library without OAuth refresh fetch", async () => {

745+

const tempDir = await mkdtemp(path.join(os.tmpdir(), "openclaw-google-vertex-authlib-"));

746+

vi.stubEnv("GOOGLE_APPLICATION_CREDENTIALS", "");

747+

vi.stubEnv("HOME", path.join(tempDir, "home"));

748+

vi.stubEnv("APPDATA", "");

749+

googleAuthGetAccessTokenMock.mockResolvedValueOnce("ya29.google-auth-token");

750+

const tokenFetchMock = vi.fn();

751+752+

await expect(resolveGoogleVertexAuthorizedUserHeaders(tokenFetchMock)).resolves.toEqual({

753+

Authorization: "Bearer ya29.google-auth-token",

754+

});

755+756+

expect(googleAuthMock).toHaveBeenCalledWith({

757+

scopes: ["https://www.googleapis.com/auth/cloud-platform"],

758+

});

759+

expect(googleAuthGetAccessTokenMock).toHaveBeenCalledTimes(1);

760+

expect(tokenFetchMock).not.toHaveBeenCalled();

761+

});

762+763+

it("uses google-auth-library bearer auth for Google Vertex credential marker requests", async () => {

764+

const tempDir = await mkdtemp(path.join(os.tmpdir(), "openclaw-google-vertex-authlib-stream-"));

765+

vi.stubEnv("GOOGLE_APPLICATION_CREDENTIALS", "");

766+

vi.stubEnv("HOME", path.join(tempDir, "home"));

767+

vi.stubEnv("APPDATA", "");

768+

vi.stubEnv("GOOGLE_CLOUD_PROJECT", "vertex-project");

769+

vi.stubEnv("GOOGLE_CLOUD_LOCATION", "us-central1");

770+

googleAuthGetAccessTokenMock.mockResolvedValueOnce("ya29.transport-token");

771+

const tokenFetchMock = vi.fn();

772+

guardedFetchMock.mockResolvedValueOnce(

773+

buildSseResponse([

774+

{

775+

candidates: [{ content: { parts: [{ text: "ok" }] }, finishReason: "STOP" }],

776+

},

777+

]),

778+

);

779+780+

const streamFn = createGoogleVertexTransportStreamFn();

781+

const stream = await Promise.resolve(

782+

streamFn(

783+

buildGoogleVertexModel(),

784+

{

785+

messages: [{ role: "user", content: "hello", timestamp: 0 }],

786+

} as Parameters<typeof streamFn>[1],

787+

{

788+

apiKey: "gcp-vertex-credentials",

789+

fetch: tokenFetchMock,

790+

} as Parameters<typeof streamFn>[2],

791+

),

792+

);

793+

await stream.result();

794+795+

expect(tokenFetchMock).not.toHaveBeenCalled();

796+

const guardedCall = requireMockCall(guardedFetchMock, 0, "guarded fetch");

797+

const guardedInit = requireRequestInit(guardedCall, "guarded fetch");

798+

expectHeaders(guardedInit, {

799+

Authorization: "Bearer ya29.transport-token",

800+

"Content-Type": "application/json",

801+

accept: "text/event-stream",

802+

});

803+

expect(new Headers(guardedInit.headers).has("x-goog-api-key")).toBe(false);

804+

});

805+698806

it("refreshes authorized_user ADC before Google Vertex requests", async () => {

699807

const tempDir = await mkdtemp(path.join(os.tmpdir(), "openclaw-google-vertex-adc-"));

700808

const credentialsPath = path.join(tempDir, "application_default_credentials.json");