@bloxystudios/bloxycode 1.0.0

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.
Files changed (344) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +256 -0
  3. package/bin/bloxycode +84 -0
  4. package/package.json +133 -0
  5. package/src/acp/README.md +164 -0
  6. package/src/acp/agent.ts +1437 -0
  7. package/src/acp/session.ts +105 -0
  8. package/src/acp/types.ts +22 -0
  9. package/src/agent/agent.ts +356 -0
  10. package/src/agent/generate.txt +75 -0
  11. package/src/agent/prompt/bloxy.txt +46 -0
  12. package/src/agent/prompt/compaction.txt +12 -0
  13. package/src/agent/prompt/explore.txt +18 -0
  14. package/src/agent/prompt/summary.txt +11 -0
  15. package/src/agent/prompt/title.txt +44 -0
  16. package/src/auth/index.ts +73 -0
  17. package/src/bloxy/event.ts +41 -0
  18. package/src/bloxy/index.ts +5 -0
  19. package/src/bloxy/parser.ts +263 -0
  20. package/src/bloxy/prompt.ts +121 -0
  21. package/src/bloxy/runner.ts +193 -0
  22. package/src/bloxy/state.ts +246 -0
  23. package/src/bun/index.ts +134 -0
  24. package/src/bus/bus-event.ts +43 -0
  25. package/src/bus/global.ts +10 -0
  26. package/src/bus/index.ts +105 -0
  27. package/src/cli/bootstrap.ts +17 -0
  28. package/src/cli/cmd/acp.ts +69 -0
  29. package/src/cli/cmd/agent.ts +257 -0
  30. package/src/cli/cmd/auth.ts +400 -0
  31. package/src/cli/cmd/cmd.ts +7 -0
  32. package/src/cli/cmd/debug/agent.ts +167 -0
  33. package/src/cli/cmd/debug/config.ts +16 -0
  34. package/src/cli/cmd/debug/file.ts +97 -0
  35. package/src/cli/cmd/debug/index.ts +48 -0
  36. package/src/cli/cmd/debug/lsp.ts +52 -0
  37. package/src/cli/cmd/debug/ripgrep.ts +87 -0
  38. package/src/cli/cmd/debug/scrap.ts +16 -0
  39. package/src/cli/cmd/debug/skill.ts +16 -0
  40. package/src/cli/cmd/debug/snapshot.ts +52 -0
  41. package/src/cli/cmd/export.ts +88 -0
  42. package/src/cli/cmd/generate.ts +38 -0
  43. package/src/cli/cmd/github.ts +1548 -0
  44. package/src/cli/cmd/import.ts +98 -0
  45. package/src/cli/cmd/mcp.ts +755 -0
  46. package/src/cli/cmd/models.ts +77 -0
  47. package/src/cli/cmd/pr.ts +112 -0
  48. package/src/cli/cmd/run.ts +395 -0
  49. package/src/cli/cmd/serve.ts +20 -0
  50. package/src/cli/cmd/session.ts +135 -0
  51. package/src/cli/cmd/stats.ts +402 -0
  52. package/src/cli/cmd/tui/app.tsx +771 -0
  53. package/src/cli/cmd/tui/attach.ts +39 -0
  54. package/src/cli/cmd/tui/component/border.tsx +21 -0
  55. package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
  56. package/src/cli/cmd/tui/component/dialog-command.tsx +148 -0
  57. package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
  58. package/src/cli/cmd/tui/component/dialog-model.tsx +234 -0
  59. package/src/cli/cmd/tui/component/dialog-provider.tsx +256 -0
  60. package/src/cli/cmd/tui/component/dialog-session-list.tsx +114 -0
  61. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
  62. package/src/cli/cmd/tui/component/dialog-stash.tsx +87 -0
  63. package/src/cli/cmd/tui/component/dialog-status.tsx +164 -0
  64. package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
  65. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
  66. package/src/cli/cmd/tui/component/logo.tsx +102 -0
  67. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +653 -0
  68. package/src/cli/cmd/tui/component/prompt/frecency.tsx +89 -0
  69. package/src/cli/cmd/tui/component/prompt/history.tsx +108 -0
  70. package/src/cli/cmd/tui/component/prompt/index.tsx +1138 -0
  71. package/src/cli/cmd/tui/component/prompt/stash.tsx +101 -0
  72. package/src/cli/cmd/tui/component/textarea-keybindings.ts +73 -0
  73. package/src/cli/cmd/tui/component/tips.tsx +153 -0
  74. package/src/cli/cmd/tui/component/todo-item.tsx +32 -0
  75. package/src/cli/cmd/tui/context/args.tsx +14 -0
  76. package/src/cli/cmd/tui/context/directory.ts +13 -0
  77. package/src/cli/cmd/tui/context/exit.tsx +23 -0
  78. package/src/cli/cmd/tui/context/helper.tsx +25 -0
  79. package/src/cli/cmd/tui/context/keybind.tsx +101 -0
  80. package/src/cli/cmd/tui/context/kv.tsx +52 -0
  81. package/src/cli/cmd/tui/context/local.tsx +402 -0
  82. package/src/cli/cmd/tui/context/prompt.tsx +18 -0
  83. package/src/cli/cmd/tui/context/route.tsx +46 -0
  84. package/src/cli/cmd/tui/context/sdk.tsx +94 -0
  85. package/src/cli/cmd/tui/context/sync.tsx +470 -0
  86. package/src/cli/cmd/tui/context/theme/aura.json +69 -0
  87. package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
  88. package/src/cli/cmd/tui/context/theme/bloxycode.json +245 -0
  89. package/src/cli/cmd/tui/context/theme/carbonfox.json +248 -0
  90. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +233 -0
  91. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
  92. package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
  93. package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
  94. package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
  95. package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
  96. package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
  97. package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
  98. package/src/cli/cmd/tui/context/theme/github.json +233 -0
  99. package/src/cli/cmd/tui/context/theme/gruvbox.json +242 -0
  100. package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
  101. package/src/cli/cmd/tui/context/theme/lucent-orng.json +237 -0
  102. package/src/cli/cmd/tui/context/theme/material.json +235 -0
  103. package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
  104. package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
  105. package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
  106. package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
  107. package/src/cli/cmd/tui/context/theme/nord.json +223 -0
  108. package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
  109. package/src/cli/cmd/tui/context/theme/orng.json +249 -0
  110. package/src/cli/cmd/tui/context/theme/osaka-jade.json +93 -0
  111. package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
  112. package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
  113. package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
  114. package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
  115. package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
  116. package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
  117. package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
  118. package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
  119. package/src/cli/cmd/tui/context/theme.tsx +1152 -0
  120. package/src/cli/cmd/tui/event.ts +48 -0
  121. package/src/cli/cmd/tui/routes/home.tsx +140 -0
  122. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +64 -0
  123. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +109 -0
  124. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +26 -0
  125. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +47 -0
  126. package/src/cli/cmd/tui/routes/session/footer.tsx +91 -0
  127. package/src/cli/cmd/tui/routes/session/header.tsx +142 -0
  128. package/src/cli/cmd/tui/routes/session/index.tsx +2048 -0
  129. package/src/cli/cmd/tui/routes/session/permission.tsx +508 -0
  130. package/src/cli/cmd/tui/routes/session/question.tsx +453 -0
  131. package/src/cli/cmd/tui/routes/session/sidebar.tsx +313 -0
  132. package/src/cli/cmd/tui/thread.ts +165 -0
  133. package/src/cli/cmd/tui/ui/dialog-alert.tsx +57 -0
  134. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +83 -0
  135. package/src/cli/cmd/tui/ui/dialog-export-options.tsx +204 -0
  136. package/src/cli/cmd/tui/ui/dialog-help.tsx +38 -0
  137. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +77 -0
  138. package/src/cli/cmd/tui/ui/dialog-select.tsx +385 -0
  139. package/src/cli/cmd/tui/ui/dialog.tsx +167 -0
  140. package/src/cli/cmd/tui/ui/link.tsx +28 -0
  141. package/src/cli/cmd/tui/ui/spinner.ts +368 -0
  142. package/src/cli/cmd/tui/ui/toast.tsx +100 -0
  143. package/src/cli/cmd/tui/util/clipboard.ts +160 -0
  144. package/src/cli/cmd/tui/util/editor.ts +32 -0
  145. package/src/cli/cmd/tui/util/signal.ts +7 -0
  146. package/src/cli/cmd/tui/util/terminal.ts +114 -0
  147. package/src/cli/cmd/tui/util/transcript.ts +98 -0
  148. package/src/cli/cmd/tui/worker.ts +152 -0
  149. package/src/cli/cmd/uninstall.ts +357 -0
  150. package/src/cli/cmd/upgrade.ts +73 -0
  151. package/src/cli/cmd/web.ts +81 -0
  152. package/src/cli/error.ts +57 -0
  153. package/src/cli/network.ts +53 -0
  154. package/src/cli/ui.ts +86 -0
  155. package/src/cli/upgrade.ts +25 -0
  156. package/src/command/index.ts +173 -0
  157. package/src/command/template/bloxy-resume.txt +15 -0
  158. package/src/command/template/bloxy-status.txt +25 -0
  159. package/src/command/template/bloxy-validate.txt +22 -0
  160. package/src/command/template/bloxy.txt +14 -0
  161. package/src/command/template/initialize.txt +10 -0
  162. package/src/command/template/review.txt +99 -0
  163. package/src/config/config.ts +1367 -0
  164. package/src/config/markdown.ts +93 -0
  165. package/src/env/index.ts +26 -0
  166. package/src/file/ignore.ts +83 -0
  167. package/src/file/index.ts +415 -0
  168. package/src/file/ripgrep.ts +407 -0
  169. package/src/file/time.ts +69 -0
  170. package/src/file/watcher.ts +127 -0
  171. package/src/flag/flag.ts +79 -0
  172. package/src/format/formatter.ts +357 -0
  173. package/src/format/index.ts +137 -0
  174. package/src/global/index.ts +55 -0
  175. package/src/id/id.ts +83 -0
  176. package/src/ide/index.ts +76 -0
  177. package/src/index.ts +159 -0
  178. package/src/installation/index.ts +246 -0
  179. package/src/lsp/client.ts +252 -0
  180. package/src/lsp/index.ts +485 -0
  181. package/src/lsp/language.ts +119 -0
  182. package/src/lsp/server.ts +2046 -0
  183. package/src/mcp/auth.ts +135 -0
  184. package/src/mcp/index.ts +934 -0
  185. package/src/mcp/oauth-callback.ts +200 -0
  186. package/src/mcp/oauth-provider.ts +154 -0
  187. package/src/patch/index.ts +680 -0
  188. package/src/permission/arity.ts +163 -0
  189. package/src/permission/index.ts +210 -0
  190. package/src/permission/next.ts +280 -0
  191. package/src/plugin/antigravity.ts +378 -0
  192. package/src/plugin/codex.ts +506 -0
  193. package/src/plugin/copilot.ts +298 -0
  194. package/src/plugin/index.ts +136 -0
  195. package/src/project/bootstrap.ts +35 -0
  196. package/src/project/instance.ts +91 -0
  197. package/src/project/project.ts +371 -0
  198. package/src/project/state.ts +66 -0
  199. package/src/project/vcs.ts +76 -0
  200. package/src/provider/auth.ts +147 -0
  201. package/src/provider/models-snapshot.ts +2 -0
  202. package/src/provider/models.ts +133 -0
  203. package/src/provider/provider.ts +1241 -0
  204. package/src/provider/sdk/openai-compatible/src/README.md +5 -0
  205. package/src/provider/sdk/openai-compatible/src/index.ts +2 -0
  206. package/src/provider/sdk/openai-compatible/src/openai-compatible-provider.ts +100 -0
  207. package/src/provider/sdk/openai-compatible/src/responses/convert-to-openai-responses-input.ts +303 -0
  208. package/src/provider/sdk/openai-compatible/src/responses/map-openai-responses-finish-reason.ts +22 -0
  209. package/src/provider/sdk/openai-compatible/src/responses/openai-config.ts +18 -0
  210. package/src/provider/sdk/openai-compatible/src/responses/openai-error.ts +22 -0
  211. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-api-types.ts +207 -0
  212. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-language-model.ts +1732 -0
  213. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-prepare-tools.ts +177 -0
  214. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-settings.ts +1 -0
  215. package/src/provider/sdk/openai-compatible/src/responses/tool/code-interpreter.ts +88 -0
  216. package/src/provider/sdk/openai-compatible/src/responses/tool/file-search.ts +128 -0
  217. package/src/provider/sdk/openai-compatible/src/responses/tool/image-generation.ts +115 -0
  218. package/src/provider/sdk/openai-compatible/src/responses/tool/local-shell.ts +65 -0
  219. package/src/provider/sdk/openai-compatible/src/responses/tool/web-search-preview.ts +104 -0
  220. package/src/provider/sdk/openai-compatible/src/responses/tool/web-search.ts +103 -0
  221. package/src/provider/transform.ts +741 -0
  222. package/src/pty/index.ts +241 -0
  223. package/src/question/index.ts +171 -0
  224. package/src/scheduler/index.ts +61 -0
  225. package/src/server/error.ts +36 -0
  226. package/src/server/event.ts +7 -0
  227. package/src/server/mdns.ts +59 -0
  228. package/src/server/routes/config.ts +92 -0
  229. package/src/server/routes/experimental.ts +208 -0
  230. package/src/server/routes/file.ts +197 -0
  231. package/src/server/routes/global.ts +135 -0
  232. package/src/server/routes/mcp.ts +225 -0
  233. package/src/server/routes/permission.ts +68 -0
  234. package/src/server/routes/project.ts +82 -0
  235. package/src/server/routes/provider.ts +165 -0
  236. package/src/server/routes/pty.ts +169 -0
  237. package/src/server/routes/question.ts +98 -0
  238. package/src/server/routes/session.ts +939 -0
  239. package/src/server/routes/tui.ts +379 -0
  240. package/src/server/server.ts +604 -0
  241. package/src/session/compaction.ts +225 -0
  242. package/src/session/fallback.ts +246 -0
  243. package/src/session/index.ts +498 -0
  244. package/src/session/instruction.ts +164 -0
  245. package/src/session/llm.ts +298 -0
  246. package/src/session/message-v2.ts +747 -0
  247. package/src/session/message.ts +189 -0
  248. package/src/session/processor.ts +450 -0
  249. package/src/session/prompt/anthropic-20250930.txt +166 -0
  250. package/src/session/prompt/anthropic.txt +105 -0
  251. package/src/session/prompt/beast.txt +147 -0
  252. package/src/session/prompt/build-switch.txt +5 -0
  253. package/src/session/prompt/codex_header.txt +79 -0
  254. package/src/session/prompt/copilot-gpt-5.txt +143 -0
  255. package/src/session/prompt/gemini.txt +155 -0
  256. package/src/session/prompt/max-steps.txt +16 -0
  257. package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
  258. package/src/session/prompt/plan.txt +26 -0
  259. package/src/session/prompt/qwen.txt +109 -0
  260. package/src/session/prompt.ts +1822 -0
  261. package/src/session/retry.ts +99 -0
  262. package/src/session/revert.ts +121 -0
  263. package/src/session/status.ts +100 -0
  264. package/src/session/summary.ts +217 -0
  265. package/src/session/system.ts +52 -0
  266. package/src/session/todo.ts +37 -0
  267. package/src/share/share-next.ts +200 -0
  268. package/src/share/share.ts +92 -0
  269. package/src/shell/shell.ts +67 -0
  270. package/src/skill/index.ts +1 -0
  271. package/src/skill/skill.ts +135 -0
  272. package/src/snapshot/index.ts +236 -0
  273. package/src/storage/storage.ts +227 -0
  274. package/src/tool/apply_patch.ts +281 -0
  275. package/src/tool/apply_patch.txt +33 -0
  276. package/src/tool/bash.ts +258 -0
  277. package/src/tool/bash.txt +115 -0
  278. package/src/tool/batch.ts +175 -0
  279. package/src/tool/batch.txt +24 -0
  280. package/src/tool/bloxy-control.ts +123 -0
  281. package/src/tool/bloxy-control.txt +13 -0
  282. package/src/tool/codesearch.ts +132 -0
  283. package/src/tool/codesearch.txt +12 -0
  284. package/src/tool/edit.ts +655 -0
  285. package/src/tool/edit.txt +10 -0
  286. package/src/tool/external-directory.ts +32 -0
  287. package/src/tool/glob.ts +77 -0
  288. package/src/tool/glob.txt +6 -0
  289. package/src/tool/grep.ts +154 -0
  290. package/src/tool/grep.txt +8 -0
  291. package/src/tool/invalid.ts +17 -0
  292. package/src/tool/ls.ts +121 -0
  293. package/src/tool/ls.txt +1 -0
  294. package/src/tool/lsp.ts +96 -0
  295. package/src/tool/lsp.txt +19 -0
  296. package/src/tool/multiedit.ts +46 -0
  297. package/src/tool/multiedit.txt +41 -0
  298. package/src/tool/plan-enter.txt +14 -0
  299. package/src/tool/plan-exit.txt +13 -0
  300. package/src/tool/plan.ts +130 -0
  301. package/src/tool/question.ts +33 -0
  302. package/src/tool/question.txt +10 -0
  303. package/src/tool/read.ts +211 -0
  304. package/src/tool/read.txt +12 -0
  305. package/src/tool/registry.ts +161 -0
  306. package/src/tool/skill.ts +82 -0
  307. package/src/tool/task.ts +191 -0
  308. package/src/tool/task.txt +60 -0
  309. package/src/tool/todo.ts +53 -0
  310. package/src/tool/todoread.txt +14 -0
  311. package/src/tool/todowrite.txt +167 -0
  312. package/src/tool/tool.ts +89 -0
  313. package/src/tool/truncation.ts +106 -0
  314. package/src/tool/webfetch.ts +188 -0
  315. package/src/tool/webfetch.txt +13 -0
  316. package/src/tool/websearch.ts +150 -0
  317. package/src/tool/websearch.txt +14 -0
  318. package/src/tool/write.ts +85 -0
  319. package/src/tool/write.txt +8 -0
  320. package/src/util/archive.ts +16 -0
  321. package/src/util/binary.ts +41 -0
  322. package/src/util/color.ts +19 -0
  323. package/src/util/context.ts +25 -0
  324. package/src/util/defer.ts +12 -0
  325. package/src/util/error.ts +54 -0
  326. package/src/util/eventloop.ts +20 -0
  327. package/src/util/filesystem.ts +93 -0
  328. package/src/util/fn.ts +11 -0
  329. package/src/util/format.ts +20 -0
  330. package/src/util/iife.ts +3 -0
  331. package/src/util/keybind.ts +103 -0
  332. package/src/util/lazy.ts +23 -0
  333. package/src/util/locale.ts +81 -0
  334. package/src/util/lock.ts +98 -0
  335. package/src/util/log.ts +180 -0
  336. package/src/util/queue.ts +32 -0
  337. package/src/util/rpc.ts +66 -0
  338. package/src/util/scrap.ts +10 -0
  339. package/src/util/signal.ts +12 -0
  340. package/src/util/slug.ts +74 -0
  341. package/src/util/timeout.ts +14 -0
  342. package/src/util/token.ts +7 -0
  343. package/src/util/wildcard.ts +56 -0
  344. package/src/worktree/index.ts +549 -0
@@ -0,0 +1,378 @@
1
+ import type { Hooks, PluginInput } from "@opencode-ai/plugin"
2
+ import { Installation } from "@/installation"
3
+ import { Global } from "@/global"
4
+ import path from "path"
5
+ import fs from "fs/promises"
6
+ import z from "zod"
7
+
8
+ // Google OAuth configuration for Antigravity
9
+ const GOOGLE_CLIENT_ID = "77185425430.apps.googleusercontent.com"
10
+ const GOOGLE_CLIENT_SECRET = "OTJgUOQcT7lO7GsGZq2G4IlT"
11
+ const GOOGLE_DEVICE_AUTH_URL = "https://oauth2.googleapis.com/device/code"
12
+ const GOOGLE_TOKEN_URL = "https://oauth2.googleapis.com/token"
13
+ const ANTIGRAVITY_API_BASE = "https://autopush-aicompanion-pa.sandbox.googleapis.com"
14
+
15
+ // Account storage schema
16
+ const AntigravityAccountSchema = z.object({
17
+ email: z.string(),
18
+ refreshToken: z.string(),
19
+ accessToken: z.string().optional(),
20
+ accessTokenExpiry: z.number().optional(),
21
+ projectId: z.string().optional(),
22
+ rateLimitedUntil: z.number().optional(),
23
+ enabled: z.boolean().default(true),
24
+ })
25
+ type AntigravityAccount = z.infer<typeof AntigravityAccountSchema>
26
+
27
+ const AntigravityAccountsSchema = z.object({
28
+ accounts: z.array(AntigravityAccountSchema),
29
+ currentIndex: z.number().default(0),
30
+ })
31
+ type AntigravityAccountsData = z.infer<typeof AntigravityAccountsSchema>
32
+
33
+ // Accounts file path
34
+ const accountsFilePath = path.join(Global.Path.config, "antigravity-accounts.json")
35
+
36
+ // Account management
37
+ async function loadAccounts(): Promise<AntigravityAccountsData> {
38
+ try {
39
+ const file = Bun.file(accountsFilePath)
40
+ if (await file.exists()) {
41
+ const data = await file.json()
42
+ return AntigravityAccountsSchema.parse(data)
43
+ }
44
+ } catch {}
45
+ return { accounts: [], currentIndex: 0 }
46
+ }
47
+
48
+ async function saveAccounts(accounts: AntigravityAccountsData): Promise<void> {
49
+ await Bun.write(accountsFilePath, JSON.stringify(accounts, null, 2))
50
+ await fs.chmod(accountsFilePath, 0o600)
51
+ }
52
+
53
+ async function refreshAccessToken(account: AntigravityAccount): Promise<string | null> {
54
+ try {
55
+ const response = await fetch(GOOGLE_TOKEN_URL, {
56
+ method: "POST",
57
+ headers: {
58
+ "Content-Type": "application/x-www-form-urlencoded",
59
+ },
60
+ body: new URLSearchParams({
61
+ client_id: GOOGLE_CLIENT_ID,
62
+ client_secret: GOOGLE_CLIENT_SECRET,
63
+ refresh_token: account.refreshToken,
64
+ grant_type: "refresh_token",
65
+ }),
66
+ })
67
+
68
+ if (!response.ok) return null
69
+
70
+ const data = (await response.json()) as {
71
+ access_token: string
72
+ expires_in: number
73
+ }
74
+
75
+ account.accessToken = data.access_token
76
+ account.accessTokenExpiry = Date.now() + data.expires_in * 1000 - 60000 // 1 min buffer
77
+
78
+ return data.access_token
79
+ } catch {
80
+ return null
81
+ }
82
+ }
83
+
84
+ async function getValidAccount(): Promise<{ account: AntigravityAccount; index: number } | null> {
85
+ const accounts = await loadAccounts()
86
+ if (accounts.accounts.length === 0) return null
87
+
88
+ const now = Date.now()
89
+ const enabledAccounts = accounts.accounts
90
+ .map((acc: AntigravityAccount, idx: number) => ({ account: acc, index: idx }))
91
+ .filter((item: { account: AntigravityAccount; index: number }) => item.account.enabled)
92
+
93
+ // Try to find an account that's not rate limited
94
+ for (const item of enabledAccounts) {
95
+ if (item.account.rateLimitedUntil && item.account.rateLimitedUntil > now) {
96
+ continue
97
+ }
98
+
99
+ // Check if access token needs refresh
100
+ if (!item.account.accessToken || (item.account.accessTokenExpiry && item.account.accessTokenExpiry < now)) {
101
+ const newToken = await refreshAccessToken(item.account)
102
+ if (!newToken) continue
103
+ await saveAccounts(accounts)
104
+ }
105
+
106
+ return item
107
+ }
108
+
109
+ // All accounts rate limited, find the one that recovers soonest
110
+ type AccountWithIndex = { account: AntigravityAccount; index: number }
111
+ const sortedByRecovery = enabledAccounts
112
+ .filter((item: AccountWithIndex) => item.account.rateLimitedUntil)
113
+ .sort((a: AccountWithIndex, b: AccountWithIndex) => (a.account.rateLimitedUntil || 0) - (b.account.rateLimitedUntil || 0))
114
+
115
+ if (sortedByRecovery.length > 0) {
116
+ const first = sortedByRecovery[0]
117
+ const waitTime = (first.account.rateLimitedUntil || 0) - now
118
+ if (waitTime > 0 && waitTime < 300000) {
119
+ // Wait up to 5 minutes
120
+ await Bun.sleep(waitTime + 1000)
121
+ first.account.rateLimitedUntil = undefined
122
+ await saveAccounts(accounts)
123
+ return first
124
+ }
125
+ }
126
+
127
+ return null
128
+ }
129
+
130
+ async function markAccountRateLimited(index: number, resetAfterMs: number = 60000): Promise<void> {
131
+ const accounts = await loadAccounts()
132
+ if (accounts.accounts[index]) {
133
+ accounts.accounts[index].rateLimitedUntil = Date.now() + resetAfterMs
134
+ await saveAccounts(accounts)
135
+ }
136
+ }
137
+
138
+ export async function AntigravityAuthPlugin(input: PluginInput): Promise<Hooks> {
139
+ return {
140
+ auth: {
141
+ provider: "antigravity",
142
+ async loader(getAuth, provider) {
143
+ const accounts = await loadAccounts()
144
+ if (accounts.accounts.length === 0) return {}
145
+
146
+ if (provider && provider.models) {
147
+ for (const model of Object.values(provider.models)) {
148
+ // Antigravity models are free (use Google quota)
149
+ model.cost = {
150
+ input: 0,
151
+ output: 0,
152
+ cache: { read: 0, write: 0 },
153
+ }
154
+ }
155
+ }
156
+
157
+ return {
158
+ apiKey: "antigravity-oauth",
159
+ async fetch(request: RequestInfo | URL, init?: RequestInit) {
160
+ const accountInfo = await getValidAccount()
161
+ if (!accountInfo) {
162
+ throw new Error("No valid Antigravity accounts available. Run 'bloxycode auth login' to add an account.")
163
+ }
164
+
165
+ const { account, index } = accountInfo
166
+
167
+ const headers: Record<string, string> = {
168
+ ...(init?.headers as Record<string, string>),
169
+ Authorization: `Bearer ${account.accessToken}`,
170
+ "User-Agent": `bloxycode/${Installation.VERSION}`,
171
+ "Content-Type": "application/json",
172
+ }
173
+
174
+ // Remove any conflicting auth headers
175
+ delete headers["x-api-key"]
176
+
177
+ const response = await fetch(request, {
178
+ ...init,
179
+ headers,
180
+ })
181
+
182
+ // Handle rate limiting
183
+ if (response.status === 429) {
184
+ const retryAfter = response.headers.get("retry-after")
185
+ const resetMs = retryAfter ? parseInt(retryAfter) * 1000 : 60000
186
+ await markAccountRateLimited(index, resetMs)
187
+
188
+ // Try with another account
189
+ const nextAccount = await getValidAccount()
190
+ if (nextAccount && nextAccount.index !== index) {
191
+ headers.Authorization = `Bearer ${nextAccount.account.accessToken}`
192
+ return fetch(request, { ...init, headers })
193
+ }
194
+ }
195
+
196
+ return response
197
+ },
198
+ }
199
+ },
200
+ methods: [
201
+ {
202
+ type: "oauth",
203
+ label: "Login with Google (Antigravity)",
204
+ async authorize() {
205
+ // Google Device Code OAuth flow
206
+ const deviceResponse = await fetch(GOOGLE_DEVICE_AUTH_URL, {
207
+ method: "POST",
208
+ headers: {
209
+ "Content-Type": "application/x-www-form-urlencoded",
210
+ },
211
+ body: new URLSearchParams({
212
+ client_id: GOOGLE_CLIENT_ID,
213
+ scope: "openid email profile https://www.googleapis.com/auth/cloud-platform",
214
+ }),
215
+ })
216
+
217
+ if (!deviceResponse.ok) {
218
+ throw new Error("Failed to initiate Google device authorization")
219
+ }
220
+
221
+ const deviceData = (await deviceResponse.json()) as {
222
+ verification_url: string
223
+ user_code: string
224
+ device_code: string
225
+ interval: number
226
+ expires_in: number
227
+ }
228
+
229
+ return {
230
+ url: deviceData.verification_url,
231
+ instructions: `Enter code: ${deviceData.user_code}`,
232
+ method: "auto" as const,
233
+ async callback() {
234
+ const startTime = Date.now()
235
+ const expiresAt = startTime + deviceData.expires_in * 1000
236
+
237
+ while (Date.now() < expiresAt) {
238
+ const response = await fetch(GOOGLE_TOKEN_URL, {
239
+ method: "POST",
240
+ headers: {
241
+ "Content-Type": "application/x-www-form-urlencoded",
242
+ },
243
+ body: new URLSearchParams({
244
+ client_id: GOOGLE_CLIENT_ID,
245
+ client_secret: GOOGLE_CLIENT_SECRET,
246
+ device_code: deviceData.device_code,
247
+ grant_type: "urn:ietf:params:oauth:grant-type:device_code",
248
+ }),
249
+ })
250
+
251
+ const data = (await response.json()) as {
252
+ access_token?: string
253
+ refresh_token?: string
254
+ expires_in?: number
255
+ id_token?: string
256
+ error?: string
257
+ }
258
+
259
+ if (data.access_token && data.refresh_token) {
260
+ // Get user email from ID token or userinfo
261
+ let email = "unknown"
262
+ try {
263
+ const userInfoResponse = await fetch("https://www.googleapis.com/oauth2/v2/userinfo", {
264
+ headers: { Authorization: `Bearer ${data.access_token}` },
265
+ })
266
+ if (userInfoResponse.ok) {
267
+ const userInfo = (await userInfoResponse.json()) as { email: string }
268
+ email = userInfo.email
269
+ }
270
+ } catch {}
271
+
272
+ // Save to accounts file
273
+ const accounts = await loadAccounts()
274
+
275
+ // Check if account already exists
276
+ const existingIndex = accounts.accounts.findIndex((a: AntigravityAccount) => a.email === email)
277
+ const newAccount: AntigravityAccount = {
278
+ email,
279
+ refreshToken: data.refresh_token,
280
+ accessToken: data.access_token,
281
+ accessTokenExpiry: Date.now() + (data.expires_in || 3600) * 1000 - 60000,
282
+ enabled: true,
283
+ }
284
+
285
+ if (existingIndex >= 0) {
286
+ accounts.accounts[existingIndex] = newAccount
287
+ } else {
288
+ accounts.accounts.push(newAccount)
289
+ }
290
+
291
+ await saveAccounts(accounts)
292
+
293
+ return {
294
+ type: "success" as const,
295
+ refresh: data.refresh_token,
296
+ access: data.access_token,
297
+ expires: Date.now() + (data.expires_in || 3600) * 1000,
298
+ accountId: email,
299
+ }
300
+ }
301
+
302
+ if (data.error === "authorization_pending") {
303
+ await Bun.sleep(deviceData.interval * 1000 + 1000)
304
+ continue
305
+ }
306
+
307
+ if (data.error === "slow_down") {
308
+ await Bun.sleep((deviceData.interval + 5) * 1000)
309
+ continue
310
+ }
311
+
312
+ if (data.error) {
313
+ return { type: "failed" as const }
314
+ }
315
+
316
+ await Bun.sleep(deviceData.interval * 1000)
317
+ }
318
+
319
+ return { type: "failed" as const }
320
+ },
321
+ }
322
+ },
323
+ },
324
+ ],
325
+ },
326
+ // Add thinking block support for Claude models via Antigravity
327
+ "chat.headers": async (input, output) => {
328
+ if (!input.model.providerID.includes("antigravity")) return
329
+
330
+ if (input.model.id.includes("claude")) {
331
+ output.headers["anthropic-beta"] = "interleaved-thinking-2025-05-14"
332
+ }
333
+ },
334
+ }
335
+ }
336
+
337
+ // Export account management functions for CLI
338
+ export const AntigravityAccounts = {
339
+ async list(): Promise<Array<{ email: string; enabled: boolean; rateLimited: boolean }>> {
340
+ const accounts = await loadAccounts()
341
+ const now = Date.now()
342
+ return accounts.accounts.map((acc: AntigravityAccount) => ({
343
+ email: acc.email,
344
+ enabled: acc.enabled,
345
+ rateLimited: acc.rateLimitedUntil ? acc.rateLimitedUntil > now : false,
346
+ }))
347
+ },
348
+
349
+ async remove(email: string): Promise<boolean> {
350
+ const accounts = await loadAccounts()
351
+ const index = accounts.accounts.findIndex((a: AntigravityAccount) => a.email === email)
352
+ if (index >= 0) {
353
+ accounts.accounts.splice(index, 1)
354
+ await saveAccounts(accounts)
355
+ return true
356
+ }
357
+ return false
358
+ },
359
+
360
+ async toggle(email: string, enabled: boolean): Promise<boolean> {
361
+ const accounts = await loadAccounts()
362
+ const account = accounts.accounts.find((a: AntigravityAccount) => a.email === email)
363
+ if (account) {
364
+ account.enabled = enabled
365
+ await saveAccounts(accounts)
366
+ return true
367
+ }
368
+ return false
369
+ },
370
+
371
+ async clearRateLimits(): Promise<void> {
372
+ const accounts = await loadAccounts()
373
+ for (const account of accounts.accounts) {
374
+ account.rateLimitedUntil = undefined
375
+ }
376
+ await saveAccounts(accounts)
377
+ },
378
+ }