@gajae-code/ai 0.1.1

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 (349) hide show
  1. package/CHANGELOG.md +2644 -0
  2. package/README.md +1181 -0
  3. package/dist/types/api-registry.d.ts +30 -0
  4. package/dist/types/auth-broker/client.d.ts +66 -0
  5. package/dist/types/auth-broker/index.d.ts +5 -0
  6. package/dist/types/auth-broker/refresher.d.ts +25 -0
  7. package/dist/types/auth-broker/remote-store.d.ts +96 -0
  8. package/dist/types/auth-broker/server.d.ts +32 -0
  9. package/dist/types/auth-broker/types.d.ts +105 -0
  10. package/dist/types/auth-broker/wire-schemas.d.ts +412 -0
  11. package/dist/types/auth-gateway/http.d.ts +39 -0
  12. package/dist/types/auth-gateway/index.d.ts +3 -0
  13. package/dist/types/auth-gateway/server.d.ts +17 -0
  14. package/dist/types/auth-gateway/types.d.ts +115 -0
  15. package/dist/types/auth-storage.d.ts +641 -0
  16. package/dist/types/cli.d.ts +2 -0
  17. package/dist/types/index.d.ts +49 -0
  18. package/dist/types/model-cache.d.ts +17 -0
  19. package/dist/types/model-manager.d.ts +62 -0
  20. package/dist/types/model-thinking.d.ts +71 -0
  21. package/dist/types/models.d.ts +12 -0
  22. package/dist/types/provider-details.d.ts +24 -0
  23. package/dist/types/provider-models/bundled-references.d.ts +4 -0
  24. package/dist/types/provider-models/descriptors.d.ts +48 -0
  25. package/dist/types/provider-models/google.d.ts +20 -0
  26. package/dist/types/provider-models/index.d.ts +5 -0
  27. package/dist/types/provider-models/ollama.d.ts +7 -0
  28. package/dist/types/provider-models/openai-compat.d.ts +237 -0
  29. package/dist/types/provider-models/special.d.ts +16 -0
  30. package/dist/types/providers/amazon-bedrock.d.ts +36 -0
  31. package/dist/types/providers/anthropic-messages-server-schema.d.ts +450 -0
  32. package/dist/types/providers/anthropic-messages-server.d.ts +17 -0
  33. package/dist/types/providers/anthropic.d.ts +188 -0
  34. package/dist/types/providers/aws-credentials.d.ts +43 -0
  35. package/dist/types/providers/aws-eventstream.d.ts +38 -0
  36. package/dist/types/providers/aws-sigv4.d.ts +55 -0
  37. package/dist/types/providers/azure-openai-responses.d.ts +15 -0
  38. package/dist/types/providers/cursor/gen/agent_pb.d.ts +13022 -0
  39. package/dist/types/providers/cursor.d.ts +42 -0
  40. package/dist/types/providers/error-message.d.ts +27 -0
  41. package/dist/types/providers/github-copilot-headers.d.ts +40 -0
  42. package/dist/types/providers/gitlab-duo.d.ts +27 -0
  43. package/dist/types/providers/google-auth.d.ts +24 -0
  44. package/dist/types/providers/google-gemini-cli.d.ts +72 -0
  45. package/dist/types/providers/google-gemini-headers.d.ts +18 -0
  46. package/dist/types/providers/google-shared.d.ts +163 -0
  47. package/dist/types/providers/google-types.d.ts +138 -0
  48. package/dist/types/providers/google-vertex.d.ts +7 -0
  49. package/dist/types/providers/google.d.ts +4 -0
  50. package/dist/types/providers/grammar.d.ts +1 -0
  51. package/dist/types/providers/kimi.d.ts +27 -0
  52. package/dist/types/providers/mock.d.ts +175 -0
  53. package/dist/types/providers/ollama.d.ts +6 -0
  54. package/dist/types/providers/openai-anthropic-shim.d.ts +31 -0
  55. package/dist/types/providers/openai-chat-server-schema.d.ts +814 -0
  56. package/dist/types/providers/openai-chat-server.d.ts +16 -0
  57. package/dist/types/providers/openai-codex/constants.d.ts +26 -0
  58. package/dist/types/providers/openai-codex/request-transformer.d.ts +49 -0
  59. package/dist/types/providers/openai-codex/response-handler.d.ts +17 -0
  60. package/dist/types/providers/openai-codex-responses.d.ts +67 -0
  61. package/dist/types/providers/openai-completions-compat.d.ts +25 -0
  62. package/dist/types/providers/openai-completions.d.ts +33 -0
  63. package/dist/types/providers/openai-responses-server-schema.d.ts +392 -0
  64. package/dist/types/providers/openai-responses-server.d.ts +17 -0
  65. package/dist/types/providers/openai-responses-shared.d.ts +89 -0
  66. package/dist/types/providers/openai-responses.d.ts +32 -0
  67. package/dist/types/providers/pi-native-client.d.ts +13 -0
  68. package/dist/types/providers/pi-native-server.d.ts +68 -0
  69. package/dist/types/providers/register-builtins.d.ts +31 -0
  70. package/dist/types/providers/synthetic.d.ts +26 -0
  71. package/dist/types/providers/transform-messages.d.ts +12 -0
  72. package/dist/types/providers/vision-guard.d.ts +8 -0
  73. package/dist/types/rate-limit-utils.d.ts +19 -0
  74. package/dist/types/stream.d.ts +24 -0
  75. package/dist/types/types.d.ts +746 -0
  76. package/dist/types/usage/claude.d.ts +3 -0
  77. package/dist/types/usage/gemini.d.ts +2 -0
  78. package/dist/types/usage/github-copilot.d.ts +7 -0
  79. package/dist/types/usage/google-antigravity.d.ts +2 -0
  80. package/dist/types/usage/kimi.d.ts +2 -0
  81. package/dist/types/usage/minimax-code.d.ts +2 -0
  82. package/dist/types/usage/openai-codex.d.ts +3 -0
  83. package/dist/types/usage/shared.d.ts +1 -0
  84. package/dist/types/usage/zai.d.ts +2 -0
  85. package/dist/types/usage.d.ts +258 -0
  86. package/dist/types/utils/abort.d.ts +19 -0
  87. package/dist/types/utils/anthropic-auth.d.ts +31 -0
  88. package/dist/types/utils/discovery/antigravity.d.ts +61 -0
  89. package/dist/types/utils/discovery/codex.d.ts +38 -0
  90. package/dist/types/utils/discovery/cursor.d.ts +23 -0
  91. package/dist/types/utils/discovery/gemini.d.ts +25 -0
  92. package/dist/types/utils/discovery/index.d.ts +4 -0
  93. package/dist/types/utils/discovery/openai-compatible.d.ts +72 -0
  94. package/dist/types/utils/event-stream.d.ts +28 -0
  95. package/dist/types/utils/fireworks-model-id.d.ts +10 -0
  96. package/dist/types/utils/foundry.d.ts +1 -0
  97. package/dist/types/utils/h2-fetch.d.ts +22 -0
  98. package/dist/types/utils/http-inspector.d.ts +31 -0
  99. package/dist/types/utils/idle-iterator.d.ts +67 -0
  100. package/dist/types/utils/json-parse.d.ts +10 -0
  101. package/dist/types/utils/oauth/alibaba-coding-plan.d.ts +18 -0
  102. package/dist/types/utils/oauth/anthropic.d.ts +22 -0
  103. package/dist/types/utils/oauth/api-key-login.d.ts +35 -0
  104. package/dist/types/utils/oauth/api-key-validation.d.ts +27 -0
  105. package/dist/types/utils/oauth/callback-server.d.ts +57 -0
  106. package/dist/types/utils/oauth/cerebras.d.ts +1 -0
  107. package/dist/types/utils/oauth/cloudflare-ai-gateway.d.ts +18 -0
  108. package/dist/types/utils/oauth/cursor.d.ts +15 -0
  109. package/dist/types/utils/oauth/deepseek.d.ts +10 -0
  110. package/dist/types/utils/oauth/firepass.d.ts +1 -0
  111. package/dist/types/utils/oauth/fireworks.d.ts +1 -0
  112. package/dist/types/utils/oauth/github-copilot.d.ts +38 -0
  113. package/dist/types/utils/oauth/gitlab-duo.d.ts +3 -0
  114. package/dist/types/utils/oauth/google-antigravity.d.ts +11 -0
  115. package/dist/types/utils/oauth/google-gemini-cli.d.ts +10 -0
  116. package/dist/types/utils/oauth/google-oauth-shared.d.ts +28 -0
  117. package/dist/types/utils/oauth/huggingface.d.ts +19 -0
  118. package/dist/types/utils/oauth/index.d.ts +38 -0
  119. package/dist/types/utils/oauth/kagi.d.ts +17 -0
  120. package/dist/types/utils/oauth/kilo.d.ts +5 -0
  121. package/dist/types/utils/oauth/kimi.d.ts +21 -0
  122. package/dist/types/utils/oauth/litellm.d.ts +18 -0
  123. package/dist/types/utils/oauth/lm-studio.d.ts +17 -0
  124. package/dist/types/utils/oauth/minimax-code.d.ts +28 -0
  125. package/dist/types/utils/oauth/moonshot.d.ts +1 -0
  126. package/dist/types/utils/oauth/nanogpt.d.ts +1 -0
  127. package/dist/types/utils/oauth/nvidia.d.ts +18 -0
  128. package/dist/types/utils/oauth/ollama-cloud.d.ts +2 -0
  129. package/dist/types/utils/oauth/ollama.d.ts +18 -0
  130. package/dist/types/utils/oauth/openai-codex.d.ts +21 -0
  131. package/dist/types/utils/oauth/opencode.d.ts +18 -0
  132. package/dist/types/utils/oauth/parallel.d.ts +17 -0
  133. package/dist/types/utils/oauth/perplexity.d.ts +9 -0
  134. package/dist/types/utils/oauth/pkce.d.ts +8 -0
  135. package/dist/types/utils/oauth/qianfan.d.ts +17 -0
  136. package/dist/types/utils/oauth/qwen-portal.d.ts +19 -0
  137. package/dist/types/utils/oauth/synthetic.d.ts +1 -0
  138. package/dist/types/utils/oauth/tavily.d.ts +17 -0
  139. package/dist/types/utils/oauth/together.d.ts +1 -0
  140. package/dist/types/utils/oauth/types.d.ts +44 -0
  141. package/dist/types/utils/oauth/venice.d.ts +18 -0
  142. package/dist/types/utils/oauth/vercel-ai-gateway.d.ts +18 -0
  143. package/dist/types/utils/oauth/vllm.d.ts +16 -0
  144. package/dist/types/utils/oauth/xiaomi.d.ts +19 -0
  145. package/dist/types/utils/oauth/zai.d.ts +18 -0
  146. package/dist/types/utils/oauth/zenmux.d.ts +1 -0
  147. package/dist/types/utils/overflow.d.ts +54 -0
  148. package/dist/types/utils/parse-bind.d.ts +23 -0
  149. package/dist/types/utils/provider-response.d.ts +3 -0
  150. package/dist/types/utils/retry-after.d.ts +3 -0
  151. package/dist/types/utils/retry.d.ts +26 -0
  152. package/dist/types/utils/schema/adapt.d.ts +24 -0
  153. package/dist/types/utils/schema/compatibility.d.ts +30 -0
  154. package/dist/types/utils/schema/dereference.d.ts +11 -0
  155. package/dist/types/utils/schema/draft.d.ts +10 -0
  156. package/dist/types/utils/schema/equality.d.ts +4 -0
  157. package/dist/types/utils/schema/fields.d.ts +49 -0
  158. package/dist/types/utils/schema/index.d.ts +13 -0
  159. package/dist/types/utils/schema/json-schema-validator.d.ts +12 -0
  160. package/dist/types/utils/schema/meta-validator.d.ts +2 -0
  161. package/dist/types/utils/schema/normalize.d.ts +93 -0
  162. package/dist/types/utils/schema/spill.d.ts +8 -0
  163. package/dist/types/utils/schema/stamps.d.ts +25 -0
  164. package/dist/types/utils/schema/types.d.ts +4 -0
  165. package/dist/types/utils/schema/wire.d.ts +54 -0
  166. package/dist/types/utils/schema/zod-decontaminate.d.ts +31 -0
  167. package/dist/types/utils/sse-debug.d.ts +10 -0
  168. package/dist/types/utils/tool-call-healing.d.ts +71 -0
  169. package/dist/types/utils/tool-choice.d.ts +50 -0
  170. package/dist/types/utils/validation.d.ts +17 -0
  171. package/dist/types/utils.d.ts +28 -0
  172. package/package.json +146 -0
  173. package/src/api-registry.ts +96 -0
  174. package/src/auth-broker/client.ts +358 -0
  175. package/src/auth-broker/index.ts +5 -0
  176. package/src/auth-broker/refresher.ts +127 -0
  177. package/src/auth-broker/remote-store.ts +623 -0
  178. package/src/auth-broker/server.ts +644 -0
  179. package/src/auth-broker/types.ts +127 -0
  180. package/src/auth-broker/wire-schemas.ts +200 -0
  181. package/src/auth-gateway/http.ts +194 -0
  182. package/src/auth-gateway/index.ts +3 -0
  183. package/src/auth-gateway/server.ts +717 -0
  184. package/src/auth-gateway/types.ts +134 -0
  185. package/src/auth-storage.ts +4104 -0
  186. package/src/cli.ts +262 -0
  187. package/src/index.ts +54 -0
  188. package/src/model-cache.ts +129 -0
  189. package/src/model-manager.ts +450 -0
  190. package/src/model-thinking.ts +691 -0
  191. package/src/models.json +73853 -0
  192. package/src/models.json.d.ts +9 -0
  193. package/src/models.ts +56 -0
  194. package/src/prompts/turn-aborted-guidance.md +4 -0
  195. package/src/provider-details.ts +90 -0
  196. package/src/provider-models/bundled-references.ts +38 -0
  197. package/src/provider-models/descriptors.ts +308 -0
  198. package/src/provider-models/google.ts +91 -0
  199. package/src/provider-models/index.ts +5 -0
  200. package/src/provider-models/ollama.ts +153 -0
  201. package/src/provider-models/openai-compat.ts +2275 -0
  202. package/src/provider-models/special.ts +67 -0
  203. package/src/providers/amazon-bedrock.ts +849 -0
  204. package/src/providers/anthropic-messages-server-schema.ts +229 -0
  205. package/src/providers/anthropic-messages-server.ts +677 -0
  206. package/src/providers/anthropic.ts +2696 -0
  207. package/src/providers/aws-credentials.ts +501 -0
  208. package/src/providers/aws-eventstream.ts +185 -0
  209. package/src/providers/aws-sigv4.ts +218 -0
  210. package/src/providers/azure-openai-responses.ts +337 -0
  211. package/src/providers/cursor/gen/agent_pb.ts +15274 -0
  212. package/src/providers/cursor/proto/agent.proto +3526 -0
  213. package/src/providers/cursor/proto/buf.gen.yaml +6 -0
  214. package/src/providers/cursor/proto/buf.yaml +17 -0
  215. package/src/providers/cursor.ts +2561 -0
  216. package/src/providers/error-message.ts +21 -0
  217. package/src/providers/github-copilot-headers.ts +140 -0
  218. package/src/providers/gitlab-duo.ts +372 -0
  219. package/src/providers/google-auth.ts +252 -0
  220. package/src/providers/google-gemini-cli.ts +795 -0
  221. package/src/providers/google-gemini-headers.ts +41 -0
  222. package/src/providers/google-shared.ts +902 -0
  223. package/src/providers/google-types.ts +167 -0
  224. package/src/providers/google-vertex.ts +88 -0
  225. package/src/providers/google.ts +41 -0
  226. package/src/providers/grammar.ts +70 -0
  227. package/src/providers/kimi.ts +52 -0
  228. package/src/providers/mock.ts +500 -0
  229. package/src/providers/ollama.ts +544 -0
  230. package/src/providers/openai-anthropic-shim.ts +138 -0
  231. package/src/providers/openai-chat-server-schema.ts +243 -0
  232. package/src/providers/openai-chat-server.ts +628 -0
  233. package/src/providers/openai-codex/constants.ts +43 -0
  234. package/src/providers/openai-codex/request-transformer.ts +161 -0
  235. package/src/providers/openai-codex/response-handler.ts +81 -0
  236. package/src/providers/openai-codex-responses.ts +2598 -0
  237. package/src/providers/openai-completions-compat.ts +279 -0
  238. package/src/providers/openai-completions.ts +1853 -0
  239. package/src/providers/openai-responses-server-schema.ts +290 -0
  240. package/src/providers/openai-responses-server.ts +1183 -0
  241. package/src/providers/openai-responses-shared.ts +800 -0
  242. package/src/providers/openai-responses.ts +621 -0
  243. package/src/providers/pi-native-client.ts +228 -0
  244. package/src/providers/pi-native-server.ts +210 -0
  245. package/src/providers/register-builtins.ts +412 -0
  246. package/src/providers/synthetic.ts +50 -0
  247. package/src/providers/transform-messages.ts +309 -0
  248. package/src/providers/vision-guard.ts +31 -0
  249. package/src/rate-limit-utils.ts +84 -0
  250. package/src/stream.ts +895 -0
  251. package/src/types.ts +884 -0
  252. package/src/usage/claude.ts +431 -0
  253. package/src/usage/gemini.ts +250 -0
  254. package/src/usage/github-copilot.ts +421 -0
  255. package/src/usage/google-antigravity.ts +201 -0
  256. package/src/usage/kimi.ts +271 -0
  257. package/src/usage/minimax-code.ts +31 -0
  258. package/src/usage/openai-codex.ts +503 -0
  259. package/src/usage/shared.ts +10 -0
  260. package/src/usage/zai.ts +247 -0
  261. package/src/usage.ts +183 -0
  262. package/src/utils/abort.ts +51 -0
  263. package/src/utils/anthropic-auth.ts +87 -0
  264. package/src/utils/discovery/antigravity.ts +261 -0
  265. package/src/utils/discovery/codex.ts +371 -0
  266. package/src/utils/discovery/cursor.ts +306 -0
  267. package/src/utils/discovery/gemini.ts +248 -0
  268. package/src/utils/discovery/index.ts +4 -0
  269. package/src/utils/discovery/openai-compatible.ts +224 -0
  270. package/src/utils/event-stream.ts +142 -0
  271. package/src/utils/fireworks-model-id.ts +30 -0
  272. package/src/utils/foundry.ts +8 -0
  273. package/src/utils/h2-fetch.ts +60 -0
  274. package/src/utils/http-inspector.ts +176 -0
  275. package/src/utils/idle-iterator.ts +250 -0
  276. package/src/utils/json-parse.ts +148 -0
  277. package/src/utils/oauth/alibaba-coding-plan.ts +59 -0
  278. package/src/utils/oauth/anthropic.ts +200 -0
  279. package/src/utils/oauth/api-key-login.ts +87 -0
  280. package/src/utils/oauth/api-key-validation.ts +92 -0
  281. package/src/utils/oauth/callback-server.ts +276 -0
  282. package/src/utils/oauth/cerebras.ts +16 -0
  283. package/src/utils/oauth/cloudflare-ai-gateway.ts +48 -0
  284. package/src/utils/oauth/cursor.ts +157 -0
  285. package/src/utils/oauth/deepseek.ts +53 -0
  286. package/src/utils/oauth/firepass.ts +24 -0
  287. package/src/utils/oauth/fireworks.ts +15 -0
  288. package/src/utils/oauth/github-copilot.ts +362 -0
  289. package/src/utils/oauth/gitlab-duo.ts +123 -0
  290. package/src/utils/oauth/google-antigravity.ts +200 -0
  291. package/src/utils/oauth/google-gemini-cli.ts +256 -0
  292. package/src/utils/oauth/google-oauth-shared.ts +110 -0
  293. package/src/utils/oauth/huggingface.ts +62 -0
  294. package/src/utils/oauth/index.ts +444 -0
  295. package/src/utils/oauth/kagi.ts +47 -0
  296. package/src/utils/oauth/kilo.ts +87 -0
  297. package/src/utils/oauth/kimi.ts +254 -0
  298. package/src/utils/oauth/litellm.ts +47 -0
  299. package/src/utils/oauth/lm-studio.ts +38 -0
  300. package/src/utils/oauth/minimax-code.ts +78 -0
  301. package/src/utils/oauth/moonshot.ts +16 -0
  302. package/src/utils/oauth/nanogpt.ts +15 -0
  303. package/src/utils/oauth/nvidia.ts +70 -0
  304. package/src/utils/oauth/oauth.html +199 -0
  305. package/src/utils/oauth/ollama-cloud.ts +28 -0
  306. package/src/utils/oauth/ollama.ts +47 -0
  307. package/src/utils/oauth/openai-codex.ts +299 -0
  308. package/src/utils/oauth/opencode.ts +49 -0
  309. package/src/utils/oauth/parallel.ts +46 -0
  310. package/src/utils/oauth/perplexity.ts +206 -0
  311. package/src/utils/oauth/pkce.ts +18 -0
  312. package/src/utils/oauth/qianfan.ts +58 -0
  313. package/src/utils/oauth/qwen-portal.ts +60 -0
  314. package/src/utils/oauth/synthetic.ts +16 -0
  315. package/src/utils/oauth/tavily.ts +46 -0
  316. package/src/utils/oauth/together.ts +16 -0
  317. package/src/utils/oauth/types.ts +94 -0
  318. package/src/utils/oauth/venice.ts +59 -0
  319. package/src/utils/oauth/vercel-ai-gateway.ts +47 -0
  320. package/src/utils/oauth/vllm.ts +40 -0
  321. package/src/utils/oauth/xiaomi.ts +137 -0
  322. package/src/utils/oauth/zai.ts +60 -0
  323. package/src/utils/oauth/zenmux.ts +15 -0
  324. package/src/utils/overflow.ts +137 -0
  325. package/src/utils/parse-bind.ts +54 -0
  326. package/src/utils/provider-response.ts +30 -0
  327. package/src/utils/retry-after.ts +110 -0
  328. package/src/utils/retry.ts +54 -0
  329. package/src/utils/schema/CONSTRAINTS.md +164 -0
  330. package/src/utils/schema/adapt.ts +36 -0
  331. package/src/utils/schema/compatibility.ts +435 -0
  332. package/src/utils/schema/dereference.ts +98 -0
  333. package/src/utils/schema/draft.ts +341 -0
  334. package/src/utils/schema/equality.ts +97 -0
  335. package/src/utils/schema/fields.ts +190 -0
  336. package/src/utils/schema/index.ts +13 -0
  337. package/src/utils/schema/json-schema-validator.ts +577 -0
  338. package/src/utils/schema/meta-validator.ts +167 -0
  339. package/src/utils/schema/normalize.ts +1588 -0
  340. package/src/utils/schema/spill.ts +43 -0
  341. package/src/utils/schema/stamps.ts +97 -0
  342. package/src/utils/schema/types.ts +11 -0
  343. package/src/utils/schema/wire.ts +213 -0
  344. package/src/utils/schema/zod-decontaminate.ts +331 -0
  345. package/src/utils/sse-debug.ts +289 -0
  346. package/src/utils/tool-call-healing.ts +271 -0
  347. package/src/utils/tool-choice.ts +99 -0
  348. package/src/utils/validation.ts +1019 -0
  349. package/src/utils.ts +166 -0
@@ -0,0 +1,306 @@
1
+ import * as http2 from "node:http2";
2
+ import { create, fromBinary, toBinary } from "@bufbuild/protobuf";
3
+ import * as z from "zod/v4";
4
+ import { getBundledModels } from "../../models";
5
+ import { GetUsableModelsRequestSchema, GetUsableModelsResponseSchema } from "../../providers/cursor/gen/agent_pb";
6
+ import type { Model } from "../../types";
7
+
8
+ const CURSOR_DEFAULT_BASE_URL = "https://api2.cursor.sh";
9
+ const CURSOR_DEFAULT_CLIENT_VERSION = "cli-2026.02.13-41ac335";
10
+ const CURSOR_GET_USABLE_MODELS_PATH = "/agent.v1.AgentService/GetUsableModels";
11
+
12
+ const DEFAULT_CONTEXT_WINDOW = 200_000;
13
+ const DEFAULT_MAX_TOKENS = 64_000;
14
+
15
+ const OptionalDisplayNameSchema = z.string().optional().catch(undefined);
16
+ const CursorAliasesSchema = z
17
+ .array(z.unknown())
18
+ .optional()
19
+ .catch([])
20
+ .transform(aliases => (aliases ?? []).filter((alias: unknown): alias is string => typeof alias === "string"));
21
+
22
+ const CursorModelDetailsSchema = z.object({
23
+ modelId: z.string(),
24
+ displayName: OptionalDisplayNameSchema,
25
+ displayNameShort: OptionalDisplayNameSchema,
26
+ displayModelId: OptionalDisplayNameSchema,
27
+ aliases: CursorAliasesSchema,
28
+ thinkingDetails: z.unknown().optional(),
29
+ });
30
+
31
+ const CursorDecodedResponseSchema = z.object({
32
+ models: z.array(z.unknown()).optional().catch([]),
33
+ });
34
+
35
+ type CursorModelDetailsValue = z.infer<typeof CursorModelDetailsSchema>;
36
+
37
+ /**
38
+ * Options for fetching dynamic Cursor models from `GetUsableModels`.
39
+ */
40
+ export interface CursorModelDiscoveryOptions {
41
+ /** Cursor access token used for bearer authentication. */
42
+ apiKey: string;
43
+ /** Optional Cursor API base URL override. */
44
+ baseUrl?: string;
45
+ /** Optional client version override sent as `x-cursor-client-version`. */
46
+ clientVersion?: string;
47
+ /** Optional request timeout in milliseconds. */
48
+ timeoutMs?: number;
49
+ /** Optional list of custom Cursor model ids to include in request context. */
50
+ customModelIds?: string[];
51
+ }
52
+
53
+ /**
54
+ * Fetches Cursor models through `GetUsableModels` and normalizes them into canonical model entries.
55
+ *
56
+ * Returns `null` on request/decode failures.
57
+ * Returns `[]` only when the endpoint responds successfully with no usable models.
58
+ */
59
+ export async function fetchCursorUsableModels(
60
+ options: CursorModelDiscoveryOptions,
61
+ ): Promise<Model<"cursor-agent">[] | null> {
62
+ const timeoutMs = options.timeoutMs ?? 5_000;
63
+ try {
64
+ const requestPayload = create(GetUsableModelsRequestSchema, {
65
+ customModelIds: normalizeCustomModelIds(options.customModelIds),
66
+ });
67
+ const body = toBinary(GetUsableModelsRequestSchema, requestPayload);
68
+ const baseUrl = (options.baseUrl ?? CURSOR_DEFAULT_BASE_URL).replace(/\/+$/, "");
69
+
70
+ const responseBuffer = await fetchViaHttp2(baseUrl, body, options, timeoutMs);
71
+
72
+ if (!responseBuffer) {
73
+ return null;
74
+ }
75
+ const decoded = decodeGetUsableModelsResponse(responseBuffer);
76
+ const parsedDecoded = CursorDecodedResponseSchema.safeParse(decoded);
77
+ if (!parsedDecoded.success) {
78
+ return null;
79
+ }
80
+
81
+ const references = createCursorReferenceMap();
82
+ return normalizeCursorModels(parsedDecoded.data.models, options.baseUrl, references);
83
+ } catch {
84
+ return null;
85
+ }
86
+ }
87
+
88
+ function buildRequestHeaders(options: CursorModelDiscoveryOptions): Record<string, string> {
89
+ return {
90
+ "content-type": "application/proto",
91
+ te: "trailers",
92
+ authorization: `Bearer ${options.apiKey}`,
93
+ "x-ghost-mode": "true",
94
+ "x-cursor-client-version": options.clientVersion ?? CURSOR_DEFAULT_CLIENT_VERSION,
95
+ "x-cursor-client-type": "cli",
96
+ };
97
+ }
98
+
99
+ /** HTTP/2 transport required by Cursor API (HTTP/1.1 is rejected with 464). */
100
+ async function fetchViaHttp2(
101
+ baseUrl: string,
102
+ body: Uint8Array,
103
+ options: CursorModelDiscoveryOptions,
104
+ timeoutMs: number,
105
+ ): Promise<Uint8Array | null> {
106
+ const { promise, resolve } = Promise.withResolvers<Uint8Array | null>();
107
+ const client = http2.connect(baseUrl);
108
+ const timer = setTimeout(() => {
109
+ client.destroy();
110
+ resolve(null);
111
+ }, timeoutMs);
112
+
113
+ client.on("error", () => {
114
+ clearTimeout(timer);
115
+ resolve(null);
116
+ });
117
+
118
+ const req = client.request({
119
+ ":method": "POST",
120
+ ":path": CURSOR_GET_USABLE_MODELS_PATH,
121
+ ...buildRequestHeaders(options),
122
+ });
123
+
124
+ const chunks: Buffer[] = [];
125
+ req.on("data", (chunk: Buffer) => chunks.push(chunk));
126
+ req.on("end", () => {
127
+ clearTimeout(timer);
128
+ client.close();
129
+ resolve(new Uint8Array(Buffer.concat(chunks)));
130
+ });
131
+ req.on("error", () => {
132
+ clearTimeout(timer);
133
+ client.close();
134
+ resolve(null);
135
+ });
136
+ req.on("response", headers => {
137
+ const status = Number(headers[":status"] ?? 0);
138
+ if (status < 200 || status >= 300) {
139
+ clearTimeout(timer);
140
+ client.close();
141
+ resolve(null);
142
+ }
143
+ });
144
+
145
+ if (body.length > 0) {
146
+ req.end(Buffer.from(body));
147
+ } else {
148
+ req.end();
149
+ }
150
+
151
+ return promise;
152
+ }
153
+
154
+ function normalizeCustomModelIds(customModelIds: readonly string[] | undefined): string[] {
155
+ if (!customModelIds) {
156
+ return [];
157
+ }
158
+ const normalized = new Set<string>();
159
+ for (const value of customModelIds) {
160
+ if (typeof value !== "string") {
161
+ continue;
162
+ }
163
+ const trimmed = value.trim();
164
+ if (!trimmed) {
165
+ continue;
166
+ }
167
+ normalized.add(trimmed);
168
+ }
169
+ return [...normalized];
170
+ }
171
+
172
+ function createCursorReferenceMap(): Map<string, Model<"cursor-agent">> {
173
+ const references = new Map<string, Model<"cursor-agent">>();
174
+ for (const model of getBundledModels("cursor") as Model<"cursor-agent">[]) {
175
+ references.set(model.id, model);
176
+ }
177
+ return references;
178
+ }
179
+
180
+ function decodeGetUsableModelsResponse(payload: Uint8Array) {
181
+ if (payload.length === 0) {
182
+ return null;
183
+ }
184
+
185
+ const framedBody = decodeConnectUnaryBody(payload);
186
+ if (framedBody) {
187
+ try {
188
+ return fromBinary(GetUsableModelsResponseSchema, framedBody);
189
+ } catch {
190
+ return null;
191
+ }
192
+ }
193
+
194
+ try {
195
+ return fromBinary(GetUsableModelsResponseSchema, payload);
196
+ } catch {
197
+ return null;
198
+ }
199
+ }
200
+
201
+ function decodeConnectUnaryBody(payload: Uint8Array): Uint8Array | null {
202
+ if (payload.length < 5) {
203
+ return null;
204
+ }
205
+
206
+ let offset = 0;
207
+ while (offset + 5 <= payload.length) {
208
+ const flags = payload[offset];
209
+ const view = new DataView(payload.buffer, payload.byteOffset + offset, payload.byteLength - offset);
210
+ const messageLength = view.getUint32(1, false);
211
+ const frameEnd = offset + 5 + messageLength;
212
+ if (frameEnd > payload.length) {
213
+ return null;
214
+ }
215
+ const compressionFlagSet = (flags & 0b0000_0001) !== 0;
216
+ if (compressionFlagSet) {
217
+ return null;
218
+ }
219
+ const endStreamFlagSet = (flags & 0b0000_0010) !== 0;
220
+ if (!endStreamFlagSet) {
221
+ return payload.subarray(offset + 5, frameEnd);
222
+ }
223
+
224
+ offset = frameEnd;
225
+ }
226
+
227
+ return null;
228
+ }
229
+
230
+ function normalizeCursorModels(
231
+ models: readonly unknown[] | undefined,
232
+ baseUrlOverride: string | undefined,
233
+ references: Map<string, Model<"cursor-agent">>,
234
+ ): Model<"cursor-agent">[] {
235
+ if (!models || models.length === 0) {
236
+ return [];
237
+ }
238
+
239
+ const byId = new Map<string, Model<"cursor-agent">>();
240
+ for (const model of models) {
241
+ const normalized = normalizeCursorModel(model, baseUrlOverride, references);
242
+ if (!normalized) {
243
+ continue;
244
+ }
245
+ byId.set(normalized.id, normalized);
246
+ }
247
+
248
+ return [...byId.values()].sort((a, b) => a.id.localeCompare(b.id));
249
+ }
250
+
251
+ function normalizeCursorModel(
252
+ model: unknown,
253
+ baseUrlOverride: string | undefined,
254
+ references: Map<string, Model<"cursor-agent">>,
255
+ ): Model<"cursor-agent"> | null {
256
+ const parsedModel = CursorModelDetailsSchema.safeParse(model);
257
+ if (!parsedModel.success) {
258
+ return null;
259
+ }
260
+
261
+ const details = parsedModel.data;
262
+ const id = details.modelId.trim();
263
+ if (!id) {
264
+ return null;
265
+ }
266
+
267
+ const name = pickModelDisplayName(details, id);
268
+ const reference = references.get(id);
269
+ const reasoning = Boolean(details.thinkingDetails) || reference?.reasoning === true;
270
+
271
+ if (reference) {
272
+ return {
273
+ ...reference,
274
+ id,
275
+ name,
276
+ baseUrl: baseUrlOverride ?? reference.baseUrl,
277
+ reasoning,
278
+ };
279
+ }
280
+ return {
281
+ id,
282
+ name,
283
+ api: "cursor-agent",
284
+ provider: "cursor",
285
+ baseUrl: baseUrlOverride ?? CURSOR_DEFAULT_BASE_URL,
286
+ reasoning,
287
+ input: ["text"],
288
+ cost: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0 },
289
+ contextWindow: DEFAULT_CONTEXT_WINDOW,
290
+ maxTokens: DEFAULT_MAX_TOKENS,
291
+ };
292
+ }
293
+
294
+ function pickModelDisplayName(model: CursorModelDetailsValue, fallbackId: string): string {
295
+ const candidates = [model.displayName, model.displayNameShort, model.displayModelId, ...model.aliases, fallbackId];
296
+ for (const candidate of candidates) {
297
+ if (typeof candidate !== "string") {
298
+ continue;
299
+ }
300
+ const trimmed = candidate.trim();
301
+ if (trimmed) {
302
+ return trimmed;
303
+ }
304
+ }
305
+ return fallbackId;
306
+ }
@@ -0,0 +1,248 @@
1
+ import { UNK_CONTEXT_WINDOW, UNK_MAX_TOKENS } from "@gajae-code/ai";
2
+ import * as z from "zod/v4";
3
+ import { getBundledModels } from "../../models";
4
+ import type { Model } from "../../types";
5
+
6
+ const GOOGLE_GENERATIVE_AI_BASE_URL = "https://generativelanguage.googleapis.com/v1beta";
7
+ const DEFAULT_PAGE_SIZE = 100;
8
+ const DEFAULT_MAX_PAGES = 25;
9
+
10
+ const geminiModelListItemSchema = z.object({
11
+ name: z.string().optional().catch(undefined),
12
+ displayName: z.string().optional().catch(undefined),
13
+ supportedGenerationMethods: z.array(z.string()).optional(),
14
+ inputTokenLimit: z.number().finite().optional().catch(undefined),
15
+ outputTokenLimit: z.number().finite().optional().catch(undefined),
16
+ });
17
+
18
+ const geminiModelListResponseSchema = z.object({
19
+ models: z
20
+ .array(z.unknown())
21
+ .optional()
22
+ .transform(items => {
23
+ if (!items) {
24
+ return [];
25
+ }
26
+ const parsedItems: GeminiModelListItem[] = [];
27
+ for (const item of items) {
28
+ const parsed = geminiModelListItemSchema.safeParse(item);
29
+ if (parsed.success) {
30
+ parsedItems.push(parsed.data);
31
+ }
32
+ }
33
+ return parsedItems;
34
+ }),
35
+ nextPageToken: z.string().optional().catch(undefined),
36
+ });
37
+
38
+ type GeminiModelListItem = z.infer<typeof geminiModelListItemSchema>;
39
+
40
+ /**
41
+ * Configuration for Google Generative AI model discovery.
42
+ */
43
+ export interface GeminiDiscoveryOptions {
44
+ /** API key for the Google Generative AI public endpoint. */
45
+ apiKey: string;
46
+ /** Optional endpoint override for testing or proxying. */
47
+ baseUrl?: string;
48
+ /** Optional requested page size for model listing. */
49
+ pageSize?: number;
50
+ /** Maximum number of pages to request before stopping pagination. */
51
+ maxPages?: number;
52
+ /** Optional abort signal for HTTP requests. */
53
+ signal?: AbortSignal;
54
+ /** Optional fetch implementation override for tests. */
55
+ fetch?: typeof fetch;
56
+ }
57
+
58
+ /**
59
+ * Fetches and normalizes Google Generative AI models from the public models endpoint.
60
+ *
61
+ * Returns `null` on transport/protocol failures.
62
+ * Returns `[]` only when the endpoint responds successfully with no usable models.
63
+ */
64
+ export async function fetchGeminiModels(
65
+ options: GeminiDiscoveryOptions,
66
+ ): Promise<Model<"google-generative-ai">[] | null> {
67
+ if (!options.apiKey.trim()) {
68
+ return null;
69
+ }
70
+
71
+ const fetchImpl = options.fetch ?? fetch;
72
+ const baseUrl = normalizeBaseUrl(options.baseUrl);
73
+ const pageSize = normalizePositiveInt(options.pageSize, DEFAULT_PAGE_SIZE);
74
+ const maxPages = normalizePositiveInt(options.maxPages, DEFAULT_MAX_PAGES);
75
+
76
+ const bundledById = new Map(
77
+ getBundledModels("google").map(model => [model.id, model as Model<"google-generative-ai">]),
78
+ );
79
+ const modelsById = new Map<string, Model<"google-generative-ai">>();
80
+ const seenTokens = new Set<string>();
81
+ let nextPageToken: string | undefined;
82
+
83
+ for (let page = 0; page < maxPages; page += 1) {
84
+ const requestUrl = buildModelsUrl(baseUrl, options.apiKey, pageSize, nextPageToken);
85
+ let response: Response;
86
+ try {
87
+ response = await fetchImpl(requestUrl, {
88
+ method: "GET",
89
+ signal: options.signal,
90
+ });
91
+ } catch {
92
+ return null;
93
+ }
94
+
95
+ if (!response.ok) {
96
+ return null;
97
+ }
98
+
99
+ let payload: unknown;
100
+ try {
101
+ payload = await response.json();
102
+ } catch {
103
+ return null;
104
+ }
105
+
106
+ const parsed = geminiModelListResponseSchema.safeParse(payload);
107
+ if (!parsed.success) {
108
+ return null;
109
+ }
110
+
111
+ for (const item of parsed.data.models) {
112
+ const model = normalizeModel(item, baseUrl, bundledById);
113
+ if (model) {
114
+ modelsById.set(model.id, model);
115
+ }
116
+ }
117
+
118
+ const token = normalizePageToken(parsed.data.nextPageToken);
119
+ if (!token) {
120
+ break;
121
+ }
122
+ if (seenTokens.has(token)) {
123
+ break;
124
+ }
125
+ seenTokens.add(token);
126
+ nextPageToken = token;
127
+ }
128
+
129
+ return Array.from(modelsById.values()).sort((left, right) => left.id.localeCompare(right.id));
130
+ }
131
+
132
+ function buildModelsUrl(baseUrl: string, apiKey: string, pageSize: number, pageToken?: string): URL {
133
+ const url = new URL(`${baseUrl}/models`);
134
+ url.searchParams.set("key", apiKey);
135
+ url.searchParams.set("pageSize", String(pageSize));
136
+ if (pageToken) {
137
+ url.searchParams.set("pageToken", pageToken);
138
+ }
139
+ return url;
140
+ }
141
+
142
+ function normalizeBaseUrl(baseUrl?: string): string {
143
+ const value = (baseUrl ?? GOOGLE_GENERATIVE_AI_BASE_URL).trim();
144
+ if (!value) {
145
+ return GOOGLE_GENERATIVE_AI_BASE_URL;
146
+ }
147
+ return value.replace(/\/+$/, "");
148
+ }
149
+
150
+ function normalizePositiveInt(value: number | undefined, fallback: number): number {
151
+ if (typeof value !== "number" || !Number.isFinite(value) || value <= 0) {
152
+ return fallback;
153
+ }
154
+ const normalized = Math.floor(value);
155
+ return normalized > 0 ? normalized : fallback;
156
+ }
157
+
158
+ function normalizePageToken(value: unknown): string | undefined {
159
+ if (typeof value !== "string") {
160
+ return undefined;
161
+ }
162
+ const token = value.trim();
163
+ return token.length > 0 ? token : undefined;
164
+ }
165
+
166
+ function normalizeModel(
167
+ item: GeminiModelListItem,
168
+ baseUrl: string,
169
+ bundledById: Map<string, Model<"google-generative-ai">>,
170
+ ): Model<"google-generative-ai"> | null {
171
+ const id = normalizeModelId(item.name);
172
+ if (!id) {
173
+ return null;
174
+ }
175
+ if (!supportsTextGeneration(item.supportedGenerationMethods)) {
176
+ return null;
177
+ }
178
+
179
+ const reference = bundledById.get(id);
180
+ const contextWindow = normalizePositiveInt(item.inputTokenLimit, reference?.contextWindow ?? UNK_CONTEXT_WINDOW);
181
+ const maxTokens = normalizePositiveInt(item.outputTokenLimit, reference?.maxTokens ?? UNK_MAX_TOKENS);
182
+ const name = normalizeModelName(item.displayName, reference?.name ?? id);
183
+
184
+ if (reference) {
185
+ return {
186
+ ...reference,
187
+ id,
188
+ name,
189
+ baseUrl,
190
+ contextWindow,
191
+ maxTokens,
192
+ };
193
+ }
194
+ return {
195
+ id,
196
+ name,
197
+ api: "google-generative-ai",
198
+ provider: "google",
199
+ baseUrl,
200
+ reasoning: inferReasoningFromGeminiId(id),
201
+ input: inferInputFromGeminiId(id),
202
+ cost: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0 },
203
+ contextWindow,
204
+ maxTokens,
205
+ };
206
+ }
207
+
208
+ function normalizeModelId(value: string | undefined): string | null {
209
+ if (!value) {
210
+ return null;
211
+ }
212
+ const trimmed = value.trim();
213
+ if (!trimmed) {
214
+ return null;
215
+ }
216
+ return trimmed.startsWith("models/") ? trimmed.slice("models/".length) : trimmed;
217
+ }
218
+
219
+ function normalizeModelName(displayName: string | undefined, id: string): string {
220
+ const trimmed = displayName?.trim();
221
+ return trimmed ? trimmed : id;
222
+ }
223
+
224
+ function supportsTextGeneration(methods: string[] | undefined): boolean {
225
+ if (!methods || methods.length === 0) {
226
+ return false;
227
+ }
228
+ return methods.some(method => method === "generateContent");
229
+ }
230
+
231
+ function inferReasoningFromGeminiId(id: string): boolean {
232
+ const normalized = id.toLowerCase();
233
+ if (normalized.includes("thinking")) {
234
+ return true;
235
+ }
236
+ if (normalized.includes("pro") || normalized.includes("2.5")) {
237
+ return true;
238
+ }
239
+ return false;
240
+ }
241
+
242
+ function inferInputFromGeminiId(id: string): ("text" | "image")[] {
243
+ const normalized = id.toLowerCase();
244
+ if (normalized.includes("vision") || normalized.includes("image") || normalized.includes("gemini")) {
245
+ return ["text", "image"];
246
+ }
247
+ return ["text"];
248
+ }
@@ -0,0 +1,4 @@
1
+ export * from "./antigravity";
2
+ export * from "./codex";
3
+ export * from "./gemini";
4
+ export * from "./openai-compatible";