@google/gemini-cli-core 0.1.10-dev.15 → 0.1.11-nightly.250713.4442e893
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +16 -2
- package/dist/google-gemini-cli-core-0.1.11.tgz +0 -0
- package/dist/src/code_assist/codeAssist.d.ts +2 -1
- package/dist/src/code_assist/codeAssist.js +4 -3
- package/dist/src/code_assist/codeAssist.js.map +1 -1
- package/dist/src/code_assist/converter.js.map +1 -1
- package/dist/src/code_assist/oauth2.d.ts +3 -8
- package/dist/src/code_assist/oauth2.js +117 -75
- package/dist/src/code_assist/oauth2.js.map +1 -1
- package/dist/src/code_assist/oauth2.test.js +134 -34
- package/dist/src/code_assist/oauth2.test.js.map +1 -1
- package/dist/src/code_assist/server.d.ts +4 -0
- package/dist/src/code_assist/server.js +68 -2
- package/dist/src/code_assist/server.js.map +1 -1
- package/dist/src/code_assist/setup.js +1 -1
- package/dist/src/code_assist/setup.js.map +1 -1
- package/dist/src/code_assist/setup.test.d.ts +6 -0
- package/dist/src/code_assist/setup.test.js +62 -0
- package/dist/src/code_assist/setup.test.js.map +1 -0
- package/dist/src/config/config.d.ts +31 -3
- package/dist/src/config/config.js +102 -56
- package/dist/src/config/config.js.map +1 -1
- package/dist/src/config/config.test.js +48 -23
- package/dist/src/config/config.test.js.map +1 -1
- package/dist/src/core/client.d.ts +23 -6
- package/dist/src/core/client.js +88 -24
- package/dist/src/core/client.js.map +1 -1
- package/dist/src/core/client.test.js +131 -12
- package/dist/src/core/client.test.js.map +1 -1
- package/dist/src/core/contentGenerator.d.ts +7 -5
- package/dist/src/core/contentGenerator.js +16 -15
- package/dist/src/core/contentGenerator.js.map +1 -1
- package/dist/src/core/contentGenerator.test.js +52 -5
- package/dist/src/core/contentGenerator.test.js.map +1 -1
- package/dist/src/core/coreToolScheduler.d.ts +9 -2
- package/dist/src/core/coreToolScheduler.js +52 -5
- package/dist/src/core/coreToolScheduler.js.map +1 -1
- package/dist/src/core/coreToolScheduler.test.js +80 -0
- package/dist/src/core/coreToolScheduler.test.js.map +1 -1
- package/dist/src/core/geminiChat.d.ts +2 -2
- package/dist/src/core/geminiChat.js +52 -32
- package/dist/src/core/geminiChat.js.map +1 -1
- package/dist/src/core/geminiChat.test.js +4 -2
- package/dist/src/core/geminiChat.test.js.map +1 -1
- package/dist/src/core/modelCheck.js +1 -1
- package/dist/src/core/modelCheck.js.map +1 -1
- package/dist/src/core/nonInteractiveToolExecutor.js +11 -2
- package/dist/src/core/nonInteractiveToolExecutor.js.map +1 -1
- package/dist/src/core/nonInteractiveToolExecutor.test.js +5 -0
- package/dist/src/core/nonInteractiveToolExecutor.test.js.map +1 -1
- package/dist/src/core/prompts.js +9 -8
- package/dist/src/core/prompts.js.map +1 -1
- package/dist/src/core/turn.d.ts +9 -3
- package/dist/src/core/turn.js +6 -2
- package/dist/src/core/turn.js.map +1 -1
- package/dist/src/core/turn.test.js +2 -2
- package/dist/src/core/turn.test.js.map +1 -1
- package/dist/src/index.d.ts +1 -0
- package/dist/src/index.js +1 -0
- package/dist/src/index.js.map +1 -1
- package/dist/src/services/fileDiscoveryService.d.ts +4 -0
- package/dist/src/services/fileDiscoveryService.js +13 -0
- package/dist/src/services/fileDiscoveryService.js.map +1 -1
- package/dist/src/telemetry/clearcut-logger/clearcut-logger.d.ts +3 -2
- package/dist/src/telemetry/clearcut-logger/clearcut-logger.js +70 -28
- package/dist/src/telemetry/clearcut-logger/clearcut-logger.js.map +1 -1
- package/dist/src/telemetry/clearcut-logger/event-metadata-key.d.ts +4 -1
- package/dist/src/telemetry/clearcut-logger/event-metadata-key.js +9 -0
- package/dist/src/telemetry/clearcut-logger/event-metadata-key.js.map +1 -1
- package/dist/src/telemetry/constants.d.ts +1 -0
- package/dist/src/telemetry/constants.js +1 -0
- package/dist/src/telemetry/constants.js.map +1 -1
- package/dist/src/telemetry/index.d.ts +2 -2
- package/dist/src/telemetry/index.js +2 -2
- package/dist/src/telemetry/index.js.map +1 -1
- package/dist/src/telemetry/loggers.d.ts +2 -1
- package/dist/src/telemetry/loggers.js +18 -1
- package/dist/src/telemetry/loggers.js.map +1 -1
- package/dist/src/telemetry/loggers.test.js +42 -9
- package/dist/src/telemetry/loggers.test.js.map +1 -1
- package/dist/src/telemetry/types.d.ts +19 -5
- package/dist/src/telemetry/types.js +30 -4
- package/dist/src/telemetry/types.js.map +1 -1
- package/dist/src/telemetry/uiTelemetry.test.js +1 -0
- package/dist/src/telemetry/uiTelemetry.test.js.map +1 -1
- package/dist/src/tools/edit.d.ts +1 -2
- package/dist/src/tools/edit.js +11 -12
- package/dist/src/tools/edit.js.map +1 -1
- package/dist/src/tools/edit.test.js +7 -9
- package/dist/src/tools/edit.test.js.map +1 -1
- package/dist/src/tools/glob.js +9 -8
- package/dist/src/tools/glob.js.map +1 -1
- package/dist/src/tools/glob.test.js +3 -3
- package/dist/src/tools/glob.test.js.map +1 -1
- package/dist/src/tools/grep.js +8 -7
- package/dist/src/tools/grep.js.map +1 -1
- package/dist/src/tools/grep.test.js +4 -4
- package/dist/src/tools/grep.test.js.map +1 -1
- package/dist/src/tools/ls.js +9 -8
- package/dist/src/tools/ls.js.map +1 -1
- package/dist/src/tools/mcp-client.d.ts +0 -9
- package/dist/src/tools/mcp-client.js +29 -48
- package/dist/src/tools/mcp-client.js.map +1 -1
- package/dist/src/tools/mcp-client.test.js +131 -5
- package/dist/src/tools/mcp-client.test.js.map +1 -1
- package/dist/src/tools/mcp-tool.d.ts +3 -3
- package/dist/src/tools/mcp-tool.js.map +1 -1
- package/dist/src/tools/memoryTool.js +3 -2
- package/dist/src/tools/memoryTool.js.map +1 -1
- package/dist/src/tools/read-file.js +9 -10
- package/dist/src/tools/read-file.js.map +1 -1
- package/dist/src/tools/read-file.test.js +4 -4
- package/dist/src/tools/read-file.test.js.map +1 -1
- package/dist/src/tools/read-many-files.js +25 -39
- package/dist/src/tools/read-many-files.js.map +1 -1
- package/dist/src/tools/read-many-files.test.js +25 -14
- package/dist/src/tools/read-many-files.test.js.map +1 -1
- package/dist/src/tools/shell.d.ts +5 -2
- package/dist/src/tools/shell.js +41 -18
- package/dist/src/tools/shell.js.map +1 -1
- package/dist/src/tools/shell.test.js +85 -65
- package/dist/src/tools/shell.test.js.map +1 -1
- package/dist/src/tools/tool-registry.d.ts +16 -2
- package/dist/src/tools/tool-registry.js +153 -20
- package/dist/src/tools/tool-registry.js.map +1 -1
- package/dist/src/tools/tool-registry.test.js +186 -75
- package/dist/src/tools/tool-registry.test.js.map +1 -1
- package/dist/src/tools/tools.d.ts +25 -4
- package/dist/src/tools/tools.js +8 -1
- package/dist/src/tools/tools.js.map +1 -1
- package/dist/src/tools/web-fetch.js +6 -5
- package/dist/src/tools/web-fetch.js.map +1 -1
- package/dist/src/tools/web-search.d.ts +1 -1
- package/dist/src/tools/web-search.js +7 -6
- package/dist/src/tools/web-search.js.map +1 -1
- package/dist/src/tools/write-file.d.ts +0 -1
- package/dist/src/tools/write-file.js +10 -11
- package/dist/src/tools/write-file.js.map +1 -1
- package/dist/src/tools/write-file.test.js +4 -4
- package/dist/src/tools/write-file.test.js.map +1 -1
- package/dist/src/utils/editCorrector.d.ts +1 -1
- package/dist/src/utils/editCorrector.js +106 -2
- package/dist/src/utils/editCorrector.js.map +1 -1
- package/dist/src/utils/editCorrector.test.js +69 -19
- package/dist/src/utils/editCorrector.test.js.map +1 -1
- package/dist/src/utils/editor.d.ts +1 -1
- package/dist/src/utils/editor.js +1 -1
- package/dist/src/utils/fileUtils.d.ts +1 -1
- package/dist/src/utils/fileUtils.js +17 -0
- package/dist/src/utils/fileUtils.js.map +1 -1
- package/dist/src/utils/fileUtils.test.js +17 -0
- package/dist/src/utils/fileUtils.test.js.map +1 -1
- package/dist/src/utils/flashFallback.integration.test.js +1 -1
- package/dist/src/utils/flashFallback.integration.test.js.map +1 -1
- package/dist/src/utils/messageInspectors.d.ts +1 -0
- package/dist/src/utils/messageInspectors.js +5 -0
- package/dist/src/utils/messageInspectors.js.map +1 -1
- package/dist/src/utils/nextSpeakerChecker.js +1 -1
- package/dist/src/utils/nextSpeakerChecker.js.map +1 -1
- package/dist/src/utils/paths.d.ts +1 -0
- package/dist/src/utils/paths.js +1 -0
- package/dist/src/utils/paths.js.map +1 -1
- package/dist/src/utils/quotaErrorDetection.d.ts +22 -0
- package/dist/src/utils/quotaErrorDetection.js +74 -0
- package/dist/src/utils/quotaErrorDetection.js.map +1 -0
- package/dist/src/utils/retry.d.ts +1 -1
- package/dist/src/utils/retry.js +58 -2
- package/dist/src/utils/retry.js.map +1 -1
- package/dist/src/utils/retry.test.js +1 -1
- package/dist/src/utils/retry.test.js.map +1 -1
- package/dist/src/utils/schemaValidator.d.ts +10 -5
- package/dist/src/utils/schemaValidator.js +44 -32
- package/dist/src/utils/schemaValidator.js.map +1 -1
- package/dist/src/utils/summarizer.d.ts +25 -0
- package/dist/src/utils/summarizer.js +79 -0
- package/dist/src/utils/summarizer.js.map +1 -0
- package/dist/src/utils/summarizer.test.d.ts +6 -0
- package/dist/src/utils/summarizer.test.js +131 -0
- package/dist/src/utils/summarizer.test.js.map +1 -0
- package/dist/src/utils/user_account.d.ts +9 -0
- package/dist/src/utils/user_account.js +99 -0
- package/dist/src/utils/user_account.js.map +1 -0
- package/dist/src/utils/user_account.test.d.ts +6 -0
- package/dist/src/utils/user_account.test.js +153 -0
- package/dist/src/utils/user_account.test.js.map +1 -0
- package/dist/src/utils/user_id.d.ts +0 -6
- package/dist/src/utils/user_id.js +0 -21
- package/dist/src/utils/user_id.js.map +1 -1
- package/dist/src/utils/user_id.test.js +1 -22
- package/dist/src/utils/user_id.test.js.map +1 -1
- package/dist/tsconfig.tsbuildinfo +1 -1
- package/package.json +7 -6
package/README.md
CHANGED
|
@@ -19,7 +19,7 @@ With the Gemini CLI you can:
|
|
|
19
19
|
|
|
20
20
|
## Quickstart
|
|
21
21
|
|
|
22
|
-
1. **Prerequisites:** Ensure you have [Node.js version
|
|
22
|
+
1. **Prerequisites:** Ensure you have [Node.js version 20](https://nodejs.org/en/download) or higher installed.
|
|
23
23
|
2. **Run the CLI:** Execute the following command in your terminal:
|
|
24
24
|
|
|
25
25
|
```bash
|
|
@@ -51,6 +51,20 @@ The Gemini API provides a free tier with [100 requests per day](https://ai.googl
|
|
|
51
51
|
|
|
52
52
|
3. (Optionally) Upgrade your Gemini API project to a paid plan on the API key page (will automatically unlock [Tier 1 rate limits](https://ai.google.dev/gemini-api/docs/rate-limits#tier-1))
|
|
53
53
|
|
|
54
|
+
### Use a Vertex AI API key:
|
|
55
|
+
|
|
56
|
+
The Vertex AI API provides a [free tier](https://cloud.google.com/vertex-ai/generative-ai/docs/start/express-mode/overview) using express mode for Gemini 2.5 Pro, control over which model you use, and access to higher rate limits with a billing account:
|
|
57
|
+
|
|
58
|
+
1. Generate a key from [Google Cloud](https://cloud.google.com/vertex-ai/generative-ai/docs/start/api-keys).
|
|
59
|
+
2. Set it as an environment variable in your terminal. Replace `YOUR_API_KEY` with your generated key and set GOOGLE_GENAI_USE_VERTEXAI to true
|
|
60
|
+
|
|
61
|
+
```bash
|
|
62
|
+
export GOOGLE_API_KEY="YOUR_API_KEY"
|
|
63
|
+
export GOOGLE_GENAI_USE_VERTEXAI=true
|
|
64
|
+
```
|
|
65
|
+
|
|
66
|
+
3. (Optionally) Add a billing account on your project to get access to [higher usage limits](https://cloud.google.com/vertex-ai/generative-ai/docs/quotas)
|
|
67
|
+
|
|
54
68
|
For other authentication methods, including Google Workspace accounts, see the [authentication](./docs/cli/authentication.md) guide.
|
|
55
69
|
|
|
56
70
|
## Examples
|
|
@@ -130,7 +144,7 @@ Use MCP servers to integrate your local system tools with your enterprise collab
|
|
|
130
144
|
```
|
|
131
145
|
|
|
132
146
|
```text
|
|
133
|
-
>
|
|
147
|
+
> Organize my PDF invoices by month of expenditure.
|
|
134
148
|
```
|
|
135
149
|
|
|
136
150
|
### Uninstall
|
|
Binary file
|
|
@@ -5,4 +5,5 @@
|
|
|
5
5
|
*/
|
|
6
6
|
import { AuthType, ContentGenerator } from '../core/contentGenerator.js';
|
|
7
7
|
import { HttpOptions } from './server.js';
|
|
8
|
-
|
|
8
|
+
import { Config } from '../config/config.js';
|
|
9
|
+
export declare function createCodeAssistContentGenerator(httpOptions: HttpOptions, authType: AuthType, config: Config, sessionId?: string): Promise<ContentGenerator>;
|
|
@@ -7,9 +7,10 @@ import { AuthType } from '../core/contentGenerator.js';
|
|
|
7
7
|
import { getOauthClient } from './oauth2.js';
|
|
8
8
|
import { setupUser } from './setup.js';
|
|
9
9
|
import { CodeAssistServer } from './server.js';
|
|
10
|
-
export async function createCodeAssistContentGenerator(httpOptions, authType, sessionId) {
|
|
11
|
-
if (authType === AuthType.LOGIN_WITH_GOOGLE
|
|
12
|
-
|
|
10
|
+
export async function createCodeAssistContentGenerator(httpOptions, authType, config, sessionId) {
|
|
11
|
+
if (authType === AuthType.LOGIN_WITH_GOOGLE ||
|
|
12
|
+
authType === AuthType.CLOUD_SHELL) {
|
|
13
|
+
const authClient = await getOauthClient(authType, config);
|
|
13
14
|
const projectId = await setupUser(authClient);
|
|
14
15
|
return new CodeAssistServer(authClient, projectId, httpOptions, sessionId);
|
|
15
16
|
}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"codeAssist.js","sourceRoot":"","sources":["../../../src/code_assist/codeAssist.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAEH,OAAO,EAAE,QAAQ,EAAoB,MAAM,6BAA6B,CAAC;AACzE,OAAO,EAAE,cAAc,EAAE,MAAM,aAAa,CAAC;AAC7C,OAAO,EAAE,SAAS,EAAE,MAAM,YAAY,CAAC;AACvC,OAAO,EAAE,gBAAgB,EAAe,MAAM,aAAa,CAAC;
|
|
1
|
+
{"version":3,"file":"codeAssist.js","sourceRoot":"","sources":["../../../src/code_assist/codeAssist.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAEH,OAAO,EAAE,QAAQ,EAAoB,MAAM,6BAA6B,CAAC;AACzE,OAAO,EAAE,cAAc,EAAE,MAAM,aAAa,CAAC;AAC7C,OAAO,EAAE,SAAS,EAAE,MAAM,YAAY,CAAC;AACvC,OAAO,EAAE,gBAAgB,EAAe,MAAM,aAAa,CAAC;AAG5D,MAAM,CAAC,KAAK,UAAU,gCAAgC,CACpD,WAAwB,EACxB,QAAkB,EAClB,MAAc,EACd,SAAkB;IAElB,IACE,QAAQ,KAAK,QAAQ,CAAC,iBAAiB;QACvC,QAAQ,KAAK,QAAQ,CAAC,WAAW,EACjC,CAAC;QACD,MAAM,UAAU,GAAG,MAAM,cAAc,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAC;QAC1D,MAAM,SAAS,GAAG,MAAM,SAAS,CAAC,UAAU,CAAC,CAAC;QAC9C,OAAO,IAAI,gBAAgB,CAAC,UAAU,EAAE,SAAS,EAAE,WAAW,EAAE,SAAS,CAAC,CAAC;IAC7E,CAAC;IAED,MAAM,IAAI,KAAK,CAAC,yBAAyB,QAAQ,EAAE,CAAC,CAAC;AACvD,CAAC"}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"converter.js","sourceRoot":"","sources":["../../../src/code_assist/converter.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAEH,OAAO,EAQL,uBAAuB,GAexB,MAAM,eAAe,CAAC;
|
|
1
|
+
{"version":3,"file":"converter.js","sourceRoot":"","sources":["../../../src/code_assist/converter.ts"],"names":[],"mappings":"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"}
|
|
@@ -4,6 +4,8 @@
|
|
|
4
4
|
* SPDX-License-Identifier: Apache-2.0
|
|
5
5
|
*/
|
|
6
6
|
import { OAuth2Client } from 'google-auth-library';
|
|
7
|
+
import { Config } from '../config/config.js';
|
|
8
|
+
import { AuthType } from '../core/contentGenerator.js';
|
|
7
9
|
/**
|
|
8
10
|
* An Authentication URL for updating the credentials of a Oauth2Client
|
|
9
11
|
* as well as a promise that will resolve when the credentials have
|
|
@@ -13,13 +15,6 @@ export interface OauthWebLogin {
|
|
|
13
15
|
authUrl: string;
|
|
14
16
|
loginCompletePromise: Promise<void>;
|
|
15
17
|
}
|
|
16
|
-
export declare function getOauthClient(): Promise<OAuth2Client>;
|
|
18
|
+
export declare function getOauthClient(authType: AuthType, config: Config): Promise<OAuth2Client>;
|
|
17
19
|
export declare function getAvailablePort(): Promise<number>;
|
|
18
|
-
export declare function getCachedGoogleAccountId(): string | null;
|
|
19
20
|
export declare function clearCachedCredentialFile(): Promise<void>;
|
|
20
|
-
/**
|
|
21
|
-
* Retrieves the authenticated user's Google Account ID from Google's UserInfo API.
|
|
22
|
-
* @param client - The authenticated OAuth2Client
|
|
23
|
-
* @returns The user's Google Account ID or null if not available
|
|
24
|
-
*/
|
|
25
|
-
export declare function getRawGoogleAccountId(client: OAuth2Client): Promise<string | null>;
|
|
@@ -3,15 +3,19 @@
|
|
|
3
3
|
* Copyright 2025 Google LLC
|
|
4
4
|
* SPDX-License-Identifier: Apache-2.0
|
|
5
5
|
*/
|
|
6
|
-
import { OAuth2Client } from 'google-auth-library';
|
|
6
|
+
import { OAuth2Client, Compute, CodeChallengeMethod, } from 'google-auth-library';
|
|
7
7
|
import * as http from 'http';
|
|
8
8
|
import url from 'url';
|
|
9
9
|
import crypto from 'crypto';
|
|
10
10
|
import * as net from 'net';
|
|
11
11
|
import open from 'open';
|
|
12
12
|
import path from 'node:path';
|
|
13
|
-
import { promises as fs
|
|
13
|
+
import { promises as fs } from 'node:fs';
|
|
14
14
|
import * as os from 'os';
|
|
15
|
+
import { getErrorMessage } from '../utils/errors.js';
|
|
16
|
+
import { cacheGoogleAccount, getCachedGoogleAccount, clearCachedGoogleAccount, } from '../utils/user_account.js';
|
|
17
|
+
import { AuthType } from '../core/contentGenerator.js';
|
|
18
|
+
import readline from 'node:readline';
|
|
15
19
|
// OAuth Client ID used to initiate OAuth2Client class.
|
|
16
20
|
const OAUTH_CLIENT_ID = '681255809395-oo8ft2oprdrnp9e3aqf6av3hmdib135j.apps.googleusercontent.com';
|
|
17
21
|
// OAuth Secret value used to initiate OAuth2Client class.
|
|
@@ -32,8 +36,7 @@ const SIGN_IN_SUCCESS_URL = 'https://developers.google.com/gemini-code-assist/au
|
|
|
32
36
|
const SIGN_IN_FAILURE_URL = 'https://developers.google.com/gemini-code-assist/auth_failure_gemini';
|
|
33
37
|
const GEMINI_DIR = '.gemini';
|
|
34
38
|
const CREDENTIAL_FILENAME = 'oauth_creds.json';
|
|
35
|
-
|
|
36
|
-
export async function getOauthClient() {
|
|
39
|
+
export async function getOauthClient(authType, config) {
|
|
37
40
|
const client = new OAuth2Client({
|
|
38
41
|
clientId: OAUTH_CLIENT_ID,
|
|
39
42
|
clientSecret: OAUTH_CLIENT_SECRET,
|
|
@@ -41,32 +44,108 @@ export async function getOauthClient() {
|
|
|
41
44
|
client.on('tokens', async (tokens) => {
|
|
42
45
|
await cacheCredentials(tokens);
|
|
43
46
|
});
|
|
47
|
+
// If there are cached creds on disk, they always take precedence
|
|
44
48
|
if (await loadCachedCredentials(client)) {
|
|
45
49
|
// Found valid cached credentials.
|
|
46
|
-
// Check if we need to retrieve Google Account ID
|
|
47
|
-
if (!
|
|
50
|
+
// Check if we need to retrieve Google Account ID or Email
|
|
51
|
+
if (!getCachedGoogleAccount()) {
|
|
48
52
|
try {
|
|
49
|
-
|
|
50
|
-
if (googleAccountId) {
|
|
51
|
-
await cacheGoogleAccountId(googleAccountId);
|
|
52
|
-
}
|
|
53
|
+
await fetchAndCacheUserInfo(client);
|
|
53
54
|
}
|
|
54
|
-
catch
|
|
55
|
-
|
|
56
|
-
// Continue with existing auth flow
|
|
55
|
+
catch {
|
|
56
|
+
// Non-fatal, continue with existing auth.
|
|
57
57
|
}
|
|
58
58
|
}
|
|
59
|
+
console.log('Loaded cached credentials.');
|
|
59
60
|
return client;
|
|
60
61
|
}
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
62
|
+
// In Google Cloud Shell, we can use Application Default Credentials (ADC)
|
|
63
|
+
// provided via its metadata server to authenticate non-interactively using
|
|
64
|
+
// the identity of the user logged into Cloud Shell.
|
|
65
|
+
if (authType === AuthType.CLOUD_SHELL) {
|
|
66
|
+
try {
|
|
67
|
+
console.log("Attempting to authenticate via Cloud Shell VM's ADC.");
|
|
68
|
+
const computeClient = new Compute({
|
|
69
|
+
// We can leave this empty, since the metadata server will provide
|
|
70
|
+
// the service account email.
|
|
71
|
+
});
|
|
72
|
+
await computeClient.getAccessToken();
|
|
73
|
+
console.log('Authentication successful.');
|
|
74
|
+
// Do not cache creds in this case; note that Compute client will handle its own refresh
|
|
75
|
+
return computeClient;
|
|
76
|
+
}
|
|
77
|
+
catch (e) {
|
|
78
|
+
throw new Error(`Could not authenticate using Cloud Shell credentials. Please select a different authentication method or ensure you are in a properly configured environment. Error: ${getErrorMessage(e)}`);
|
|
79
|
+
}
|
|
80
|
+
}
|
|
81
|
+
if (config.getNoBrowser()) {
|
|
82
|
+
let success = false;
|
|
83
|
+
const maxRetries = 2;
|
|
84
|
+
for (let i = 0; !success && i < maxRetries; i++) {
|
|
85
|
+
success = await authWithUserCode(client);
|
|
86
|
+
if (!success) {
|
|
87
|
+
console.error('\nFailed to authenticate with user code.', i === maxRetries - 1 ? '' : 'Retrying...\n');
|
|
88
|
+
}
|
|
89
|
+
}
|
|
90
|
+
if (!success) {
|
|
91
|
+
process.exit(1);
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
else {
|
|
95
|
+
const webLogin = await authWithWeb(client);
|
|
96
|
+
// This does basically nothing, as it isn't show to the user.
|
|
97
|
+
console.log(`\n\nCode Assist login required.\n` +
|
|
98
|
+
`Attempting to open authentication page in your browser.\n` +
|
|
99
|
+
`Otherwise navigate to:\n\n${webLogin.authUrl}\n\n`);
|
|
100
|
+
await open(webLogin.authUrl);
|
|
101
|
+
console.log('Waiting for authentication...');
|
|
102
|
+
await webLogin.loginCompletePromise;
|
|
103
|
+
}
|
|
68
104
|
return client;
|
|
69
105
|
}
|
|
106
|
+
async function authWithUserCode(client) {
|
|
107
|
+
const redirectUri = 'https://sdk.cloud.google.com/authcode_cloudcode.html';
|
|
108
|
+
const codeVerifier = await client.generateCodeVerifierAsync();
|
|
109
|
+
const state = crypto.randomBytes(32).toString('hex');
|
|
110
|
+
const authUrl = client.generateAuthUrl({
|
|
111
|
+
redirect_uri: redirectUri,
|
|
112
|
+
access_type: 'offline',
|
|
113
|
+
scope: OAUTH_SCOPE,
|
|
114
|
+
code_challenge_method: CodeChallengeMethod.S256,
|
|
115
|
+
code_challenge: codeVerifier.codeChallenge,
|
|
116
|
+
state,
|
|
117
|
+
});
|
|
118
|
+
console.log('Please visit the following URL to authorize the application:');
|
|
119
|
+
console.log('');
|
|
120
|
+
console.log(authUrl);
|
|
121
|
+
console.log('');
|
|
122
|
+
const code = await new Promise((resolve) => {
|
|
123
|
+
const rl = readline.createInterface({
|
|
124
|
+
input: process.stdin,
|
|
125
|
+
output: process.stdout,
|
|
126
|
+
});
|
|
127
|
+
rl.question('Enter the authorization code: ', (code) => {
|
|
128
|
+
rl.close();
|
|
129
|
+
resolve(code.trim());
|
|
130
|
+
});
|
|
131
|
+
});
|
|
132
|
+
if (!code) {
|
|
133
|
+
console.error('Authorization code is required.');
|
|
134
|
+
return false;
|
|
135
|
+
}
|
|
136
|
+
try {
|
|
137
|
+
const { tokens } = await client.getToken({
|
|
138
|
+
code,
|
|
139
|
+
codeVerifier: codeVerifier.codeVerifier,
|
|
140
|
+
redirect_uri: redirectUri,
|
|
141
|
+
});
|
|
142
|
+
client.setCredentials(tokens);
|
|
143
|
+
}
|
|
144
|
+
catch (_error) {
|
|
145
|
+
return false;
|
|
146
|
+
}
|
|
147
|
+
return true;
|
|
148
|
+
}
|
|
70
149
|
async function authWithWeb(client) {
|
|
71
150
|
const port = await getAvailablePort();
|
|
72
151
|
const redirectUri = `http://localhost:${port}/oauth2callback`;
|
|
@@ -104,10 +183,7 @@ async function authWithWeb(client) {
|
|
|
104
183
|
client.setCredentials(tokens);
|
|
105
184
|
// Retrieve and cache Google Account ID during authentication
|
|
106
185
|
try {
|
|
107
|
-
|
|
108
|
-
if (googleAccountId) {
|
|
109
|
-
await cacheGoogleAccountId(googleAccountId);
|
|
110
|
-
}
|
|
186
|
+
await fetchAndCacheUserInfo(client);
|
|
111
187
|
}
|
|
112
188
|
catch (error) {
|
|
113
189
|
console.error('Failed to retrieve Google Account ID during authentication:', error);
|
|
@@ -183,72 +259,38 @@ async function cacheCredentials(credentials) {
|
|
|
183
259
|
function getCachedCredentialPath() {
|
|
184
260
|
return path.join(os.homedir(), GEMINI_DIR, CREDENTIAL_FILENAME);
|
|
185
261
|
}
|
|
186
|
-
function getGoogleAccountIdCachePath() {
|
|
187
|
-
return path.join(os.homedir(), GEMINI_DIR, GOOGLE_ACCOUNT_ID_FILENAME);
|
|
188
|
-
}
|
|
189
|
-
async function cacheGoogleAccountId(googleAccountId) {
|
|
190
|
-
const filePath = getGoogleAccountIdCachePath();
|
|
191
|
-
await fs.mkdir(path.dirname(filePath), { recursive: true });
|
|
192
|
-
await fs.writeFile(filePath, googleAccountId, 'utf-8');
|
|
193
|
-
}
|
|
194
|
-
export function getCachedGoogleAccountId() {
|
|
195
|
-
try {
|
|
196
|
-
const filePath = getGoogleAccountIdCachePath();
|
|
197
|
-
if (existsSync(filePath)) {
|
|
198
|
-
return readFileSync(filePath, 'utf-8').trim() || null;
|
|
199
|
-
}
|
|
200
|
-
return null;
|
|
201
|
-
}
|
|
202
|
-
catch (error) {
|
|
203
|
-
console.debug('Error reading cached Google Account ID:', error);
|
|
204
|
-
return null;
|
|
205
|
-
}
|
|
206
|
-
}
|
|
207
262
|
export async function clearCachedCredentialFile() {
|
|
208
263
|
try {
|
|
209
264
|
await fs.rm(getCachedCredentialPath(), { force: true });
|
|
210
265
|
// Clear the Google Account ID cache when credentials are cleared
|
|
211
|
-
await
|
|
266
|
+
await clearCachedGoogleAccount();
|
|
212
267
|
}
|
|
213
268
|
catch (_) {
|
|
214
269
|
/* empty */
|
|
215
270
|
}
|
|
216
271
|
}
|
|
217
|
-
|
|
218
|
-
* Retrieves the authenticated user's Google Account ID from Google's UserInfo API.
|
|
219
|
-
* @param client - The authenticated OAuth2Client
|
|
220
|
-
* @returns The user's Google Account ID or null if not available
|
|
221
|
-
*/
|
|
222
|
-
export async function getRawGoogleAccountId(client) {
|
|
272
|
+
async function fetchAndCacheUserInfo(client) {
|
|
223
273
|
try {
|
|
224
|
-
|
|
225
|
-
|
|
226
|
-
|
|
227
|
-
if (err) {
|
|
228
|
-
return reject(err);
|
|
229
|
-
}
|
|
230
|
-
resolve(tokens ?? null);
|
|
231
|
-
});
|
|
232
|
-
});
|
|
233
|
-
if (!refreshedTokens?.id_token) {
|
|
234
|
-
console.warn('No id_token obtained after refreshing tokens.');
|
|
235
|
-
return null;
|
|
274
|
+
const { token } = await client.getAccessToken();
|
|
275
|
+
if (!token) {
|
|
276
|
+
return;
|
|
236
277
|
}
|
|
237
|
-
|
|
238
|
-
|
|
239
|
-
|
|
240
|
-
|
|
278
|
+
const response = await fetch('https://www.googleapis.com/oauth2/v2/userinfo', {
|
|
279
|
+
headers: {
|
|
280
|
+
Authorization: `Bearer ${token}`,
|
|
281
|
+
},
|
|
241
282
|
});
|
|
242
|
-
|
|
243
|
-
|
|
244
|
-
|
|
245
|
-
|
|
283
|
+
if (!response.ok) {
|
|
284
|
+
console.error('Failed to fetch user info:', response.status, response.statusText);
|
|
285
|
+
return;
|
|
286
|
+
}
|
|
287
|
+
const userInfo = await response.json();
|
|
288
|
+
if (userInfo.email) {
|
|
289
|
+
await cacheGoogleAccount(userInfo.email);
|
|
246
290
|
}
|
|
247
|
-
return payload.sub;
|
|
248
291
|
}
|
|
249
292
|
catch (error) {
|
|
250
|
-
console.error('Error retrieving
|
|
251
|
-
return null;
|
|
293
|
+
console.error('Error retrieving user info:', error);
|
|
252
294
|
}
|
|
253
295
|
}
|
|
254
296
|
//# sourceMappingURL=oauth2.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"oauth2.js","sourceRoot":"","sources":["../../../src/code_assist/oauth2.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAEH,OAAO,
|
|
1
|
+
{"version":3,"file":"oauth2.js","sourceRoot":"","sources":["../../../src/code_assist/oauth2.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAEH,OAAO,EACL,YAAY,EAEZ,OAAO,EACP,mBAAmB,GACpB,MAAM,qBAAqB,CAAC;AAC7B,OAAO,KAAK,IAAI,MAAM,MAAM,CAAC;AAC7B,OAAO,GAAG,MAAM,KAAK,CAAC;AACtB,OAAO,MAAM,MAAM,QAAQ,CAAC;AAC5B,OAAO,KAAK,GAAG,MAAM,KAAK,CAAC;AAC3B,OAAO,IAAI,MAAM,MAAM,CAAC;AACxB,OAAO,IAAI,MAAM,WAAW,CAAC;AAC7B,OAAO,EAAE,QAAQ,IAAI,EAAE,EAAE,MAAM,SAAS,CAAC;AACzC,OAAO,KAAK,EAAE,MAAM,IAAI,CAAC;AAEzB,OAAO,EAAE,eAAe,EAAE,MAAM,oBAAoB,CAAC;AACrD,OAAO,EACL,kBAAkB,EAClB,sBAAsB,EACtB,wBAAwB,GACzB,MAAM,0BAA0B,CAAC;AAClC,OAAO,EAAE,QAAQ,EAAE,MAAM,6BAA6B,CAAC;AACvD,OAAO,QAAQ,MAAM,eAAe,CAAC;AAErC,wDAAwD;AACxD,MAAM,eAAe,GACnB,0EAA0E,CAAC;AAE7E,0DAA0D;AAC1D,6EAA6E;AAC7E,uFAAuF;AACvF,2EAA2E;AAC3E,4EAA4E;AAC5E,4DAA4D;AAC5D,MAAM,mBAAmB,GAAG,qCAAqC,CAAC;AAElE,6CAA6C;AAC7C,MAAM,WAAW,GAAG;IAClB,gDAAgD;IAChD,gDAAgD;IAChD,kDAAkD;CACnD,CAAC;AAEF,MAAM,aAAa,GAAG,GAAG,CAAC;AAC1B,MAAM,mBAAmB,GACvB,sEAAsE,CAAC;AACzE,MAAM,mBAAmB,GACvB,sEAAsE,CAAC;AAEzE,MAAM,UAAU,GAAG,SAAS,CAAC;AAC7B,MAAM,mBAAmB,GAAG,kBAAkB,CAAC;AAY/C,MAAM,CAAC,KAAK,UAAU,cAAc,CAClC,QAAkB,EAClB,MAAc;IAEd,MAAM,MAAM,GAAG,IAAI,YAAY,CAAC;QAC9B,QAAQ,EAAE,eAAe;QACzB,YAAY,EAAE,mBAAmB;KAClC,CAAC,CAAC;IAEH,MAAM,CAAC,EAAE,CAAC,QAAQ,EAAE,KAAK,EAAE,MAAmB,EAAE,EAAE;QAChD,MAAM,gBAAgB,CAAC,MAAM,CAAC,CAAC;IACjC,CAAC,CAAC,CAAC;IAEH,iEAAiE;IACjE,IAAI,MAAM,qBAAqB,CAAC,MAAM,CAAC,EAAE,CAAC;QACxC,kCAAkC;QAClC,0DAA0D;QAC1D,IAAI,CAAC,sBAAsB,EAAE,EAAE,CAAC;YAC9B,IAAI,CAAC;gBACH,MAAM,qBAAqB,CAAC,MAAM,CAAC,CAAC;YACtC,CAAC;YAAC,MAAM,CAAC;gBACP,0CAA0C;YAC5C,CAAC;QACH,CAAC;QACD,OAAO,CAAC,GAAG,CAAC,4BAA4B,CAAC,CAAC;QAC1C,OAAO,MAAM,CAAC;IAChB,CAAC;IAED,0EAA0E;IAC1E,2EAA2E;IAC3E,oDAAoD;IACpD,IAAI,QAAQ,KAAK,QAAQ,CAAC,WAAW,EAAE,CAAC;QACtC,IAAI,CAAC;YACH,OAAO,CAAC,GAAG,CAAC,sDAAsD,CAAC,CAAC;YACpE,MAAM,aAAa,GAAG,IAAI,OAAO,CAAC;YAChC,kEAAkE;YAClE,6BAA6B;aAC9B,CAAC,CAAC;YACH,MAAM,aAAa,CAAC,cAAc,EAAE,CAAC;YACrC,OAAO,CAAC,GAAG,CAAC,4BAA4B,CAAC,CAAC;YAE1C,wFAAwF;YACxF,OAAO,aAAa,CAAC;QACvB,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,MAAM,IAAI,KAAK,CACb,wKAAwK,eAAe,CACrL,CAAC,CACF,EAAE,CACJ,CAAC;QACJ,CAAC;IACH,CAAC;IAED,IAAI,MAAM,CAAC,YAAY,EAAE,EAAE,CAAC;QAC1B,IAAI,OAAO,GAAG,KAAK,CAAC;QACpB,MAAM,UAAU,GAAG,CAAC,CAAC;QACrB,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,OAAO,IAAI,CAAC,GAAG,UAAU,EAAE,CAAC,EAAE,EAAE,CAAC;YAChD,OAAO,GAAG,MAAM,gBAAgB,CAAC,MAAM,CAAC,CAAC;YACzC,IAAI,CAAC,OAAO,EAAE,CAAC;gBACb,OAAO,CAAC,KAAK,CACX,0CAA0C,EAC1C,CAAC,KAAK,UAAU,GAAG,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,eAAe,CAC5C,CAAC;YACJ,CAAC;QACH,CAAC;QACD,IAAI,CAAC,OAAO,EAAE,CAAC;YACb,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;QAClB,CAAC;IACH,CAAC;SAAM,CAAC;QACN,MAAM,QAAQ,GAAG,MAAM,WAAW,CAAC,MAAM,CAAC,CAAC;QAE3C,6DAA6D;QAC7D,OAAO,CAAC,GAAG,CACT,mCAAmC;YACjC,2DAA2D;YAC3D,6BAA6B,QAAQ,CAAC,OAAO,MAAM,CACtD,CAAC;QACF,MAAM,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;QAC7B,OAAO,CAAC,GAAG,CAAC,+BAA+B,CAAC,CAAC;QAE7C,MAAM,QAAQ,CAAC,oBAAoB,CAAC;IACtC,CAAC;IAED,OAAO,MAAM,CAAC;AAChB,CAAC;AAED,KAAK,UAAU,gBAAgB,CAAC,MAAoB;IAClD,MAAM,WAAW,GAAG,sDAAsD,CAAC;IAC3E,MAAM,YAAY,GAAG,MAAM,MAAM,CAAC,yBAAyB,EAAE,CAAC;IAC9D,MAAM,KAAK,GAAG,MAAM,CAAC,WAAW,CAAC,EAAE,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;IACrD,MAAM,OAAO,GAAW,MAAM,CAAC,eAAe,CAAC;QAC7C,YAAY,EAAE,WAAW;QACzB,WAAW,EAAE,SAAS;QACtB,KAAK,EAAE,WAAW;QAClB,qBAAqB,EAAE,mBAAmB,CAAC,IAAI;QAC/C,cAAc,EAAE,YAAY,CAAC,aAAa;QAC1C,KAAK;KACN,CAAC,CAAC;IACH,OAAO,CAAC,GAAG,CAAC,8DAA8D,CAAC,CAAC;IAC5E,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;IAChB,OAAO,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;IACrB,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;IAEhB,MAAM,IAAI,GAAG,MAAM,IAAI,OAAO,CAAS,CAAC,OAAO,EAAE,EAAE;QACjD,MAAM,EAAE,GAAG,QAAQ,CAAC,eAAe,CAAC;YAClC,KAAK,EAAE,OAAO,CAAC,KAAK;YACpB,MAAM,EAAE,OAAO,CAAC,MAAM;SACvB,CAAC,CAAC;QACH,EAAE,CAAC,QAAQ,CAAC,gCAAgC,EAAE,CAAC,IAAI,EAAE,EAAE;YACrD,EAAE,CAAC,KAAK,EAAE,CAAC;YACX,OAAO,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC;QACvB,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,IAAI,CAAC,IAAI,EAAE,CAAC;QACV,OAAO,CAAC,KAAK,CAAC,iCAAiC,CAAC,CAAC;QACjD,OAAO,KAAK,CAAC;IACf,CAAC;IAED,IAAI,CAAC;QACH,MAAM,EAAE,MAAM,EAAE,GAAG,MAAM,MAAM,CAAC,QAAQ,CAAC;YACvC,IAAI;YACJ,YAAY,EAAE,YAAY,CAAC,YAAY;YACvC,YAAY,EAAE,WAAW;SAC1B,CAAC,CAAC;QACH,MAAM,CAAC,cAAc,CAAC,MAAM,CAAC,CAAC;IAChC,CAAC;IAAC,OAAO,MAAM,EAAE,CAAC;QAChB,OAAO,KAAK,CAAC;IACf,CAAC;IACD,OAAO,IAAI,CAAC;AACd,CAAC;AAED,KAAK,UAAU,WAAW,CAAC,MAAoB;IAC7C,MAAM,IAAI,GAAG,MAAM,gBAAgB,EAAE,CAAC;IACtC,MAAM,WAAW,GAAG,oBAAoB,IAAI,iBAAiB,CAAC;IAC9D,MAAM,KAAK,GAAG,MAAM,CAAC,WAAW,CAAC,EAAE,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;IACrD,MAAM,OAAO,GAAG,MAAM,CAAC,eAAe,CAAC;QACrC,YAAY,EAAE,WAAW;QACzB,WAAW,EAAE,SAAS;QACtB,KAAK,EAAE,WAAW;QAClB,KAAK;KACN,CAAC,CAAC;IAEH,MAAM,oBAAoB,GAAG,IAAI,OAAO,CAAO,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;QACjE,MAAM,MAAM,GAAG,IAAI,CAAC,YAAY,CAAC,KAAK,EAAE,GAAG,EAAE,GAAG,EAAE,EAAE;YAClD,IAAI,CAAC;gBACH,IAAI,GAAG,CAAC,GAAI,CAAC,OAAO,CAAC,iBAAiB,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC;oBAC/C,GAAG,CAAC,SAAS,CAAC,aAAa,EAAE,EAAE,QAAQ,EAAE,mBAAmB,EAAE,CAAC,CAAC;oBAChE,GAAG,CAAC,GAAG,EAAE,CAAC;oBACV,MAAM,CAAC,IAAI,KAAK,CAAC,sBAAsB,GAAG,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC;gBACtD,CAAC;gBACD,mEAAmE;gBACnE,MAAM,EAAE,GAAG,IAAI,GAAG,CAAC,GAAG,CAAC,GAAG,CAAC,GAAI,EAAE,uBAAuB,CAAC,CAAC,YAAY,CAAC;gBACvE,IAAI,EAAE,CAAC,GAAG,CAAC,OAAO,CAAC,EAAE,CAAC;oBACpB,GAAG,CAAC,SAAS,CAAC,aAAa,EAAE,EAAE,QAAQ,EAAE,mBAAmB,EAAE,CAAC,CAAC;oBAChE,GAAG,CAAC,GAAG,EAAE,CAAC;oBAEV,MAAM,CAAC,IAAI,KAAK,CAAC,gCAAgC,EAAE,CAAC,GAAG,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC,CAAC;gBACvE,CAAC;qBAAM,IAAI,EAAE,CAAC,GAAG,CAAC,OAAO,CAAC,KAAK,KAAK,EAAE,CAAC;oBACrC,GAAG,CAAC,GAAG,CAAC,sCAAsC,CAAC,CAAC;oBAEhD,MAAM,CAAC,IAAI,KAAK,CAAC,sCAAsC,CAAC,CAAC,CAAC;gBAC5D,CAAC;qBAAM,IAAI,EAAE,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC;oBAC1B,MAAM,EAAE,MAAM,EAAE,GAAG,MAAM,MAAM,CAAC,QAAQ,CAAC;wBACvC,IAAI,EAAE,EAAE,CAAC,GAAG,CAAC,MAAM,CAAE;wBACrB,YAAY,EAAE,WAAW;qBAC1B,CAAC,CAAC;oBACH,MAAM,CAAC,cAAc,CAAC,MAAM,CAAC,CAAC;oBAC9B,6DAA6D;oBAC7D,IAAI,CAAC;wBACH,MAAM,qBAAqB,CAAC,MAAM,CAAC,CAAC;oBACtC,CAAC;oBAAC,OAAO,KAAK,EAAE,CAAC;wBACf,OAAO,CAAC,KAAK,CACX,6DAA6D,EAC7D,KAAK,CACN,CAAC;wBACF,gEAAgE;oBAClE,CAAC;oBAED,GAAG,CAAC,SAAS,CAAC,aAAa,EAAE,EAAE,QAAQ,EAAE,mBAAmB,EAAE,CAAC,CAAC;oBAChE,GAAG,CAAC,GAAG,EAAE,CAAC;oBACV,OAAO,EAAE,CAAC;gBACZ,CAAC;qBAAM,CAAC;oBACN,MAAM,CAAC,IAAI,KAAK,CAAC,0BAA0B,CAAC,CAAC,CAAC;gBAChD,CAAC;YACH,CAAC;YAAC,OAAO,CAAC,EAAE,CAAC;gBACX,MAAM,CAAC,CAAC,CAAC,CAAC;YACZ,CAAC;oBAAS,CAAC;gBACT,MAAM,CAAC,KAAK,EAAE,CAAC;YACjB,CAAC;QACH,CAAC,CAAC,CAAC;QACH,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;IACtB,CAAC,CAAC,CAAC;IAEH,OAAO;QACL,OAAO;QACP,oBAAoB;KACrB,CAAC;AACJ,CAAC;AAED,MAAM,UAAU,gBAAgB;IAC9B,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;QACrC,IAAI,IAAI,GAAG,CAAC,CAAC;QACb,IAAI,CAAC;YACH,MAAM,MAAM,GAAG,GAAG,CAAC,YAAY,EAAE,CAAC;YAClC,MAAM,CAAC,MAAM,CAAC,CAAC,EAAE,GAAG,EAAE;gBACpB,MAAM,OAAO,GAAG,MAAM,CAAC,OAAO,EAAsB,CAAC;gBACrD,IAAI,GAAG,OAAO,CAAC,IAAI,CAAC;YACtB,CAAC,CAAC,CAAC;YACH,MAAM,CAAC,EAAE,CAAC,WAAW,EAAE,GAAG,EAAE;gBAC1B,MAAM,CAAC,KAAK,EAAE,CAAC;gBACf,MAAM,CAAC,KAAK,EAAE,CAAC;YACjB,CAAC,CAAC,CAAC;YACH,MAAM,CAAC,EAAE,CAAC,OAAO,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC;YACrC,MAAM,CAAC,EAAE,CAAC,OAAO,EAAE,GAAG,EAAE,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC;QAC1C,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,MAAM,CAAC,CAAC,CAAC,CAAC;QACZ,CAAC;IACH,CAAC,CAAC,CAAC;AACL,CAAC;AAED,KAAK,UAAU,qBAAqB,CAAC,MAAoB;IACvD,IAAI,CAAC;QACH,MAAM,OAAO,GACX,OAAO,CAAC,GAAG,CAAC,8BAA8B,IAAI,uBAAuB,EAAE,CAAC;QAE1E,MAAM,KAAK,GAAG,MAAM,EAAE,CAAC,QAAQ,CAAC,OAAO,EAAE,OAAO,CAAC,CAAC;QAClD,MAAM,CAAC,cAAc,CAAC,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC;QAEzC,2DAA2D;QAC3D,MAAM,EAAE,KAAK,EAAE,GAAG,MAAM,MAAM,CAAC,cAAc,EAAE,CAAC;QAChD,IAAI,CAAC,KAAK,EAAE,CAAC;YACX,OAAO,KAAK,CAAC;QACf,CAAC;QAED,oEAAoE;QACpE,MAAM,MAAM,CAAC,YAAY,CAAC,KAAK,CAAC,CAAC;QAEjC,OAAO,IAAI,CAAC;IACd,CAAC;IAAC,OAAO,CAAC,EAAE,CAAC;QACX,OAAO,KAAK,CAAC;IACf,CAAC;AACH,CAAC;AAED,KAAK,UAAU,gBAAgB,CAAC,WAAwB;IACtD,MAAM,QAAQ,GAAG,uBAAuB,EAAE,CAAC;IAC3C,MAAM,EAAE,CAAC,KAAK,CAAC,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;IAE5D,MAAM,UAAU,GAAG,IAAI,CAAC,SAAS,CAAC,WAAW,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC;IACxD,MAAM,EAAE,CAAC,SAAS,CAAC,QAAQ,EAAE,UAAU,CAAC,CAAC;AAC3C,CAAC;AAED,SAAS,uBAAuB;IAC9B,OAAO,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,OAAO,EAAE,EAAE,UAAU,EAAE,mBAAmB,CAAC,CAAC;AAClE,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,yBAAyB;IAC7C,IAAI,CAAC;QACH,MAAM,EAAE,CAAC,EAAE,CAAC,uBAAuB,EAAE,EAAE,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC,CAAC;QACxD,iEAAiE;QACjE,MAAM,wBAAwB,EAAE,CAAC;IACnC,CAAC;IAAC,OAAO,CAAC,EAAE,CAAC;QACX,WAAW;IACb,CAAC;AACH,CAAC;AAED,KAAK,UAAU,qBAAqB,CAAC,MAAoB;IACvD,IAAI,CAAC;QACH,MAAM,EAAE,KAAK,EAAE,GAAG,MAAM,MAAM,CAAC,cAAc,EAAE,CAAC;QAChD,IAAI,CAAC,KAAK,EAAE,CAAC;YACX,OAAO;QACT,CAAC;QAED,MAAM,QAAQ,GAAG,MAAM,KAAK,CAC1B,+CAA+C,EAC/C;YACE,OAAO,EAAE;gBACP,aAAa,EAAE,UAAU,KAAK,EAAE;aACjC;SACF,CACF,CAAC;QAEF,IAAI,CAAC,QAAQ,CAAC,EAAE,EAAE,CAAC;YACjB,OAAO,CAAC,KAAK,CACX,4BAA4B,EAC5B,QAAQ,CAAC,MAAM,EACf,QAAQ,CAAC,UAAU,CACpB,CAAC;YACF,OAAO;QACT,CAAC;QAED,MAAM,QAAQ,GAAG,MAAM,QAAQ,CAAC,IAAI,EAAE,CAAC;QACvC,IAAI,QAAQ,CAAC,KAAK,EAAE,CAAC;YACnB,MAAM,kBAAkB,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;QAC3C,CAAC;IACH,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QACf,OAAO,CAAC,KAAK,CAAC,6BAA6B,EAAE,KAAK,CAAC,CAAC;IACtD,CAAC;AACH,CAAC"}
|
|
@@ -4,14 +4,17 @@
|
|
|
4
4
|
* SPDX-License-Identifier: Apache-2.0
|
|
5
5
|
*/
|
|
6
6
|
import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest';
|
|
7
|
-
import { getOauthClient
|
|
8
|
-
import {
|
|
7
|
+
import { getOauthClient } from './oauth2.js';
|
|
8
|
+
import { getCachedGoogleAccount } from '../utils/user_account.js';
|
|
9
|
+
import { OAuth2Client, Compute } from 'google-auth-library';
|
|
9
10
|
import * as fs from 'fs';
|
|
10
11
|
import * as path from 'path';
|
|
11
12
|
import http from 'http';
|
|
12
13
|
import open from 'open';
|
|
13
14
|
import crypto from 'crypto';
|
|
14
15
|
import * as os from 'os';
|
|
16
|
+
import { AuthType } from '../core/contentGenerator.js';
|
|
17
|
+
import readline from 'node:readline';
|
|
15
18
|
vi.mock('os', async (importOriginal) => {
|
|
16
19
|
const os = await importOriginal();
|
|
17
20
|
return {
|
|
@@ -23,16 +26,22 @@ vi.mock('google-auth-library');
|
|
|
23
26
|
vi.mock('http');
|
|
24
27
|
vi.mock('open');
|
|
25
28
|
vi.mock('crypto');
|
|
29
|
+
vi.mock('node:readline');
|
|
30
|
+
const mockConfig = {
|
|
31
|
+
getNoBrowser: () => false,
|
|
32
|
+
};
|
|
26
33
|
// Mock fetch globally
|
|
27
34
|
global.fetch = vi.fn();
|
|
28
35
|
describe('oauth2', () => {
|
|
29
36
|
let tempHomeDir;
|
|
30
37
|
beforeEach(() => {
|
|
31
38
|
tempHomeDir = fs.mkdtempSync(path.join(os.tmpdir(), 'gemini-cli-test-home-'));
|
|
32
|
-
|
|
39
|
+
os.homedir.mockReturnValue(tempHomeDir);
|
|
33
40
|
});
|
|
34
41
|
afterEach(() => {
|
|
35
42
|
fs.rmSync(tempHomeDir, { recursive: true, force: true });
|
|
43
|
+
vi.clearAllMocks();
|
|
44
|
+
delete process.env.CLOUD_SHELL;
|
|
36
45
|
});
|
|
37
46
|
it('should perform a web login', async () => {
|
|
38
47
|
const mockAuthUrl = 'https://example.com/auth';
|
|
@@ -48,39 +57,23 @@ describe('oauth2', () => {
|
|
|
48
57
|
const mockGetAccessToken = vi
|
|
49
58
|
.fn()
|
|
50
59
|
.mockResolvedValue({ token: 'mock-access-token' });
|
|
51
|
-
const mockRefreshAccessToken = vi.fn().mockImplementation((callback) => {
|
|
52
|
-
// Mock the callback-style refreshAccessToken method
|
|
53
|
-
const mockTokensWithIdToken = {
|
|
54
|
-
access_token: 'test-access-token',
|
|
55
|
-
refresh_token: 'test-refresh-token',
|
|
56
|
-
id_token: 'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJzdWIiOiJ0ZXN0LWdvb2dsZS1hY2NvdW50LWlkLTEyMyJ9.signature', // Mock JWT with sub: test-google-account-id-123
|
|
57
|
-
};
|
|
58
|
-
callback(null, mockTokensWithIdToken);
|
|
59
|
-
});
|
|
60
|
-
const mockVerifyIdToken = vi.fn().mockResolvedValue({
|
|
61
|
-
getPayload: () => ({
|
|
62
|
-
sub: 'test-google-account-id-123',
|
|
63
|
-
aud: 'test-audience',
|
|
64
|
-
iss: 'https://accounts.google.com',
|
|
65
|
-
}),
|
|
66
|
-
});
|
|
67
60
|
const mockOAuth2Client = {
|
|
68
61
|
generateAuthUrl: mockGenerateAuthUrl,
|
|
69
62
|
getToken: mockGetToken,
|
|
70
63
|
setCredentials: mockSetCredentials,
|
|
71
64
|
getAccessToken: mockGetAccessToken,
|
|
72
|
-
refreshAccessToken: mockRefreshAccessToken,
|
|
73
|
-
verifyIdToken: mockVerifyIdToken,
|
|
74
65
|
credentials: mockTokens,
|
|
75
66
|
on: vi.fn(),
|
|
76
67
|
};
|
|
77
|
-
|
|
68
|
+
OAuth2Client.mockImplementation(() => mockOAuth2Client);
|
|
78
69
|
vi.spyOn(crypto, 'randomBytes').mockReturnValue(mockState);
|
|
79
|
-
|
|
70
|
+
open.mockImplementation(async () => ({}));
|
|
80
71
|
// Mock the UserInfo API response
|
|
81
|
-
|
|
72
|
+
global.fetch.mockResolvedValue({
|
|
82
73
|
ok: true,
|
|
83
|
-
json: vi
|
|
74
|
+
json: vi
|
|
75
|
+
.fn()
|
|
76
|
+
.mockResolvedValue({ email: 'test-google-account@gmail.com' }),
|
|
84
77
|
});
|
|
85
78
|
let requestCallback;
|
|
86
79
|
let serverListeningCallback;
|
|
@@ -102,11 +95,11 @@ describe('oauth2', () => {
|
|
|
102
95
|
on: vi.fn(),
|
|
103
96
|
address: () => ({ port: capturedPort }),
|
|
104
97
|
};
|
|
105
|
-
|
|
98
|
+
http.createServer.mockImplementation((cb) => {
|
|
106
99
|
requestCallback = cb;
|
|
107
100
|
return mockHttpServer;
|
|
108
101
|
});
|
|
109
|
-
const clientPromise = getOauthClient();
|
|
102
|
+
const clientPromise = getOauthClient(AuthType.LOGIN_WITH_GOOGLE, mockConfig);
|
|
110
103
|
// wait for server to start listening.
|
|
111
104
|
await serverListeningPromise;
|
|
112
105
|
const mockReq = {
|
|
@@ -125,13 +118,120 @@ describe('oauth2', () => {
|
|
|
125
118
|
redirect_uri: `http://localhost:${capturedPort}/oauth2callback`,
|
|
126
119
|
});
|
|
127
120
|
expect(mockSetCredentials).toHaveBeenCalledWith(mockTokens);
|
|
128
|
-
// Verify Google Account
|
|
129
|
-
const
|
|
130
|
-
expect(fs.existsSync(
|
|
131
|
-
const
|
|
132
|
-
expect(
|
|
133
|
-
|
|
134
|
-
|
|
121
|
+
// Verify Google Account was cached
|
|
122
|
+
const googleAccountPath = path.join(tempHomeDir, '.gemini', 'google_accounts.json');
|
|
123
|
+
expect(fs.existsSync(googleAccountPath)).toBe(true);
|
|
124
|
+
const cachedGoogleAccount = fs.readFileSync(googleAccountPath, 'utf-8');
|
|
125
|
+
expect(JSON.parse(cachedGoogleAccount)).toEqual({
|
|
126
|
+
active: 'test-google-account@gmail.com',
|
|
127
|
+
old: [],
|
|
128
|
+
});
|
|
129
|
+
// Verify the getCachedGoogleAccount function works
|
|
130
|
+
expect(getCachedGoogleAccount()).toBe('test-google-account@gmail.com');
|
|
131
|
+
});
|
|
132
|
+
it('should perform login with user code', async () => {
|
|
133
|
+
const mockConfigWithNoBrowser = {
|
|
134
|
+
getNoBrowser: () => true,
|
|
135
|
+
};
|
|
136
|
+
const mockCodeVerifier = {
|
|
137
|
+
codeChallenge: 'test-challenge',
|
|
138
|
+
codeVerifier: 'test-verifier',
|
|
139
|
+
};
|
|
140
|
+
const mockAuthUrl = 'https://example.com/auth-user-code';
|
|
141
|
+
const mockCode = 'test-user-code';
|
|
142
|
+
const mockTokens = {
|
|
143
|
+
access_token: 'test-access-token-user-code',
|
|
144
|
+
refresh_token: 'test-refresh-token-user-code',
|
|
145
|
+
};
|
|
146
|
+
const mockGenerateAuthUrl = vi.fn().mockReturnValue(mockAuthUrl);
|
|
147
|
+
const mockGetToken = vi.fn().mockResolvedValue({ tokens: mockTokens });
|
|
148
|
+
const mockSetCredentials = vi.fn();
|
|
149
|
+
const mockGenerateCodeVerifierAsync = vi
|
|
150
|
+
.fn()
|
|
151
|
+
.mockResolvedValue(mockCodeVerifier);
|
|
152
|
+
const mockOAuth2Client = {
|
|
153
|
+
generateAuthUrl: mockGenerateAuthUrl,
|
|
154
|
+
getToken: mockGetToken,
|
|
155
|
+
setCredentials: mockSetCredentials,
|
|
156
|
+
generateCodeVerifierAsync: mockGenerateCodeVerifierAsync,
|
|
157
|
+
on: vi.fn(),
|
|
158
|
+
};
|
|
159
|
+
OAuth2Client.mockImplementation(() => mockOAuth2Client);
|
|
160
|
+
const mockReadline = {
|
|
161
|
+
question: vi.fn((_query, callback) => callback(mockCode)),
|
|
162
|
+
close: vi.fn(),
|
|
163
|
+
};
|
|
164
|
+
readline.createInterface.mockReturnValue(mockReadline);
|
|
165
|
+
const consoleLogSpy = vi.spyOn(console, 'log').mockImplementation(() => { });
|
|
166
|
+
const client = await getOauthClient(AuthType.LOGIN_WITH_GOOGLE, mockConfigWithNoBrowser);
|
|
167
|
+
expect(client).toBe(mockOAuth2Client);
|
|
168
|
+
// Verify the auth flow
|
|
169
|
+
expect(mockGenerateCodeVerifierAsync).toHaveBeenCalled();
|
|
170
|
+
expect(mockGenerateAuthUrl).toHaveBeenCalled();
|
|
171
|
+
expect(consoleLogSpy).toHaveBeenCalledWith(expect.stringContaining(mockAuthUrl));
|
|
172
|
+
expect(mockReadline.question).toHaveBeenCalledWith('Enter the authorization code: ', expect.any(Function));
|
|
173
|
+
expect(mockGetToken).toHaveBeenCalledWith({
|
|
174
|
+
code: mockCode,
|
|
175
|
+
codeVerifier: mockCodeVerifier.codeVerifier,
|
|
176
|
+
redirect_uri: 'https://sdk.cloud.google.com/authcode_cloudcode.html',
|
|
177
|
+
});
|
|
178
|
+
expect(mockSetCredentials).toHaveBeenCalledWith(mockTokens);
|
|
179
|
+
consoleLogSpy.mockRestore();
|
|
180
|
+
});
|
|
181
|
+
describe('in Cloud Shell', () => {
|
|
182
|
+
const mockGetAccessToken = vi.fn();
|
|
183
|
+
let mockComputeClient;
|
|
184
|
+
beforeEach(() => {
|
|
185
|
+
vi.spyOn(os, 'homedir').mockReturnValue('/user/home');
|
|
186
|
+
vi.spyOn(fs.promises, 'mkdir').mockResolvedValue(undefined);
|
|
187
|
+
vi.spyOn(fs.promises, 'writeFile').mockResolvedValue(undefined);
|
|
188
|
+
vi.spyOn(fs.promises, 'readFile').mockRejectedValue(new Error('File not found')); // Default to no cached creds
|
|
189
|
+
mockGetAccessToken.mockResolvedValue({ token: 'test-access-token' });
|
|
190
|
+
mockComputeClient = {
|
|
191
|
+
credentials: { refresh_token: 'test-refresh-token' },
|
|
192
|
+
getAccessToken: mockGetAccessToken,
|
|
193
|
+
};
|
|
194
|
+
Compute.mockImplementation(() => mockComputeClient);
|
|
195
|
+
});
|
|
196
|
+
it('should attempt to load cached credentials first', async () => {
|
|
197
|
+
const cachedCreds = { refresh_token: 'cached-token' };
|
|
198
|
+
vi.spyOn(fs.promises, 'readFile').mockResolvedValue(JSON.stringify(cachedCreds));
|
|
199
|
+
const mockClient = {
|
|
200
|
+
setCredentials: vi.fn(),
|
|
201
|
+
getAccessToken: vi.fn().mockResolvedValue({ token: 'test-token' }),
|
|
202
|
+
getTokenInfo: vi.fn().mockResolvedValue({}),
|
|
203
|
+
on: vi.fn(),
|
|
204
|
+
};
|
|
205
|
+
// To mock the new OAuth2Client() inside the function
|
|
206
|
+
OAuth2Client.mockImplementation(() => mockClient);
|
|
207
|
+
await getOauthClient(AuthType.LOGIN_WITH_GOOGLE, mockConfig);
|
|
208
|
+
expect(fs.promises.readFile).toHaveBeenCalledWith('/user/home/.gemini/oauth_creds.json', 'utf-8');
|
|
209
|
+
expect(mockClient.setCredentials).toHaveBeenCalledWith(cachedCreds);
|
|
210
|
+
expect(mockClient.getAccessToken).toHaveBeenCalled();
|
|
211
|
+
expect(mockClient.getTokenInfo).toHaveBeenCalled();
|
|
212
|
+
expect(Compute).not.toHaveBeenCalled(); // Should not fetch new client if cache is valid
|
|
213
|
+
});
|
|
214
|
+
it('should use Compute to get a client if no cached credentials exist', async () => {
|
|
215
|
+
await getOauthClient(AuthType.CLOUD_SHELL, mockConfig);
|
|
216
|
+
expect(Compute).toHaveBeenCalledWith({});
|
|
217
|
+
expect(mockGetAccessToken).toHaveBeenCalled();
|
|
218
|
+
});
|
|
219
|
+
it('should not cache the credentials after fetching them via ADC', async () => {
|
|
220
|
+
const newCredentials = { refresh_token: 'new-adc-token' };
|
|
221
|
+
mockComputeClient.credentials = newCredentials;
|
|
222
|
+
mockGetAccessToken.mockResolvedValue({ token: 'new-adc-token' });
|
|
223
|
+
await getOauthClient(AuthType.CLOUD_SHELL, mockConfig);
|
|
224
|
+
expect(fs.promises.writeFile).not.toHaveBeenCalled();
|
|
225
|
+
});
|
|
226
|
+
it('should return the Compute client on successful ADC authentication', async () => {
|
|
227
|
+
const client = await getOauthClient(AuthType.CLOUD_SHELL, mockConfig);
|
|
228
|
+
expect(client).toBe(mockComputeClient);
|
|
229
|
+
});
|
|
230
|
+
it('should throw an error if ADC fails', async () => {
|
|
231
|
+
const testError = new Error('ADC Failed');
|
|
232
|
+
mockGetAccessToken.mockRejectedValue(testError);
|
|
233
|
+
await expect(getOauthClient(AuthType.CLOUD_SHELL, mockConfig)).rejects.toThrow('Could not authenticate using Cloud Shell credentials. Please select a different authentication method or ensure you are in a properly configured environment. Error: ADC Failed');
|
|
234
|
+
});
|
|
135
235
|
});
|
|
136
236
|
});
|
|
137
237
|
//# sourceMappingURL=oauth2.test.js.map
|