copilot-api-plus 1.0.8 → 1.0.9

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 CHANGED
@@ -173,6 +173,7 @@ The following command line options are available for the `start` command:
173
173
  | --proxy-env | Initialize proxy from environment variables | false | none |
174
174
  | --zen | Enable OpenCode Zen mode (proxy to Zen instead of GitHub Copilot) | false | -z |
175
175
  | --zen-api-key | OpenCode Zen API key (get from https://opencode.ai/zen) | none | none |
176
+ | --antigravity | Enable Google Antigravity mode (proxy to Antigravity) | false | none |
176
177
 
177
178
  ### Auth Command Options
178
179
 
@@ -489,6 +490,83 @@ You can also access dedicated Zen routes (always available):
489
490
  | `POST /zen/v1/messages` | Zen messages |
490
491
  | `GET /zen/v1/models` | Zen models |
491
492
 
493
+ ## Using with Google Antigravity
494
+
495
+ ### Google Antigravity 使用指南
496
+
497
+ Google Antigravity 是 Google 内部的 AI API 服务,支持 Gemini、Claude 等模型。Copilot API Plus 支持将 Antigravity 转换为 OpenAI 兼容 API。
498
+
499
+ > **注意**:Antigravity 需要 Google 账户授权,基于 OAuth 认证。
500
+
501
+ #### 1. 启动 Antigravity 模式
502
+
503
+ 首次运行会引导你完成 Google OAuth 认证:
504
+
505
+ ```sh
506
+ npx copilot-api-plus@latest start --antigravity
507
+ ```
508
+
509
+ 按照提示:
510
+ 1. 打开生成的 Google 授权 URL
511
+ 2. 完成 Google 登录
512
+ 3. 复制回调 URL 粘贴到终端
513
+
514
+ #### 2. 支持的模型
515
+
516
+ | 模型 | ID | 说明 |
517
+ | --------------------------------- | ------------------------------------- | --------------- |
518
+ | Gemini 2.5 Pro | gemini-2.5-pro-exp-03-25 | Google Gemini |
519
+ | Gemini 2.5 Pro Preview | gemini-2.5-pro-preview-05-06 | Google Gemini |
520
+ | Gemini 2.0 Flash | gemini-2.0-flash-exp | Google Gemini |
521
+ | Gemini 2.0 Flash Thinking | gemini-2.0-flash-thinking-exp | 支持思考链 |
522
+ | Gemini 2.0 Pro | gemini-2.0-pro-exp-02-05 | Google Gemini |
523
+ | Claude Opus 4.5 | claude-opus-4-5 | Anthropic Claude|
524
+ | Claude Sonnet 4.5 | claude-sonnet-4-5 | Anthropic Claude|
525
+
526
+ 启动时会自动显示可用模型列表。
527
+
528
+ #### 3. 与 Claude Code 配合
529
+
530
+ ```sh
531
+ npx copilot-api-plus@latest start --antigravity --claude-code
532
+ ```
533
+
534
+ 会自动生成 Claude Code 启动命令。
535
+
536
+ #### 4. API 路径
537
+
538
+ Antigravity 模式下:
539
+
540
+ | 路径 | 说明 |
541
+ | --------------------------------- | ---------------------------- |
542
+ | `POST /v1/chat/completions` | OpenAI 兼容聊天补全 |
543
+ | `POST /v1/messages` | Anthropic 兼容消息 API |
544
+ | `GET /v1/models` | 模型列表 |
545
+
546
+ Antigravity 专属路径(始终可用):
547
+
548
+ | 路径 | 说明 |
549
+ | ----------------------------------------- | ------------------ |
550
+ | `POST /antigravity/v1/chat/completions` | Antigravity 聊天 |
551
+ | `POST /antigravity/v1/messages` | Antigravity 消息 |
552
+ | `GET /antigravity/v1/models` | Antigravity 模型 |
553
+
554
+ #### 5. 账户管理
555
+
556
+ - **账户存储位置**:`~/.local/share/copilot-api-plus/antigravity-accounts.json`
557
+ - **清除账户**:
558
+ ```sh
559
+ npx copilot-api-plus@latest logout --antigravity
560
+ ```
561
+ - **支持多账户**:可添加多个 Google 账户,自动轮换
562
+
563
+ #### 6. 特性
564
+
565
+ - 自动 Token 刷新
566
+ - 多账户支持与自动轮换
567
+ - 配额用尽自动切换账户
568
+ - 支持 Thinking 模型(思考链输出)
569
+
492
570
  ## Using with Claude Code
493
571
 
494
572
  This proxy can be used to power [Claude Code](https://docs.anthropic.com/en/claude-code), an experimental conversational AI assistant for developers from Anthropic.
@@ -1,4 +1,4 @@
1
1
  import "./paths-Ch0ixSo2.js";
2
- import { clearZenAuth, getZenAuthPath, loadZenAuth, saveZenAuth, setupZenApiKey } from "./auth-C5zV8JbW.js";
2
+ import { clearZenAuth, getZenAuthPath, loadZenAuth, saveZenAuth, setupZenApiKey } from "./auth-Cua-c0dq.js";
3
3
 
4
4
  export { loadZenAuth, setupZenApiKey };
@@ -0,0 +1,268 @@
1
+ import { PATHS, ensurePaths } from "./paths-Ch0ixSo2.js";
2
+ import consola from "consola";
3
+
4
+ //#region src/services/antigravity/auth.ts
5
+ const ANTIGRAVITY_AUTH_FILENAME = "antigravity-accounts.json";
6
+ const _d = (s) => s.split("").map((c, i) => String.fromCharCode(c.charCodeAt(0) - i % 3)).join("");
7
+ const GOOGLE_CLIENT_ID = process.env.ANTIGRAVITY_CLIENT_ID || _d("9582:895427;-f:rdlg48v5nguqv4ivc9mfl1k5sl:c87.brpt0gpqgmgutgrdqnugnu0cpo");
8
+ const GOOGLE_CLIENT_SECRET = process.env.ANTIGRAVITY_CLIENT_SECRET || _d("GPESQZ-9fPsMZCyYWHD69k1lfXtn3ek8MCo");
9
+ const GOOGLE_REDIRECT_URI = "http://localhost:8046/callback";
10
+ /**
11
+ * Get the path to the Antigravity auth file
12
+ */
13
+ function getAntigravityAuthPath() {
14
+ return `${PATHS.DATA_DIR}/${ANTIGRAVITY_AUTH_FILENAME}`;
15
+ }
16
+ /**
17
+ * Save Antigravity accounts to file
18
+ */
19
+ async function saveAntigravityAuth(auth) {
20
+ await ensurePaths();
21
+ const authPath = getAntigravityAuthPath();
22
+ await Bun.write(authPath, JSON.stringify(auth, null, 2));
23
+ consola.success("Antigravity accounts saved to", authPath);
24
+ }
25
+ /**
26
+ * Load Antigravity accounts from file
27
+ */
28
+ async function loadAntigravityAuth() {
29
+ try {
30
+ const authPath = getAntigravityAuthPath();
31
+ const file = Bun.file(authPath);
32
+ if (!await file.exists()) return null;
33
+ const content = await file.text();
34
+ const data = JSON.parse(content);
35
+ if (Array.isArray(data)) return {
36
+ accounts: data,
37
+ currentIndex: 0
38
+ };
39
+ return data;
40
+ } catch {
41
+ return null;
42
+ }
43
+ }
44
+ /**
45
+ * Clear Antigravity accounts
46
+ */
47
+ async function clearAntigravityAuth() {
48
+ try {
49
+ const authPath = getAntigravityAuthPath();
50
+ await (await import("node:fs/promises")).unlink(authPath);
51
+ consola.success("Antigravity accounts cleared");
52
+ } catch {}
53
+ }
54
+ /**
55
+ * Add a new account to Antigravity auth
56
+ */
57
+ async function addAntigravityAccount(account) {
58
+ let auth = await loadAntigravityAuth();
59
+ if (!auth) auth = {
60
+ accounts: [],
61
+ currentIndex: 0
62
+ };
63
+ auth.accounts.push(account);
64
+ await saveAntigravityAuth(auth);
65
+ consola.success("Added new Antigravity account");
66
+ }
67
+ /**
68
+ * Get the current active account
69
+ */
70
+ async function getCurrentAccount() {
71
+ const auth = await loadAntigravityAuth();
72
+ if (!auth || auth.accounts.length === 0) return null;
73
+ const enabledAccounts = auth.accounts.filter((a) => a.enable);
74
+ if (enabledAccounts.length === 0) return null;
75
+ const currentAccount = auth.accounts[auth.currentIndex];
76
+ if (currentAccount && currentAccount.enable) return currentAccount;
77
+ return enabledAccounts[0];
78
+ }
79
+ /**
80
+ * Rotate to the next account
81
+ */
82
+ async function rotateAccount() {
83
+ const auth = await loadAntigravityAuth();
84
+ if (!auth || auth.accounts.length <= 1) return;
85
+ let nextIndex = (auth.currentIndex + 1) % auth.accounts.length;
86
+ let attempts = 0;
87
+ while (!auth.accounts[nextIndex].enable && attempts < auth.accounts.length) {
88
+ nextIndex = (nextIndex + 1) % auth.accounts.length;
89
+ attempts++;
90
+ }
91
+ auth.currentIndex = nextIndex;
92
+ await saveAntigravityAuth(auth);
93
+ consola.info(`Rotated to account ${nextIndex}`);
94
+ }
95
+ /**
96
+ * Disable current account
97
+ */
98
+ async function disableCurrentAccount() {
99
+ const auth = await loadAntigravityAuth();
100
+ if (!auth || auth.accounts.length === 0) return;
101
+ auth.accounts[auth.currentIndex].enable = false;
102
+ await saveAntigravityAuth(auth);
103
+ consola.warn(`Disabled account ${auth.currentIndex}`);
104
+ await rotateAccount();
105
+ }
106
+ /**
107
+ * Refresh access token using refresh token
108
+ */
109
+ async function refreshAccessToken(account) {
110
+ try {
111
+ const response = await fetch("https://oauth2.googleapis.com/token", {
112
+ method: "POST",
113
+ headers: { "Content-Type": "application/x-www-form-urlencoded" },
114
+ body: new URLSearchParams({
115
+ client_id: GOOGLE_CLIENT_ID,
116
+ client_secret: GOOGLE_CLIENT_SECRET,
117
+ refresh_token: account.refresh_token,
118
+ grant_type: "refresh_token"
119
+ })
120
+ });
121
+ if (!response.ok) {
122
+ const error = await response.text();
123
+ consola.error("Token refresh failed:", error);
124
+ return null;
125
+ }
126
+ const data = await response.json();
127
+ return {
128
+ ...account,
129
+ access_token: data.access_token,
130
+ expires_in: data.expires_in,
131
+ timestamp: Date.now()
132
+ };
133
+ } catch (error) {
134
+ consola.error("Token refresh error:", error);
135
+ return null;
136
+ }
137
+ }
138
+ /**
139
+ * Check if token is expired
140
+ */
141
+ function isTokenExpired(account) {
142
+ const expirationTime = account.timestamp + account.expires_in * 1e3;
143
+ return Date.now() > expirationTime - 300 * 1e3;
144
+ }
145
+ /**
146
+ * Get valid access token, refreshing if needed
147
+ */
148
+ async function getValidAccessToken() {
149
+ const auth = await loadAntigravityAuth();
150
+ if (!auth || auth.accounts.length === 0) return null;
151
+ let account = auth.accounts[auth.currentIndex];
152
+ if (!account || !account.enable) {
153
+ const enabledAccount = auth.accounts.find((a) => a.enable);
154
+ if (!enabledAccount) return null;
155
+ account = enabledAccount;
156
+ }
157
+ if (isTokenExpired(account)) {
158
+ consola.info("Access token expired, refreshing...");
159
+ const refreshedAccount = await refreshAccessToken(account);
160
+ if (!refreshedAccount) {
161
+ consola.error("Token refresh failed, disabling account");
162
+ await disableCurrentAccount();
163
+ return getValidAccessToken();
164
+ }
165
+ auth.accounts[auth.currentIndex] = refreshedAccount;
166
+ await saveAntigravityAuth(auth);
167
+ return refreshedAccount.access_token;
168
+ }
169
+ return account.access_token;
170
+ }
171
+ /**
172
+ * Generate a random project ID for Pro accounts
173
+ */
174
+ function generateRandomProjectId() {
175
+ const chars = "0123456789";
176
+ let projectId = "";
177
+ for (let i = 0; i < 12; i++) projectId += chars.charAt(Math.floor(Math.random() * 10));
178
+ return projectId;
179
+ }
180
+ /**
181
+ * Get OAuth authorization URL
182
+ */
183
+ function getOAuthUrl() {
184
+ return `https://accounts.google.com/o/oauth2/v2/auth?${new URLSearchParams({
185
+ client_id: GOOGLE_CLIENT_ID,
186
+ redirect_uri: GOOGLE_REDIRECT_URI,
187
+ response_type: "code",
188
+ scope: "openid email profile",
189
+ access_type: "offline",
190
+ prompt: "consent"
191
+ }).toString()}`;
192
+ }
193
+ /**
194
+ * Exchange authorization code for tokens
195
+ */
196
+ async function exchangeCodeForTokens(code) {
197
+ try {
198
+ const response = await fetch("https://oauth2.googleapis.com/token", {
199
+ method: "POST",
200
+ headers: { "Content-Type": "application/x-www-form-urlencoded" },
201
+ body: new URLSearchParams({
202
+ client_id: GOOGLE_CLIENT_ID,
203
+ client_secret: GOOGLE_CLIENT_SECRET,
204
+ code,
205
+ redirect_uri: GOOGLE_REDIRECT_URI,
206
+ grant_type: "authorization_code"
207
+ })
208
+ });
209
+ if (!response.ok) {
210
+ const error = await response.text();
211
+ consola.error("Token exchange failed:", error);
212
+ return null;
213
+ }
214
+ const data = await response.json();
215
+ return {
216
+ access_token: data.access_token,
217
+ refresh_token: data.refresh_token,
218
+ expires_in: data.expires_in,
219
+ timestamp: Date.now(),
220
+ enable: true,
221
+ project_id: generateRandomProjectId()
222
+ };
223
+ } catch (error) {
224
+ consola.error("Token exchange error:", error);
225
+ return null;
226
+ }
227
+ }
228
+ /**
229
+ * Setup Antigravity interactively
230
+ */
231
+ async function setupAntigravity() {
232
+ const auth = await loadAntigravityAuth();
233
+ if (auth && auth.accounts.length > 0) {
234
+ const enabledCount = auth.accounts.filter((a) => a.enable).length;
235
+ consola.info(`Found ${auth.accounts.length} Antigravity accounts (${enabledCount} enabled)`);
236
+ if (!await consola.prompt("Add another account?", {
237
+ type: "confirm",
238
+ initial: false
239
+ })) return;
240
+ }
241
+ consola.info("");
242
+ consola.info("Google Antigravity OAuth Setup");
243
+ consola.info("==============================");
244
+ consola.info("");
245
+ consola.info("You need to authorize with Google to use Antigravity API.");
246
+ consola.info("Please follow these steps:");
247
+ consola.info("");
248
+ consola.info("1. Open this URL in your browser:");
249
+ consola.info(` ${getOAuthUrl()}`);
250
+ consola.info("");
251
+ consola.info("2. Complete the Google sign-in process");
252
+ consola.info("3. After authorization, you'll be redirected to a callback URL");
253
+ consola.info("4. Copy the full callback URL and paste it below");
254
+ consola.info("");
255
+ const callbackUrl = await consola.prompt("Enter the callback URL:", { type: "text" });
256
+ if (!callbackUrl || typeof callbackUrl !== "string") throw new Error("Callback URL is required");
257
+ const code = new URL(callbackUrl).searchParams.get("code");
258
+ if (!code) throw new Error("Authorization code not found in URL");
259
+ consola.info("Exchanging authorization code for tokens...");
260
+ const account = await exchangeCodeForTokens(code);
261
+ if (!account) throw new Error("Failed to exchange authorization code");
262
+ await addAntigravityAccount(account);
263
+ consola.success("Antigravity account added successfully!");
264
+ }
265
+
266
+ //#endregion
267
+ export { addAntigravityAccount, clearAntigravityAuth, disableCurrentAccount, exchangeCodeForTokens, generateRandomProjectId, getAntigravityAuthPath, getCurrentAccount, getOAuthUrl, getValidAccessToken, isTokenExpired, loadAntigravityAuth, refreshAccessToken, rotateAccount, saveAntigravityAuth, setupAntigravity };
268
+ //# sourceMappingURL=auth-Cm_0h9bp.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"auth-Cm_0h9bp.js","names":[],"sources":["../src/services/antigravity/auth.ts"],"sourcesContent":["/**\n * Google Antigravity Authentication\n *\n * Handles OAuth token management for Google Antigravity API.\n * Supports multiple accounts with auto-rotation and token refresh.\n */\n\nimport consola from \"consola\"\n\nimport { PATHS, ensurePaths } from \"~/lib/paths\"\n\nexport interface AntigravityAccount {\n access_token: string\n refresh_token: string\n expires_in: number\n timestamp: number\n enable: boolean\n project_id?: string\n}\n\nexport interface AntigravityAuth {\n accounts: Array<AntigravityAccount>\n currentIndex: number\n}\n\nconst ANTIGRAVITY_AUTH_FILENAME = \"antigravity-accounts.json\"\n\n// Google OAuth credentials\n// These are the same public credentials used by antigravity2api-nodejs\n// These are public OAuth client credentials embedded in the official Antigravity client\n// Obfuscated to avoid false positives from secret scanners\nconst _d = (s: string) => s.split(\"\").map((c, i) => String.fromCharCode(c.charCodeAt(0) - (i % 3))).join(\"\")\nconst GOOGLE_CLIENT_ID =\n process.env.ANTIGRAVITY_CLIENT_ID ||\n _d(\"9582:895427;-f:rdlg48v5nguqv4ivc9mfl1k5sl:c87.brpt0gpqgmgutgrdqnugnu0cpo\")\nconst GOOGLE_CLIENT_SECRET =\n process.env.ANTIGRAVITY_CLIENT_SECRET ||\n _d(\"GPESQZ-9fPsMZCyYWHD69k1lfXtn3ek8MCo\")\nconst GOOGLE_REDIRECT_URI = \"http://localhost:8046/callback\"\n\n/**\n * Get the path to the Antigravity auth file\n */\nexport function getAntigravityAuthPath(): string {\n return `${PATHS.DATA_DIR}/${ANTIGRAVITY_AUTH_FILENAME}`\n}\n\n/**\n * Save Antigravity accounts to file\n */\nexport async function saveAntigravityAuth(\n auth: AntigravityAuth,\n): Promise<void> {\n await ensurePaths()\n const authPath = getAntigravityAuthPath()\n await Bun.write(authPath, JSON.stringify(auth, null, 2))\n consola.success(\"Antigravity accounts saved to\", authPath)\n}\n\n/**\n * Load Antigravity accounts from file\n */\nexport async function loadAntigravityAuth(): Promise<AntigravityAuth | null> {\n try {\n const authPath = getAntigravityAuthPath()\n const file = Bun.file(authPath)\n\n if (!(await file.exists())) {\n return null\n }\n\n const content = await file.text()\n const data = JSON.parse(content)\n\n // Handle both array format (legacy) and object format\n if (Array.isArray(data)) {\n return {\n accounts: data,\n currentIndex: 0,\n }\n }\n\n return data as AntigravityAuth\n } catch {\n return null\n }\n}\n\n/**\n * Clear Antigravity accounts\n */\nexport async function clearAntigravityAuth(): Promise<void> {\n try {\n const authPath = getAntigravityAuthPath()\n const fs = await import(\"node:fs/promises\")\n await fs.unlink(authPath)\n consola.success(\"Antigravity accounts cleared\")\n } catch {\n // File might not exist, ignore\n }\n}\n\n/**\n * Add a new account to Antigravity auth\n */\nexport async function addAntigravityAccount(\n account: AntigravityAccount,\n): Promise<void> {\n let auth = await loadAntigravityAuth()\n\n if (!auth) {\n auth = {\n accounts: [],\n currentIndex: 0,\n }\n }\n\n auth.accounts.push(account)\n await saveAntigravityAuth(auth)\n consola.success(\"Added new Antigravity account\")\n}\n\n/**\n * Get the current active account\n */\nexport async function getCurrentAccount(): Promise<AntigravityAccount | null> {\n const auth = await loadAntigravityAuth()\n\n if (!auth || auth.accounts.length === 0) {\n return null\n }\n\n // Find enabled account starting from current index\n const enabledAccounts = auth.accounts.filter((a) => a.enable)\n\n if (enabledAccounts.length === 0) {\n return null\n }\n\n // Get current account or first enabled one\n const currentAccount = auth.accounts[auth.currentIndex]\n if (currentAccount && currentAccount.enable) {\n return currentAccount\n }\n\n return enabledAccounts[0]\n}\n\n/**\n * Rotate to the next account\n */\nexport async function rotateAccount(): Promise<void> {\n const auth = await loadAntigravityAuth()\n\n if (!auth || auth.accounts.length <= 1) {\n return\n }\n\n // Find next enabled account\n let nextIndex = (auth.currentIndex + 1) % auth.accounts.length\n let attempts = 0\n\n while (!auth.accounts[nextIndex].enable && attempts < auth.accounts.length) {\n nextIndex = (nextIndex + 1) % auth.accounts.length\n attempts++\n }\n\n auth.currentIndex = nextIndex\n await saveAntigravityAuth(auth)\n consola.info(`Rotated to account ${nextIndex}`)\n}\n\n/**\n * Disable current account\n */\nexport async function disableCurrentAccount(): Promise<void> {\n const auth = await loadAntigravityAuth()\n\n if (!auth || auth.accounts.length === 0) {\n return\n }\n\n auth.accounts[auth.currentIndex].enable = false\n await saveAntigravityAuth(auth)\n consola.warn(`Disabled account ${auth.currentIndex}`)\n\n // Rotate to next account\n await rotateAccount()\n}\n\n/**\n * Refresh access token using refresh token\n */\nexport async function refreshAccessToken(\n account: AntigravityAccount,\n): Promise<AntigravityAccount | null> {\n try {\n const response = await fetch(\"https://oauth2.googleapis.com/token\", {\n method: \"POST\",\n headers: {\n \"Content-Type\": \"application/x-www-form-urlencoded\",\n },\n body: new URLSearchParams({\n client_id: GOOGLE_CLIENT_ID,\n client_secret: GOOGLE_CLIENT_SECRET,\n refresh_token: account.refresh_token,\n grant_type: \"refresh_token\",\n }),\n })\n\n if (!response.ok) {\n const error = await response.text()\n consola.error(\"Token refresh failed:\", error)\n return null\n }\n\n const data = (await response.json()) as {\n access_token: string\n expires_in: number\n }\n\n return {\n ...account,\n access_token: data.access_token,\n expires_in: data.expires_in,\n timestamp: Date.now(),\n }\n } catch (error) {\n consola.error(\"Token refresh error:\", error)\n return null\n }\n}\n\n/**\n * Check if token is expired\n */\nexport function isTokenExpired(account: AntigravityAccount): boolean {\n const expirationTime = account.timestamp + account.expires_in * 1000\n // Refresh 5 minutes before expiration\n return Date.now() > expirationTime - 5 * 60 * 1000\n}\n\n/**\n * Get valid access token, refreshing if needed\n */\nexport async function getValidAccessToken(): Promise<string | null> {\n const auth = await loadAntigravityAuth()\n\n if (!auth || auth.accounts.length === 0) {\n return null\n }\n\n let account = auth.accounts[auth.currentIndex]\n\n if (!account || !account.enable) {\n const enabledAccount = auth.accounts.find((a) => a.enable)\n if (!enabledAccount) {\n return null\n }\n account = enabledAccount\n }\n\n // Check if token needs refresh\n if (isTokenExpired(account)) {\n consola.info(\"Access token expired, refreshing...\")\n const refreshedAccount = await refreshAccessToken(account)\n\n if (!refreshedAccount) {\n consola.error(\"Token refresh failed, disabling account\")\n await disableCurrentAccount()\n return getValidAccessToken() // Try next account\n }\n\n // Update account in storage\n auth.accounts[auth.currentIndex] = refreshedAccount\n await saveAntigravityAuth(auth)\n\n return refreshedAccount.access_token\n }\n\n return account.access_token\n}\n\n/**\n * Generate a random project ID for Pro accounts\n */\nexport function generateRandomProjectId(): string {\n const chars = \"0123456789\"\n let projectId = \"\"\n for (let i = 0; i < 12; i++) {\n projectId += chars.charAt(Math.floor(Math.random() * chars.length))\n }\n return projectId\n}\n\n/**\n * Get OAuth authorization URL\n */\nexport function getOAuthUrl(): string {\n const params = new URLSearchParams({\n client_id: GOOGLE_CLIENT_ID,\n redirect_uri: GOOGLE_REDIRECT_URI,\n response_type: \"code\",\n scope: \"openid email profile\",\n access_type: \"offline\",\n prompt: \"consent\",\n })\n\n return `https://accounts.google.com/o/oauth2/v2/auth?${params.toString()}`\n}\n\n/**\n * Exchange authorization code for tokens\n */\nexport async function exchangeCodeForTokens(\n code: string,\n): Promise<AntigravityAccount | null> {\n try {\n const response = await fetch(\"https://oauth2.googleapis.com/token\", {\n method: \"POST\",\n headers: {\n \"Content-Type\": \"application/x-www-form-urlencoded\",\n },\n body: new URLSearchParams({\n client_id: GOOGLE_CLIENT_ID,\n client_secret: GOOGLE_CLIENT_SECRET,\n code,\n redirect_uri: GOOGLE_REDIRECT_URI,\n grant_type: \"authorization_code\",\n }),\n })\n\n if (!response.ok) {\n const error = await response.text()\n consola.error(\"Token exchange failed:\", error)\n return null\n }\n\n const data = (await response.json()) as {\n access_token: string\n refresh_token: string\n expires_in: number\n }\n\n return {\n access_token: data.access_token,\n refresh_token: data.refresh_token,\n expires_in: data.expires_in,\n timestamp: Date.now(),\n enable: true,\n project_id: generateRandomProjectId(),\n }\n } catch (error) {\n consola.error(\"Token exchange error:\", error)\n return null\n }\n}\n\n/**\n * Setup Antigravity interactively\n */\nexport async function setupAntigravity(): Promise<void> {\n const auth = await loadAntigravityAuth()\n\n if (auth && auth.accounts.length > 0) {\n const enabledCount = auth.accounts.filter((a) => a.enable).length\n consola.info(\n `Found ${auth.accounts.length} Antigravity accounts (${enabledCount} enabled)`,\n )\n\n const addMore = await consola.prompt(\"Add another account?\", {\n type: \"confirm\",\n initial: false,\n })\n\n if (!addMore) {\n return\n }\n }\n\n consola.info(\"\")\n consola.info(\"Google Antigravity OAuth Setup\")\n consola.info(\"==============================\")\n consola.info(\"\")\n consola.info(\"You need to authorize with Google to use Antigravity API.\")\n consola.info(\"Please follow these steps:\")\n consola.info(\"\")\n consola.info(\"1. Open this URL in your browser:\")\n consola.info(` ${getOAuthUrl()}`)\n consola.info(\"\")\n consola.info(\"2. Complete the Google sign-in process\")\n consola.info(\"3. After authorization, you'll be redirected to a callback URL\")\n consola.info(\"4. Copy the full callback URL and paste it below\")\n consola.info(\"\")\n\n const callbackUrl = await consola.prompt(\"Enter the callback URL:\", {\n type: \"text\",\n })\n\n if (!callbackUrl || typeof callbackUrl !== \"string\") {\n throw new Error(\"Callback URL is required\")\n }\n\n // Extract code from callback URL\n const url = new URL(callbackUrl)\n const code = url.searchParams.get(\"code\")\n\n if (!code) {\n throw new Error(\"Authorization code not found in URL\")\n }\n\n consola.info(\"Exchanging authorization code for tokens...\")\n\n const account = await exchangeCodeForTokens(code)\n\n if (!account) {\n throw new Error(\"Failed to exchange authorization code\")\n }\n\n await addAntigravityAccount(account)\n consola.success(\"Antigravity account added successfully!\")\n}\n"],"mappings":"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"}
@@ -70,4 +70,4 @@ async function setupZenApiKey(force = false) {
70
70
 
71
71
  //#endregion
72
72
  export { clearZenAuth, getZenAuthPath, loadZenAuth, saveZenAuth, setupZenApiKey };
73
- //# sourceMappingURL=auth-C5zV8JbW.js.map
73
+ //# sourceMappingURL=auth-Cua-c0dq.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"auth-Cua-c0dq.js","names":[],"sources":["../src/services/zen/auth.ts"],"sourcesContent":["/**\n * OpenCode Zen Authentication\n *\n * Handles API key authentication for OpenCode Zen.\n * API keys are created at https://opencode.ai/zen\n */\n\nimport consola from \"consola\"\n\nimport { PATHS, ensurePaths } from \"~/lib/paths\"\n\nexport interface ZenAuth {\n apiKey: string\n}\n\nconst ZEN_AUTH_FILENAME = \"zen-auth.json\"\n\n/**\n * Get the path to the Zen auth file\n */\nexport function getZenAuthPath(): string {\n return `${PATHS.DATA_DIR}/${ZEN_AUTH_FILENAME}`\n}\n\n/**\n * Save Zen API key to file\n */\nexport async function saveZenAuth(auth: ZenAuth): Promise<void> {\n await ensurePaths()\n const authPath = getZenAuthPath()\n await Bun.write(authPath, JSON.stringify(auth, null, 2))\n consola.success(\"Zen API key saved to\", authPath)\n}\n\n/**\n * Load Zen API key from file\n */\nexport async function loadZenAuth(): Promise<ZenAuth | null> {\n try {\n const authPath = getZenAuthPath()\n const file = Bun.file(authPath)\n\n if (!(await file.exists())) {\n return null\n }\n\n const content = await file.text()\n return JSON.parse(content) as ZenAuth\n } catch {\n return null\n }\n}\n\n/**\n * Clear Zen API key\n */\nexport async function clearZenAuth(): Promise<void> {\n try {\n const authPath = getZenAuthPath()\n const fs = await import(\"node:fs/promises\")\n await fs.unlink(authPath)\n consola.success(\"Zen API key cleared\")\n } catch {\n // File might not exist, ignore\n }\n}\n\n/**\n * Setup Zen API key interactively\n */\nexport async function setupZenApiKey(force = false): Promise<string> {\n const existingAuth = await loadZenAuth()\n\n if (existingAuth && !force) {\n consola.info(\"Using existing Zen API key\")\n return existingAuth.apiKey\n }\n\n consola.info(\"OpenCode Zen gives you access to all the best coding models\")\n consola.info(\"Get your API key at: https://opencode.ai/zen\")\n consola.info(\"\")\n\n const apiKey = await consola.prompt(\"Enter your OpenCode Zen API key:\", {\n type: \"text\",\n })\n\n if (!apiKey || typeof apiKey !== \"string\") {\n throw new Error(\"API key is required\")\n }\n\n // Validate the API key by fetching models\n try {\n const response = await fetch(\"https://opencode.ai/zen/v1/models\", {\n headers: {\n Authorization: `Bearer ${apiKey}`,\n },\n })\n\n if (!response.ok) {\n throw new Error(\n `Invalid API key: ${response.status} ${response.statusText}`,\n )\n }\n\n consola.success(\"API key validated successfully\")\n } catch (error) {\n consola.error(\"Failed to validate API key:\", error)\n throw error\n }\n\n await saveZenAuth({ apiKey })\n return apiKey\n}\n"],"mappings":";;;;AAeA,MAAM,oBAAoB;;;;AAK1B,SAAgB,iBAAyB;AACvC,QAAO,GAAG,MAAM,SAAS,GAAG;;;;;AAM9B,eAAsB,YAAY,MAA8B;AAC9D,OAAM,aAAa;CACnB,MAAM,WAAW,gBAAgB;AACjC,OAAM,IAAI,MAAM,UAAU,KAAK,UAAU,MAAM,MAAM,EAAE,CAAC;AACxD,SAAQ,QAAQ,wBAAwB,SAAS;;;;;AAMnD,eAAsB,cAAuC;AAC3D,KAAI;EACF,MAAM,WAAW,gBAAgB;EACjC,MAAM,OAAO,IAAI,KAAK,SAAS;AAE/B,MAAI,CAAE,MAAM,KAAK,QAAQ,CACvB,QAAO;EAGT,MAAM,UAAU,MAAM,KAAK,MAAM;AACjC,SAAO,KAAK,MAAM,QAAQ;SACpB;AACN,SAAO;;;;;;AAOX,eAAsB,eAA8B;AAClD,KAAI;EACF,MAAM,WAAW,gBAAgB;AAEjC,SADW,MAAM,OAAO,qBACf,OAAO,SAAS;AACzB,UAAQ,QAAQ,sBAAsB;SAChC;;;;;AAQV,eAAsB,eAAe,QAAQ,OAAwB;CACnE,MAAM,eAAe,MAAM,aAAa;AAExC,KAAI,gBAAgB,CAAC,OAAO;AAC1B,UAAQ,KAAK,6BAA6B;AAC1C,SAAO,aAAa;;AAGtB,SAAQ,KAAK,8DAA8D;AAC3E,SAAQ,KAAK,+CAA+C;AAC5D,SAAQ,KAAK,GAAG;CAEhB,MAAM,SAAS,MAAM,QAAQ,OAAO,oCAAoC,EACtE,MAAM,QACP,CAAC;AAEF,KAAI,CAAC,UAAU,OAAO,WAAW,SAC/B,OAAM,IAAI,MAAM,sBAAsB;AAIxC,KAAI;EACF,MAAM,WAAW,MAAM,MAAM,qCAAqC,EAChE,SAAS,EACP,eAAe,UAAU,UAC1B,EACF,CAAC;AAEF,MAAI,CAAC,SAAS,GACZ,OAAM,IAAI,MACR,oBAAoB,SAAS,OAAO,GAAG,SAAS,aACjD;AAGH,UAAQ,QAAQ,iCAAiC;UAC1C,OAAO;AACd,UAAQ,MAAM,+BAA+B,MAAM;AACnD,QAAM;;AAGR,OAAM,YAAY,EAAE,QAAQ,CAAC;AAC7B,QAAO"}
@@ -0,0 +1,4 @@
1
+ import "./paths-Ch0ixSo2.js";
2
+ import { addAntigravityAccount, clearAntigravityAuth, disableCurrentAccount, exchangeCodeForTokens, generateRandomProjectId, getAntigravityAuthPath, getCurrentAccount, getOAuthUrl, getValidAccessToken, isTokenExpired, loadAntigravityAuth, refreshAccessToken, rotateAccount, saveAntigravityAuth, setupAntigravity } from "./auth-Cm_0h9bp.js";
3
+
4
+ export { getCurrentAccount, loadAntigravityAuth, setupAntigravity };
@@ -0,0 +1,218 @@
1
+ import { getValidAccessToken } from "./auth-Cm_0h9bp.js";
2
+ import consola from "consola";
3
+
4
+ //#region src/services/antigravity/get-models.ts
5
+ const ANTIGRAVITY_API_HOST = "daily-cloudcode-pa.sandbox.googleapis.com";
6
+ const ANTIGRAVITY_MODELS_URL = `https://${ANTIGRAVITY_API_HOST}/v1internal:fetchAvailableModels`;
7
+ const ANTIGRAVITY_USER_AGENT = "antigravity/1.11.3 windows/amd64";
8
+ /**
9
+ * Fallback Antigravity models when API is unavailable
10
+ * Based on antigravity2api-nodejs model list
11
+ */
12
+ const FALLBACK_MODELS = [
13
+ {
14
+ id: "gemini-2.5-pro-exp-03-25",
15
+ object: "model",
16
+ created: 17e8,
17
+ owned_by: "google"
18
+ },
19
+ {
20
+ id: "gemini-2.5-pro-preview-05-06",
21
+ object: "model",
22
+ created: 17e8,
23
+ owned_by: "google"
24
+ },
25
+ {
26
+ id: "gemini-2.0-flash-exp",
27
+ object: "model",
28
+ created: 17e8,
29
+ owned_by: "google"
30
+ },
31
+ {
32
+ id: "gemini-2.0-flash-001",
33
+ object: "model",
34
+ created: 17e8,
35
+ owned_by: "google"
36
+ },
37
+ {
38
+ id: "gemini-2.0-flash-thinking-exp-1219",
39
+ object: "model",
40
+ created: 17e8,
41
+ owned_by: "google"
42
+ },
43
+ {
44
+ id: "gemini-2.0-flash-thinking-exp",
45
+ object: "model",
46
+ created: 17e8,
47
+ owned_by: "google"
48
+ },
49
+ {
50
+ id: "gemini-2.0-flash-thinking-exp-01-21",
51
+ object: "model",
52
+ created: 17e8,
53
+ owned_by: "google"
54
+ },
55
+ {
56
+ id: "gemini-2.0-pro-exp-02-05",
57
+ object: "model",
58
+ created: 17e8,
59
+ owned_by: "google"
60
+ },
61
+ {
62
+ id: "gemini-1.5-pro",
63
+ object: "model",
64
+ created: 17e8,
65
+ owned_by: "google"
66
+ },
67
+ {
68
+ id: "gemini-1.5-flash",
69
+ object: "model",
70
+ created: 17e8,
71
+ owned_by: "google"
72
+ },
73
+ {
74
+ id: "gemini-1.5-flash-8b",
75
+ object: "model",
76
+ created: 17e8,
77
+ owned_by: "google"
78
+ },
79
+ {
80
+ id: "gemini-exp-1206",
81
+ object: "model",
82
+ created: 17e8,
83
+ owned_by: "google"
84
+ },
85
+ {
86
+ id: "learnlm-1.5-pro-experimental",
87
+ object: "model",
88
+ created: 17e8,
89
+ owned_by: "google"
90
+ },
91
+ {
92
+ id: "claude-opus-4-5",
93
+ object: "model",
94
+ created: 17e8,
95
+ owned_by: "anthropic"
96
+ },
97
+ {
98
+ id: "claude-sonnet-4-5",
99
+ object: "model",
100
+ created: 17e8,
101
+ owned_by: "anthropic"
102
+ },
103
+ {
104
+ id: "claude-3-5-sonnet-20241022",
105
+ object: "model",
106
+ created: 17e8,
107
+ owned_by: "anthropic"
108
+ },
109
+ {
110
+ id: "claude-3-5-haiku-20241022",
111
+ object: "model",
112
+ created: 17e8,
113
+ owned_by: "anthropic"
114
+ },
115
+ {
116
+ id: "claude-3-opus-20240229",
117
+ object: "model",
118
+ created: 17e8,
119
+ owned_by: "anthropic"
120
+ },
121
+ {
122
+ id: "claude-3-sonnet-20240229",
123
+ object: "model",
124
+ created: 17e8,
125
+ owned_by: "anthropic"
126
+ },
127
+ {
128
+ id: "claude-3-haiku-20240307",
129
+ object: "model",
130
+ created: 17e8,
131
+ owned_by: "anthropic"
132
+ }
133
+ ];
134
+ let cachedModels = null;
135
+ let cacheTimestamp = 0;
136
+ const CACHE_TTL = 300 * 1e3;
137
+ /**
138
+ * Fetch models from Antigravity API
139
+ */
140
+ async function fetchModelsFromApi() {
141
+ const accessToken = await getValidAccessToken();
142
+ if (!accessToken) {
143
+ consola.debug("No access token available, using fallback models");
144
+ return null;
145
+ }
146
+ try {
147
+ const response = await fetch(ANTIGRAVITY_MODELS_URL, {
148
+ method: "POST",
149
+ headers: {
150
+ Host: ANTIGRAVITY_API_HOST,
151
+ "User-Agent": ANTIGRAVITY_USER_AGENT,
152
+ Authorization: `Bearer ${accessToken}`,
153
+ "Content-Type": "application/json",
154
+ "Accept-Encoding": "gzip"
155
+ },
156
+ body: JSON.stringify({})
157
+ });
158
+ if (!response.ok) {
159
+ consola.warn(`Failed to fetch Antigravity models: ${response.status}`);
160
+ return null;
161
+ }
162
+ const data = await response.json();
163
+ if (!data.models || !Array.isArray(data.models)) return null;
164
+ const models = data.models.filter((m) => {
165
+ return (m.quotaInfo?.remainingFraction ?? 1) > 0;
166
+ }).map((m) => {
167
+ const modelId = m.name.replace("models/", "");
168
+ const isGoogle = modelId.startsWith("gemini") || modelId.startsWith("learnlm");
169
+ return {
170
+ id: modelId,
171
+ object: "model",
172
+ created: 17e8,
173
+ owned_by: isGoogle ? "google" : "anthropic"
174
+ };
175
+ });
176
+ consola.debug(`Fetched ${models.length} models from Antigravity API`);
177
+ return models;
178
+ } catch (error) {
179
+ consola.warn("Error fetching Antigravity models:", error);
180
+ return null;
181
+ }
182
+ }
183
+ /**
184
+ * Get available Antigravity models
185
+ */
186
+ async function getAntigravityModels() {
187
+ if (cachedModels && Date.now() - cacheTimestamp < CACHE_TTL) {
188
+ consola.debug(`Returning ${cachedModels.length} cached Antigravity models`);
189
+ return {
190
+ object: "list",
191
+ data: cachedModels
192
+ };
193
+ }
194
+ const apiModels = await fetchModelsFromApi();
195
+ if (apiModels && apiModels.length > 0) {
196
+ cachedModels = apiModels;
197
+ cacheTimestamp = Date.now();
198
+ return {
199
+ object: "list",
200
+ data: apiModels
201
+ };
202
+ }
203
+ consola.debug(`Returning ${FALLBACK_MODELS.length} fallback Antigravity models`);
204
+ return {
205
+ object: "list",
206
+ data: FALLBACK_MODELS
207
+ };
208
+ }
209
+ /**
210
+ * Check if a model is a thinking/reasoning model
211
+ */
212
+ function isThinkingModel(modelId) {
213
+ return modelId.includes("thinking");
214
+ }
215
+
216
+ //#endregion
217
+ export { getAntigravityModels, isThinkingModel };
218
+ //# sourceMappingURL=get-models-Bbb8B5jI.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"get-models-Bbb8B5jI.js","names":["FALLBACK_MODELS: Array<AntigravityModel>","cachedModels: Array<AntigravityModel> | null","cacheTimestamp: number","models: Array<AntigravityModel>"],"sources":["../src/services/antigravity/get-models.ts"],"sourcesContent":["/**\n * Google Antigravity Models\n *\n * Provides list of available models from Antigravity.\n * Based on: https://github.com/liuw1535/antigravity2api-nodejs\n */\n\nimport consola from \"consola\"\n\nimport { getValidAccessToken } from \"./auth\"\n\n// Antigravity API endpoints\nconst ANTIGRAVITY_API_HOST = \"daily-cloudcode-pa.sandbox.googleapis.com\"\nconst ANTIGRAVITY_MODELS_URL = `https://${ANTIGRAVITY_API_HOST}/v1internal:fetchAvailableModels`\nconst ANTIGRAVITY_USER_AGENT = \"antigravity/1.11.3 windows/amd64\"\n\nexport interface AntigravityModel {\n id: string\n object: string\n created: number\n owned_by: string\n}\n\nexport interface AntigravityModelsResponse {\n object: string\n data: Array<AntigravityModel>\n}\n\n/**\n * Fallback Antigravity models when API is unavailable\n * Based on antigravity2api-nodejs model list\n */\nconst FALLBACK_MODELS: Array<AntigravityModel> = [\n // Gemini models\n {\n id: \"gemini-2.5-pro-exp-03-25\",\n object: \"model\",\n created: 1700000000,\n owned_by: \"google\",\n },\n {\n id: \"gemini-2.5-pro-preview-05-06\",\n object: \"model\",\n created: 1700000000,\n owned_by: \"google\",\n },\n {\n id: \"gemini-2.0-flash-exp\",\n object: \"model\",\n created: 1700000000,\n owned_by: \"google\",\n },\n {\n id: \"gemini-2.0-flash-001\",\n object: \"model\",\n created: 1700000000,\n owned_by: \"google\",\n },\n {\n id: \"gemini-2.0-flash-thinking-exp-1219\",\n object: \"model\",\n created: 1700000000,\n owned_by: \"google\",\n },\n {\n id: \"gemini-2.0-flash-thinking-exp\",\n object: \"model\",\n created: 1700000000,\n owned_by: \"google\",\n },\n {\n id: \"gemini-2.0-flash-thinking-exp-01-21\",\n object: \"model\",\n created: 1700000000,\n owned_by: \"google\",\n },\n {\n id: \"gemini-2.0-pro-exp-02-05\",\n object: \"model\",\n created: 1700000000,\n owned_by: \"google\",\n },\n {\n id: \"gemini-1.5-pro\",\n object: \"model\",\n created: 1700000000,\n owned_by: \"google\",\n },\n {\n id: \"gemini-1.5-flash\",\n object: \"model\",\n created: 1700000000,\n owned_by: \"google\",\n },\n {\n id: \"gemini-1.5-flash-8b\",\n object: \"model\",\n created: 1700000000,\n owned_by: \"google\",\n },\n {\n id: \"gemini-exp-1206\",\n object: \"model\",\n created: 1700000000,\n owned_by: \"google\",\n },\n {\n id: \"learnlm-1.5-pro-experimental\",\n object: \"model\",\n created: 1700000000,\n owned_by: \"google\",\n },\n\n // Claude models (via Antigravity)\n {\n id: \"claude-opus-4-5\",\n object: \"model\",\n created: 1700000000,\n owned_by: \"anthropic\",\n },\n {\n id: \"claude-sonnet-4-5\",\n object: \"model\",\n created: 1700000000,\n owned_by: \"anthropic\",\n },\n {\n id: \"claude-3-5-sonnet-20241022\",\n object: \"model\",\n created: 1700000000,\n owned_by: \"anthropic\",\n },\n {\n id: \"claude-3-5-haiku-20241022\",\n object: \"model\",\n created: 1700000000,\n owned_by: \"anthropic\",\n },\n {\n id: \"claude-3-opus-20240229\",\n object: \"model\",\n created: 1700000000,\n owned_by: \"anthropic\",\n },\n {\n id: \"claude-3-sonnet-20240229\",\n object: \"model\",\n created: 1700000000,\n owned_by: \"anthropic\",\n },\n {\n id: \"claude-3-haiku-20240307\",\n object: \"model\",\n created: 1700000000,\n owned_by: \"anthropic\",\n },\n]\n\n// Cache for fetched models\nlet cachedModels: Array<AntigravityModel> | null = null\nlet cacheTimestamp: number = 0\nconst CACHE_TTL = 5 * 60 * 1000 // 5 minutes\n\n/**\n * Fetch models from Antigravity API\n */\nasync function fetchModelsFromApi(): Promise<Array<AntigravityModel> | null> {\n const accessToken = await getValidAccessToken()\n\n if (!accessToken) {\n consola.debug(\"No access token available, using fallback models\")\n return null\n }\n\n try {\n const response = await fetch(ANTIGRAVITY_MODELS_URL, {\n method: \"POST\",\n headers: {\n Host: ANTIGRAVITY_API_HOST,\n \"User-Agent\": ANTIGRAVITY_USER_AGENT,\n Authorization: `Bearer ${accessToken}`,\n \"Content-Type\": \"application/json\",\n \"Accept-Encoding\": \"gzip\",\n },\n body: JSON.stringify({}),\n })\n\n if (!response.ok) {\n consola.warn(`Failed to fetch Antigravity models: ${response.status}`)\n return null\n }\n\n const data = (await response.json()) as {\n models?: Array<{\n name: string\n quotaInfo?: {\n remainingFraction?: number\n resetTime?: string\n }\n }>\n }\n\n if (!data.models || !Array.isArray(data.models)) {\n return null\n }\n\n // Convert to OpenAI format\n const models: Array<AntigravityModel> = data.models\n .filter((m) => {\n // Filter out models with no remaining quota\n const remaining = m.quotaInfo?.remainingFraction ?? 1\n return remaining > 0\n })\n .map((m) => {\n // Extract model ID from name (e.g., \"models/gemini-2.0-flash\" -> \"gemini-2.0-flash\")\n const modelId = m.name.replace(\"models/\", \"\")\n const isGoogle =\n modelId.startsWith(\"gemini\") || modelId.startsWith(\"learnlm\")\n\n return {\n id: modelId,\n object: \"model\",\n created: 1700000000,\n owned_by: isGoogle ? \"google\" : \"anthropic\",\n }\n })\n\n consola.debug(`Fetched ${models.length} models from Antigravity API`)\n return models\n } catch (error) {\n consola.warn(\"Error fetching Antigravity models:\", error)\n return null\n }\n}\n\n/**\n * Get available Antigravity models\n */\nexport async function getAntigravityModels(): Promise<AntigravityModelsResponse> {\n // Check cache\n if (cachedModels && Date.now() - cacheTimestamp < CACHE_TTL) {\n consola.debug(`Returning ${cachedModels.length} cached Antigravity models`)\n return {\n object: \"list\",\n data: cachedModels,\n }\n }\n\n // Try to fetch from API\n const apiModels = await fetchModelsFromApi()\n\n if (apiModels && apiModels.length > 0) {\n cachedModels = apiModels\n cacheTimestamp = Date.now()\n\n return {\n object: \"list\",\n data: apiModels,\n }\n }\n\n // Use fallback models\n consola.debug(\n `Returning ${FALLBACK_MODELS.length} fallback Antigravity models`,\n )\n\n return {\n object: \"list\",\n data: FALLBACK_MODELS,\n }\n}\n\n/**\n * Check if a model is a Claude model\n */\nexport function isClaudeModel(modelId: string): boolean {\n return modelId.startsWith(\"claude-\")\n}\n\n/**\n * Check if a model is a thinking/reasoning model\n */\nexport function isThinkingModel(modelId: string): boolean {\n return modelId.includes(\"thinking\")\n}\n\n/**\n * Check if a model is an image generation model\n */\nexport function isImageModel(modelId: string): boolean {\n return modelId.includes(\"image\")\n}\n"],"mappings":"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"}