gsd-pi 2.76.0 → 2.77.0-dev.1d17f366c

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 (932) hide show
  1. package/README.md +32 -30
  2. package/dist/claude-cli-check.js +32 -3
  3. package/dist/cli-web-branch.d.ts +1 -0
  4. package/dist/cli-web-branch.js +3 -0
  5. package/dist/cli.js +38 -2
  6. package/dist/extension-discovery.d.ts +6 -0
  7. package/dist/extension-discovery.js +37 -0
  8. package/dist/extension-registry.d.ts +3 -0
  9. package/dist/extension-sort.d.ts +18 -0
  10. package/dist/extension-sort.js +114 -0
  11. package/dist/extension-validator.d.ts +47 -0
  12. package/dist/extension-validator.js +127 -0
  13. package/dist/loader.js +35 -7
  14. package/dist/mcp-server.d.ts +7 -0
  15. package/dist/mcp-server.js +35 -1
  16. package/dist/onboarding.js +45 -0
  17. package/dist/provider-migrations.d.ts +18 -0
  18. package/dist/provider-migrations.js +14 -0
  19. package/dist/resource-loader.d.ts +1 -1
  20. package/dist/resource-loader.js +2 -8
  21. package/dist/resources/agents/researcher.md +1 -1
  22. package/dist/resources/extensions/claude-code-cli/readiness.js +31 -8
  23. package/dist/resources/extensions/claude-code-cli/stream-adapter.js +144 -63
  24. package/dist/resources/extensions/cmux/index.js +20 -0
  25. package/dist/resources/extensions/github-sync/templates.js +103 -0
  26. package/dist/resources/extensions/google-search/extension-manifest.json +5 -4
  27. package/dist/resources/extensions/google-search/index.js +3 -375
  28. package/dist/resources/extensions/gsd/abandon-detect.js +44 -0
  29. package/dist/resources/extensions/gsd/auto/loop.js +90 -2
  30. package/dist/resources/extensions/gsd/auto/phases.js +145 -28
  31. package/dist/resources/extensions/gsd/auto/resolve.js +24 -0
  32. package/dist/resources/extensions/gsd/auto/run-unit.js +48 -4
  33. package/dist/resources/extensions/gsd/auto/session.js +22 -1
  34. package/dist/resources/extensions/gsd/auto/turn-epoch.js +95 -0
  35. package/dist/resources/extensions/gsd/auto-dispatch.js +115 -17
  36. package/dist/resources/extensions/gsd/auto-loop.js +1 -1
  37. package/dist/resources/extensions/gsd/auto-model-selection.js +53 -16
  38. package/dist/resources/extensions/gsd/auto-post-unit.js +90 -2
  39. package/dist/resources/extensions/gsd/auto-prompts.js +14 -0
  40. package/dist/resources/extensions/gsd/auto-recovery.js +46 -1
  41. package/dist/resources/extensions/gsd/auto-start.js +84 -60
  42. package/dist/resources/extensions/gsd/auto-timeout-recovery.js +11 -5
  43. package/dist/resources/extensions/gsd/auto-unit-closeout.js +11 -2
  44. package/dist/resources/extensions/gsd/auto-verification.js +33 -0
  45. package/dist/resources/extensions/gsd/auto-worktree.js +109 -61
  46. package/dist/resources/extensions/gsd/auto.js +107 -38
  47. package/dist/resources/extensions/gsd/blocked-models.js +68 -0
  48. package/dist/resources/extensions/gsd/bootstrap/agent-end-recovery.js +103 -1
  49. package/dist/resources/extensions/gsd/bootstrap/db-tools.js +39 -9
  50. package/dist/resources/extensions/gsd/bootstrap/exec-tools.js +93 -0
  51. package/dist/resources/extensions/gsd/bootstrap/memory-tools.js +3 -0
  52. package/dist/resources/extensions/gsd/bootstrap/provider-error-resume.js +4 -2
  53. package/dist/resources/extensions/gsd/bootstrap/register-extension.js +23 -0
  54. package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +52 -6
  55. package/dist/resources/extensions/gsd/bootstrap/system-context.js +95 -29
  56. package/dist/resources/extensions/gsd/bootstrap/write-gate.js +34 -2
  57. package/dist/resources/extensions/gsd/clean-root-preflight.js +93 -0
  58. package/dist/resources/extensions/gsd/commands/handlers/workflow.js +31 -4
  59. package/dist/resources/extensions/gsd/commands-cmux.js +9 -6
  60. package/dist/resources/extensions/gsd/commands-extensions.js +634 -43
  61. package/dist/resources/extensions/gsd/commands-extract-learnings.js +54 -89
  62. package/dist/resources/extensions/gsd/commands-prefs-wizard.js +968 -23
  63. package/dist/resources/extensions/gsd/compaction-snapshot.js +121 -0
  64. package/dist/resources/extensions/gsd/complexity-classifier.js +5 -3
  65. package/dist/resources/extensions/gsd/db-writer.js +88 -16
  66. package/dist/resources/extensions/gsd/dispatch-guard.js +29 -3
  67. package/dist/resources/extensions/gsd/doctor-git-checks.js +23 -29
  68. package/dist/resources/extensions/gsd/doctor-providers.js +51 -5
  69. package/dist/resources/extensions/gsd/ecosystem/gsd-extension-api.js +1 -0
  70. package/dist/resources/extensions/gsd/error-classifier.js +31 -3
  71. package/dist/resources/extensions/gsd/exec-history.js +120 -0
  72. package/dist/resources/extensions/gsd/exec-sandbox.js +258 -0
  73. package/dist/resources/extensions/gsd/file-lock.js +49 -9
  74. package/dist/resources/extensions/gsd/git-service.js +1 -0
  75. package/dist/resources/extensions/gsd/gitignore.js +2 -0
  76. package/dist/resources/extensions/gsd/gsd-db.js +168 -23
  77. package/dist/resources/extensions/gsd/guided-flow-queue.js +4 -1
  78. package/dist/resources/extensions/gsd/guided-flow.js +212 -9
  79. package/dist/resources/extensions/gsd/health-widget.js +4 -1
  80. package/dist/resources/extensions/gsd/hook-emitter.js +108 -0
  81. package/dist/resources/extensions/gsd/init-wizard.js +15 -1
  82. package/dist/resources/extensions/gsd/journal.js +17 -2
  83. package/dist/resources/extensions/gsd/key-manager.js +28 -0
  84. package/dist/resources/extensions/gsd/memory-backfill.js +126 -0
  85. package/dist/resources/extensions/gsd/memory-store.js +19 -0
  86. package/dist/resources/extensions/gsd/milestone-actions.js +15 -0
  87. package/dist/resources/extensions/gsd/milestone-summary-classifier.js +37 -0
  88. package/dist/resources/extensions/gsd/model-router.js +36 -3
  89. package/dist/resources/extensions/gsd/notifications.js +30 -16
  90. package/dist/resources/extensions/gsd/pre-execution-checks.js +44 -9
  91. package/dist/resources/extensions/gsd/preferences-types.js +9 -0
  92. package/dist/resources/extensions/gsd/preferences-validation.js +83 -0
  93. package/dist/resources/extensions/gsd/preferences.js +17 -17
  94. package/dist/resources/extensions/gsd/prompt-loader.js +22 -7
  95. package/dist/resources/extensions/gsd/prompts/complete-milestone.md +1 -1
  96. package/dist/resources/extensions/gsd/prompts/complete-slice.md +2 -2
  97. package/dist/resources/extensions/gsd/prompts/debug-diagnose.md +2 -0
  98. package/dist/resources/extensions/gsd/prompts/discuss-headless.md +29 -2
  99. package/dist/resources/extensions/gsd/prompts/discuss.md +29 -2
  100. package/dist/resources/extensions/gsd/prompts/doctor-heal.md +5 -4
  101. package/dist/resources/extensions/gsd/prompts/execute-task.md +3 -2
  102. package/dist/resources/extensions/gsd/prompts/parallel-research-slices.md +5 -2
  103. package/dist/resources/extensions/gsd/prompts/plan-slice.md +1 -0
  104. package/dist/resources/extensions/gsd/prompts/research-slice.md +1 -0
  105. package/dist/resources/extensions/gsd/prompts/system.md +1 -0
  106. package/dist/resources/extensions/gsd/reports.js +5 -4
  107. package/dist/resources/extensions/gsd/safety/evidence-collector.js +96 -0
  108. package/dist/resources/extensions/gsd/safety/file-change-validator.js +13 -5
  109. package/dist/resources/extensions/gsd/safety/safety-harness.js +5 -1
  110. package/dist/resources/extensions/gsd/state-transition-matrix.js +118 -0
  111. package/dist/resources/extensions/gsd/state.js +68 -29
  112. package/dist/resources/extensions/gsd/token-counter.js +22 -5
  113. package/dist/resources/extensions/gsd/tools/complete-milestone.js +16 -10
  114. package/dist/resources/extensions/gsd/tools/complete-slice.js +21 -0
  115. package/dist/resources/extensions/gsd/tools/complete-task.js +31 -0
  116. package/dist/resources/extensions/gsd/tools/exec-search-tool.js +59 -0
  117. package/dist/resources/extensions/gsd/tools/exec-tool.js +126 -0
  118. package/dist/resources/extensions/gsd/tools/memory-tools.js +26 -1
  119. package/dist/resources/extensions/gsd/tools/resume-tool.js +23 -0
  120. package/dist/resources/extensions/gsd/uok/audit.js +18 -2
  121. package/dist/resources/extensions/gsd/uok/dispatch-envelope.js +33 -0
  122. package/dist/resources/extensions/gsd/uok/execution-graph.js +10 -0
  123. package/dist/resources/extensions/gsd/uok/gitops.js +2 -1
  124. package/dist/resources/extensions/gsd/uok/loop-adapter.js +37 -10
  125. package/dist/resources/extensions/gsd/uok/parity-report.js +58 -0
  126. package/dist/resources/extensions/gsd/uok/plan-v2.js +30 -7
  127. package/dist/resources/extensions/gsd/uok/writer.js +82 -0
  128. package/dist/resources/extensions/gsd/workflow-logger.js +10 -2
  129. package/dist/resources/extensions/gsd/workflow-mcp.js +3 -0
  130. package/dist/resources/extensions/gsd/workflow-templates/spike.md +6 -0
  131. package/dist/resources/extensions/gsd/worktree-manager.js +1 -0
  132. package/dist/resources/extensions/gsd/worktree-resolver.js +50 -10
  133. package/dist/resources/extensions/mcp-client/auth.js +10 -1
  134. package/dist/resources/extensions/mcp-client/index.js +118 -9
  135. package/dist/resources/extensions/search-the-web/command-search-provider.js +5 -4
  136. package/dist/resources/extensions/search-the-web/native-search.js +45 -13
  137. package/dist/resources/extensions/shared/cmux-events.js +12 -0
  138. package/dist/resources/extensions/shared/rtk-session-stats.js +1 -2
  139. package/dist/resources/skills/api-design/SKILL.md +190 -0
  140. package/dist/resources/skills/create-mcp-server/SKILL.md +121 -0
  141. package/dist/resources/skills/create-skill/SKILL.md +2 -2
  142. package/dist/resources/skills/create-skill/references/gsd-skill-ecosystem.md +4 -4
  143. package/dist/resources/skills/create-skill/workflows/audit-skill.md +4 -4
  144. package/dist/resources/skills/create-skill/workflows/create-new-skill.md +5 -5
  145. package/dist/resources/skills/decompose-into-slices/SKILL.md +139 -0
  146. package/dist/resources/skills/dependency-upgrade/SKILL.md +158 -0
  147. package/dist/resources/skills/design-an-interface/SKILL.md +102 -0
  148. package/dist/resources/skills/forensics/SKILL.md +153 -0
  149. package/dist/resources/skills/grill-me/SKILL.md +93 -0
  150. package/dist/resources/skills/handoff/SKILL.md +121 -0
  151. package/dist/resources/skills/observability/SKILL.md +174 -0
  152. package/dist/resources/skills/security-review/SKILL.md +181 -0
  153. package/dist/resources/skills/spike-wrap-up/SKILL.md +138 -0
  154. package/dist/resources/skills/tdd/SKILL.md +112 -0
  155. package/dist/resources/skills/verify-before-complete/SKILL.md +98 -0
  156. package/dist/resources/skills/write-docs/SKILL.md +82 -0
  157. package/dist/resources/skills/write-milestone-brief/SKILL.md +135 -0
  158. package/dist/tsconfig.extensions.tsbuildinfo +1 -1
  159. package/dist/web/standalone/.next/BUILD_ID +1 -1
  160. package/dist/web/standalone/.next/app-path-routes-manifest.json +13 -13
  161. package/dist/web/standalone/.next/build-manifest.json +4 -4
  162. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  163. package/dist/web/standalone/.next/react-loadable-manifest.json +1 -1
  164. package/dist/web/standalone/.next/required-server-files.json +4 -4
  165. package/dist/web/standalone/.next/server/app/_global-error/page.js +3 -3
  166. package/dist/web/standalone/.next/server/app/_global-error/page_client-reference-manifest.js +1 -1
  167. package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
  168. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  169. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  170. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  171. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  172. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  173. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  174. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  175. package/dist/web/standalone/.next/server/app/_not-found/page.js +2 -2
  176. package/dist/web/standalone/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
  177. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  178. package/dist/web/standalone/.next/server/app/_not-found.rsc +3 -3
  179. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +3 -3
  180. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  181. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +3 -3
  182. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  183. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  184. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  185. package/dist/web/standalone/.next/server/app/api/boot/route.js +1 -1
  186. package/dist/web/standalone/.next/server/app/api/boot/route.js.nft.json +1 -1
  187. package/dist/web/standalone/.next/server/app/api/boot/route_client-reference-manifest.js +1 -1
  188. package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route.js +1 -1
  189. package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route.js.nft.json +1 -1
  190. package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route_client-reference-manifest.js +1 -1
  191. package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route.js +1 -1
  192. package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route.js.nft.json +1 -1
  193. package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route_client-reference-manifest.js +1 -1
  194. package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route.js +2 -2
  195. package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route.js.nft.json +1 -1
  196. package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route_client-reference-manifest.js +1 -1
  197. package/dist/web/standalone/.next/server/app/api/browse-directories/route.js +1 -1
  198. package/dist/web/standalone/.next/server/app/api/browse-directories/route_client-reference-manifest.js +1 -1
  199. package/dist/web/standalone/.next/server/app/api/captures/route.js +1 -1
  200. package/dist/web/standalone/.next/server/app/api/captures/route.js.nft.json +1 -1
  201. package/dist/web/standalone/.next/server/app/api/captures/route_client-reference-manifest.js +1 -1
  202. package/dist/web/standalone/.next/server/app/api/cleanup/route.js +1 -1
  203. package/dist/web/standalone/.next/server/app/api/cleanup/route.js.nft.json +1 -1
  204. package/dist/web/standalone/.next/server/app/api/cleanup/route_client-reference-manifest.js +1 -1
  205. package/dist/web/standalone/.next/server/app/api/dev-mode/route.js +1 -1
  206. package/dist/web/standalone/.next/server/app/api/dev-mode/route_client-reference-manifest.js +1 -1
  207. package/dist/web/standalone/.next/server/app/api/doctor/route.js +1 -1
  208. package/dist/web/standalone/.next/server/app/api/doctor/route.js.nft.json +1 -1
  209. package/dist/web/standalone/.next/server/app/api/doctor/route_client-reference-manifest.js +1 -1
  210. package/dist/web/standalone/.next/server/app/api/experimental/route.js +2 -2
  211. package/dist/web/standalone/.next/server/app/api/experimental/route_client-reference-manifest.js +1 -1
  212. package/dist/web/standalone/.next/server/app/api/export-data/route.js +1 -1
  213. package/dist/web/standalone/.next/server/app/api/export-data/route.js.nft.json +1 -1
  214. package/dist/web/standalone/.next/server/app/api/export-data/route_client-reference-manifest.js +1 -1
  215. package/dist/web/standalone/.next/server/app/api/files/route.js +1 -1
  216. package/dist/web/standalone/.next/server/app/api/files/route.js.nft.json +1 -1
  217. package/dist/web/standalone/.next/server/app/api/files/route_client-reference-manifest.js +1 -1
  218. package/dist/web/standalone/.next/server/app/api/forensics/route.js +1 -1
  219. package/dist/web/standalone/.next/server/app/api/forensics/route.js.nft.json +1 -1
  220. package/dist/web/standalone/.next/server/app/api/forensics/route_client-reference-manifest.js +1 -1
  221. package/dist/web/standalone/.next/server/app/api/git/route.js +1 -1
  222. package/dist/web/standalone/.next/server/app/api/git/route.js.nft.json +1 -1
  223. package/dist/web/standalone/.next/server/app/api/git/route_client-reference-manifest.js +1 -1
  224. package/dist/web/standalone/.next/server/app/api/history/route.js +1 -1
  225. package/dist/web/standalone/.next/server/app/api/history/route.js.nft.json +1 -1
  226. package/dist/web/standalone/.next/server/app/api/history/route_client-reference-manifest.js +1 -1
  227. package/dist/web/standalone/.next/server/app/api/hooks/route.js +1 -1
  228. package/dist/web/standalone/.next/server/app/api/hooks/route.js.nft.json +1 -1
  229. package/dist/web/standalone/.next/server/app/api/hooks/route_client-reference-manifest.js +1 -1
  230. package/dist/web/standalone/.next/server/app/api/inspect/route.js +1 -1
  231. package/dist/web/standalone/.next/server/app/api/inspect/route.js.nft.json +1 -1
  232. package/dist/web/standalone/.next/server/app/api/inspect/route_client-reference-manifest.js +1 -1
  233. package/dist/web/standalone/.next/server/app/api/knowledge/route.js +1 -1
  234. package/dist/web/standalone/.next/server/app/api/knowledge/route.js.nft.json +1 -1
  235. package/dist/web/standalone/.next/server/app/api/knowledge/route_client-reference-manifest.js +1 -1
  236. package/dist/web/standalone/.next/server/app/api/live-state/route.js +1 -1
  237. package/dist/web/standalone/.next/server/app/api/live-state/route.js.nft.json +1 -1
  238. package/dist/web/standalone/.next/server/app/api/live-state/route_client-reference-manifest.js +1 -1
  239. package/dist/web/standalone/.next/server/app/api/notifications/route.js +2 -2
  240. package/dist/web/standalone/.next/server/app/api/notifications/route.js.nft.json +1 -1
  241. package/dist/web/standalone/.next/server/app/api/notifications/route_client-reference-manifest.js +1 -1
  242. package/dist/web/standalone/.next/server/app/api/onboarding/route.js +1 -1
  243. package/dist/web/standalone/.next/server/app/api/onboarding/route.js.nft.json +1 -1
  244. package/dist/web/standalone/.next/server/app/api/onboarding/route_client-reference-manifest.js +1 -1
  245. package/dist/web/standalone/.next/server/app/api/preferences/route.js +1 -1
  246. package/dist/web/standalone/.next/server/app/api/preferences/route_client-reference-manifest.js +1 -1
  247. package/dist/web/standalone/.next/server/app/api/projects/route.js +1 -1
  248. package/dist/web/standalone/.next/server/app/api/projects/route.js.nft.json +1 -1
  249. package/dist/web/standalone/.next/server/app/api/projects/route_client-reference-manifest.js +1 -1
  250. package/dist/web/standalone/.next/server/app/api/recovery/route.js +1 -1
  251. package/dist/web/standalone/.next/server/app/api/recovery/route.js.nft.json +1 -1
  252. package/dist/web/standalone/.next/server/app/api/recovery/route_client-reference-manifest.js +1 -1
  253. package/dist/web/standalone/.next/server/app/api/remote-questions/route.js +2 -2
  254. package/dist/web/standalone/.next/server/app/api/remote-questions/route_client-reference-manifest.js +1 -1
  255. package/dist/web/standalone/.next/server/app/api/session/browser/route.js +1 -1
  256. package/dist/web/standalone/.next/server/app/api/session/browser/route.js.nft.json +1 -1
  257. package/dist/web/standalone/.next/server/app/api/session/browser/route_client-reference-manifest.js +1 -1
  258. package/dist/web/standalone/.next/server/app/api/session/command/route.js +1 -1
  259. package/dist/web/standalone/.next/server/app/api/session/command/route.js.nft.json +1 -1
  260. package/dist/web/standalone/.next/server/app/api/session/command/route_client-reference-manifest.js +1 -1
  261. package/dist/web/standalone/.next/server/app/api/session/events/route.js +2 -2
  262. package/dist/web/standalone/.next/server/app/api/session/events/route.js.nft.json +1 -1
  263. package/dist/web/standalone/.next/server/app/api/session/events/route_client-reference-manifest.js +1 -1
  264. package/dist/web/standalone/.next/server/app/api/session/manage/route.js +1 -1
  265. package/dist/web/standalone/.next/server/app/api/session/manage/route.js.nft.json +1 -1
  266. package/dist/web/standalone/.next/server/app/api/session/manage/route_client-reference-manifest.js +1 -1
  267. package/dist/web/standalone/.next/server/app/api/settings-data/route.js +1 -1
  268. package/dist/web/standalone/.next/server/app/api/settings-data/route.js.nft.json +1 -1
  269. package/dist/web/standalone/.next/server/app/api/settings-data/route_client-reference-manifest.js +1 -1
  270. package/dist/web/standalone/.next/server/app/api/shutdown/route.js +1 -1
  271. package/dist/web/standalone/.next/server/app/api/shutdown/route_client-reference-manifest.js +1 -1
  272. package/dist/web/standalone/.next/server/app/api/skill-health/route.js +1 -1
  273. package/dist/web/standalone/.next/server/app/api/skill-health/route.js.nft.json +1 -1
  274. package/dist/web/standalone/.next/server/app/api/skill-health/route_client-reference-manifest.js +1 -1
  275. package/dist/web/standalone/.next/server/app/api/steer/route.js +1 -1
  276. package/dist/web/standalone/.next/server/app/api/steer/route.js.nft.json +1 -1
  277. package/dist/web/standalone/.next/server/app/api/steer/route_client-reference-manifest.js +1 -1
  278. package/dist/web/standalone/.next/server/app/api/switch-root/route.js +1 -1
  279. package/dist/web/standalone/.next/server/app/api/switch-root/route.js.nft.json +1 -1
  280. package/dist/web/standalone/.next/server/app/api/switch-root/route_client-reference-manifest.js +1 -1
  281. package/dist/web/standalone/.next/server/app/api/terminal/input/route.js +2 -2
  282. package/dist/web/standalone/.next/server/app/api/terminal/input/route_client-reference-manifest.js +1 -1
  283. package/dist/web/standalone/.next/server/app/api/terminal/resize/route.js +2 -2
  284. package/dist/web/standalone/.next/server/app/api/terminal/resize/route_client-reference-manifest.js +1 -1
  285. package/dist/web/standalone/.next/server/app/api/terminal/sessions/route.js +2 -2
  286. package/dist/web/standalone/.next/server/app/api/terminal/sessions/route.js.nft.json +1 -1
  287. package/dist/web/standalone/.next/server/app/api/terminal/sessions/route_client-reference-manifest.js +1 -1
  288. package/dist/web/standalone/.next/server/app/api/terminal/stream/route.js +4 -4
  289. package/dist/web/standalone/.next/server/app/api/terminal/stream/route.js.nft.json +1 -1
  290. package/dist/web/standalone/.next/server/app/api/terminal/stream/route_client-reference-manifest.js +1 -1
  291. package/dist/web/standalone/.next/server/app/api/terminal/upload/route.js +1 -1
  292. package/dist/web/standalone/.next/server/app/api/terminal/upload/route_client-reference-manifest.js +1 -1
  293. package/dist/web/standalone/.next/server/app/api/undo/route.js +1 -1
  294. package/dist/web/standalone/.next/server/app/api/undo/route.js.nft.json +1 -1
  295. package/dist/web/standalone/.next/server/app/api/undo/route_client-reference-manifest.js +1 -1
  296. package/dist/web/standalone/.next/server/app/api/update/route.js +1 -1
  297. package/dist/web/standalone/.next/server/app/api/update/route_client-reference-manifest.js +1 -1
  298. package/dist/web/standalone/.next/server/app/api/visualizer/route.js +1 -1
  299. package/dist/web/standalone/.next/server/app/api/visualizer/route.js.nft.json +1 -1
  300. package/dist/web/standalone/.next/server/app/api/visualizer/route_client-reference-manifest.js +1 -1
  301. package/dist/web/standalone/.next/server/app/index.html +1 -1
  302. package/dist/web/standalone/.next/server/app/index.rsc +4 -4
  303. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +2 -2
  304. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +4 -4
  305. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  306. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +3 -3
  307. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  308. package/dist/web/standalone/.next/server/app/page.js +2 -2
  309. package/dist/web/standalone/.next/server/app/page_client-reference-manifest.js +1 -1
  310. package/dist/web/standalone/.next/server/app-paths-manifest.json +13 -13
  311. package/dist/web/standalone/.next/server/chunks/1926.js +1 -0
  312. package/dist/web/standalone/.next/server/chunks/63.js +3 -3
  313. package/dist/web/standalone/.next/server/chunks/6897.js +3 -3
  314. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  315. package/dist/web/standalone/.next/server/middleware-manifest.json +5 -5
  316. package/dist/web/standalone/.next/server/middleware-react-loadable-manifest.js +1 -1
  317. package/dist/web/standalone/.next/server/middleware.js +2 -2
  318. package/dist/web/standalone/.next/server/next-font-manifest.js +1 -1
  319. package/dist/web/standalone/.next/server/next-font-manifest.json +1 -1
  320. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  321. package/dist/web/standalone/.next/server/pages/500.html +1 -1
  322. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  323. package/dist/web/standalone/.next/static/chunks/2826.e9f5195e91f9cad2.js +11 -0
  324. package/dist/web/standalone/.next/static/chunks/app/_not-found/{page-2f24283c162b6ab3.js → page-f2a7482d42a5614b.js} +1 -1
  325. package/dist/web/standalone/.next/static/chunks/app/{layout-9ecfd95f343793f0.js → layout-a16c7a7ecdf0c2cf.js} +1 -1
  326. package/dist/web/standalone/.next/static/chunks/app/page-5b113fd32bc2a1c3.js +1 -0
  327. package/dist/web/standalone/.next/static/chunks/main-app-fdab67f7802d7832.js +1 -0
  328. package/dist/web/standalone/.next/static/chunks/next/dist/client/components/builtin/global-error-459824ffb8c323dd.js +1 -0
  329. package/dist/web/standalone/.next/static/chunks/{webpack-5fc74f13a25fa1bb.js → webpack-2e68521d7c82f7c2.js} +1 -1
  330. package/dist/web/standalone/node_modules/node-pty/build/Makefile +2 -2
  331. package/dist/web/standalone/node_modules/node-pty/build/Release/pty.node +0 -0
  332. package/dist/web/standalone/node_modules/node-pty/build/pty.target.mk +14 -14
  333. package/dist/web/standalone/node_modules/node-pty/node-addon-api/node_addon_api.target.mk +14 -14
  334. package/dist/web/standalone/node_modules/node-pty/node-addon-api/node_addon_api_except.target.mk +14 -14
  335. package/dist/web/standalone/node_modules/node-pty/node-addon-api/node_addon_api_maybe.target.mk +14 -14
  336. package/dist/web/standalone/server.js +1 -1
  337. package/dist/welcome-screen.js +6 -1
  338. package/dist/wizard.js +2 -0
  339. package/package.json +16 -14
  340. package/packages/daemon/package.json +2 -2
  341. package/packages/mcp-server/README.md +3 -3
  342. package/packages/mcp-server/dist/env-writer.d.ts +1 -0
  343. package/packages/mcp-server/dist/env-writer.d.ts.map +1 -1
  344. package/packages/mcp-server/dist/env-writer.js +74 -6
  345. package/packages/mcp-server/dist/env-writer.js.map +1 -1
  346. package/packages/mcp-server/dist/remote-questions.d.ts +45 -0
  347. package/packages/mcp-server/dist/remote-questions.d.ts.map +1 -0
  348. package/packages/mcp-server/dist/remote-questions.js +732 -0
  349. package/packages/mcp-server/dist/remote-questions.js.map +1 -0
  350. package/packages/mcp-server/dist/server.d.ts +7 -0
  351. package/packages/mcp-server/dist/server.d.ts.map +1 -1
  352. package/packages/mcp-server/dist/server.js +95 -10
  353. package/packages/mcp-server/dist/server.js.map +1 -1
  354. package/packages/mcp-server/dist/session-manager.d.ts +14 -0
  355. package/packages/mcp-server/dist/session-manager.d.ts.map +1 -1
  356. package/packages/mcp-server/dist/session-manager.js +49 -1
  357. package/packages/mcp-server/dist/session-manager.js.map +1 -1
  358. package/packages/mcp-server/dist/workflow-tools.d.ts +1 -1
  359. package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
  360. package/packages/mcp-server/dist/workflow-tools.js +178 -31
  361. package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
  362. package/packages/mcp-server/package.json +9 -3
  363. package/packages/mcp-server/src/env-writer.test.ts +79 -1
  364. package/packages/mcp-server/src/env-writer.ts +76 -6
  365. package/packages/mcp-server/src/mcp-server.test.ts +67 -0
  366. package/packages/mcp-server/src/readers/readers.test.ts +5 -1
  367. package/packages/mcp-server/src/remote-questions.test.ts +294 -0
  368. package/packages/mcp-server/src/remote-questions.ts +916 -0
  369. package/packages/mcp-server/src/server.ts +118 -16
  370. package/packages/mcp-server/src/session-manager.ts +43 -1
  371. package/packages/mcp-server/src/workflow-tools.test.ts +190 -1
  372. package/packages/mcp-server/src/workflow-tools.ts +234 -49
  373. package/packages/mcp-server/tsconfig.test.json +19 -0
  374. package/packages/mcp-server/tsconfig.tsbuildinfo +1 -1
  375. package/packages/native/package.json +6 -1
  376. package/packages/native/src/__tests__/clipboard.test.mjs +69 -23
  377. package/packages/native/tsconfig.tsbuildinfo +1 -1
  378. package/packages/pi-agent-core/dist/agent-loop.js +12 -0
  379. package/packages/pi-agent-core/dist/agent-loop.js.map +1 -1
  380. package/packages/pi-agent-core/dist/types.d.ts +30 -0
  381. package/packages/pi-agent-core/dist/types.d.ts.map +1 -1
  382. package/packages/pi-agent-core/dist/types.js.map +1 -1
  383. package/packages/pi-agent-core/package.json +6 -1
  384. package/packages/pi-agent-core/src/agent-loop.test.ts +220 -15
  385. package/packages/pi-agent-core/src/agent-loop.ts +14 -0
  386. package/packages/pi-agent-core/src/types.ts +34 -0
  387. package/packages/pi-agent-core/tsconfig.tsbuildinfo +1 -1
  388. package/packages/pi-ai/dist/models/custom.d.ts +38 -0
  389. package/packages/pi-ai/dist/models/custom.d.ts.map +1 -1
  390. package/packages/pi-ai/dist/models/custom.js +41 -0
  391. package/packages/pi-ai/dist/models/custom.js.map +1 -1
  392. package/packages/pi-ai/dist/providers/anthropic-auth.test.js +1 -1
  393. package/packages/pi-ai/dist/providers/anthropic-auth.test.js.map +1 -1
  394. package/packages/pi-ai/dist/providers/anthropic-bearer-auth.test.d.ts +2 -0
  395. package/packages/pi-ai/dist/providers/anthropic-bearer-auth.test.d.ts.map +1 -0
  396. package/packages/pi-ai/dist/providers/anthropic-bearer-auth.test.js +13 -0
  397. package/packages/pi-ai/dist/providers/anthropic-bearer-auth.test.js.map +1 -0
  398. package/packages/pi-ai/dist/providers/anthropic-shared.d.ts.map +1 -1
  399. package/packages/pi-ai/dist/providers/anthropic-shared.js +27 -4
  400. package/packages/pi-ai/dist/providers/anthropic-shared.js.map +1 -1
  401. package/packages/pi-ai/dist/providers/anthropic.d.ts.map +1 -1
  402. package/packages/pi-ai/dist/providers/anthropic.js +13 -4
  403. package/packages/pi-ai/dist/providers/anthropic.js.map +1 -1
  404. package/packages/pi-ai/dist/providers/minimax-tool-name.test.d.ts +2 -0
  405. package/packages/pi-ai/dist/providers/minimax-tool-name.test.d.ts.map +1 -0
  406. package/packages/pi-ai/dist/providers/minimax-tool-name.test.js +80 -0
  407. package/packages/pi-ai/dist/providers/minimax-tool-name.test.js.map +1 -0
  408. package/packages/pi-ai/dist/providers/openai-completions.d.ts.map +1 -1
  409. package/packages/pi-ai/dist/providers/openai-completions.js +60 -15
  410. package/packages/pi-ai/dist/providers/openai-completions.js.map +1 -1
  411. package/packages/pi-ai/dist/providers/simple-options.d.ts +10 -0
  412. package/packages/pi-ai/dist/providers/simple-options.d.ts.map +1 -1
  413. package/packages/pi-ai/dist/providers/simple-options.js +16 -1
  414. package/packages/pi-ai/dist/providers/simple-options.js.map +1 -1
  415. package/packages/pi-ai/dist/providers/think-tag-parser.d.ts +17 -0
  416. package/packages/pi-ai/dist/providers/think-tag-parser.d.ts.map +1 -0
  417. package/packages/pi-ai/dist/providers/think-tag-parser.js +75 -0
  418. package/packages/pi-ai/dist/providers/think-tag-parser.js.map +1 -0
  419. package/packages/pi-ai/dist/providers/think-tag-parser.test.d.ts +2 -0
  420. package/packages/pi-ai/dist/providers/think-tag-parser.test.d.ts.map +1 -0
  421. package/packages/pi-ai/dist/providers/think-tag-parser.test.js +41 -0
  422. package/packages/pi-ai/dist/providers/think-tag-parser.test.js.map +1 -0
  423. package/packages/pi-ai/dist/utils/oauth/github-copilot.d.ts.map +1 -1
  424. package/packages/pi-ai/dist/utils/oauth/github-copilot.js +12 -2
  425. package/packages/pi-ai/dist/utils/oauth/github-copilot.js.map +1 -1
  426. package/packages/pi-ai/dist/utils/oauth/github-copilot.test.js +164 -14
  427. package/packages/pi-ai/dist/utils/oauth/github-copilot.test.js.map +1 -1
  428. package/packages/pi-ai/dist/utils/oauth/google-antigravity.d.ts.map +1 -1
  429. package/packages/pi-ai/dist/utils/oauth/google-antigravity.js +15 -3
  430. package/packages/pi-ai/dist/utils/oauth/google-antigravity.js.map +1 -1
  431. package/packages/pi-ai/dist/utils/oauth/google-antigravity.test.d.ts +2 -0
  432. package/packages/pi-ai/dist/utils/oauth/google-antigravity.test.d.ts.map +1 -0
  433. package/packages/pi-ai/dist/utils/oauth/google-antigravity.test.js +67 -0
  434. package/packages/pi-ai/dist/utils/oauth/google-antigravity.test.js.map +1 -0
  435. package/packages/pi-ai/dist/utils/oauth/google-gemini-cli.d.ts.map +1 -1
  436. package/packages/pi-ai/dist/utils/oauth/google-gemini-cli.js +16 -3
  437. package/packages/pi-ai/dist/utils/oauth/google-gemini-cli.js.map +1 -1
  438. package/packages/pi-ai/dist/utils/oauth/google-gemini-cli.test.d.ts +2 -0
  439. package/packages/pi-ai/dist/utils/oauth/google-gemini-cli.test.d.ts.map +1 -0
  440. package/packages/pi-ai/dist/utils/oauth/google-gemini-cli.test.js +67 -0
  441. package/packages/pi-ai/dist/utils/oauth/google-gemini-cli.test.js.map +1 -0
  442. package/packages/pi-ai/dist/utils/oauth/oauth-providers.test.d.ts +2 -0
  443. package/packages/pi-ai/dist/utils/oauth/oauth-providers.test.d.ts.map +1 -0
  444. package/packages/pi-ai/dist/utils/oauth/oauth-providers.test.js +289 -0
  445. package/packages/pi-ai/dist/utils/oauth/oauth-providers.test.js.map +1 -0
  446. package/packages/pi-ai/package.json +6 -1
  447. package/packages/pi-ai/src/models/custom.ts +42 -0
  448. package/packages/pi-ai/src/providers/anthropic-auth.test.ts +1 -1
  449. package/packages/pi-ai/src/providers/anthropic-bearer-auth.test.ts +26 -0
  450. package/packages/pi-ai/src/providers/anthropic-shared.ts +26 -5
  451. package/packages/pi-ai/src/providers/anthropic.ts +15 -4
  452. package/packages/pi-ai/src/providers/minimax-tool-name.test.ts +98 -0
  453. package/packages/pi-ai/src/providers/openai-completions.ts +57 -16
  454. package/packages/pi-ai/src/providers/simple-options.ts +17 -1
  455. package/packages/pi-ai/src/providers/think-tag-parser.test.ts +44 -0
  456. package/packages/pi-ai/src/providers/think-tag-parser.ts +94 -0
  457. package/packages/pi-ai/src/utils/oauth/github-copilot.test.ts +200 -23
  458. package/packages/pi-ai/src/utils/oauth/github-copilot.ts +12 -2
  459. package/packages/pi-ai/src/utils/oauth/google-antigravity.test.ts +84 -0
  460. package/packages/pi-ai/src/utils/oauth/google-antigravity.ts +15 -5
  461. package/packages/pi-ai/src/utils/oauth/google-gemini-cli.test.ts +84 -0
  462. package/packages/pi-ai/src/utils/oauth/google-gemini-cli.ts +16 -5
  463. package/packages/pi-ai/src/utils/oauth/oauth-providers.test.ts +363 -0
  464. package/packages/pi-ai/tsconfig.tsbuildinfo +1 -1
  465. package/packages/pi-coding-agent/dist/core/agent-session-abort-order.test.js +3 -2
  466. package/packages/pi-coding-agent/dist/core/agent-session-abort-order.test.js.map +1 -1
  467. package/packages/pi-coding-agent/dist/core/agent-session.d.ts +2 -0
  468. package/packages/pi-coding-agent/dist/core/agent-session.d.ts.map +1 -1
  469. package/packages/pi-coding-agent/dist/core/agent-session.js +32 -2
  470. package/packages/pi-coding-agent/dist/core/agent-session.js.map +1 -1
  471. package/packages/pi-coding-agent/dist/core/compaction/compaction.d.ts +25 -0
  472. package/packages/pi-coding-agent/dist/core/compaction/compaction.d.ts.map +1 -1
  473. package/packages/pi-coding-agent/dist/core/compaction/compaction.js +105 -6
  474. package/packages/pi-coding-agent/dist/core/compaction/compaction.js.map +1 -1
  475. package/packages/pi-coding-agent/dist/core/compaction/compaction.test.js +230 -28
  476. package/packages/pi-coding-agent/dist/core/compaction/compaction.test.js.map +1 -1
  477. package/packages/pi-coding-agent/dist/core/compaction/utils.d.ts +30 -2
  478. package/packages/pi-coding-agent/dist/core/compaction/utils.d.ts.map +1 -1
  479. package/packages/pi-coding-agent/dist/core/compaction/utils.js +113 -12
  480. package/packages/pi-coding-agent/dist/core/compaction/utils.js.map +1 -1
  481. package/packages/pi-coding-agent/dist/core/compaction-orchestrator.d.ts +1 -0
  482. package/packages/pi-coding-agent/dist/core/compaction-orchestrator.d.ts.map +1 -1
  483. package/packages/pi-coding-agent/dist/core/compaction-orchestrator.js +29 -18
  484. package/packages/pi-coding-agent/dist/core/compaction-orchestrator.js.map +1 -1
  485. package/packages/pi-coding-agent/dist/core/compaction-orchestrator.test.d.ts +2 -0
  486. package/packages/pi-coding-agent/dist/core/compaction-orchestrator.test.d.ts.map +1 -0
  487. package/packages/pi-coding-agent/dist/core/compaction-orchestrator.test.js +130 -0
  488. package/packages/pi-coding-agent/dist/core/compaction-orchestrator.test.js.map +1 -0
  489. package/packages/pi-coding-agent/dist/core/compaction-utils.test.js +56 -1
  490. package/packages/pi-coding-agent/dist/core/compaction-utils.test.js.map +1 -1
  491. package/packages/pi-coding-agent/dist/core/discovery-cache.test.js +8 -15
  492. package/packages/pi-coding-agent/dist/core/discovery-cache.test.js.map +1 -1
  493. package/packages/pi-coding-agent/dist/core/extensions/extension-discovery.d.ts +25 -0
  494. package/packages/pi-coding-agent/dist/core/extensions/extension-discovery.d.ts.map +1 -0
  495. package/packages/pi-coding-agent/dist/core/extensions/extension-discovery.js +109 -0
  496. package/packages/pi-coding-agent/dist/core/extensions/extension-discovery.js.map +1 -0
  497. package/packages/pi-coding-agent/dist/core/extensions/extension-registry.d.ts +67 -0
  498. package/packages/pi-coding-agent/dist/core/extensions/extension-registry.d.ts.map +1 -0
  499. package/packages/pi-coding-agent/dist/core/extensions/extension-registry.js +167 -0
  500. package/packages/pi-coding-agent/dist/core/extensions/extension-registry.js.map +1 -0
  501. package/packages/pi-coding-agent/dist/core/extensions/index.d.ts +1 -1
  502. package/packages/pi-coding-agent/dist/core/extensions/index.d.ts.map +1 -1
  503. package/packages/pi-coding-agent/dist/core/extensions/index.js.map +1 -1
  504. package/packages/pi-coding-agent/dist/core/extensions/loader.d.ts +3 -2
  505. package/packages/pi-coding-agent/dist/core/extensions/loader.d.ts.map +1 -1
  506. package/packages/pi-coding-agent/dist/core/extensions/loader.js +28 -8
  507. package/packages/pi-coding-agent/dist/core/extensions/loader.js.map +1 -1
  508. package/packages/pi-coding-agent/dist/core/extensions/runner.d.ts +35 -2
  509. package/packages/pi-coding-agent/dist/core/extensions/runner.d.ts.map +1 -1
  510. package/packages/pi-coding-agent/dist/core/extensions/runner.js +233 -0
  511. package/packages/pi-coding-agent/dist/core/extensions/runner.js.map +1 -1
  512. package/packages/pi-coding-agent/dist/core/extensions/types.d.ts +212 -2
  513. package/packages/pi-coding-agent/dist/core/extensions/types.d.ts.map +1 -1
  514. package/packages/pi-coding-agent/dist/core/extensions/types.js.map +1 -1
  515. package/packages/pi-coding-agent/dist/core/hooks-runner.d.ts +53 -0
  516. package/packages/pi-coding-agent/dist/core/hooks-runner.d.ts.map +1 -0
  517. package/packages/pi-coding-agent/dist/core/hooks-runner.js +337 -0
  518. package/packages/pi-coding-agent/dist/core/hooks-runner.js.map +1 -0
  519. package/packages/pi-coding-agent/dist/core/hooks-runner.test.d.ts +2 -0
  520. package/packages/pi-coding-agent/dist/core/hooks-runner.test.d.ts.map +1 -0
  521. package/packages/pi-coding-agent/dist/core/hooks-runner.test.js +234 -0
  522. package/packages/pi-coding-agent/dist/core/hooks-runner.test.js.map +1 -0
  523. package/packages/pi-coding-agent/dist/core/index.d.ts +1 -0
  524. package/packages/pi-coding-agent/dist/core/index.d.ts.map +1 -1
  525. package/packages/pi-coding-agent/dist/core/index.js +1 -0
  526. package/packages/pi-coding-agent/dist/core/index.js.map +1 -1
  527. package/packages/pi-coding-agent/dist/core/lsp/lsp-integration.test.js +11 -0
  528. package/packages/pi-coding-agent/dist/core/lsp/lsp-integration.test.js.map +1 -1
  529. package/packages/pi-coding-agent/dist/core/model-discovery.d.ts +3 -1
  530. package/packages/pi-coding-agent/dist/core/model-discovery.d.ts.map +1 -1
  531. package/packages/pi-coding-agent/dist/core/model-discovery.js +92 -12
  532. package/packages/pi-coding-agent/dist/core/model-discovery.js.map +1 -1
  533. package/packages/pi-coding-agent/dist/core/model-discovery.test.js +16 -1
  534. package/packages/pi-coding-agent/dist/core/model-discovery.test.js.map +1 -1
  535. package/packages/pi-coding-agent/dist/core/model-registry-auth-header.test.d.ts +2 -0
  536. package/packages/pi-coding-agent/dist/core/model-registry-auth-header.test.d.ts.map +1 -0
  537. package/packages/pi-coding-agent/dist/core/model-registry-auth-header.test.js +40 -0
  538. package/packages/pi-coding-agent/dist/core/model-registry-auth-header.test.js.map +1 -0
  539. package/packages/pi-coding-agent/dist/core/model-registry-auth-mode.test.js +2 -2
  540. package/packages/pi-coding-agent/dist/core/model-registry-auth-mode.test.js.map +1 -1
  541. package/packages/pi-coding-agent/dist/core/model-registry-custom-caps.test.d.ts +2 -0
  542. package/packages/pi-coding-agent/dist/core/model-registry-custom-caps.test.d.ts.map +1 -0
  543. package/packages/pi-coding-agent/dist/core/model-registry-custom-caps.test.js +203 -0
  544. package/packages/pi-coding-agent/dist/core/model-registry-custom-caps.test.js.map +1 -0
  545. package/packages/pi-coding-agent/dist/core/model-registry-discovery.test.js +61 -1
  546. package/packages/pi-coding-agent/dist/core/model-registry-discovery.test.js.map +1 -1
  547. package/packages/pi-coding-agent/dist/core/model-registry.d.ts +5 -0
  548. package/packages/pi-coding-agent/dist/core/model-registry.d.ts.map +1 -1
  549. package/packages/pi-coding-agent/dist/core/model-registry.js +90 -10
  550. package/packages/pi-coding-agent/dist/core/model-registry.js.map +1 -1
  551. package/packages/pi-coding-agent/dist/core/redact-secrets.d.ts +2 -0
  552. package/packages/pi-coding-agent/dist/core/redact-secrets.d.ts.map +1 -0
  553. package/packages/pi-coding-agent/dist/core/redact-secrets.js +49 -0
  554. package/packages/pi-coding-agent/dist/core/redact-secrets.js.map +1 -0
  555. package/packages/pi-coding-agent/dist/core/redact-secrets.test.d.ts +2 -0
  556. package/packages/pi-coding-agent/dist/core/redact-secrets.test.d.ts.map +1 -0
  557. package/packages/pi-coding-agent/dist/core/redact-secrets.test.js +67 -0
  558. package/packages/pi-coding-agent/dist/core/redact-secrets.test.js.map +1 -0
  559. package/packages/pi-coding-agent/dist/core/resource-loader.js +1 -1
  560. package/packages/pi-coding-agent/dist/core/resource-loader.js.map +1 -1
  561. package/packages/pi-coding-agent/dist/core/sdk.d.ts +1 -0
  562. package/packages/pi-coding-agent/dist/core/sdk.d.ts.map +1 -1
  563. package/packages/pi-coding-agent/dist/core/sdk.js +4 -1
  564. package/packages/pi-coding-agent/dist/core/sdk.js.map +1 -1
  565. package/packages/pi-coding-agent/dist/core/sdk.test.js +19 -1
  566. package/packages/pi-coding-agent/dist/core/sdk.test.js.map +1 -1
  567. package/packages/pi-coding-agent/dist/core/session-manager.d.ts.map +1 -1
  568. package/packages/pi-coding-agent/dist/core/session-manager.js +10 -6
  569. package/packages/pi-coding-agent/dist/core/session-manager.js.map +1 -1
  570. package/packages/pi-coding-agent/dist/core/session-manager.test.js +45 -1
  571. package/packages/pi-coding-agent/dist/core/session-manager.test.js.map +1 -1
  572. package/packages/pi-coding-agent/dist/core/settings-manager.d.ts +55 -0
  573. package/packages/pi-coding-agent/dist/core/settings-manager.d.ts.map +1 -1
  574. package/packages/pi-coding-agent/dist/core/settings-manager.js.map +1 -1
  575. package/packages/pi-coding-agent/dist/core/system-prompt.js +3 -3
  576. package/packages/pi-coding-agent/dist/core/system-prompt.js.map +1 -1
  577. package/packages/pi-coding-agent/dist/core/tools/path-utils.test.js +2 -1
  578. package/packages/pi-coding-agent/dist/core/tools/path-utils.test.js.map +1 -1
  579. package/packages/pi-coding-agent/dist/index.d.ts +1 -1
  580. package/packages/pi-coding-agent/dist/index.d.ts.map +1 -1
  581. package/packages/pi-coding-agent/dist/index.js.map +1 -1
  582. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/provider-display-name.test.js +15 -6
  583. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/provider-display-name.test.js.map +1 -1
  584. package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.d.ts +1 -1
  585. package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.d.ts.map +1 -1
  586. package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.js +5 -4
  587. package/packages/pi-coding-agent/dist/modes/interactive/components/chat-frame.js.map +1 -1
  588. package/packages/pi-coding-agent/dist/modes/interactive/components/footer.d.ts.map +1 -1
  589. package/packages/pi-coding-agent/dist/modes/interactive/components/footer.js +14 -5
  590. package/packages/pi-coding-agent/dist/modes/interactive/components/footer.js.map +1 -1
  591. package/packages/pi-coding-agent/dist/modes/interactive/components/model-selector.d.ts +7 -1
  592. package/packages/pi-coding-agent/dist/modes/interactive/components/model-selector.d.ts.map +1 -1
  593. package/packages/pi-coding-agent/dist/modes/interactive/components/model-selector.js +31 -9
  594. package/packages/pi-coding-agent/dist/modes/interactive/components/model-selector.js.map +1 -1
  595. package/packages/pi-coding-agent/dist/modes/interactive/components/provider-manager.d.ts.map +1 -1
  596. package/packages/pi-coding-agent/dist/modes/interactive/components/provider-manager.js +13 -7
  597. package/packages/pi-coding-agent/dist/modes/interactive/components/provider-manager.js.map +1 -1
  598. package/packages/pi-coding-agent/dist/modes/interactive/components/skill-invocation-message.d.ts +7 -6
  599. package/packages/pi-coding-agent/dist/modes/interactive/components/skill-invocation-message.d.ts.map +1 -1
  600. package/packages/pi-coding-agent/dist/modes/interactive/components/skill-invocation-message.js +29 -21
  601. package/packages/pi-coding-agent/dist/modes/interactive/components/skill-invocation-message.js.map +1 -1
  602. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.test.js +14 -0
  603. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.test.js.map +1 -1
  604. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
  605. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js +13 -1
  606. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js.map +1 -1
  607. package/packages/pi-coding-agent/package.json +6 -1
  608. package/packages/pi-coding-agent/src/core/agent-session-abort-order.test.ts +3 -2
  609. package/packages/pi-coding-agent/src/core/agent-session.ts +38 -2
  610. package/packages/pi-coding-agent/src/core/compaction/compaction.test.ts +368 -28
  611. package/packages/pi-coding-agent/src/core/compaction/compaction.ts +122 -6
  612. package/packages/pi-coding-agent/src/core/compaction/utils.ts +111 -13
  613. package/packages/pi-coding-agent/src/core/compaction-orchestrator.test.ts +154 -0
  614. package/packages/pi-coding-agent/src/core/compaction-orchestrator.ts +32 -18
  615. package/packages/pi-coding-agent/src/core/compaction-utils.test.ts +68 -1
  616. package/packages/pi-coding-agent/src/core/discovery-cache.test.ts +9 -18
  617. package/packages/pi-coding-agent/src/core/extensions/extension-discovery.ts +119 -0
  618. package/packages/pi-coding-agent/src/core/extensions/extension-registry.ts +222 -0
  619. package/packages/pi-coding-agent/src/core/extensions/index.ts +16 -0
  620. package/packages/pi-coding-agent/src/core/extensions/loader.ts +29 -11
  621. package/packages/pi-coding-agent/src/core/extensions/runner.ts +351 -0
  622. package/packages/pi-coding-agent/src/core/extensions/types.ts +266 -0
  623. package/packages/pi-coding-agent/src/core/hooks-runner.test.ts +269 -0
  624. package/packages/pi-coding-agent/src/core/hooks-runner.ts +460 -0
  625. package/packages/pi-coding-agent/src/core/index.ts +10 -0
  626. package/packages/pi-coding-agent/src/core/lsp/lsp-integration.test.ts +13 -0
  627. package/packages/pi-coding-agent/src/core/model-discovery.test.ts +19 -0
  628. package/packages/pi-coding-agent/src/core/model-discovery.ts +99 -12
  629. package/packages/pi-coding-agent/src/core/model-registry-auth-header.test.ts +44 -0
  630. package/packages/pi-coding-agent/src/core/model-registry-auth-mode.test.ts +2 -2
  631. package/packages/pi-coding-agent/src/core/model-registry-custom-caps.test.ts +245 -0
  632. package/packages/pi-coding-agent/src/core/model-registry-discovery.test.ts +75 -0
  633. package/packages/pi-coding-agent/src/core/model-registry.ts +102 -10
  634. package/packages/pi-coding-agent/src/core/redact-secrets.test.ts +86 -0
  635. package/packages/pi-coding-agent/src/core/redact-secrets.ts +58 -0
  636. package/packages/pi-coding-agent/src/core/resource-loader.ts +1 -1
  637. package/packages/pi-coding-agent/src/core/sdk.test.ts +25 -1
  638. package/packages/pi-coding-agent/src/core/sdk.ts +10 -3
  639. package/packages/pi-coding-agent/src/core/session-manager.test.ts +65 -1
  640. package/packages/pi-coding-agent/src/core/session-manager.ts +10 -6
  641. package/packages/pi-coding-agent/src/core/settings-manager.ts +57 -0
  642. package/packages/pi-coding-agent/src/core/system-prompt.ts +3 -3
  643. package/packages/pi-coding-agent/src/core/tools/path-utils.test.ts +2 -1
  644. package/packages/pi-coding-agent/src/index.ts +16 -0
  645. package/packages/pi-coding-agent/src/modes/interactive/components/__tests__/provider-display-name.test.ts +17 -7
  646. package/packages/pi-coding-agent/src/modes/interactive/components/chat-frame.ts +6 -6
  647. package/packages/pi-coding-agent/src/modes/interactive/components/footer.ts +14 -5
  648. package/packages/pi-coding-agent/src/modes/interactive/components/model-selector.ts +45 -11
  649. package/packages/pi-coding-agent/src/modes/interactive/components/provider-manager.ts +16 -7
  650. package/packages/pi-coding-agent/src/modes/interactive/components/skill-invocation-message.ts +36 -22
  651. package/packages/pi-coding-agent/src/modes/interactive/controllers/input-controller.test.ts +14 -0
  652. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode.ts +13 -1
  653. package/packages/pi-coding-agent/tsconfig.tsbuildinfo +1 -1
  654. package/packages/pi-tui/dist/__tests__/autocomplete.test.js +12 -5
  655. package/packages/pi-tui/dist/__tests__/autocomplete.test.js.map +1 -1
  656. package/packages/pi-tui/dist/__tests__/stdin-buffer.test.js +21 -0
  657. package/packages/pi-tui/dist/__tests__/stdin-buffer.test.js.map +1 -1
  658. package/packages/pi-tui/dist/stdin-buffer.d.ts +7 -0
  659. package/packages/pi-tui/dist/stdin-buffer.d.ts.map +1 -1
  660. package/packages/pi-tui/dist/stdin-buffer.js +20 -0
  661. package/packages/pi-tui/dist/stdin-buffer.js.map +1 -1
  662. package/packages/pi-tui/package.json +6 -1
  663. package/packages/pi-tui/src/__tests__/autocomplete.test.ts +20 -5
  664. package/packages/pi-tui/src/__tests__/stdin-buffer.test.ts +27 -0
  665. package/packages/pi-tui/src/stdin-buffer.ts +26 -0
  666. package/packages/pi-tui/tsconfig.tsbuildinfo +1 -1
  667. package/packages/rpc-client/package.json +6 -1
  668. package/pkg/package.json +1 -1
  669. package/scripts/install.js +512 -0
  670. package/scripts/lib/workspace-manifest.cjs +86 -0
  671. package/scripts/link-workspace-packages.cjs +5 -16
  672. package/scripts/postinstall.js +9 -178
  673. package/src/resources/agents/researcher.md +1 -1
  674. package/src/resources/extensions/claude-code-cli/readiness.ts +32 -8
  675. package/src/resources/extensions/claude-code-cli/stream-adapter.ts +158 -67
  676. package/src/resources/extensions/claude-code-cli/tests/stream-adapter.test.ts +251 -14
  677. package/src/resources/extensions/cmux/index.ts +35 -10
  678. package/src/resources/extensions/github-sync/templates.ts +151 -0
  679. package/src/resources/extensions/github-sync/tests/templates.test.ts +59 -0
  680. package/src/resources/extensions/google-search/extension-manifest.json +5 -4
  681. package/src/resources/extensions/google-search/index.ts +9 -470
  682. package/src/resources/extensions/gsd/abandon-detect.ts +62 -0
  683. package/src/resources/extensions/gsd/auto/loop-deps.ts +15 -1
  684. package/src/resources/extensions/gsd/auto/loop.ts +104 -2
  685. package/src/resources/extensions/gsd/auto/phases.ts +176 -27
  686. package/src/resources/extensions/gsd/auto/resolve.ts +29 -0
  687. package/src/resources/extensions/gsd/auto/run-unit.ts +56 -4
  688. package/src/resources/extensions/gsd/auto/session.ts +35 -2
  689. package/src/resources/extensions/gsd/auto/turn-epoch.ts +108 -0
  690. package/src/resources/extensions/gsd/auto/types.ts +1 -1
  691. package/src/resources/extensions/gsd/auto-dispatch.ts +117 -16
  692. package/src/resources/extensions/gsd/auto-loop.ts +1 -1
  693. package/src/resources/extensions/gsd/auto-model-selection.ts +71 -15
  694. package/src/resources/extensions/gsd/auto-post-unit.ts +92 -3
  695. package/src/resources/extensions/gsd/auto-prompts.ts +28 -1
  696. package/src/resources/extensions/gsd/auto-recovery.ts +40 -1
  697. package/src/resources/extensions/gsd/auto-start.ts +88 -74
  698. package/src/resources/extensions/gsd/auto-timeout-recovery.ts +12 -5
  699. package/src/resources/extensions/gsd/auto-unit-closeout.ts +14 -3
  700. package/src/resources/extensions/gsd/auto-verification.ts +33 -0
  701. package/src/resources/extensions/gsd/auto-worktree.ts +122 -68
  702. package/src/resources/extensions/gsd/auto.ts +115 -42
  703. package/src/resources/extensions/gsd/blocked-models.ts +98 -0
  704. package/src/resources/extensions/gsd/bootstrap/agent-end-recovery.ts +131 -1
  705. package/src/resources/extensions/gsd/bootstrap/db-tools.ts +40 -9
  706. package/src/resources/extensions/gsd/bootstrap/exec-tools.ts +109 -0
  707. package/src/resources/extensions/gsd/bootstrap/memory-tools.ts +5 -0
  708. package/src/resources/extensions/gsd/bootstrap/provider-error-resume.ts +6 -2
  709. package/src/resources/extensions/gsd/bootstrap/register-extension.ts +26 -0
  710. package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +54 -6
  711. package/src/resources/extensions/gsd/bootstrap/system-context.ts +102 -35
  712. package/src/resources/extensions/gsd/bootstrap/write-gate.ts +35 -2
  713. package/src/resources/extensions/gsd/clean-root-preflight.ts +111 -0
  714. package/src/resources/extensions/gsd/commands/handlers/workflow.ts +27 -8
  715. package/src/resources/extensions/gsd/commands-cmux.ts +10 -6
  716. package/src/resources/extensions/gsd/commands-extensions.ts +747 -41
  717. package/src/resources/extensions/gsd/commands-extract-learnings.ts +55 -90
  718. package/src/resources/extensions/gsd/commands-prefs-wizard.ts +898 -32
  719. package/src/resources/extensions/gsd/compaction-snapshot.ts +165 -0
  720. package/src/resources/extensions/gsd/complexity-classifier.ts +5 -3
  721. package/src/resources/extensions/gsd/db-writer.ts +88 -17
  722. package/src/resources/extensions/gsd/dispatch-guard.ts +26 -2
  723. package/src/resources/extensions/gsd/doctor-git-checks.ts +23 -27
  724. package/src/resources/extensions/gsd/doctor-providers.ts +59 -6
  725. package/src/resources/extensions/gsd/ecosystem/gsd-extension-api.ts +2 -0
  726. package/src/resources/extensions/gsd/error-classifier.ts +36 -3
  727. package/src/resources/extensions/gsd/exec-history.ts +153 -0
  728. package/src/resources/extensions/gsd/exec-sandbox.ts +326 -0
  729. package/src/resources/extensions/gsd/file-lock.ts +84 -11
  730. package/src/resources/extensions/gsd/git-service.ts +1 -0
  731. package/src/resources/extensions/gsd/gitignore.ts +2 -1
  732. package/src/resources/extensions/gsd/gsd-db.ts +186 -23
  733. package/src/resources/extensions/gsd/guided-flow-queue.ts +4 -1
  734. package/src/resources/extensions/gsd/guided-flow.ts +259 -10
  735. package/src/resources/extensions/gsd/health-widget.ts +3 -1
  736. package/src/resources/extensions/gsd/hook-emitter.ts +188 -0
  737. package/src/resources/extensions/gsd/init-wizard.ts +15 -1
  738. package/src/resources/extensions/gsd/journal.ts +29 -3
  739. package/src/resources/extensions/gsd/key-manager.ts +28 -0
  740. package/src/resources/extensions/gsd/memory-backfill.ts +140 -0
  741. package/src/resources/extensions/gsd/memory-store.ts +26 -0
  742. package/src/resources/extensions/gsd/milestone-actions.ts +18 -0
  743. package/src/resources/extensions/gsd/milestone-summary-classifier.ts +42 -0
  744. package/src/resources/extensions/gsd/model-router.ts +42 -1
  745. package/src/resources/extensions/gsd/notifications.ts +27 -15
  746. package/src/resources/extensions/gsd/pre-execution-checks.ts +46 -10
  747. package/src/resources/extensions/gsd/preferences-types.ts +46 -0
  748. package/src/resources/extensions/gsd/preferences-validation.ts +79 -0
  749. package/src/resources/extensions/gsd/preferences.ts +17 -17
  750. package/src/resources/extensions/gsd/prompt-loader.ts +30 -7
  751. package/src/resources/extensions/gsd/prompts/complete-milestone.md +1 -1
  752. package/src/resources/extensions/gsd/prompts/complete-slice.md +2 -2
  753. package/src/resources/extensions/gsd/prompts/debug-diagnose.md +2 -0
  754. package/src/resources/extensions/gsd/prompts/discuss-headless.md +29 -2
  755. package/src/resources/extensions/gsd/prompts/discuss.md +29 -2
  756. package/src/resources/extensions/gsd/prompts/doctor-heal.md +5 -4
  757. package/src/resources/extensions/gsd/prompts/execute-task.md +3 -2
  758. package/src/resources/extensions/gsd/prompts/parallel-research-slices.md +5 -2
  759. package/src/resources/extensions/gsd/prompts/plan-slice.md +1 -0
  760. package/src/resources/extensions/gsd/prompts/research-slice.md +1 -0
  761. package/src/resources/extensions/gsd/prompts/system.md +1 -0
  762. package/src/resources/extensions/gsd/reports.ts +5 -4
  763. package/src/resources/extensions/gsd/safety/evidence-collector.ts +119 -0
  764. package/src/resources/extensions/gsd/safety/file-change-validator.ts +17 -4
  765. package/src/resources/extensions/gsd/safety/safety-harness.ts +9 -0
  766. package/src/resources/extensions/gsd/state-transition-matrix.ts +152 -0
  767. package/src/resources/extensions/gsd/state.ts +80 -34
  768. package/src/resources/extensions/gsd/tests/auto-loop.test.ts +238 -4
  769. package/src/resources/extensions/gsd/tests/auto-mode-guards.test.ts +79 -0
  770. package/src/resources/extensions/gsd/tests/auto-model-selection.test.ts +95 -1
  771. package/src/resources/extensions/gsd/tests/auto-paused-session-validation.test.ts +12 -0
  772. package/src/resources/extensions/gsd/tests/auto-recovery.test.ts +122 -0
  773. package/src/resources/extensions/gsd/tests/auto-start-bootstrap-await-3420.test.ts +141 -0
  774. package/src/resources/extensions/gsd/tests/auto-start-clean-runtime-db-gated.test.ts +63 -0
  775. package/src/resources/extensions/gsd/tests/auto-start-model-capture.test.ts +33 -3
  776. package/src/resources/extensions/gsd/tests/auto-thinking-restore.test.ts +38 -0
  777. package/src/resources/extensions/gsd/tests/auto-wrapup-inflight-guard.test.ts +23 -0
  778. package/src/resources/extensions/gsd/tests/blocked-models.test.ts +98 -0
  779. package/src/resources/extensions/gsd/tests/bundled-skill-triggers.test.ts +54 -0
  780. package/src/resources/extensions/gsd/tests/clean-root-preflight.test.ts +186 -0
  781. package/src/resources/extensions/gsd/tests/cmux.test.ts +5 -9
  782. package/src/resources/extensions/gsd/tests/commands-extract-learnings.test.ts +68 -66
  783. package/src/resources/extensions/gsd/tests/compaction-snapshot.test.ts +123 -0
  784. package/src/resources/extensions/gsd/tests/complete-milestone-false-merge.test.ts +15 -0
  785. package/src/resources/extensions/gsd/tests/complete-milestone.test.ts +61 -1
  786. package/src/resources/extensions/gsd/tests/complete-slice.test.ts +2 -2
  787. package/src/resources/extensions/gsd/tests/complete-task.test.ts +2 -2
  788. package/src/resources/extensions/gsd/tests/complexity-classifier.test.ts +3 -3
  789. package/src/resources/extensions/gsd/tests/custom-engine-loop-integration.test.ts +161 -0
  790. package/src/resources/extensions/gsd/tests/db-access-guardrails.test.ts +1 -0
  791. package/src/resources/extensions/gsd/tests/derive-state-db.test.ts +42 -0
  792. package/src/resources/extensions/gsd/tests/derive-state-helpers.test.ts +8 -4
  793. package/src/resources/extensions/gsd/tests/derive-state.test.ts +1 -2
  794. package/src/resources/extensions/gsd/tests/dispatch-complete-milestone-guard.test.ts +14 -9
  795. package/src/resources/extensions/gsd/tests/dispatch-guard-summary-db-mismatch.test.ts +77 -0
  796. package/src/resources/extensions/gsd/tests/dispatch-guard.test.ts +25 -0
  797. package/src/resources/extensions/gsd/tests/dispatch-missing-task-plans.test.ts +14 -0
  798. package/src/resources/extensions/gsd/tests/doctor-providers.test.ts +148 -3
  799. package/src/resources/extensions/gsd/tests/double-merge-guard.test.ts +1 -1
  800. package/src/resources/extensions/gsd/tests/ensure-db-open.test.ts +306 -1
  801. package/src/resources/extensions/gsd/tests/escalation.test.ts +1 -1
  802. package/src/resources/extensions/gsd/tests/exec-history.test.ts +237 -0
  803. package/src/resources/extensions/gsd/tests/exec-sandbox.test.ts +210 -0
  804. package/src/resources/extensions/gsd/tests/execution-entry-missing-context-4671.test.ts +173 -0
  805. package/src/resources/extensions/gsd/tests/file-change-validator.test.ts +58 -0
  806. package/src/resources/extensions/gsd/tests/file-lock.test.ts +86 -12
  807. package/src/resources/extensions/gsd/tests/flat-rate-routing-guard.test.ts +40 -9
  808. package/src/resources/extensions/gsd/tests/freeform-decisions.test.ts +62 -0
  809. package/src/resources/extensions/gsd/tests/google-search-stub.test.ts +131 -0
  810. package/src/resources/extensions/gsd/tests/gsd-db.test.ts +447 -1
  811. package/src/resources/extensions/gsd/tests/headless-milestone-parity.test.ts +117 -0
  812. package/src/resources/extensions/gsd/tests/init-wizard.test.ts +27 -0
  813. package/src/resources/extensions/gsd/tests/integration/auto-worktree-milestone-merge.test.ts +30 -0
  814. package/src/resources/extensions/gsd/tests/integration/doctor-git-symlink-cwd.test.ts +11 -0
  815. package/src/resources/extensions/gsd/tests/integration/doctor-git.test.ts +78 -0
  816. package/src/resources/extensions/gsd/tests/integration/git-service.test.ts +4 -2
  817. package/src/resources/extensions/gsd/tests/integration/gitignore-tracked-gsd.test.ts +1 -0
  818. package/src/resources/extensions/gsd/tests/integration/idle-recovery.test.ts +30 -0
  819. package/src/resources/extensions/gsd/tests/integration/worktree-e2e.test.ts +11 -0
  820. package/src/resources/extensions/gsd/tests/interactive-routing-bypass.test.ts +1 -1
  821. package/src/resources/extensions/gsd/tests/isolation-none-branch-guard.test.ts +1 -1
  822. package/src/resources/extensions/gsd/tests/issue-4540-regressions.test.ts +288 -0
  823. package/src/resources/extensions/gsd/tests/journal-integration.test.ts +101 -0
  824. package/src/resources/extensions/gsd/tests/key-manager.test.ts +9 -0
  825. package/src/resources/extensions/gsd/tests/load-memory-block.test.ts +36 -0
  826. package/src/resources/extensions/gsd/tests/mcp-client-security.test.ts +76 -0
  827. package/src/resources/extensions/gsd/tests/md-importer.test.ts +1 -1
  828. package/src/resources/extensions/gsd/tests/memory-pressure-stuck-state.test.ts +12 -0
  829. package/src/resources/extensions/gsd/tests/memory-store.test.ts +2 -2
  830. package/src/resources/extensions/gsd/tests/milestone-status-authoritative.test.ts +3 -3
  831. package/src/resources/extensions/gsd/tests/milestone-summary-classifier.test.ts +30 -0
  832. package/src/resources/extensions/gsd/tests/milestone-transition-state-rebuild.test.ts +4 -2
  833. package/src/resources/extensions/gsd/tests/parallel-commit-scope.test.ts +5 -0
  834. package/src/resources/extensions/gsd/tests/parallel-research-dispatch.test.ts +19 -0
  835. package/src/resources/extensions/gsd/tests/parallel-skill-prompt-integration.test.ts +150 -0
  836. package/src/resources/extensions/gsd/tests/plan-gate-failed-doctor-heal-hint.test.ts +37 -0
  837. package/src/resources/extensions/gsd/tests/pre-exec-backtick-strip.test.ts +14 -0
  838. package/src/resources/extensions/gsd/tests/pre-exec-gate-loop.test.ts +272 -0
  839. package/src/resources/extensions/gsd/tests/pre-execution-checks.test.ts +356 -0
  840. package/src/resources/extensions/gsd/tests/preferences.test.ts +110 -0
  841. package/src/resources/extensions/gsd/tests/prefs-wizard-coverage.test.ts +44 -0
  842. package/src/resources/extensions/gsd/tests/prompt-loader-extension-dir.test.ts +49 -0
  843. package/src/resources/extensions/gsd/tests/provider-errors.test.ts +142 -29
  844. package/src/resources/extensions/gsd/tests/queue-auto-guard.test.ts +181 -0
  845. package/src/resources/extensions/gsd/tests/quick-auto-guard.test.ts +13 -7
  846. package/src/resources/extensions/gsd/tests/ready-phrase-no-files-4573.test.ts +388 -0
  847. package/src/resources/extensions/gsd/tests/require-slice-discussion-dispatch.test.ts +170 -0
  848. package/src/resources/extensions/gsd/tests/restore-tools-after-discuss.test.ts +9 -3
  849. package/src/resources/extensions/gsd/tests/resume-dispatch-worktree.test.ts +230 -0
  850. package/src/resources/extensions/gsd/tests/rewrite-docs-abandon-detect.test.ts +195 -0
  851. package/src/resources/extensions/gsd/tests/safety-harness-false-positives.test.ts +205 -0
  852. package/src/resources/extensions/gsd/tests/save-gate-result-render.test.ts +95 -0
  853. package/src/resources/extensions/gsd/tests/schema-v21-sequence.test.ts +413 -0
  854. package/src/resources/extensions/gsd/tests/session-start-footer.test.ts +32 -40
  855. package/src/resources/extensions/gsd/tests/stale-dirlistcache-4648.test.ts +112 -0
  856. package/src/resources/extensions/gsd/tests/stash-queued-context-files.test.ts +56 -0
  857. package/src/resources/extensions/gsd/tests/state-machine-full-walkthrough.test.ts +24 -0
  858. package/src/resources/extensions/gsd/tests/state-transition-matrix.test.ts +44 -0
  859. package/src/resources/extensions/gsd/tests/stuck-detection-coverage.test.ts +2 -2
  860. package/src/resources/extensions/gsd/tests/token-counter.test.ts +105 -1
  861. package/src/resources/extensions/gsd/tests/tool-compatibility.test.ts +107 -0
  862. package/src/resources/extensions/gsd/tests/triage-resolution.test.ts +50 -2
  863. package/src/resources/extensions/gsd/tests/turn-epoch.test.ts +162 -0
  864. package/src/resources/extensions/gsd/tests/uok-contracts.test.ts +51 -0
  865. package/src/resources/extensions/gsd/tests/uok-execution-graph.test.ts +16 -0
  866. package/src/resources/extensions/gsd/tests/uok-loop-adapter-writer.test.ts +65 -0
  867. package/src/resources/extensions/gsd/tests/uok-parity-report.test.ts +42 -0
  868. package/src/resources/extensions/gsd/tests/uok-plan-v2-wiring.test.ts +42 -2
  869. package/src/resources/extensions/gsd/tests/uok-writer.test.ts +75 -0
  870. package/src/resources/extensions/gsd/tests/validate-extension-package.test.ts +168 -0
  871. package/src/resources/extensions/gsd/tests/validate-milestone.test.ts +147 -8
  872. package/src/resources/extensions/gsd/tests/workflow-mcp.test.ts +25 -2
  873. package/src/resources/extensions/gsd/tests/workflow-tool-executors.test.ts +65 -2
  874. package/src/resources/extensions/gsd/tests/worktree-db.test.ts +35 -0
  875. package/src/resources/extensions/gsd/tests/worktree-journal-events.test.ts +6 -1
  876. package/src/resources/extensions/gsd/tests/worktree-resolver.test.ts +78 -5
  877. package/src/resources/extensions/gsd/tests/write-gate.test.ts +64 -0
  878. package/src/resources/extensions/gsd/tests/zombie-gsd-state.test.ts +3 -1
  879. package/src/resources/extensions/gsd/token-counter.ts +22 -5
  880. package/src/resources/extensions/gsd/tools/complete-milestone.ts +15 -9
  881. package/src/resources/extensions/gsd/tools/complete-slice.ts +38 -0
  882. package/src/resources/extensions/gsd/tools/complete-task.ts +49 -0
  883. package/src/resources/extensions/gsd/tools/exec-search-tool.ts +81 -0
  884. package/src/resources/extensions/gsd/tools/exec-tool.ts +183 -0
  885. package/src/resources/extensions/gsd/tools/memory-tools.ts +31 -1
  886. package/src/resources/extensions/gsd/tools/resume-tool.ts +40 -0
  887. package/src/resources/extensions/gsd/uok/audit.ts +20 -2
  888. package/src/resources/extensions/gsd/uok/contracts.ts +65 -0
  889. package/src/resources/extensions/gsd/uok/dispatch-envelope.ts +56 -0
  890. package/src/resources/extensions/gsd/uok/execution-graph.ts +22 -0
  891. package/src/resources/extensions/gsd/uok/gitops.ts +6 -1
  892. package/src/resources/extensions/gsd/uok/loop-adapter.ts +45 -10
  893. package/src/resources/extensions/gsd/uok/parity-report.ts +84 -0
  894. package/src/resources/extensions/gsd/uok/plan-v2.ts +39 -8
  895. package/src/resources/extensions/gsd/uok/writer.ts +113 -0
  896. package/src/resources/extensions/gsd/workflow-logger.ts +25 -3
  897. package/src/resources/extensions/gsd/workflow-mcp.ts +3 -0
  898. package/src/resources/extensions/gsd/workflow-templates/spike.md +6 -0
  899. package/src/resources/extensions/gsd/worktree-manager.ts +1 -0
  900. package/src/resources/extensions/gsd/worktree-resolver.ts +54 -9
  901. package/src/resources/extensions/mcp-client/auth.ts +12 -1
  902. package/src/resources/extensions/mcp-client/index.ts +129 -10
  903. package/src/resources/extensions/search-the-web/command-search-provider.ts +5 -4
  904. package/src/resources/extensions/search-the-web/native-search.ts +48 -12
  905. package/src/resources/extensions/shared/cmux-events.ts +59 -0
  906. package/src/resources/extensions/shared/rtk-session-stats.ts +1 -2
  907. package/src/resources/skills/api-design/SKILL.md +190 -0
  908. package/src/resources/skills/create-mcp-server/SKILL.md +121 -0
  909. package/src/resources/skills/create-skill/SKILL.md +2 -2
  910. package/src/resources/skills/create-skill/references/gsd-skill-ecosystem.md +4 -4
  911. package/src/resources/skills/create-skill/workflows/audit-skill.md +4 -4
  912. package/src/resources/skills/create-skill/workflows/create-new-skill.md +5 -5
  913. package/src/resources/skills/decompose-into-slices/SKILL.md +139 -0
  914. package/src/resources/skills/dependency-upgrade/SKILL.md +158 -0
  915. package/src/resources/skills/design-an-interface/SKILL.md +102 -0
  916. package/src/resources/skills/forensics/SKILL.md +153 -0
  917. package/src/resources/skills/grill-me/SKILL.md +93 -0
  918. package/src/resources/skills/handoff/SKILL.md +121 -0
  919. package/src/resources/skills/observability/SKILL.md +174 -0
  920. package/src/resources/skills/security-review/SKILL.md +181 -0
  921. package/src/resources/skills/spike-wrap-up/SKILL.md +138 -0
  922. package/src/resources/skills/tdd/SKILL.md +112 -0
  923. package/src/resources/skills/verify-before-complete/SKILL.md +98 -0
  924. package/src/resources/skills/write-docs/SKILL.md +82 -0
  925. package/src/resources/skills/write-milestone-brief/SKILL.md +135 -0
  926. package/dist/web/standalone/.next/server/chunks/7461.js +0 -1
  927. package/dist/web/standalone/.next/static/chunks/2826.e59e8578e2e28639.js +0 -9
  928. package/dist/web/standalone/.next/static/chunks/app/page-151349214571e2b6.js +0 -1
  929. package/dist/web/standalone/.next/static/chunks/main-app-d3d4c336195465f9.js +0 -1
  930. package/dist/web/standalone/.next/static/chunks/next/dist/client/components/builtin/global-error-ab5a8926e07ec673.js +0 -1
  931. /package/dist/web/standalone/.next/static/{ssX7BLv3Dw9Fb4CtrCGeR → vidAVJkURvTJ0_V2-64ro}/_buildManifest.js +0 -0
  932. /package/dist/web/standalone/.next/static/{ssX7BLv3Dw9Fb4CtrCGeR → vidAVJkURvTJ0_V2-64ro}/_ssgManifest.js +0 -0
@@ -1 +1 @@
1
- {"version":3,"file":"google-gemini-cli.d.ts","sourceRoot":"","sources":["../../../src/utils/oauth/google-gemini-cli.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AAUH,OAAO,KAAK,EAAE,gBAAgB,EAAuB,sBAAsB,EAAE,MAAM,YAAY,CAAC;AAoPhG;;GAEG;AACH,wBAAsB,uBAAuB,CAAC,YAAY,EAAE,MAAM,EAAE,SAAS,EAAE,MAAM,GAAG,OAAO,CAAC,gBAAgB,CAAC,CAEhH;AAED;;;;;;;GAOG;AACH,wBAAsB,cAAc,CACnC,MAAM,EAAE,CAAC,IAAI,EAAE;IAAE,GAAG,EAAE,MAAM,CAAC;IAAC,YAAY,CAAC,EAAE,MAAM,CAAA;CAAE,KAAK,IAAI,EAC9D,UAAU,CAAC,EAAE,CAAC,OAAO,EAAE,MAAM,KAAK,IAAI,EACtC,iBAAiB,CAAC,EAAE,MAAM,OAAO,CAAC,MAAM,CAAC,GACvC,OAAO,CAAC,gBAAgB,CAAC,CA0J3B;AAED,eAAO,MAAM,sBAAsB,EAAE,sBAqBpC,CAAC"}
1
+ {"version":3,"file":"google-gemini-cli.d.ts","sourceRoot":"","sources":["../../../src/utils/oauth/google-gemini-cli.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AAUH,OAAO,KAAK,EAAE,gBAAgB,EAAuB,sBAAsB,EAAE,MAAM,YAAY,CAAC;AA+PhG;;GAEG;AACH,wBAAsB,uBAAuB,CAAC,YAAY,EAAE,MAAM,EAAE,SAAS,EAAE,MAAM,GAAG,OAAO,CAAC,gBAAgB,CAAC,CAEhH;AAED;;;;;;;GAOG;AACH,wBAAsB,cAAc,CACnC,MAAM,EAAE,CAAC,IAAI,EAAE;IAAE,GAAG,EAAE,MAAM,CAAC;IAAC,YAAY,CAAC,EAAE,MAAM,CAAA;CAAE,KAAK,IAAI,EAC9D,UAAU,CAAC,EAAE,CAAC,OAAO,EAAE,MAAM,KAAK,IAAI,EACtC,iBAAiB,CAAC,EAAE,MAAM,OAAO,CAAC,MAAM,CAAC,GACvC,OAAO,CAAC,gBAAgB,CAAC,CA0J3B;AAED,eAAO,MAAM,sBAAsB,EAAE,sBAqBpC,CAAC"}
@@ -7,9 +7,22 @@
7
7
  */
8
8
  import { getGoogleUserEmail, parseRedirectUrl, refreshGoogleOAuthToken, startCallbackServer, } from "./google-oauth-utils.js";
9
9
  import { generatePKCE } from "./pkce.js";
10
- const decode = (s) => atob(s);
11
- const CLIENT_ID = decode("NjgxMjU1ODA5Mzk1LW9vOGZ0Mm9wcmRybnA5ZTNhcWY2YXYzaG1kaWIxMzVqLmFwcHMuZ29vZ2xldXNlcmNvbnRlbnQuY29t");
12
- const CLIENT_SECRET = decode("R09DU1BYLTR1SGdNUG0tMW83U2stZ2VWNkN1NWNsWEZzeGw=");
10
+ // Gemini CLI OAuth credentials (Google Cloud Code Assist)
11
+ //
12
+ // NOTE: These credentials are public in the source code. They should NOT be
13
+ // obfuscated because security scanners flag obfuscated data as potentially
14
+ // malicious (see: https://socket.dev/npm/package/gsd-pi/alerts/2.70.1?alert_name=obfuscatedFile)
15
+ //
16
+ // Google's OAuth implementation requires client_secret for Desktop App OAuth
17
+ // clients even though it cannot be kept secret in distributed applications.
18
+ // The actual security relies on:
19
+ // - PKCE (Proof Key for Code Exchange) via code_verifier
20
+ // - Redirect URI validation (localhost only)
21
+ // - User consent and token scope limits
22
+ //
23
+ // See: https://developers.google.com/identity/protocols/oauth2/native-app
24
+ const CLIENT_ID = "681255809395-oo8ft2oprdrnp9e3aqf6av3hmdib135j.apps.googleusercontent.com";
25
+ const CLIENT_SECRET = "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";
13
26
  const REDIRECT_URI = "http://localhost:8085/oauth2callback";
14
27
  const SCOPES = [
15
28
  "https://www.googleapis.com/auth/cloud-platform",
@@ -1 +1 @@
1
- {"version":3,"file":"google-gemini-cli.js","sourceRoot":"","sources":["../../../src/utils/oauth/google-gemini-cli.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Gemini CLI OAuth flow (Google Cloud Code Assist)\n * Standard Gemini models only (gemini-2.0-flash, gemini-2.5-*)\n *\n * NOTE: This module uses Node.js http.createServer for the OAuth callback.\n * It is only intended for CLI use, not browser environments.\n */\n\nimport {\n\ttype CallbackServerInfo,\n\tgetGoogleUserEmail,\n\tparseRedirectUrl,\n\trefreshGoogleOAuthToken,\n\tstartCallbackServer,\n} from \"./google-oauth-utils.js\";\nimport { generatePKCE } from \"./pkce.js\";\nimport type { OAuthCredentials, OAuthLoginCallbacks, OAuthProviderInterface } from \"./types.js\";\n\ntype GeminiCredentials = OAuthCredentials & {\n\tprojectId: string;\n};\n\nconst decode = (s: string) => atob(s);\nconst CLIENT_ID = decode(\n\t\"NjgxMjU1ODA5Mzk1LW9vOGZ0Mm9wcmRybnA5ZTNhcWY2YXYzaG1kaWIxMzVqLmFwcHMuZ29vZ2xldXNlcmNvbnRlbnQuY29t\",\n);\nconst CLIENT_SECRET = decode(\"R09DU1BYLTR1SGdNUG0tMW83U2stZ2VWNkN1NWNsWEZzeGw=\");\nconst REDIRECT_URI = \"http://localhost:8085/oauth2callback\";\nconst SCOPES = [\n\t\"https://www.googleapis.com/auth/cloud-platform\",\n\t\"https://www.googleapis.com/auth/userinfo.email\",\n\t\"https://www.googleapis.com/auth/userinfo.profile\",\n];\nconst AUTH_URL = \"https://accounts.google.com/o/oauth2/v2/auth\";\nconst TOKEN_URL = \"https://oauth2.googleapis.com/token\";\nconst CODE_ASSIST_ENDPOINT = \"https://cloudcode-pa.googleapis.com\";\n\n// Callback server configuration\nconst CALLBACK_PORT = 8085;\nconst CALLBACK_PATH = \"/oauth2callback\";\n\ninterface LoadCodeAssistPayload {\n\tcloudaicompanionProject?: string;\n\tcurrentTier?: { id?: string };\n\tallowedTiers?: Array<{ id?: string; isDefault?: boolean }>;\n}\n\n/**\n * Long-running operation response from onboardUser\n */\ninterface LongRunningOperationResponse {\n\tname?: string;\n\tdone?: boolean;\n\tresponse?: {\n\t\tcloudaicompanionProject?: { id?: string };\n\t};\n}\n\n// Tier IDs as used by the Cloud Code API\nconst TIER_FREE = \"free-tier\";\nconst TIER_LEGACY = \"legacy-tier\";\nconst TIER_STANDARD = \"standard-tier\";\n\ninterface GoogleRpcErrorResponse {\n\terror?: {\n\t\tdetails?: Array<{ reason?: string }>;\n\t};\n}\n\n/**\n * Wait helper for onboarding retries\n */\nfunction wait(ms: number): Promise<void> {\n\treturn new Promise((resolve) => setTimeout(resolve, ms));\n}\n\n/**\n * Get default tier from allowed tiers\n */\nfunction getDefaultTier(allowedTiers?: Array<{ id?: string; isDefault?: boolean }>): { id?: string } {\n\tif (!allowedTiers || allowedTiers.length === 0) return { id: TIER_LEGACY };\n\tconst defaultTier = allowedTiers.find((t) => t.isDefault);\n\treturn defaultTier ?? { id: TIER_LEGACY };\n}\n\nfunction isVpcScAffectedUser(payload: unknown): boolean {\n\tif (!payload || typeof payload !== \"object\") return false;\n\tif (!(\"error\" in payload)) return false;\n\tconst error = (payload as GoogleRpcErrorResponse).error;\n\tif (!error?.details || !Array.isArray(error.details)) return false;\n\treturn error.details.some((detail) => detail.reason === \"SECURITY_POLICY_VIOLATED\");\n}\n\n/**\n * Poll a long-running operation until completion\n */\nasync function pollOperation(\n\toperationName: string,\n\theaders: Record<string, string>,\n\tonProgress?: (message: string) => void,\n): Promise<LongRunningOperationResponse> {\n\tlet attempt = 0;\n\twhile (true) {\n\t\tif (attempt > 0) {\n\t\t\tonProgress?.(`Waiting for project provisioning (attempt ${attempt + 1})...`);\n\t\t\tawait wait(5000);\n\t\t}\n\n\t\tconst response = await fetch(`${CODE_ASSIST_ENDPOINT}/v1internal/${operationName}`, {\n\t\t\tmethod: \"GET\",\n\t\t\theaders,\n\t\t\tsignal: AbortSignal.timeout(30_000),\n\t\t});\n\n\t\tif (!response.ok) {\n\t\t\tthrow new Error(`Failed to poll operation: ${response.status} ${response.statusText}`);\n\t\t}\n\n\t\tconst data = (await response.json()) as LongRunningOperationResponse;\n\t\tif (data.done) {\n\t\t\treturn data;\n\t\t}\n\n\t\tattempt += 1;\n\t}\n}\n\n/**\n * Discover or provision a Google Cloud project for the user\n */\nasync function discoverProject(accessToken: string, onProgress?: (message: string) => void): Promise<string> {\n\t// Check for user-provided project ID via environment variable\n\tconst envProjectId = process.env.GOOGLE_CLOUD_PROJECT || process.env.GOOGLE_CLOUD_PROJECT_ID;\n\n\tconst headers = {\n\t\tAuthorization: `Bearer ${accessToken}`,\n\t\t\"Content-Type\": \"application/json\",\n\t\t\"User-Agent\": \"google-api-nodejs-client/9.15.1\",\n\t\t\"X-Goog-Api-Client\": \"gl-node/22.17.0\",\n\t};\n\n\t// Try to load existing project via loadCodeAssist\n\tonProgress?.(\"Checking for existing Cloud Code Assist project...\");\n\tconst loadResponse = await fetch(`${CODE_ASSIST_ENDPOINT}/v1internal:loadCodeAssist`, {\n\t\tmethod: \"POST\",\n\t\theaders,\n\t\tbody: JSON.stringify({\n\t\t\tcloudaicompanionProject: envProjectId,\n\t\t\tmetadata: {\n\t\t\t\tideType: \"IDE_UNSPECIFIED\",\n\t\t\t\tplatform: \"PLATFORM_UNSPECIFIED\",\n\t\t\t\tpluginType: \"GEMINI\",\n\t\t\t\tduetProject: envProjectId,\n\t\t\t},\n\t\t}),\n\t\tsignal: AbortSignal.timeout(30_000),\n\t});\n\n\tlet data: LoadCodeAssistPayload;\n\n\tif (!loadResponse.ok) {\n\t\tlet errorPayload: unknown;\n\t\ttry {\n\t\t\terrorPayload = await loadResponse.clone().json();\n\t\t} catch {\n\t\t\terrorPayload = undefined;\n\t\t}\n\n\t\tif (isVpcScAffectedUser(errorPayload)) {\n\t\t\tdata = { currentTier: { id: TIER_STANDARD } };\n\t\t} else {\n\t\t\tconst errorText = await loadResponse.text();\n\t\t\tthrow new Error(`loadCodeAssist failed: ${loadResponse.status} ${loadResponse.statusText}: ${errorText}`);\n\t\t}\n\t} else {\n\t\tdata = (await loadResponse.json()) as LoadCodeAssistPayload;\n\t}\n\n\t// If user already has a current tier and project, use it\n\tif (data.currentTier) {\n\t\tif (data.cloudaicompanionProject) {\n\t\t\treturn data.cloudaicompanionProject;\n\t\t}\n\t\t// User has a tier but no managed project - they need to provide one via env var\n\t\tif (envProjectId) {\n\t\t\treturn envProjectId;\n\t\t}\n\t\tthrow new Error(\n\t\t\t\"This account requires setting the GOOGLE_CLOUD_PROJECT or GOOGLE_CLOUD_PROJECT_ID environment variable. \" +\n\t\t\t\t\"See https://goo.gle/gemini-cli-auth-docs#workspace-gca\",\n\t\t);\n\t}\n\n\t// User needs to be onboarded - get the default tier\n\tconst tier = getDefaultTier(data.allowedTiers);\n\tconst tierId = tier?.id ?? TIER_FREE;\n\n\tif (tierId !== TIER_FREE && !envProjectId) {\n\t\tthrow new Error(\n\t\t\t\"This account requires setting the GOOGLE_CLOUD_PROJECT or GOOGLE_CLOUD_PROJECT_ID environment variable. \" +\n\t\t\t\t\"See https://goo.gle/gemini-cli-auth-docs#workspace-gca\",\n\t\t);\n\t}\n\n\tonProgress?.(\"Provisioning Cloud Code Assist project (this may take a moment)...\");\n\n\t// Build onboard request - for free tier, don't include project ID (Google provisions one)\n\t// For other tiers, include the user's project ID if available\n\tconst onboardBody: Record<string, unknown> = {\n\t\ttierId,\n\t\tmetadata: {\n\t\t\tideType: \"IDE_UNSPECIFIED\",\n\t\t\tplatform: \"PLATFORM_UNSPECIFIED\",\n\t\t\tpluginType: \"GEMINI\",\n\t\t},\n\t};\n\n\tif (tierId !== TIER_FREE && envProjectId) {\n\t\tonboardBody.cloudaicompanionProject = envProjectId;\n\t\t(onboardBody.metadata as Record<string, unknown>).duetProject = envProjectId;\n\t}\n\n\t// Start onboarding - this returns a long-running operation\n\tconst onboardResponse = await fetch(`${CODE_ASSIST_ENDPOINT}/v1internal:onboardUser`, {\n\t\tmethod: \"POST\",\n\t\theaders,\n\t\tbody: JSON.stringify(onboardBody),\n\t\tsignal: AbortSignal.timeout(30_000),\n\t});\n\n\tif (!onboardResponse.ok) {\n\t\tconst errorText = await onboardResponse.text();\n\t\tthrow new Error(`onboardUser failed: ${onboardResponse.status} ${onboardResponse.statusText}: ${errorText}`);\n\t}\n\n\tlet lroData = (await onboardResponse.json()) as LongRunningOperationResponse;\n\n\t// If the operation isn't done yet, poll until completion\n\tif (!lroData.done && lroData.name) {\n\t\tlroData = await pollOperation(lroData.name, headers, onProgress);\n\t}\n\n\t// Try to get project ID from the response\n\tconst projectId = lroData.response?.cloudaicompanionProject?.id;\n\tif (projectId) {\n\t\treturn projectId;\n\t}\n\n\t// If no project ID from onboarding, fall back to env var\n\tif (envProjectId) {\n\t\treturn envProjectId;\n\t}\n\n\tthrow new Error(\n\t\t\"Could not discover or provision a Google Cloud project. \" +\n\t\t\t\"Try setting the GOOGLE_CLOUD_PROJECT or GOOGLE_CLOUD_PROJECT_ID environment variable. \" +\n\t\t\t\"See https://goo.gle/gemini-cli-auth-docs#workspace-gca\",\n\t);\n}\n\n/**\n * Refresh Google Cloud Code Assist token\n */\nexport async function refreshGoogleCloudToken(refreshToken: string, projectId: string): Promise<OAuthCredentials> {\n\treturn refreshGoogleOAuthToken(refreshToken, CLIENT_ID, CLIENT_SECRET, \"Google Cloud\", { projectId });\n}\n\n/**\n * Login with Gemini CLI (Google Cloud Code Assist) OAuth\n *\n * @param onAuth - Callback with URL and optional instructions\n * @param onProgress - Optional progress callback\n * @param onManualCodeInput - Optional promise that resolves with user-pasted redirect URL.\n * Races with browser callback - whichever completes first wins.\n */\nexport async function loginGeminiCli(\n\tonAuth: (info: { url: string; instructions?: string }) => void,\n\tonProgress?: (message: string) => void,\n\tonManualCodeInput?: () => Promise<string>,\n): Promise<OAuthCredentials> {\n\tconst { verifier, challenge } = await generatePKCE();\n\n\t// Start local server for callback\n\tonProgress?.(\"Starting local server for OAuth callback...\");\n\tconst server: CallbackServerInfo = await startCallbackServer(CALLBACK_PORT, CALLBACK_PATH, \"Gemini CLI\");\n\n\tlet code: string | undefined;\n\n\ttry {\n\t\t// Build authorization URL\n\t\tconst authParams = new URLSearchParams({\n\t\t\tclient_id: CLIENT_ID,\n\t\t\tresponse_type: \"code\",\n\t\t\tredirect_uri: REDIRECT_URI,\n\t\t\tscope: SCOPES.join(\" \"),\n\t\t\tcode_challenge: challenge,\n\t\t\tcode_challenge_method: \"S256\",\n\t\t\tstate: verifier,\n\t\t\taccess_type: \"offline\",\n\t\t\tprompt: \"consent\",\n\t\t});\n\n\t\tconst authUrl = `${AUTH_URL}?${authParams.toString()}`;\n\n\t\t// Notify caller with URL to open\n\t\tonAuth({\n\t\t\turl: authUrl,\n\t\t\tinstructions: \"Complete the sign-in in your browser.\",\n\t\t});\n\n\t\t// Wait for the callback, racing with manual input if provided\n\t\tonProgress?.(\"Waiting for OAuth callback...\");\n\n\t\tif (onManualCodeInput) {\n\t\t\t// Race between browser callback and manual input\n\t\t\tlet manualInput: string | undefined;\n\t\t\tlet manualError: Error | undefined;\n\t\t\tconst manualPromise = onManualCodeInput()\n\t\t\t\t.then((input) => {\n\t\t\t\t\tmanualInput = input;\n\t\t\t\t\tserver.cancelWait();\n\t\t\t\t})\n\t\t\t\t.catch((err) => {\n\t\t\t\t\tmanualError = err instanceof Error ? err : new Error(String(err));\n\t\t\t\t\tserver.cancelWait();\n\t\t\t\t});\n\n\t\t\tconst result = await server.waitForCode();\n\n\t\t\t// If manual input was cancelled, throw that error\n\t\t\tif (manualError) {\n\t\t\t\tthrow manualError;\n\t\t\t}\n\n\t\t\tif (result?.code) {\n\t\t\t\t// Browser callback won - verify state\n\t\t\t\tif (result.state !== verifier) {\n\t\t\t\t\tthrow new Error(\"OAuth state mismatch - possible CSRF attack\");\n\t\t\t\t}\n\t\t\t\tcode = result.code;\n\t\t\t} else if (manualInput) {\n\t\t\t\t// Manual input won\n\t\t\t\tconst parsed = parseRedirectUrl(manualInput);\n\t\t\t\tif (parsed.state && parsed.state !== verifier) {\n\t\t\t\t\tthrow new Error(\"OAuth state mismatch - possible CSRF attack\");\n\t\t\t\t}\n\t\t\t\tcode = parsed.code;\n\t\t\t}\n\n\t\t\t// If still no code, wait for manual promise and try that\n\t\t\tif (!code) {\n\t\t\t\tawait manualPromise;\n\t\t\t\tif (manualError) {\n\t\t\t\t\tthrow manualError;\n\t\t\t\t}\n\t\t\t\tif (manualInput) {\n\t\t\t\t\tconst parsed = parseRedirectUrl(manualInput);\n\t\t\t\t\tif (parsed.state && parsed.state !== verifier) {\n\t\t\t\t\t\tthrow new Error(\"OAuth state mismatch - possible CSRF attack\");\n\t\t\t\t\t}\n\t\t\t\t\tcode = parsed.code;\n\t\t\t\t}\n\t\t\t}\n\t\t} else {\n\t\t\t// Original flow: just wait for callback\n\t\t\tconst result = await server.waitForCode();\n\t\t\tif (result?.code) {\n\t\t\t\tif (result.state !== verifier) {\n\t\t\t\t\tthrow new Error(\"OAuth state mismatch - possible CSRF attack\");\n\t\t\t\t}\n\t\t\t\tcode = result.code;\n\t\t\t}\n\t\t}\n\n\t\tif (!code) {\n\t\t\tthrow new Error(\"No authorization code received\");\n\t\t}\n\n\t\t// Exchange code for tokens\n\t\tonProgress?.(\"Exchanging authorization code for tokens...\");\n\t\tconst tokenResponse = await fetch(TOKEN_URL, {\n\t\t\tmethod: \"POST\",\n\t\t\theaders: {\n\t\t\t\t\"Content-Type\": \"application/x-www-form-urlencoded\",\n\t\t\t},\n\t\t\tbody: new URLSearchParams({\n\t\t\t\tclient_id: CLIENT_ID,\n\t\t\t\tclient_secret: CLIENT_SECRET,\n\t\t\t\tcode,\n\t\t\t\tgrant_type: \"authorization_code\",\n\t\t\t\tredirect_uri: REDIRECT_URI,\n\t\t\t\tcode_verifier: verifier,\n\t\t\t}),\n\t\t\tsignal: AbortSignal.timeout(30_000),\n\t\t});\n\n\t\tif (!tokenResponse.ok) {\n\t\t\tconst error = await tokenResponse.text();\n\t\t\tthrow new Error(`Token exchange failed: ${error}`);\n\t\t}\n\n\t\tconst tokenData = (await tokenResponse.json()) as {\n\t\t\taccess_token: string;\n\t\t\trefresh_token: string;\n\t\t\texpires_in: number;\n\t\t};\n\n\t\tif (!tokenData.refresh_token) {\n\t\t\tthrow new Error(\"No refresh token received. Please try again.\");\n\t\t}\n\n\t\t// Get user email\n\t\tonProgress?.(\"Getting user info...\");\n\t\tconst email = await getGoogleUserEmail(tokenData.access_token);\n\n\t\t// Discover project\n\t\tconst projectId = await discoverProject(tokenData.access_token, onProgress);\n\n\t\t// Calculate expiry time (current time + expires_in seconds - 5 min buffer)\n\t\tconst expiresAt = Date.now() + tokenData.expires_in * 1000 - 5 * 60 * 1000;\n\n\t\tconst credentials: OAuthCredentials = {\n\t\t\trefresh: tokenData.refresh_token,\n\t\t\taccess: tokenData.access_token,\n\t\t\texpires: expiresAt,\n\t\t\tprojectId,\n\t\t\temail,\n\t\t};\n\n\t\treturn credentials;\n\t} finally {\n\t\tserver.server.close();\n\t}\n}\n\nexport const geminiCliOAuthProvider: OAuthProviderInterface = {\n\tid: \"google-gemini-cli\",\n\tname: \"Google Cloud Code Assist (Gemini CLI)\",\n\tusesCallbackServer: true,\n\n\tasync login(callbacks: OAuthLoginCallbacks): Promise<OAuthCredentials> {\n\t\treturn loginGeminiCli(callbacks.onAuth, callbacks.onProgress, callbacks.onManualCodeInput);\n\t},\n\n\tasync refreshToken(credentials: OAuthCredentials): Promise<OAuthCredentials> {\n\t\tconst creds = credentials as GeminiCredentials;\n\t\tif (!creds.projectId) {\n\t\t\tthrow new Error(\"Google Cloud credentials missing projectId\");\n\t\t}\n\t\treturn refreshGoogleCloudToken(creds.refresh, creds.projectId);\n\t},\n\n\tgetApiKey(credentials: OAuthCredentials): string {\n\t\tconst creds = credentials as GeminiCredentials;\n\t\treturn JSON.stringify({ token: creds.access, projectId: creds.projectId });\n\t},\n};\n"]}
1
+ {"version":3,"file":"google-gemini-cli.js","sourceRoot":"","sources":["../../../src/utils/oauth/google-gemini-cli.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AAEH,OAAO,EAEN,kBAAkB,EAClB,gBAAgB,EAChB,uBAAuB,EACvB,mBAAmB,GACnB,MAAM,yBAAyB,CAAC;AACjC,OAAO,EAAE,YAAY,EAAE,MAAM,WAAW,CAAC;AAOzC,0DAA0D;AAC1D,EAAE;AACF,4EAA4E;AAC5E,2EAA2E;AAC3E,iGAAiG;AACjG,EAAE;AACF,6EAA6E;AAC7E,4EAA4E;AAC5E,iCAAiC;AACjC,yDAAyD;AACzD,6CAA6C;AAC7C,wCAAwC;AACxC,EAAE;AACF,0EAA0E;AAC1E,MAAM,SAAS,GAAG,0EAA0E,CAAC;AAC7F,MAAM,aAAa,GAAG,qCAAqC,CAAC;AAC5D,MAAM,YAAY,GAAG,sCAAsC,CAAC;AAC5D,MAAM,MAAM,GAAG;IACd,gDAAgD;IAChD,gDAAgD;IAChD,kDAAkD;CAClD,CAAC;AACF,MAAM,QAAQ,GAAG,8CAA8C,CAAC;AAChE,MAAM,SAAS,GAAG,qCAAqC,CAAC;AACxD,MAAM,oBAAoB,GAAG,qCAAqC,CAAC;AAEnE,gCAAgC;AAChC,MAAM,aAAa,GAAG,IAAI,CAAC;AAC3B,MAAM,aAAa,GAAG,iBAAiB,CAAC;AAmBxC,yCAAyC;AACzC,MAAM,SAAS,GAAG,WAAW,CAAC;AAC9B,MAAM,WAAW,GAAG,aAAa,CAAC;AAClC,MAAM,aAAa,GAAG,eAAe,CAAC;AAQtC;;GAEG;AACH,SAAS,IAAI,CAAC,EAAU;IACvB,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,EAAE,CAAC,UAAU,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC,CAAC;AAC1D,CAAC;AAED;;GAEG;AACH,SAAS,cAAc,CAAC,YAA0D;IACjF,IAAI,CAAC,YAAY,IAAI,YAAY,CAAC,MAAM,KAAK,CAAC;QAAE,OAAO,EAAE,EAAE,EAAE,WAAW,EAAE,CAAC;IAC3E,MAAM,WAAW,GAAG,YAAY,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC;IAC1D,OAAO,WAAW,IAAI,EAAE,EAAE,EAAE,WAAW,EAAE,CAAC;AAC3C,CAAC;AAED,SAAS,mBAAmB,CAAC,OAAgB;IAC5C,IAAI,CAAC,OAAO,IAAI,OAAO,OAAO,KAAK,QAAQ;QAAE,OAAO,KAAK,CAAC;IAC1D,IAAI,CAAC,CAAC,OAAO,IAAI,OAAO,CAAC;QAAE,OAAO,KAAK,CAAC;IACxC,MAAM,KAAK,GAAI,OAAkC,CAAC,KAAK,CAAC;IACxD,IAAI,CAAC,KAAK,EAAE,OAAO,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,OAAO,CAAC;QAAE,OAAO,KAAK,CAAC;IACnE,OAAO,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,MAAM,CAAC,MAAM,KAAK,0BAA0B,CAAC,CAAC;AACrF,CAAC;AAED;;GAEG;AACH,KAAK,UAAU,aAAa,CAC3B,aAAqB,EACrB,OAA+B,EAC/B,UAAsC;IAEtC,IAAI,OAAO,GAAG,CAAC,CAAC;IAChB,OAAO,IAAI,EAAE,CAAC;QACb,IAAI,OAAO,GAAG,CAAC,EAAE,CAAC;YACjB,UAAU,EAAE,CAAC,6CAA6C,OAAO,GAAG,CAAC,MAAM,CAAC,CAAC;YAC7E,MAAM,IAAI,CAAC,IAAI,CAAC,CAAC;QAClB,CAAC;QAED,MAAM,QAAQ,GAAG,MAAM,KAAK,CAAC,GAAG,oBAAoB,eAAe,aAAa,EAAE,EAAE;YACnF,MAAM,EAAE,KAAK;YACb,OAAO;YACP,MAAM,EAAE,WAAW,CAAC,OAAO,CAAC,MAAM,CAAC;SACnC,CAAC,CAAC;QAEH,IAAI,CAAC,QAAQ,CAAC,EAAE,EAAE,CAAC;YAClB,MAAM,IAAI,KAAK,CAAC,6BAA6B,QAAQ,CAAC,MAAM,IAAI,QAAQ,CAAC,UAAU,EAAE,CAAC,CAAC;QACxF,CAAC;QAED,MAAM,IAAI,GAAG,CAAC,MAAM,QAAQ,CAAC,IAAI,EAAE,CAAiC,CAAC;QACrE,IAAI,IAAI,CAAC,IAAI,EAAE,CAAC;YACf,OAAO,IAAI,CAAC;QACb,CAAC;QAED,OAAO,IAAI,CAAC,CAAC;IACd,CAAC;AACF,CAAC;AAED;;GAEG;AACH,KAAK,UAAU,eAAe,CAAC,WAAmB,EAAE,UAAsC;IACzF,8DAA8D;IAC9D,MAAM,YAAY,GAAG,OAAO,CAAC,GAAG,CAAC,oBAAoB,IAAI,OAAO,CAAC,GAAG,CAAC,uBAAuB,CAAC;IAE7F,MAAM,OAAO,GAAG;QACf,aAAa,EAAE,UAAU,WAAW,EAAE;QACtC,cAAc,EAAE,kBAAkB;QAClC,YAAY,EAAE,iCAAiC;QAC/C,mBAAmB,EAAE,iBAAiB;KACtC,CAAC;IAEF,kDAAkD;IAClD,UAAU,EAAE,CAAC,oDAAoD,CAAC,CAAC;IACnE,MAAM,YAAY,GAAG,MAAM,KAAK,CAAC,GAAG,oBAAoB,4BAA4B,EAAE;QACrF,MAAM,EAAE,MAAM;QACd,OAAO;QACP,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC;YACpB,uBAAuB,EAAE,YAAY;YACrC,QAAQ,EAAE;gBACT,OAAO,EAAE,iBAAiB;gBAC1B,QAAQ,EAAE,sBAAsB;gBAChC,UAAU,EAAE,QAAQ;gBACpB,WAAW,EAAE,YAAY;aACzB;SACD,CAAC;QACF,MAAM,EAAE,WAAW,CAAC,OAAO,CAAC,MAAM,CAAC;KACnC,CAAC,CAAC;IAEH,IAAI,IAA2B,CAAC;IAEhC,IAAI,CAAC,YAAY,CAAC,EAAE,EAAE,CAAC;QACtB,IAAI,YAAqB,CAAC;QAC1B,IAAI,CAAC;YACJ,YAAY,GAAG,MAAM,YAAY,CAAC,KAAK,EAAE,CAAC,IAAI,EAAE,CAAC;QAClD,CAAC;QAAC,MAAM,CAAC;YACR,YAAY,GAAG,SAAS,CAAC;QAC1B,CAAC;QAED,IAAI,mBAAmB,CAAC,YAAY,CAAC,EAAE,CAAC;YACvC,IAAI,GAAG,EAAE,WAAW,EAAE,EAAE,EAAE,EAAE,aAAa,EAAE,EAAE,CAAC;QAC/C,CAAC;aAAM,CAAC;YACP,MAAM,SAAS,GAAG,MAAM,YAAY,CAAC,IAAI,EAAE,CAAC;YAC5C,MAAM,IAAI,KAAK,CAAC,0BAA0B,YAAY,CAAC,MAAM,IAAI,YAAY,CAAC,UAAU,KAAK,SAAS,EAAE,CAAC,CAAC;QAC3G,CAAC;IACF,CAAC;SAAM,CAAC;QACP,IAAI,GAAG,CAAC,MAAM,YAAY,CAAC,IAAI,EAAE,CAA0B,CAAC;IAC7D,CAAC;IAED,yDAAyD;IACzD,IAAI,IAAI,CAAC,WAAW,EAAE,CAAC;QACtB,IAAI,IAAI,CAAC,uBAAuB,EAAE,CAAC;YAClC,OAAO,IAAI,CAAC,uBAAuB,CAAC;QACrC,CAAC;QACD,gFAAgF;QAChF,IAAI,YAAY,EAAE,CAAC;YAClB,OAAO,YAAY,CAAC;QACrB,CAAC;QACD,MAAM,IAAI,KAAK,CACd,0GAA0G;YACzG,wDAAwD,CACzD,CAAC;IACH,CAAC;IAED,oDAAoD;IACpD,MAAM,IAAI,GAAG,cAAc,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;IAC/C,MAAM,MAAM,GAAG,IAAI,EAAE,EAAE,IAAI,SAAS,CAAC;IAErC,IAAI,MAAM,KAAK,SAAS,IAAI,CAAC,YAAY,EAAE,CAAC;QAC3C,MAAM,IAAI,KAAK,CACd,0GAA0G;YACzG,wDAAwD,CACzD,CAAC;IACH,CAAC;IAED,UAAU,EAAE,CAAC,oEAAoE,CAAC,CAAC;IAEnF,0FAA0F;IAC1F,8DAA8D;IAC9D,MAAM,WAAW,GAA4B;QAC5C,MAAM;QACN,QAAQ,EAAE;YACT,OAAO,EAAE,iBAAiB;YAC1B,QAAQ,EAAE,sBAAsB;YAChC,UAAU,EAAE,QAAQ;SACpB;KACD,CAAC;IAEF,IAAI,MAAM,KAAK,SAAS,IAAI,YAAY,EAAE,CAAC;QAC1C,WAAW,CAAC,uBAAuB,GAAG,YAAY,CAAC;QAClD,WAAW,CAAC,QAAoC,CAAC,WAAW,GAAG,YAAY,CAAC;IAC9E,CAAC;IAED,2DAA2D;IAC3D,MAAM,eAAe,GAAG,MAAM,KAAK,CAAC,GAAG,oBAAoB,yBAAyB,EAAE;QACrF,MAAM,EAAE,MAAM;QACd,OAAO;QACP,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,WAAW,CAAC;QACjC,MAAM,EAAE,WAAW,CAAC,OAAO,CAAC,MAAM,CAAC;KACnC,CAAC,CAAC;IAEH,IAAI,CAAC,eAAe,CAAC,EAAE,EAAE,CAAC;QACzB,MAAM,SAAS,GAAG,MAAM,eAAe,CAAC,IAAI,EAAE,CAAC;QAC/C,MAAM,IAAI,KAAK,CAAC,uBAAuB,eAAe,CAAC,MAAM,IAAI,eAAe,CAAC,UAAU,KAAK,SAAS,EAAE,CAAC,CAAC;IAC9G,CAAC;IAED,IAAI,OAAO,GAAG,CAAC,MAAM,eAAe,CAAC,IAAI,EAAE,CAAiC,CAAC;IAE7E,yDAAyD;IACzD,IAAI,CAAC,OAAO,CAAC,IAAI,IAAI,OAAO,CAAC,IAAI,EAAE,CAAC;QACnC,OAAO,GAAG,MAAM,aAAa,CAAC,OAAO,CAAC,IAAI,EAAE,OAAO,EAAE,UAAU,CAAC,CAAC;IAClE,CAAC;IAED,0CAA0C;IAC1C,MAAM,SAAS,GAAG,OAAO,CAAC,QAAQ,EAAE,uBAAuB,EAAE,EAAE,CAAC;IAChE,IAAI,SAAS,EAAE,CAAC;QACf,OAAO,SAAS,CAAC;IAClB,CAAC;IAED,yDAAyD;IACzD,IAAI,YAAY,EAAE,CAAC;QAClB,OAAO,YAAY,CAAC;IACrB,CAAC;IAED,MAAM,IAAI,KAAK,CACd,0DAA0D;QACzD,wFAAwF;QACxF,wDAAwD,CACzD,CAAC;AACH,CAAC;AAED;;GAEG;AACH,MAAM,CAAC,KAAK,UAAU,uBAAuB,CAAC,YAAoB,EAAE,SAAiB;IACpF,OAAO,uBAAuB,CAAC,YAAY,EAAE,SAAS,EAAE,aAAa,EAAE,cAAc,EAAE,EAAE,SAAS,EAAE,CAAC,CAAC;AACvG,CAAC;AAED;;;;;;;GAOG;AACH,MAAM,CAAC,KAAK,UAAU,cAAc,CACnC,MAA8D,EAC9D,UAAsC,EACtC,iBAAyC;IAEzC,MAAM,EAAE,QAAQ,EAAE,SAAS,EAAE,GAAG,MAAM,YAAY,EAAE,CAAC;IAErD,kCAAkC;IAClC,UAAU,EAAE,CAAC,6CAA6C,CAAC,CAAC;IAC5D,MAAM,MAAM,GAAuB,MAAM,mBAAmB,CAAC,aAAa,EAAE,aAAa,EAAE,YAAY,CAAC,CAAC;IAEzG,IAAI,IAAwB,CAAC;IAE7B,IAAI,CAAC;QACJ,0BAA0B;QAC1B,MAAM,UAAU,GAAG,IAAI,eAAe,CAAC;YACtC,SAAS,EAAE,SAAS;YACpB,aAAa,EAAE,MAAM;YACrB,YAAY,EAAE,YAAY;YAC1B,KAAK,EAAE,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC;YACvB,cAAc,EAAE,SAAS;YACzB,qBAAqB,EAAE,MAAM;YAC7B,KAAK,EAAE,QAAQ;YACf,WAAW,EAAE,SAAS;YACtB,MAAM,EAAE,SAAS;SACjB,CAAC,CAAC;QAEH,MAAM,OAAO,GAAG,GAAG,QAAQ,IAAI,UAAU,CAAC,QAAQ,EAAE,EAAE,CAAC;QAEvD,iCAAiC;QACjC,MAAM,CAAC;YACN,GAAG,EAAE,OAAO;YACZ,YAAY,EAAE,uCAAuC;SACrD,CAAC,CAAC;QAEH,8DAA8D;QAC9D,UAAU,EAAE,CAAC,+BAA+B,CAAC,CAAC;QAE9C,IAAI,iBAAiB,EAAE,CAAC;YACvB,iDAAiD;YACjD,IAAI,WAA+B,CAAC;YACpC,IAAI,WAA8B,CAAC;YACnC,MAAM,aAAa,GAAG,iBAAiB,EAAE;iBACvC,IAAI,CAAC,CAAC,KAAK,EAAE,EAAE;gBACf,WAAW,GAAG,KAAK,CAAC;gBACpB,MAAM,CAAC,UAAU,EAAE,CAAC;YACrB,CAAC,CAAC;iBACD,KAAK,CAAC,CAAC,GAAG,EAAE,EAAE;gBACd,WAAW,GAAG,GAAG,YAAY,KAAK,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,IAAI,KAAK,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC;gBAClE,MAAM,CAAC,UAAU,EAAE,CAAC;YACrB,CAAC,CAAC,CAAC;YAEJ,MAAM,MAAM,GAAG,MAAM,MAAM,CAAC,WAAW,EAAE,CAAC;YAE1C,kDAAkD;YAClD,IAAI,WAAW,EAAE,CAAC;gBACjB,MAAM,WAAW,CAAC;YACnB,CAAC;YAED,IAAI,MAAM,EAAE,IAAI,EAAE,CAAC;gBAClB,sCAAsC;gBACtC,IAAI,MAAM,CAAC,KAAK,KAAK,QAAQ,EAAE,CAAC;oBAC/B,MAAM,IAAI,KAAK,CAAC,6CAA6C,CAAC,CAAC;gBAChE,CAAC;gBACD,IAAI,GAAG,MAAM,CAAC,IAAI,CAAC;YACpB,CAAC;iBAAM,IAAI,WAAW,EAAE,CAAC;gBACxB,mBAAmB;gBACnB,MAAM,MAAM,GAAG,gBAAgB,CAAC,WAAW,CAAC,CAAC;gBAC7C,IAAI,MAAM,CAAC,KAAK,IAAI,MAAM,CAAC,KAAK,KAAK,QAAQ,EAAE,CAAC;oBAC/C,MAAM,IAAI,KAAK,CAAC,6CAA6C,CAAC,CAAC;gBAChE,CAAC;gBACD,IAAI,GAAG,MAAM,CAAC,IAAI,CAAC;YACpB,CAAC;YAED,yDAAyD;YACzD,IAAI,CAAC,IAAI,EAAE,CAAC;gBACX,MAAM,aAAa,CAAC;gBACpB,IAAI,WAAW,EAAE,CAAC;oBACjB,MAAM,WAAW,CAAC;gBACnB,CAAC;gBACD,IAAI,WAAW,EAAE,CAAC;oBACjB,MAAM,MAAM,GAAG,gBAAgB,CAAC,WAAW,CAAC,CAAC;oBAC7C,IAAI,MAAM,CAAC,KAAK,IAAI,MAAM,CAAC,KAAK,KAAK,QAAQ,EAAE,CAAC;wBAC/C,MAAM,IAAI,KAAK,CAAC,6CAA6C,CAAC,CAAC;oBAChE,CAAC;oBACD,IAAI,GAAG,MAAM,CAAC,IAAI,CAAC;gBACpB,CAAC;YACF,CAAC;QACF,CAAC;aAAM,CAAC;YACP,wCAAwC;YACxC,MAAM,MAAM,GAAG,MAAM,MAAM,CAAC,WAAW,EAAE,CAAC;YAC1C,IAAI,MAAM,EAAE,IAAI,EAAE,CAAC;gBAClB,IAAI,MAAM,CAAC,KAAK,KAAK,QAAQ,EAAE,CAAC;oBAC/B,MAAM,IAAI,KAAK,CAAC,6CAA6C,CAAC,CAAC;gBAChE,CAAC;gBACD,IAAI,GAAG,MAAM,CAAC,IAAI,CAAC;YACpB,CAAC;QACF,CAAC;QAED,IAAI,CAAC,IAAI,EAAE,CAAC;YACX,MAAM,IAAI,KAAK,CAAC,gCAAgC,CAAC,CAAC;QACnD,CAAC;QAED,2BAA2B;QAC3B,UAAU,EAAE,CAAC,6CAA6C,CAAC,CAAC;QAC5D,MAAM,aAAa,GAAG,MAAM,KAAK,CAAC,SAAS,EAAE;YAC5C,MAAM,EAAE,MAAM;YACd,OAAO,EAAE;gBACR,cAAc,EAAE,mCAAmC;aACnD;YACD,IAAI,EAAE,IAAI,eAAe,CAAC;gBACzB,SAAS,EAAE,SAAS;gBACpB,aAAa,EAAE,aAAa;gBAC5B,IAAI;gBACJ,UAAU,EAAE,oBAAoB;gBAChC,YAAY,EAAE,YAAY;gBAC1B,aAAa,EAAE,QAAQ;aACvB,CAAC;YACF,MAAM,EAAE,WAAW,CAAC,OAAO,CAAC,MAAM,CAAC;SACnC,CAAC,CAAC;QAEH,IAAI,CAAC,aAAa,CAAC,EAAE,EAAE,CAAC;YACvB,MAAM,KAAK,GAAG,MAAM,aAAa,CAAC,IAAI,EAAE,CAAC;YACzC,MAAM,IAAI,KAAK,CAAC,0BAA0B,KAAK,EAAE,CAAC,CAAC;QACpD,CAAC;QAED,MAAM,SAAS,GAAG,CAAC,MAAM,aAAa,CAAC,IAAI,EAAE,CAI5C,CAAC;QAEF,IAAI,CAAC,SAAS,CAAC,aAAa,EAAE,CAAC;YAC9B,MAAM,IAAI,KAAK,CAAC,8CAA8C,CAAC,CAAC;QACjE,CAAC;QAED,iBAAiB;QACjB,UAAU,EAAE,CAAC,sBAAsB,CAAC,CAAC;QACrC,MAAM,KAAK,GAAG,MAAM,kBAAkB,CAAC,SAAS,CAAC,YAAY,CAAC,CAAC;QAE/D,mBAAmB;QACnB,MAAM,SAAS,GAAG,MAAM,eAAe,CAAC,SAAS,CAAC,YAAY,EAAE,UAAU,CAAC,CAAC;QAE5E,2EAA2E;QAC3E,MAAM,SAAS,GAAG,IAAI,CAAC,GAAG,EAAE,GAAG,SAAS,CAAC,UAAU,GAAG,IAAI,GAAG,CAAC,GAAG,EAAE,GAAG,IAAI,CAAC;QAE3E,MAAM,WAAW,GAAqB;YACrC,OAAO,EAAE,SAAS,CAAC,aAAa;YAChC,MAAM,EAAE,SAAS,CAAC,YAAY;YAC9B,OAAO,EAAE,SAAS;YAClB,SAAS;YACT,KAAK;SACL,CAAC;QAEF,OAAO,WAAW,CAAC;IACpB,CAAC;YAAS,CAAC;QACV,MAAM,CAAC,MAAM,CAAC,KAAK,EAAE,CAAC;IACvB,CAAC;AACF,CAAC;AAED,MAAM,CAAC,MAAM,sBAAsB,GAA2B;IAC7D,EAAE,EAAE,mBAAmB;IACvB,IAAI,EAAE,uCAAuC;IAC7C,kBAAkB,EAAE,IAAI;IAExB,KAAK,CAAC,KAAK,CAAC,SAA8B;QACzC,OAAO,cAAc,CAAC,SAAS,CAAC,MAAM,EAAE,SAAS,CAAC,UAAU,EAAE,SAAS,CAAC,iBAAiB,CAAC,CAAC;IAC5F,CAAC;IAED,KAAK,CAAC,YAAY,CAAC,WAA6B;QAC/C,MAAM,KAAK,GAAG,WAAgC,CAAC;QAC/C,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,CAAC;YACtB,MAAM,IAAI,KAAK,CAAC,4CAA4C,CAAC,CAAC;QAC/D,CAAC;QACD,OAAO,uBAAuB,CAAC,KAAK,CAAC,OAAO,EAAE,KAAK,CAAC,SAAS,CAAC,CAAC;IAChE,CAAC;IAED,SAAS,CAAC,WAA6B;QACtC,MAAM,KAAK,GAAG,WAAgC,CAAC;QAC/C,OAAO,IAAI,CAAC,SAAS,CAAC,EAAE,KAAK,EAAE,KAAK,CAAC,MAAM,EAAE,SAAS,EAAE,KAAK,CAAC,SAAS,EAAE,CAAC,CAAC;IAC5E,CAAC;CACD,CAAC","sourcesContent":["/**\n * Gemini CLI OAuth flow (Google Cloud Code Assist)\n * Standard Gemini models only (gemini-2.0-flash, gemini-2.5-*)\n *\n * NOTE: This module uses Node.js http.createServer for the OAuth callback.\n * It is only intended for CLI use, not browser environments.\n */\n\nimport {\n\ttype CallbackServerInfo,\n\tgetGoogleUserEmail,\n\tparseRedirectUrl,\n\trefreshGoogleOAuthToken,\n\tstartCallbackServer,\n} from \"./google-oauth-utils.js\";\nimport { generatePKCE } from \"./pkce.js\";\nimport type { OAuthCredentials, OAuthLoginCallbacks, OAuthProviderInterface } from \"./types.js\";\n\ntype GeminiCredentials = OAuthCredentials & {\n\tprojectId: string;\n};\n\n// Gemini CLI OAuth credentials (Google Cloud Code Assist)\n//\n// NOTE: These credentials are public in the source code. They should NOT be\n// obfuscated because security scanners flag obfuscated data as potentially\n// malicious (see: https://socket.dev/npm/package/gsd-pi/alerts/2.70.1?alert_name=obfuscatedFile)\n//\n// Google's OAuth implementation requires client_secret for Desktop App OAuth\n// clients even though it cannot be kept secret in distributed applications.\n// The actual security relies on:\n// - PKCE (Proof Key for Code Exchange) via code_verifier\n// - Redirect URI validation (localhost only)\n// - User consent and token scope limits\n//\n// See: https://developers.google.com/identity/protocols/oauth2/native-app\nconst CLIENT_ID = \"681255809395-oo8ft2oprdrnp9e3aqf6av3hmdib135j.apps.googleusercontent.com\";\nconst CLIENT_SECRET = \"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\";\nconst REDIRECT_URI = \"http://localhost:8085/oauth2callback\";\nconst SCOPES = [\n\t\"https://www.googleapis.com/auth/cloud-platform\",\n\t\"https://www.googleapis.com/auth/userinfo.email\",\n\t\"https://www.googleapis.com/auth/userinfo.profile\",\n];\nconst AUTH_URL = \"https://accounts.google.com/o/oauth2/v2/auth\";\nconst TOKEN_URL = \"https://oauth2.googleapis.com/token\";\nconst CODE_ASSIST_ENDPOINT = \"https://cloudcode-pa.googleapis.com\";\n\n// Callback server configuration\nconst CALLBACK_PORT = 8085;\nconst CALLBACK_PATH = \"/oauth2callback\";\n\ninterface LoadCodeAssistPayload {\n\tcloudaicompanionProject?: string;\n\tcurrentTier?: { id?: string };\n\tallowedTiers?: Array<{ id?: string; isDefault?: boolean }>;\n}\n\n/**\n * Long-running operation response from onboardUser\n */\ninterface LongRunningOperationResponse {\n\tname?: string;\n\tdone?: boolean;\n\tresponse?: {\n\t\tcloudaicompanionProject?: { id?: string };\n\t};\n}\n\n// Tier IDs as used by the Cloud Code API\nconst TIER_FREE = \"free-tier\";\nconst TIER_LEGACY = \"legacy-tier\";\nconst TIER_STANDARD = \"standard-tier\";\n\ninterface GoogleRpcErrorResponse {\n\terror?: {\n\t\tdetails?: Array<{ reason?: string }>;\n\t};\n}\n\n/**\n * Wait helper for onboarding retries\n */\nfunction wait(ms: number): Promise<void> {\n\treturn new Promise((resolve) => setTimeout(resolve, ms));\n}\n\n/**\n * Get default tier from allowed tiers\n */\nfunction getDefaultTier(allowedTiers?: Array<{ id?: string; isDefault?: boolean }>): { id?: string } {\n\tif (!allowedTiers || allowedTiers.length === 0) return { id: TIER_LEGACY };\n\tconst defaultTier = allowedTiers.find((t) => t.isDefault);\n\treturn defaultTier ?? { id: TIER_LEGACY };\n}\n\nfunction isVpcScAffectedUser(payload: unknown): boolean {\n\tif (!payload || typeof payload !== \"object\") return false;\n\tif (!(\"error\" in payload)) return false;\n\tconst error = (payload as GoogleRpcErrorResponse).error;\n\tif (!error?.details || !Array.isArray(error.details)) return false;\n\treturn error.details.some((detail) => detail.reason === \"SECURITY_POLICY_VIOLATED\");\n}\n\n/**\n * Poll a long-running operation until completion\n */\nasync function pollOperation(\n\toperationName: string,\n\theaders: Record<string, string>,\n\tonProgress?: (message: string) => void,\n): Promise<LongRunningOperationResponse> {\n\tlet attempt = 0;\n\twhile (true) {\n\t\tif (attempt > 0) {\n\t\t\tonProgress?.(`Waiting for project provisioning (attempt ${attempt + 1})...`);\n\t\t\tawait wait(5000);\n\t\t}\n\n\t\tconst response = await fetch(`${CODE_ASSIST_ENDPOINT}/v1internal/${operationName}`, {\n\t\t\tmethod: \"GET\",\n\t\t\theaders,\n\t\t\tsignal: AbortSignal.timeout(30_000),\n\t\t});\n\n\t\tif (!response.ok) {\n\t\t\tthrow new Error(`Failed to poll operation: ${response.status} ${response.statusText}`);\n\t\t}\n\n\t\tconst data = (await response.json()) as LongRunningOperationResponse;\n\t\tif (data.done) {\n\t\t\treturn data;\n\t\t}\n\n\t\tattempt += 1;\n\t}\n}\n\n/**\n * Discover or provision a Google Cloud project for the user\n */\nasync function discoverProject(accessToken: string, onProgress?: (message: string) => void): Promise<string> {\n\t// Check for user-provided project ID via environment variable\n\tconst envProjectId = process.env.GOOGLE_CLOUD_PROJECT || process.env.GOOGLE_CLOUD_PROJECT_ID;\n\n\tconst headers = {\n\t\tAuthorization: `Bearer ${accessToken}`,\n\t\t\"Content-Type\": \"application/json\",\n\t\t\"User-Agent\": \"google-api-nodejs-client/9.15.1\",\n\t\t\"X-Goog-Api-Client\": \"gl-node/22.17.0\",\n\t};\n\n\t// Try to load existing project via loadCodeAssist\n\tonProgress?.(\"Checking for existing Cloud Code Assist project...\");\n\tconst loadResponse = await fetch(`${CODE_ASSIST_ENDPOINT}/v1internal:loadCodeAssist`, {\n\t\tmethod: \"POST\",\n\t\theaders,\n\t\tbody: JSON.stringify({\n\t\t\tcloudaicompanionProject: envProjectId,\n\t\t\tmetadata: {\n\t\t\t\tideType: \"IDE_UNSPECIFIED\",\n\t\t\t\tplatform: \"PLATFORM_UNSPECIFIED\",\n\t\t\t\tpluginType: \"GEMINI\",\n\t\t\t\tduetProject: envProjectId,\n\t\t\t},\n\t\t}),\n\t\tsignal: AbortSignal.timeout(30_000),\n\t});\n\n\tlet data: LoadCodeAssistPayload;\n\n\tif (!loadResponse.ok) {\n\t\tlet errorPayload: unknown;\n\t\ttry {\n\t\t\terrorPayload = await loadResponse.clone().json();\n\t\t} catch {\n\t\t\terrorPayload = undefined;\n\t\t}\n\n\t\tif (isVpcScAffectedUser(errorPayload)) {\n\t\t\tdata = { currentTier: { id: TIER_STANDARD } };\n\t\t} else {\n\t\t\tconst errorText = await loadResponse.text();\n\t\t\tthrow new Error(`loadCodeAssist failed: ${loadResponse.status} ${loadResponse.statusText}: ${errorText}`);\n\t\t}\n\t} else {\n\t\tdata = (await loadResponse.json()) as LoadCodeAssistPayload;\n\t}\n\n\t// If user already has a current tier and project, use it\n\tif (data.currentTier) {\n\t\tif (data.cloudaicompanionProject) {\n\t\t\treturn data.cloudaicompanionProject;\n\t\t}\n\t\t// User has a tier but no managed project - they need to provide one via env var\n\t\tif (envProjectId) {\n\t\t\treturn envProjectId;\n\t\t}\n\t\tthrow new Error(\n\t\t\t\"This account requires setting the GOOGLE_CLOUD_PROJECT or GOOGLE_CLOUD_PROJECT_ID environment variable. \" +\n\t\t\t\t\"See https://goo.gle/gemini-cli-auth-docs#workspace-gca\",\n\t\t);\n\t}\n\n\t// User needs to be onboarded - get the default tier\n\tconst tier = getDefaultTier(data.allowedTiers);\n\tconst tierId = tier?.id ?? TIER_FREE;\n\n\tif (tierId !== TIER_FREE && !envProjectId) {\n\t\tthrow new Error(\n\t\t\t\"This account requires setting the GOOGLE_CLOUD_PROJECT or GOOGLE_CLOUD_PROJECT_ID environment variable. \" +\n\t\t\t\t\"See https://goo.gle/gemini-cli-auth-docs#workspace-gca\",\n\t\t);\n\t}\n\n\tonProgress?.(\"Provisioning Cloud Code Assist project (this may take a moment)...\");\n\n\t// Build onboard request - for free tier, don't include project ID (Google provisions one)\n\t// For other tiers, include the user's project ID if available\n\tconst onboardBody: Record<string, unknown> = {\n\t\ttierId,\n\t\tmetadata: {\n\t\t\tideType: \"IDE_UNSPECIFIED\",\n\t\t\tplatform: \"PLATFORM_UNSPECIFIED\",\n\t\t\tpluginType: \"GEMINI\",\n\t\t},\n\t};\n\n\tif (tierId !== TIER_FREE && envProjectId) {\n\t\tonboardBody.cloudaicompanionProject = envProjectId;\n\t\t(onboardBody.metadata as Record<string, unknown>).duetProject = envProjectId;\n\t}\n\n\t// Start onboarding - this returns a long-running operation\n\tconst onboardResponse = await fetch(`${CODE_ASSIST_ENDPOINT}/v1internal:onboardUser`, {\n\t\tmethod: \"POST\",\n\t\theaders,\n\t\tbody: JSON.stringify(onboardBody),\n\t\tsignal: AbortSignal.timeout(30_000),\n\t});\n\n\tif (!onboardResponse.ok) {\n\t\tconst errorText = await onboardResponse.text();\n\t\tthrow new Error(`onboardUser failed: ${onboardResponse.status} ${onboardResponse.statusText}: ${errorText}`);\n\t}\n\n\tlet lroData = (await onboardResponse.json()) as LongRunningOperationResponse;\n\n\t// If the operation isn't done yet, poll until completion\n\tif (!lroData.done && lroData.name) {\n\t\tlroData = await pollOperation(lroData.name, headers, onProgress);\n\t}\n\n\t// Try to get project ID from the response\n\tconst projectId = lroData.response?.cloudaicompanionProject?.id;\n\tif (projectId) {\n\t\treturn projectId;\n\t}\n\n\t// If no project ID from onboarding, fall back to env var\n\tif (envProjectId) {\n\t\treturn envProjectId;\n\t}\n\n\tthrow new Error(\n\t\t\"Could not discover or provision a Google Cloud project. \" +\n\t\t\t\"Try setting the GOOGLE_CLOUD_PROJECT or GOOGLE_CLOUD_PROJECT_ID environment variable. \" +\n\t\t\t\"See https://goo.gle/gemini-cli-auth-docs#workspace-gca\",\n\t);\n}\n\n/**\n * Refresh Google Cloud Code Assist token\n */\nexport async function refreshGoogleCloudToken(refreshToken: string, projectId: string): Promise<OAuthCredentials> {\n\treturn refreshGoogleOAuthToken(refreshToken, CLIENT_ID, CLIENT_SECRET, \"Google Cloud\", { projectId });\n}\n\n/**\n * Login with Gemini CLI (Google Cloud Code Assist) OAuth\n *\n * @param onAuth - Callback with URL and optional instructions\n * @param onProgress - Optional progress callback\n * @param onManualCodeInput - Optional promise that resolves with user-pasted redirect URL.\n * Races with browser callback - whichever completes first wins.\n */\nexport async function loginGeminiCli(\n\tonAuth: (info: { url: string; instructions?: string }) => void,\n\tonProgress?: (message: string) => void,\n\tonManualCodeInput?: () => Promise<string>,\n): Promise<OAuthCredentials> {\n\tconst { verifier, challenge } = await generatePKCE();\n\n\t// Start local server for callback\n\tonProgress?.(\"Starting local server for OAuth callback...\");\n\tconst server: CallbackServerInfo = await startCallbackServer(CALLBACK_PORT, CALLBACK_PATH, \"Gemini CLI\");\n\n\tlet code: string | undefined;\n\n\ttry {\n\t\t// Build authorization URL\n\t\tconst authParams = new URLSearchParams({\n\t\t\tclient_id: CLIENT_ID,\n\t\t\tresponse_type: \"code\",\n\t\t\tredirect_uri: REDIRECT_URI,\n\t\t\tscope: SCOPES.join(\" \"),\n\t\t\tcode_challenge: challenge,\n\t\t\tcode_challenge_method: \"S256\",\n\t\t\tstate: verifier,\n\t\t\taccess_type: \"offline\",\n\t\t\tprompt: \"consent\",\n\t\t});\n\n\t\tconst authUrl = `${AUTH_URL}?${authParams.toString()}`;\n\n\t\t// Notify caller with URL to open\n\t\tonAuth({\n\t\t\turl: authUrl,\n\t\t\tinstructions: \"Complete the sign-in in your browser.\",\n\t\t});\n\n\t\t// Wait for the callback, racing with manual input if provided\n\t\tonProgress?.(\"Waiting for OAuth callback...\");\n\n\t\tif (onManualCodeInput) {\n\t\t\t// Race between browser callback and manual input\n\t\t\tlet manualInput: string | undefined;\n\t\t\tlet manualError: Error | undefined;\n\t\t\tconst manualPromise = onManualCodeInput()\n\t\t\t\t.then((input) => {\n\t\t\t\t\tmanualInput = input;\n\t\t\t\t\tserver.cancelWait();\n\t\t\t\t})\n\t\t\t\t.catch((err) => {\n\t\t\t\t\tmanualError = err instanceof Error ? err : new Error(String(err));\n\t\t\t\t\tserver.cancelWait();\n\t\t\t\t});\n\n\t\t\tconst result = await server.waitForCode();\n\n\t\t\t// If manual input was cancelled, throw that error\n\t\t\tif (manualError) {\n\t\t\t\tthrow manualError;\n\t\t\t}\n\n\t\t\tif (result?.code) {\n\t\t\t\t// Browser callback won - verify state\n\t\t\t\tif (result.state !== verifier) {\n\t\t\t\t\tthrow new Error(\"OAuth state mismatch - possible CSRF attack\");\n\t\t\t\t}\n\t\t\t\tcode = result.code;\n\t\t\t} else if (manualInput) {\n\t\t\t\t// Manual input won\n\t\t\t\tconst parsed = parseRedirectUrl(manualInput);\n\t\t\t\tif (parsed.state && parsed.state !== verifier) {\n\t\t\t\t\tthrow new Error(\"OAuth state mismatch - possible CSRF attack\");\n\t\t\t\t}\n\t\t\t\tcode = parsed.code;\n\t\t\t}\n\n\t\t\t// If still no code, wait for manual promise and try that\n\t\t\tif (!code) {\n\t\t\t\tawait manualPromise;\n\t\t\t\tif (manualError) {\n\t\t\t\t\tthrow manualError;\n\t\t\t\t}\n\t\t\t\tif (manualInput) {\n\t\t\t\t\tconst parsed = parseRedirectUrl(manualInput);\n\t\t\t\t\tif (parsed.state && parsed.state !== verifier) {\n\t\t\t\t\t\tthrow new Error(\"OAuth state mismatch - possible CSRF attack\");\n\t\t\t\t\t}\n\t\t\t\t\tcode = parsed.code;\n\t\t\t\t}\n\t\t\t}\n\t\t} else {\n\t\t\t// Original flow: just wait for callback\n\t\t\tconst result = await server.waitForCode();\n\t\t\tif (result?.code) {\n\t\t\t\tif (result.state !== verifier) {\n\t\t\t\t\tthrow new Error(\"OAuth state mismatch - possible CSRF attack\");\n\t\t\t\t}\n\t\t\t\tcode = result.code;\n\t\t\t}\n\t\t}\n\n\t\tif (!code) {\n\t\t\tthrow new Error(\"No authorization code received\");\n\t\t}\n\n\t\t// Exchange code for tokens\n\t\tonProgress?.(\"Exchanging authorization code for tokens...\");\n\t\tconst tokenResponse = await fetch(TOKEN_URL, {\n\t\t\tmethod: \"POST\",\n\t\t\theaders: {\n\t\t\t\t\"Content-Type\": \"application/x-www-form-urlencoded\",\n\t\t\t},\n\t\t\tbody: new URLSearchParams({\n\t\t\t\tclient_id: CLIENT_ID,\n\t\t\t\tclient_secret: CLIENT_SECRET,\n\t\t\t\tcode,\n\t\t\t\tgrant_type: \"authorization_code\",\n\t\t\t\tredirect_uri: REDIRECT_URI,\n\t\t\t\tcode_verifier: verifier,\n\t\t\t}),\n\t\t\tsignal: AbortSignal.timeout(30_000),\n\t\t});\n\n\t\tif (!tokenResponse.ok) {\n\t\t\tconst error = await tokenResponse.text();\n\t\t\tthrow new Error(`Token exchange failed: ${error}`);\n\t\t}\n\n\t\tconst tokenData = (await tokenResponse.json()) as {\n\t\t\taccess_token: string;\n\t\t\trefresh_token: string;\n\t\t\texpires_in: number;\n\t\t};\n\n\t\tif (!tokenData.refresh_token) {\n\t\t\tthrow new Error(\"No refresh token received. Please try again.\");\n\t\t}\n\n\t\t// Get user email\n\t\tonProgress?.(\"Getting user info...\");\n\t\tconst email = await getGoogleUserEmail(tokenData.access_token);\n\n\t\t// Discover project\n\t\tconst projectId = await discoverProject(tokenData.access_token, onProgress);\n\n\t\t// Calculate expiry time (current time + expires_in seconds - 5 min buffer)\n\t\tconst expiresAt = Date.now() + tokenData.expires_in * 1000 - 5 * 60 * 1000;\n\n\t\tconst credentials: OAuthCredentials = {\n\t\t\trefresh: tokenData.refresh_token,\n\t\t\taccess: tokenData.access_token,\n\t\t\texpires: expiresAt,\n\t\t\tprojectId,\n\t\t\temail,\n\t\t};\n\n\t\treturn credentials;\n\t} finally {\n\t\tserver.server.close();\n\t}\n}\n\nexport const geminiCliOAuthProvider: OAuthProviderInterface = {\n\tid: \"google-gemini-cli\",\n\tname: \"Google Cloud Code Assist (Gemini CLI)\",\n\tusesCallbackServer: true,\n\n\tasync login(callbacks: OAuthLoginCallbacks): Promise<OAuthCredentials> {\n\t\treturn loginGeminiCli(callbacks.onAuth, callbacks.onProgress, callbacks.onManualCodeInput);\n\t},\n\n\tasync refreshToken(credentials: OAuthCredentials): Promise<OAuthCredentials> {\n\t\tconst creds = credentials as GeminiCredentials;\n\t\tif (!creds.projectId) {\n\t\t\tthrow new Error(\"Google Cloud credentials missing projectId\");\n\t\t}\n\t\treturn refreshGoogleCloudToken(creds.refresh, creds.projectId);\n\t},\n\n\tgetApiKey(credentials: OAuthCredentials): string {\n\t\tconst creds = credentials as GeminiCredentials;\n\t\treturn JSON.stringify({ token: creds.access, projectId: creds.projectId });\n\t},\n};\n"]}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=google-gemini-cli.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"google-gemini-cli.test.d.ts","sourceRoot":"","sources":["../../../src/utils/oauth/google-gemini-cli.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,67 @@
1
+ import { describe, test } from "node:test";
2
+ import assert from "node:assert/strict";
3
+ import { existsSync, readFileSync } from "node:fs";
4
+ import { join } from "node:path";
5
+ import { fileURLToPath } from "node:url";
6
+ import { geminiCliOAuthProvider } from "./google-gemini-cli.js";
7
+ const __dirname = fileURLToPath(new URL(".", import.meta.url));
8
+ const packageRoot = join(__dirname, "..", "..", "..");
9
+ const sourceDir = existsSync(join(__dirname, "google-gemini-cli.ts"))
10
+ ? __dirname
11
+ : join(packageRoot, "src", "utils", "oauth");
12
+ function readSourceFile(name) {
13
+ return readFileSync(join(sourceDir, name), "utf-8");
14
+ }
15
+ describe("Gemini CLI OAuth — provider structure", () => {
16
+ test("has correct id and name", () => {
17
+ assert.equal(geminiCliOAuthProvider.id, "google-gemini-cli");
18
+ assert.equal(geminiCliOAuthProvider.name, "Google Cloud Code Assist (Gemini CLI)");
19
+ });
20
+ test("uses callback server", () => {
21
+ assert.equal(geminiCliOAuthProvider.usesCallbackServer, true);
22
+ });
23
+ test("has required methods", () => {
24
+ assert.equal(typeof geminiCliOAuthProvider.login, "function");
25
+ assert.equal(typeof geminiCliOAuthProvider.refreshToken, "function");
26
+ assert.equal(typeof geminiCliOAuthProvider.getApiKey, "function");
27
+ });
28
+ test("getApiKey returns JSON with token and projectId", () => {
29
+ const credentials = {
30
+ access: "test-access-token",
31
+ refresh: "test-refresh-token",
32
+ expires: Date.now() + 3600000,
33
+ projectId: "test-project-456",
34
+ email: "test@example.com",
35
+ };
36
+ const apiKey = geminiCliOAuthProvider.getApiKey(credentials);
37
+ assert.equal(typeof apiKey, "string");
38
+ const parsed = JSON.parse(apiKey);
39
+ assert.equal(parsed.token, "test-access-token");
40
+ assert.equal(parsed.projectId, "test-project-456");
41
+ });
42
+ test("refreshToken throws when projectId is missing", async () => {
43
+ const credentials = {
44
+ access: "test-access-token",
45
+ refresh: "test-refresh-token",
46
+ expires: Date.now() + 3600000,
47
+ };
48
+ await assert.rejects(geminiCliOAuthProvider.refreshToken(credentials), /Google Cloud credentials missing projectId/);
49
+ });
50
+ });
51
+ describe("Gemini CLI OAuth — credential regression", () => {
52
+ test("module imports successfully", () => {
53
+ assert.ok(geminiCliOAuthProvider);
54
+ });
55
+ test("CLIENT_ID and CLIENT_SECRET are plaintext", () => {
56
+ const content = readSourceFile("google-gemini-cli.ts");
57
+ assert.ok(content.includes('CLIENT_ID = "681255809395-oo8ft2oprdrnp9e3aqf6av3hmdib135j.apps.googleusercontent.com"'));
58
+ assert.ok(content.includes('CLIENT_SECRET = "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"'));
59
+ assert.ok(!content.includes("atob("));
60
+ });
61
+ test("security explanation comments are present", () => {
62
+ const content = readSourceFile("google-gemini-cli.ts");
63
+ assert.ok(content.includes("NOTE: These credentials are public"));
64
+ assert.ok(content.includes("obfuscated") || content.includes("security scanners"));
65
+ });
66
+ });
67
+ //# sourceMappingURL=google-gemini-cli.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"google-gemini-cli.test.js","sourceRoot":"","sources":["../../../src/utils/oauth/google-gemini-cli.test.ts"],"names":[],"mappings":"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","sourcesContent":["import { describe, test } from \"node:test\";\nimport assert from \"node:assert/strict\";\nimport { existsSync, readFileSync } from \"node:fs\";\nimport { join } from \"node:path\";\nimport { fileURLToPath } from \"node:url\";\nimport type { OAuthCredentials } from \"./types.js\";\nimport { geminiCliOAuthProvider } from \"./google-gemini-cli.js\";\n\nconst __dirname = fileURLToPath(new URL(\".\", import.meta.url));\nconst packageRoot = join(__dirname, \"..\", \"..\", \"..\");\nconst sourceDir = existsSync(join(__dirname, \"google-gemini-cli.ts\"))\n\t? __dirname\n\t: join(packageRoot, \"src\", \"utils\", \"oauth\");\n\nfunction readSourceFile(name: string): string {\n\treturn readFileSync(join(sourceDir, name), \"utf-8\");\n}\n\ndescribe(\"Gemini CLI OAuth — provider structure\", () => {\n\ttest(\"has correct id and name\", () => {\n\t\tassert.equal(geminiCliOAuthProvider.id, \"google-gemini-cli\");\n\t\tassert.equal(geminiCliOAuthProvider.name, \"Google Cloud Code Assist (Gemini CLI)\");\n\t});\n\n\ttest(\"uses callback server\", () => {\n\t\tassert.equal(geminiCliOAuthProvider.usesCallbackServer, true);\n\t});\n\n\ttest(\"has required methods\", () => {\n\t\tassert.equal(typeof geminiCliOAuthProvider.login, \"function\");\n\t\tassert.equal(typeof geminiCliOAuthProvider.refreshToken, \"function\");\n\t\tassert.equal(typeof geminiCliOAuthProvider.getApiKey, \"function\");\n\t});\n\n\ttest(\"getApiKey returns JSON with token and projectId\", () => {\n\t\tconst credentials: OAuthCredentials = {\n\t\t\taccess: \"test-access-token\",\n\t\t\trefresh: \"test-refresh-token\",\n\t\t\texpires: Date.now() + 3600000,\n\t\t\tprojectId: \"test-project-456\",\n\t\t\temail: \"test@example.com\",\n\t\t};\n\t\tconst apiKey = geminiCliOAuthProvider.getApiKey(credentials);\n\t\tassert.equal(typeof apiKey, \"string\");\n\t\tconst parsed = JSON.parse(apiKey);\n\t\tassert.equal(parsed.token, \"test-access-token\");\n\t\tassert.equal(parsed.projectId, \"test-project-456\");\n\t});\n\n\ttest(\"refreshToken throws when projectId is missing\", async () => {\n\t\tconst credentials: OAuthCredentials = {\n\t\t\taccess: \"test-access-token\",\n\t\t\trefresh: \"test-refresh-token\",\n\t\t\texpires: Date.now() + 3600000,\n\t\t};\n\t\tawait assert.rejects(\n\t\t\tgeminiCliOAuthProvider.refreshToken(credentials),\n\t\t\t/Google Cloud credentials missing projectId/,\n\t\t);\n\t});\n});\n\ndescribe(\"Gemini CLI OAuth — credential regression\", () => {\n\ttest(\"module imports successfully\", () => {\n\t\tassert.ok(geminiCliOAuthProvider);\n\t});\n\n\ttest(\"CLIENT_ID and CLIENT_SECRET are plaintext\", () => {\n\t\tconst content = readSourceFile(\"google-gemini-cli.ts\");\n\t\tassert.ok(\n\t\t\tcontent.includes(\n\t\t\t\t'CLIENT_ID = \"681255809395-oo8ft2oprdrnp9e3aqf6av3hmdib135j.apps.googleusercontent.com\"',\n\t\t\t),\n\t\t);\n\t\tassert.ok(content.includes('CLIENT_SECRET = \"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\"'));\n\t\tassert.ok(!content.includes(\"atob(\"));\n\t});\n\n\ttest(\"security explanation comments are present\", () => {\n\t\tconst content = readSourceFile(\"google-gemini-cli.ts\");\n\t\tassert.ok(content.includes(\"NOTE: These credentials are public\"));\n\t\tassert.ok(content.includes(\"obfuscated\") || content.includes(\"security scanners\"));\n\t});\n});"]}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=oauth-providers.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"oauth-providers.test.d.ts","sourceRoot":"","sources":["../../../src/utils/oauth/oauth-providers.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,289 @@
1
+ import { describe, test } from "node:test";
2
+ import assert from "node:assert/strict";
3
+ import { readFileSync } from "node:fs";
4
+ import { join } from "node:path";
5
+ import { fileURLToPath } from "node:url";
6
+ // Test files that need to import from the OAuth modules
7
+ import { githubCopilotOAuthProvider, normalizeDomain, getGitHubCopilotBaseUrl, } from "./github-copilot.js";
8
+ import { antigravityOAuthProvider } from "./google-antigravity.js";
9
+ import { geminiCliOAuthProvider } from "./google-gemini-cli.js";
10
+ const __dirname = fileURLToPath(new URL(".", import.meta.url));
11
+ // Helper to create a minimal model for testing
12
+ function createModel(overrides = {}) {
13
+ return {
14
+ id: "test-model",
15
+ name: "Test Model",
16
+ api: "openai-completions",
17
+ provider: "test-provider",
18
+ baseUrl: "https://example.com",
19
+ reasoning: false,
20
+ input: ["text"],
21
+ cost: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0 },
22
+ contextWindow: 128000,
23
+ maxTokens: 8192,
24
+ ...overrides,
25
+ };
26
+ }
27
+ // ═══════════════════════════════════════════════════════════════════════════
28
+ // GitHub Copilot OAuth Provider Tests
29
+ // ═══════════════════════════════════════════════════════════════════════════
30
+ describe("GitHub Copilot OAuth — normalizeDomain", () => {
31
+ test("returns null for empty input", () => {
32
+ assert.equal(normalizeDomain(""), null);
33
+ assert.equal(normalizeDomain(" "), null);
34
+ });
35
+ test("returns null for invalid domain", () => {
36
+ assert.equal(normalizeDomain("not a domain!@#"), null);
37
+ });
38
+ test("extracts hostname from full URL", () => {
39
+ assert.equal(normalizeDomain("https://github.com"), "github.com");
40
+ assert.equal(normalizeDomain("https://company.ghe.com"), "company.ghe.com");
41
+ assert.equal(normalizeDomain("http://example.com/path"), "example.com");
42
+ });
43
+ test("returns domain as-is when no protocol", () => {
44
+ assert.equal(normalizeDomain("github.com"), "github.com");
45
+ assert.equal(normalizeDomain("company.ghe.com"), "company.ghe.com");
46
+ });
47
+ test("trims whitespace", () => {
48
+ assert.equal(normalizeDomain(" github.com "), "github.com");
49
+ });
50
+ });
51
+ describe("GitHub Copilot OAuth — getBaseUrlFromToken", () => {
52
+ test("extracts API URL from token with proxy-ep", () => {
53
+ // Token format: tid=...;exp=...;proxy-ep=proxy.individual.githubcopilot.com;...
54
+ const token = "tid=123;exp=1234567890;proxy-ep=proxy.individual.githubcopilot.com;other=value";
55
+ const baseUrl = getGitHubCopilotBaseUrl(token);
56
+ assert.equal(baseUrl, "https://api.individual.githubcopilot.com");
57
+ });
58
+ test("extracts API URL from enterprise proxy-ep", () => {
59
+ const token = "tid=123;exp=1234567890;proxy-ep=proxy.company.ghe.com;other=value";
60
+ const baseUrl = getGitHubCopilotBaseUrl(token);
61
+ assert.equal(baseUrl, "https://api.company.ghe.com");
62
+ });
63
+ test("falls back to default when no token provided", () => {
64
+ const baseUrl = getGitHubCopilotBaseUrl();
65
+ assert.equal(baseUrl, "https://api.individual.githubcopilot.com");
66
+ });
67
+ test("falls back to default when token has no proxy-ep", () => {
68
+ const token = "tid=123;exp=1234567890;other=value";
69
+ const baseUrl = getGitHubCopilotBaseUrl(token);
70
+ assert.equal(baseUrl, "https://api.individual.githubcopilot.com");
71
+ });
72
+ test("uses enterprise domain when provided", () => {
73
+ const baseUrl = getGitHubCopilotBaseUrl(undefined, "company.ghe.com");
74
+ assert.equal(baseUrl, "https://copilot-api.company.ghe.com");
75
+ });
76
+ test("prioritizes token proxy-ep over enterprise domain", () => {
77
+ const token = "tid=123;exp=1234567890;proxy-ep=proxy.individual.githubcopilot.com;other=value";
78
+ const baseUrl = getGitHubCopilotBaseUrl(token, "company.ghe.com");
79
+ assert.equal(baseUrl, "https://api.individual.githubcopilot.com");
80
+ });
81
+ });
82
+ describe("GitHub Copilot OAuth — provider structure", () => {
83
+ test("has correct id and name", () => {
84
+ assert.equal(githubCopilotOAuthProvider.id, "github-copilot");
85
+ assert.equal(githubCopilotOAuthProvider.name, "GitHub Copilot");
86
+ });
87
+ test("has required methods", () => {
88
+ assert.equal(typeof githubCopilotOAuthProvider.login, "function");
89
+ assert.equal(typeof githubCopilotOAuthProvider.refreshToken, "function");
90
+ assert.equal(typeof githubCopilotOAuthProvider.getApiKey, "function");
91
+ });
92
+ test("getApiKey returns access token", () => {
93
+ const credentials = {
94
+ access: "test-access-token",
95
+ refresh: "test-refresh-token",
96
+ expires: Date.now() + 3600000,
97
+ };
98
+ const apiKey = githubCopilotOAuthProvider.getApiKey(credentials);
99
+ assert.equal(apiKey, "test-access-token");
100
+ });
101
+ test("modifyModels preserves non-Copilot models", () => {
102
+ if (!githubCopilotOAuthProvider.modifyModels)
103
+ return;
104
+ const models = [createModel({ id: "gpt-4", provider: "openai" })];
105
+ const credentials = {
106
+ access: "test-token",
107
+ refresh: "test-refresh",
108
+ expires: Date.now() + 3600000,
109
+ };
110
+ const result = githubCopilotOAuthProvider.modifyModels(models, credentials);
111
+ assert.deepEqual(result, models);
112
+ });
113
+ test("modifyModels updates Copilot model baseUrl when token has proxy-ep", () => {
114
+ if (!githubCopilotOAuthProvider.modifyModels)
115
+ return;
116
+ const models = [
117
+ createModel({
118
+ id: "claude-3.5-sonnet",
119
+ provider: "github-copilot",
120
+ baseUrl: "https://api.default.com",
121
+ }),
122
+ ];
123
+ const credentials = {
124
+ access: "tid=123;exp=1234567890;proxy-ep=proxy.individual.githubcopilot.com;",
125
+ refresh: "test-refresh",
126
+ expires: Date.now() + 3600000,
127
+ };
128
+ const result = githubCopilotOAuthProvider.modifyModels(models, credentials);
129
+ assert.equal(result[0].baseUrl, "https://api.individual.githubcopilot.com");
130
+ });
131
+ test("modifyModels applies model limits when available", () => {
132
+ if (!githubCopilotOAuthProvider.modifyModels)
133
+ return;
134
+ const models = [
135
+ createModel({
136
+ id: "claude-3.5-sonnet",
137
+ provider: "github-copilot",
138
+ baseUrl: "https://api.default.com",
139
+ }),
140
+ ];
141
+ const credentials = {
142
+ access: "test-token",
143
+ refresh: "test-refresh",
144
+ expires: Date.now() + 3600000,
145
+ modelLimits: {
146
+ "claude-3.5-sonnet": { contextWindow: 123456, maxTokens: 4096 },
147
+ },
148
+ };
149
+ const result = githubCopilotOAuthProvider.modifyModels(models, credentials);
150
+ assert.equal(result[0].contextWindow, 123456);
151
+ assert.equal(result[0].maxTokens, 4096);
152
+ });
153
+ });
154
+ // ═══════════════════════════════════════════════════════════════════════════
155
+ // Antigravity OAuth Provider Tests
156
+ // ═══════════════════════════════════════════════════════════════════════════
157
+ describe("Antigravity OAuth — provider structure", () => {
158
+ test("has correct id and name", () => {
159
+ assert.equal(antigravityOAuthProvider.id, "google-antigravity");
160
+ assert.equal(antigravityOAuthProvider.name, "Antigravity (Gemini 3, Claude, GPT-OSS)");
161
+ });
162
+ test("uses callback server", () => {
163
+ assert.equal(antigravityOAuthProvider.usesCallbackServer, true);
164
+ });
165
+ test("has required methods", () => {
166
+ assert.equal(typeof antigravityOAuthProvider.login, "function");
167
+ assert.equal(typeof antigravityOAuthProvider.refreshToken, "function");
168
+ assert.equal(typeof antigravityOAuthProvider.getApiKey, "function");
169
+ });
170
+ test("getApiKey returns JSON with token and projectId", () => {
171
+ const credentials = {
172
+ access: "test-access-token",
173
+ refresh: "test-refresh-token",
174
+ expires: Date.now() + 3600000,
175
+ projectId: "test-project-123",
176
+ email: "test@example.com",
177
+ };
178
+ const apiKey = antigravityOAuthProvider.getApiKey(credentials);
179
+ assert.equal(typeof apiKey, "string");
180
+ const parsed = JSON.parse(apiKey);
181
+ assert.equal(parsed.token, "test-access-token");
182
+ assert.equal(parsed.projectId, "test-project-123");
183
+ });
184
+ test("refreshToken throws when projectId is missing", async () => {
185
+ const credentials = {
186
+ access: "test-access-token",
187
+ refresh: "test-refresh-token",
188
+ expires: Date.now() + 3600000,
189
+ };
190
+ await assert.rejects(antigravityOAuthProvider.refreshToken(credentials), /Antigravity credentials missing projectId/);
191
+ });
192
+ });
193
+ // ═══════════════════════════════════════════════════════════════════════════
194
+ // Gemini CLI OAuth Provider Tests
195
+ // ═══════════════════════════════════════════════════════════════════════════
196
+ describe("Gemini CLI OAuth — provider structure", () => {
197
+ test("has correct id and name", () => {
198
+ assert.equal(geminiCliOAuthProvider.id, "google-gemini-cli");
199
+ assert.equal(geminiCliOAuthProvider.name, "Google Cloud Code Assist (Gemini CLI)");
200
+ });
201
+ test("uses callback server", () => {
202
+ assert.equal(geminiCliOAuthProvider.usesCallbackServer, true);
203
+ });
204
+ test("has required methods", () => {
205
+ assert.equal(typeof geminiCliOAuthProvider.login, "function");
206
+ assert.equal(typeof geminiCliOAuthProvider.refreshToken, "function");
207
+ assert.equal(typeof geminiCliOAuthProvider.getApiKey, "function");
208
+ });
209
+ test("getApiKey returns JSON with token and projectId", () => {
210
+ const credentials = {
211
+ access: "test-access-token",
212
+ refresh: "test-refresh-token",
213
+ expires: Date.now() + 3600000,
214
+ projectId: "test-project-456",
215
+ email: "test@example.com",
216
+ };
217
+ const apiKey = geminiCliOAuthProvider.getApiKey(credentials);
218
+ assert.equal(typeof apiKey, "string");
219
+ const parsed = JSON.parse(apiKey);
220
+ assert.equal(parsed.token, "test-access-token");
221
+ assert.equal(parsed.projectId, "test-project-456");
222
+ });
223
+ test("refreshToken throws when projectId is missing", async () => {
224
+ const credentials = {
225
+ access: "test-access-token",
226
+ refresh: "test-refresh-token",
227
+ expires: Date.now() + 3600000,
228
+ };
229
+ await assert.rejects(geminiCliOAuthProvider.refreshToken(credentials), /Google Cloud credentials missing projectId/);
230
+ });
231
+ });
232
+ // ═══════════════════════════════════════════════════════════════════════════
233
+ // Regression tests for credential de-obfuscation
234
+ //
235
+ // These tests verify that OAuth credentials are stored in plaintext
236
+ // (not base64-encoded) after the refactor in commit 289bb325c.
237
+ // ═══════════════════════════════════════════════════════════════════════════
238
+ describe("OAuth credentials — de-obfuscated regression", () => {
239
+ // Verify we can import the modules without decoding errors
240
+ test("GitHub Copilot module imports successfully", () => {
241
+ // If the module imports, the CLIENT_ID is valid plaintext
242
+ assert.ok(githubCopilotOAuthProvider);
243
+ });
244
+ test("Antigravity module imports successfully", () => {
245
+ assert.ok(antigravityOAuthProvider);
246
+ });
247
+ test("Gemini CLI module imports successfully", () => {
248
+ assert.ok(geminiCliOAuthProvider);
249
+ });
250
+ test("GitHub Copilot CLIENT_ID is plaintext (not base64)", () => {
251
+ const content = readFileSync(join(__dirname, "github-copilot.ts"), "utf-8");
252
+ // Should contain plaintext CLIENT_ID
253
+ assert.ok(content.includes('CLIENT_ID = "Iv1.b507a08c87ecfe98"'));
254
+ // Should NOT contain atob calls for de-obfuscation
255
+ assert.ok(!content.includes("atob("));
256
+ });
257
+ test("Antigravity CLIENT_ID and CLIENT_SECRET are plaintext", () => {
258
+ const content = readFileSync(join(__dirname, "google-antigravity.ts"), "utf-8");
259
+ // Should contain plaintext credentials
260
+ assert.ok(content.includes('CLIENT_ID = "1071006060591-tmhssin2h21lcre235vtolojh4g403ep.apps.googleusercontent.com"'));
261
+ assert.ok(content.includes('CLIENT_SECRET = "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"'));
262
+ // Should NOT contain atob calls for de-obfuscation
263
+ assert.ok(!content.includes("atob("));
264
+ });
265
+ test("Gemini CLI CLIENT_ID and CLIENT_SECRET are plaintext", () => {
266
+ const content = readFileSync(join(__dirname, "google-gemini-cli.ts"), "utf-8");
267
+ // Should contain plaintext credentials
268
+ assert.ok(content.includes('CLIENT_ID = "681255809395-oo8ft2oprdrnp9e3aqf6av3hmdib135j.apps.googleusercontent.com"'));
269
+ assert.ok(content.includes('CLIENT_SECRET = "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"'));
270
+ // Should NOT contain atob calls for de-obfuscation
271
+ assert.ok(!content.includes("atob("));
272
+ });
273
+ test("all OAuth files include security explanation comments", () => {
274
+ const files = [
275
+ "github-copilot.ts",
276
+ "google-antigravity.ts",
277
+ "google-gemini-cli.ts",
278
+ ];
279
+ for (const file of files) {
280
+ const content = readFileSync(join(__dirname, file), "utf-8");
281
+ // Should include explanation of why credentials are public
282
+ assert.ok(content.includes("NOTE: This credential is public") ||
283
+ content.includes("NOTE: These credentials are public"), `${file} should explain why credentials are public`);
284
+ assert.ok(content.includes("obfuscated") ||
285
+ content.includes("security scanners"), `${file} should reference the security scanner issue`);
286
+ }
287
+ });
288
+ });
289
+ //# sourceMappingURL=oauth-providers.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"oauth-providers.test.js","sourceRoot":"","sources":["../../../src/utils/oauth/oauth-providers.test.ts"],"names":[],"mappings":"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","sourcesContent":["import { describe, test } from \"node:test\";\nimport assert from \"node:assert/strict\";\nimport { readFileSync } from \"node:fs\";\nimport { join } from \"node:path\";\nimport { fileURLToPath } from \"node:url\";\nimport type { Model, Api } from \"../../types.js\";\nimport type { OAuthCredentials } from \"./types.js\";\n\n// Test files that need to import from the OAuth modules\nimport {\n\tgithubCopilotOAuthProvider,\n\tnormalizeDomain,\n\tgetGitHubCopilotBaseUrl,\n} from \"./github-copilot.js\";\nimport { antigravityOAuthProvider } from \"./google-antigravity.js\";\nimport { geminiCliOAuthProvider } from \"./google-gemini-cli.js\";\n\nconst __dirname = fileURLToPath(new URL(\".\", import.meta.url));\n\n// Local type for Copilot credentials (includes optional fields)\ntype CopilotCredentials = OAuthCredentials & {\n\tenterpriseUrl?: string;\n\tmodelLimits?: Record<string, { contextWindow: number; maxTokens: number }>;\n};\n\n// Helper to create a minimal model for testing\nfunction createModel(overrides: Partial<Model<Api>> = {}): Model<Api> {\n\treturn {\n\t\tid: \"test-model\",\n\t\tname: \"Test Model\",\n\t\tapi: \"openai-completions\",\n\t\tprovider: \"test-provider\",\n\t\tbaseUrl: \"https://example.com\",\n\t\treasoning: false,\n\t\tinput: [\"text\"],\n\t\tcost: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0 },\n\t\tcontextWindow: 128000,\n\t\tmaxTokens: 8192,\n\t\t...overrides,\n\t} as Model<Api>;\n}\n\n// ═══════════════════════════════════════════════════════════════════════════\n// GitHub Copilot OAuth Provider Tests\n// ═══════════════════════════════════════════════════════════════════════════\n\ndescribe(\"GitHub Copilot OAuth — normalizeDomain\", () => {\n\ttest(\"returns null for empty input\", () => {\n\t\tassert.equal(normalizeDomain(\"\"), null);\n\t\tassert.equal(normalizeDomain(\" \"), null);\n\t});\n\n\ttest(\"returns null for invalid domain\", () => {\n\t\tassert.equal(normalizeDomain(\"not a domain!@#\"), null);\n\t});\n\n\ttest(\"extracts hostname from full URL\", () => {\n\t\tassert.equal(normalizeDomain(\"https://github.com\"), \"github.com\");\n\t\tassert.equal(normalizeDomain(\"https://company.ghe.com\"), \"company.ghe.com\");\n\t\tassert.equal(normalizeDomain(\"http://example.com/path\"), \"example.com\");\n\t});\n\n\ttest(\"returns domain as-is when no protocol\", () => {\n\t\tassert.equal(normalizeDomain(\"github.com\"), \"github.com\");\n\t\tassert.equal(normalizeDomain(\"company.ghe.com\"), \"company.ghe.com\");\n\t});\n\n\ttest(\"trims whitespace\", () => {\n\t\tassert.equal(normalizeDomain(\" github.com \"), \"github.com\");\n\t});\n});\n\ndescribe(\"GitHub Copilot OAuth — getBaseUrlFromToken\", () => {\n\ttest(\"extracts API URL from token with proxy-ep\", () => {\n\t\t// Token format: tid=...;exp=...;proxy-ep=proxy.individual.githubcopilot.com;...\n\t\tconst token = \"tid=123;exp=1234567890;proxy-ep=proxy.individual.githubcopilot.com;other=value\";\n\t\tconst baseUrl = getGitHubCopilotBaseUrl(token);\n\t\tassert.equal(baseUrl, \"https://api.individual.githubcopilot.com\");\n\t});\n\n\ttest(\"extracts API URL from enterprise proxy-ep\", () => {\n\t\tconst token = \"tid=123;exp=1234567890;proxy-ep=proxy.company.ghe.com;other=value\";\n\t\tconst baseUrl = getGitHubCopilotBaseUrl(token);\n\t\tassert.equal(baseUrl, \"https://api.company.ghe.com\");\n\t});\n\n\ttest(\"falls back to default when no token provided\", () => {\n\t\tconst baseUrl = getGitHubCopilotBaseUrl();\n\t\tassert.equal(baseUrl, \"https://api.individual.githubcopilot.com\");\n\t});\n\n\ttest(\"falls back to default when token has no proxy-ep\", () => {\n\t\tconst token = \"tid=123;exp=1234567890;other=value\";\n\t\tconst baseUrl = getGitHubCopilotBaseUrl(token);\n\t\tassert.equal(baseUrl, \"https://api.individual.githubcopilot.com\");\n\t});\n\n\ttest(\"uses enterprise domain when provided\", () => {\n\t\tconst baseUrl = getGitHubCopilotBaseUrl(undefined, \"company.ghe.com\");\n\t\tassert.equal(baseUrl, \"https://copilot-api.company.ghe.com\");\n\t});\n\n\ttest(\"prioritizes token proxy-ep over enterprise domain\", () => {\n\t\tconst token = \"tid=123;exp=1234567890;proxy-ep=proxy.individual.githubcopilot.com;other=value\";\n\t\tconst baseUrl = getGitHubCopilotBaseUrl(token, \"company.ghe.com\");\n\t\tassert.equal(baseUrl, \"https://api.individual.githubcopilot.com\");\n\t});\n});\n\ndescribe(\"GitHub Copilot OAuth — provider structure\", () => {\n\ttest(\"has correct id and name\", () => {\n\t\tassert.equal(githubCopilotOAuthProvider.id, \"github-copilot\");\n\t\tassert.equal(githubCopilotOAuthProvider.name, \"GitHub Copilot\");\n\t});\n\n\ttest(\"has required methods\", () => {\n\t\tassert.equal(typeof githubCopilotOAuthProvider.login, \"function\");\n\t\tassert.equal(typeof githubCopilotOAuthProvider.refreshToken, \"function\");\n\t\tassert.equal(typeof githubCopilotOAuthProvider.getApiKey, \"function\");\n\t});\n\n\ttest(\"getApiKey returns access token\", () => {\n\t\tconst credentials: OAuthCredentials = {\n\t\t\taccess: \"test-access-token\",\n\t\t\trefresh: \"test-refresh-token\",\n\t\t\texpires: Date.now() + 3600000,\n\t\t};\n\t\tconst apiKey = githubCopilotOAuthProvider.getApiKey(credentials);\n\t\tassert.equal(apiKey, \"test-access-token\");\n\t});\n\n\ttest(\"modifyModels preserves non-Copilot models\", () => {\n\t\tif (!githubCopilotOAuthProvider.modifyModels) return;\n\t\tconst models = [createModel({ id: \"gpt-4\", provider: \"openai\" })];\n\t\tconst credentials: OAuthCredentials = {\n\t\t\taccess: \"test-token\",\n\t\t\trefresh: \"test-refresh\",\n\t\t\texpires: Date.now() + 3600000,\n\t\t};\n\t\tconst result = githubCopilotOAuthProvider.modifyModels(models, credentials);\n\t\tassert.deepEqual(result, models);\n\t});\n\n\ttest(\"modifyModels updates Copilot model baseUrl when token has proxy-ep\", () => {\n\t\tif (!githubCopilotOAuthProvider.modifyModels) return;\n\t\tconst models = [\n\t\t\tcreateModel({\n\t\t\t\tid: \"claude-3.5-sonnet\",\n\t\t\t\tprovider: \"github-copilot\",\n\t\t\t\tbaseUrl: \"https://api.default.com\",\n\t\t\t}),\n\t\t];\n\t\tconst credentials: CopilotCredentials = {\n\t\t\taccess: \"tid=123;exp=1234567890;proxy-ep=proxy.individual.githubcopilot.com;\",\n\t\t\trefresh: \"test-refresh\",\n\t\t\texpires: Date.now() + 3600000,\n\t\t};\n\t\tconst result = githubCopilotOAuthProvider.modifyModels(models, credentials);\n\t\tassert.equal(result[0].baseUrl, \"https://api.individual.githubcopilot.com\");\n\t});\n\n\ttest(\"modifyModels applies model limits when available\", () => {\n\t\tif (!githubCopilotOAuthProvider.modifyModels) return;\n\t\tconst models = [\n\t\t\tcreateModel({\n\t\t\t\tid: \"claude-3.5-sonnet\",\n\t\t\t\tprovider: \"github-copilot\",\n\t\t\t\tbaseUrl: \"https://api.default.com\",\n\t\t\t}),\n\t\t];\n\t\tconst credentials: CopilotCredentials = {\n\t\t\taccess: \"test-token\",\n\t\t\trefresh: \"test-refresh\",\n\t\t\texpires: Date.now() + 3600000,\n\t\t\tmodelLimits: {\n\t\t\t\t\"claude-3.5-sonnet\": { contextWindow: 123456, maxTokens: 4096 },\n\t\t\t},\n\t\t};\n\t\tconst result = githubCopilotOAuthProvider.modifyModels(models, credentials);\n\t\tassert.equal(result[0].contextWindow, 123456);\n\t\tassert.equal(result[0].maxTokens, 4096);\n\t});\n});\n\n// ═══════════════════════════════════════════════════════════════════════════\n// Antigravity OAuth Provider Tests\n// ═══════════════════════════════════════════════════════════════════════════\n\ndescribe(\"Antigravity OAuth — provider structure\", () => {\n\ttest(\"has correct id and name\", () => {\n\t\tassert.equal(antigravityOAuthProvider.id, \"google-antigravity\");\n\t\tassert.equal(antigravityOAuthProvider.name, \"Antigravity (Gemini 3, Claude, GPT-OSS)\");\n\t});\n\n\ttest(\"uses callback server\", () => {\n\t\tassert.equal(antigravityOAuthProvider.usesCallbackServer, true);\n\t});\n\n\ttest(\"has required methods\", () => {\n\t\tassert.equal(typeof antigravityOAuthProvider.login, \"function\");\n\t\tassert.equal(typeof antigravityOAuthProvider.refreshToken, \"function\");\n\t\tassert.equal(typeof antigravityOAuthProvider.getApiKey, \"function\");\n\t});\n\n\ttest(\"getApiKey returns JSON with token and projectId\", () => {\n\t\tconst credentials: OAuthCredentials = {\n\t\t\taccess: \"test-access-token\",\n\t\t\trefresh: \"test-refresh-token\",\n\t\t\texpires: Date.now() + 3600000,\n\t\t\tprojectId: \"test-project-123\",\n\t\t\temail: \"test@example.com\",\n\t\t};\n\t\tconst apiKey = antigravityOAuthProvider.getApiKey(credentials);\n\t\tassert.equal(typeof apiKey, \"string\");\n\t\tconst parsed = JSON.parse(apiKey);\n\t\tassert.equal(parsed.token, \"test-access-token\");\n\t\tassert.equal(parsed.projectId, \"test-project-123\");\n\t});\n\n\ttest(\"refreshToken throws when projectId is missing\", async () => {\n\t\tconst credentials: OAuthCredentials = {\n\t\t\taccess: \"test-access-token\",\n\t\t\trefresh: \"test-refresh-token\",\n\t\t\texpires: Date.now() + 3600000,\n\t\t};\n\t\tawait assert.rejects(\n\t\t\tantigravityOAuthProvider.refreshToken(credentials),\n\t\t\t/Antigravity credentials missing projectId/,\n\t\t);\n\t});\n});\n\n// ═══════════════════════════════════════════════════════════════════════════\n// Gemini CLI OAuth Provider Tests\n// ═══════════════════════════════════════════════════════════════════════════\n\ndescribe(\"Gemini CLI OAuth — provider structure\", () => {\n\ttest(\"has correct id and name\", () => {\n\t\tassert.equal(geminiCliOAuthProvider.id, \"google-gemini-cli\");\n\t\tassert.equal(geminiCliOAuthProvider.name, \"Google Cloud Code Assist (Gemini CLI)\");\n\t});\n\n\ttest(\"uses callback server\", () => {\n\t\tassert.equal(geminiCliOAuthProvider.usesCallbackServer, true);\n\t});\n\n\ttest(\"has required methods\", () => {\n\t\tassert.equal(typeof geminiCliOAuthProvider.login, \"function\");\n\t\tassert.equal(typeof geminiCliOAuthProvider.refreshToken, \"function\");\n\t\tassert.equal(typeof geminiCliOAuthProvider.getApiKey, \"function\");\n\t});\n\n\ttest(\"getApiKey returns JSON with token and projectId\", () => {\n\t\tconst credentials: OAuthCredentials = {\n\t\t\taccess: \"test-access-token\",\n\t\t\trefresh: \"test-refresh-token\",\n\t\t\texpires: Date.now() + 3600000,\n\t\t\tprojectId: \"test-project-456\",\n\t\t\temail: \"test@example.com\",\n\t\t};\n\t\tconst apiKey = geminiCliOAuthProvider.getApiKey(credentials);\n\t\tassert.equal(typeof apiKey, \"string\");\n\t\tconst parsed = JSON.parse(apiKey);\n\t\tassert.equal(parsed.token, \"test-access-token\");\n\t\tassert.equal(parsed.projectId, \"test-project-456\");\n\t});\n\n\ttest(\"refreshToken throws when projectId is missing\", async () => {\n\t\tconst credentials: OAuthCredentials = {\n\t\t\taccess: \"test-access-token\",\n\t\t\trefresh: \"test-refresh-token\",\n\t\t\texpires: Date.now() + 3600000,\n\t\t};\n\t\tawait assert.rejects(\n\t\t\tgeminiCliOAuthProvider.refreshToken(credentials),\n\t\t\t/Google Cloud credentials missing projectId/,\n\t\t);\n\t});\n});\n\n// ═══════════════════════════════════════════════════════════════════════════\n// Regression tests for credential de-obfuscation\n//\n// These tests verify that OAuth credentials are stored in plaintext\n// (not base64-encoded) after the refactor in commit 289bb325c.\n// ═══════════════════════════════════════════════════════════════════════════\n\ndescribe(\"OAuth credentials — de-obfuscated regression\", () => {\n\t// Verify we can import the modules without decoding errors\n\ttest(\"GitHub Copilot module imports successfully\", () => {\n\t\t// If the module imports, the CLIENT_ID is valid plaintext\n\t\tassert.ok(githubCopilotOAuthProvider);\n\t});\n\n\ttest(\"Antigravity module imports successfully\", () => {\n\t\tassert.ok(antigravityOAuthProvider);\n\t});\n\n\ttest(\"Gemini CLI module imports successfully\", () => {\n\t\tassert.ok(geminiCliOAuthProvider);\n\t});\n\n\ttest(\"GitHub Copilot CLIENT_ID is plaintext (not base64)\", () => {\n\t\tconst content = readFileSync(join(__dirname, \"github-copilot.ts\"), \"utf-8\");\n\t\t// Should contain plaintext CLIENT_ID\n\t\tassert.ok(content.includes('CLIENT_ID = \"Iv1.b507a08c87ecfe98\"'));\n\t\t// Should NOT contain atob calls for de-obfuscation\n\t\tassert.ok(!content.includes(\"atob(\"));\n\t});\n\n\ttest(\"Antigravity CLIENT_ID and CLIENT_SECRET are plaintext\", () => {\n\t\tconst content = readFileSync(join(__dirname, \"google-antigravity.ts\"), \"utf-8\");\n\t\t// Should contain plaintext credentials\n\t\tassert.ok(\n\t\t\tcontent.includes(\n\t\t\t\t'CLIENT_ID = \"1071006060591-tmhssin2h21lcre235vtolojh4g403ep.apps.googleusercontent.com\"',\n\t\t\t),\n\t\t);\n\t\tassert.ok(\n\t\t\tcontent.includes('CLIENT_SECRET = \"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\"'),\n\t\t);\n\t\t// Should NOT contain atob calls for de-obfuscation\n\t\tassert.ok(!content.includes(\"atob(\"));\n\t});\n\n\ttest(\"Gemini CLI CLIENT_ID and CLIENT_SECRET are plaintext\", () => {\n\t\tconst content = readFileSync(join(__dirname, \"google-gemini-cli.ts\"), \"utf-8\");\n\t\t// Should contain plaintext credentials\n\t\tassert.ok(\n\t\t\tcontent.includes(\n\t\t\t\t'CLIENT_ID = \"681255809395-oo8ft2oprdrnp9e3aqf6av3hmdib135j.apps.googleusercontent.com\"',\n\t\t\t),\n\t\t);\n\t\tassert.ok(\n\t\t\tcontent.includes('CLIENT_SECRET = \"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\"'),\n\t\t);\n\t\t// Should NOT contain atob calls for de-obfuscation\n\t\tassert.ok(!content.includes(\"atob(\"));\n\t});\n\n\ttest(\"all OAuth files include security explanation comments\", () => {\n\t\tconst files = [\n\t\t\t\"github-copilot.ts\",\n\t\t\t\"google-antigravity.ts\",\n\t\t\t\"google-gemini-cli.ts\",\n\t\t];\n\n\t\tfor (const file of files) {\n\t\t\tconst content = readFileSync(join(__dirname, file), \"utf-8\");\n\t\t\t// Should include explanation of why credentials are public\n\t\t\tassert.ok(\n\t\t\t\tcontent.includes(\"NOTE: This credential is public\") ||\n\t\t\t\t\tcontent.includes(\"NOTE: These credentials are public\"),\n\t\t\t\t`${file} should explain why credentials are public`,\n\t\t\t);\n\t\t\tassert.ok(\n\t\t\t\tcontent.includes(\"obfuscated\") ||\n\t\t\t\t\tcontent.includes(\"security scanners\"),\n\t\t\t\t`${file} should reference the security scanner issue`,\n\t\t\t);\n\t\t}\n\t});\n});\n"]}