@prometheus-ai/ai 0.5.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (369) hide show
  1. package/CHANGELOG.md +7 -0
  2. package/README.md +1184 -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 +6 -0
  6. package/dist/types/auth-broker/refresher.d.ts +25 -0
  7. package/dist/types/auth-broker/remote-store.d.ts +101 -0
  8. package/dist/types/auth-broker/server.d.ts +32 -0
  9. package/dist/types/auth-broker/snapshot-cache.d.ts +17 -0
  10. package/dist/types/auth-broker/types.d.ts +107 -0
  11. package/dist/types/auth-broker/wire-schemas.d.ts +412 -0
  12. package/dist/types/auth-gateway/http.d.ts +39 -0
  13. package/dist/types/auth-gateway/index.d.ts +3 -0
  14. package/dist/types/auth-gateway/server.d.ts +36 -0
  15. package/dist/types/auth-gateway/types.d.ts +117 -0
  16. package/dist/types/auth-storage.d.ts +762 -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 +64 -0
  20. package/dist/types/model-thinking.d.ts +100 -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 +50 -0
  25. package/dist/types/provider-models/google.d.ts +24 -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 +323 -0
  29. package/dist/types/provider-models/special.d.ts +16 -0
  30. package/dist/types/providers/amazon-bedrock.d.ts +38 -0
  31. package/dist/types/providers/anthropic-client.d.ts +99 -0
  32. package/dist/types/providers/anthropic-messages-server-schema.d.ts +465 -0
  33. package/dist/types/providers/anthropic-messages-server.d.ts +17 -0
  34. package/dist/types/providers/anthropic-wire.d.ts +262 -0
  35. package/dist/types/providers/anthropic.d.ts +206 -0
  36. package/dist/types/providers/aws-credentials.d.ts +43 -0
  37. package/dist/types/providers/aws-eventstream.d.ts +38 -0
  38. package/dist/types/providers/aws-sigv4.d.ts +55 -0
  39. package/dist/types/providers/azure-openai-responses.d.ts +15 -0
  40. package/dist/types/providers/cursor/gen/agent_pb.d.ts +13022 -0
  41. package/dist/types/providers/cursor.d.ts +43 -0
  42. package/dist/types/providers/error-message.d.ts +27 -0
  43. package/dist/types/providers/github-copilot-headers.d.ts +40 -0
  44. package/dist/types/providers/gitlab-duo.d.ts +27 -0
  45. package/dist/types/providers/google-auth.d.ts +24 -0
  46. package/dist/types/providers/google-gemini-cli.d.ts +81 -0
  47. package/dist/types/providers/google-gemini-headers.d.ts +18 -0
  48. package/dist/types/providers/google-shared.d.ts +171 -0
  49. package/dist/types/providers/google-types.d.ts +138 -0
  50. package/dist/types/providers/google-vertex.d.ts +7 -0
  51. package/dist/types/providers/google.d.ts +4 -0
  52. package/dist/types/providers/grammar.d.ts +1 -0
  53. package/dist/types/providers/kimi.d.ts +27 -0
  54. package/dist/types/providers/mock.d.ts +173 -0
  55. package/dist/types/providers/ollama.d.ts +6 -0
  56. package/dist/types/providers/openai-anthropic-shim.d.ts +31 -0
  57. package/dist/types/providers/openai-chat-server-schema.d.ts +817 -0
  58. package/dist/types/providers/openai-chat-server.d.ts +16 -0
  59. package/dist/types/providers/openai-codex/constants.d.ts +26 -0
  60. package/dist/types/providers/openai-codex/request-transformer.d.ts +49 -0
  61. package/dist/types/providers/openai-codex/response-handler.d.ts +17 -0
  62. package/dist/types/providers/openai-codex-responses.d.ts +67 -0
  63. package/dist/types/providers/openai-completions-compat.d.ts +27 -0
  64. package/dist/types/providers/openai-completions.d.ts +54 -0
  65. package/dist/types/providers/openai-responses-server-schema.d.ts +392 -0
  66. package/dist/types/providers/openai-responses-server.d.ts +17 -0
  67. package/dist/types/providers/openai-responses-shared.d.ts +105 -0
  68. package/dist/types/providers/openai-responses.d.ts +66 -0
  69. package/dist/types/providers/prometheus-native-client.d.ts +13 -0
  70. package/dist/types/providers/prometheus-native-server.d.ts +68 -0
  71. package/dist/types/providers/register-builtins.d.ts +31 -0
  72. package/dist/types/providers/synthetic.d.ts +26 -0
  73. package/dist/types/providers/transform-messages.d.ts +12 -0
  74. package/dist/types/providers/vision-guard.d.ts +20 -0
  75. package/dist/types/providers/xai-responses.d.ts +23 -0
  76. package/dist/types/rate-limit-utils.d.ts +19 -0
  77. package/dist/types/stream.d.ts +28 -0
  78. package/dist/types/types.d.ts +819 -0
  79. package/dist/types/usage/claude.d.ts +4 -0
  80. package/dist/types/usage/gemini.d.ts +2 -0
  81. package/dist/types/usage/github-copilot.d.ts +7 -0
  82. package/dist/types/usage/google-antigravity.d.ts +2 -0
  83. package/dist/types/usage/kimi.d.ts +2 -0
  84. package/dist/types/usage/minimax-code.d.ts +2 -0
  85. package/dist/types/usage/openai-codex.d.ts +3 -0
  86. package/dist/types/usage/shared.d.ts +1 -0
  87. package/dist/types/usage/zai.d.ts +2 -0
  88. package/dist/types/usage.d.ts +260 -0
  89. package/dist/types/utils/abort.d.ts +19 -0
  90. package/dist/types/utils/abortable-iterator.d.ts +4 -0
  91. package/dist/types/utils/anthropic-auth.d.ts +35 -0
  92. package/dist/types/utils/discovery/antigravity.d.ts +61 -0
  93. package/dist/types/utils/discovery/codex.d.ts +38 -0
  94. package/dist/types/utils/discovery/cursor.d.ts +23 -0
  95. package/dist/types/utils/discovery/gemini.d.ts +25 -0
  96. package/dist/types/utils/discovery/index.d.ts +4 -0
  97. package/dist/types/utils/discovery/openai-compatible.d.ts +72 -0
  98. package/dist/types/utils/event-stream.d.ts +28 -0
  99. package/dist/types/utils/fireworks-model-id.d.ts +10 -0
  100. package/dist/types/utils/foundry.d.ts +1 -0
  101. package/dist/types/utils/http-inspector.d.ts +31 -0
  102. package/dist/types/utils/idle-iterator.d.ts +78 -0
  103. package/dist/types/utils/json-parse.d.ts +37 -0
  104. package/dist/types/utils/oauth/__tests__/xai-oauth.test.d.ts +1 -0
  105. package/dist/types/utils/oauth/alibaba-coding-plan.d.ts +18 -0
  106. package/dist/types/utils/oauth/anthropic.d.ts +22 -0
  107. package/dist/types/utils/oauth/api-key-login.d.ts +35 -0
  108. package/dist/types/utils/oauth/api-key-validation.d.ts +27 -0
  109. package/dist/types/utils/oauth/callback-server.d.ts +57 -0
  110. package/dist/types/utils/oauth/cerebras.d.ts +1 -0
  111. package/dist/types/utils/oauth/cloudflare-ai-gateway.d.ts +18 -0
  112. package/dist/types/utils/oauth/cursor.d.ts +15 -0
  113. package/dist/types/utils/oauth/deepseek.d.ts +10 -0
  114. package/dist/types/utils/oauth/firepass.d.ts +1 -0
  115. package/dist/types/utils/oauth/fireworks.d.ts +1 -0
  116. package/dist/types/utils/oauth/github-copilot.d.ts +38 -0
  117. package/dist/types/utils/oauth/gitlab-duo.d.ts +3 -0
  118. package/dist/types/utils/oauth/google-antigravity.d.ts +11 -0
  119. package/dist/types/utils/oauth/google-gemini-cli.d.ts +10 -0
  120. package/dist/types/utils/oauth/google-oauth-shared.d.ts +28 -0
  121. package/dist/types/utils/oauth/huggingface.d.ts +19 -0
  122. package/dist/types/utils/oauth/index.d.ts +38 -0
  123. package/dist/types/utils/oauth/kagi.d.ts +17 -0
  124. package/dist/types/utils/oauth/kilo.d.ts +5 -0
  125. package/dist/types/utils/oauth/kimi.d.ts +21 -0
  126. package/dist/types/utils/oauth/litellm.d.ts +18 -0
  127. package/dist/types/utils/oauth/lm-studio.d.ts +17 -0
  128. package/dist/types/utils/oauth/minimax-code.d.ts +28 -0
  129. package/dist/types/utils/oauth/moonshot.d.ts +1 -0
  130. package/dist/types/utils/oauth/nanogpt.d.ts +1 -0
  131. package/dist/types/utils/oauth/nvidia.d.ts +18 -0
  132. package/dist/types/utils/oauth/ollama-cloud.d.ts +2 -0
  133. package/dist/types/utils/oauth/ollama.d.ts +18 -0
  134. package/dist/types/utils/oauth/openai-codex.d.ts +21 -0
  135. package/dist/types/utils/oauth/opencode.d.ts +18 -0
  136. package/dist/types/utils/oauth/openrouter.d.ts +1 -0
  137. package/dist/types/utils/oauth/parallel.d.ts +17 -0
  138. package/dist/types/utils/oauth/perplexity.d.ts +9 -0
  139. package/dist/types/utils/oauth/pkce.d.ts +8 -0
  140. package/dist/types/utils/oauth/qianfan.d.ts +17 -0
  141. package/dist/types/utils/oauth/qwen-portal.d.ts +19 -0
  142. package/dist/types/utils/oauth/synthetic.d.ts +1 -0
  143. package/dist/types/utils/oauth/tavily.d.ts +17 -0
  144. package/dist/types/utils/oauth/together.d.ts +1 -0
  145. package/dist/types/utils/oauth/types.d.ts +44 -0
  146. package/dist/types/utils/oauth/venice.d.ts +18 -0
  147. package/dist/types/utils/oauth/vercel-ai-gateway.d.ts +18 -0
  148. package/dist/types/utils/oauth/vllm.d.ts +16 -0
  149. package/dist/types/utils/oauth/wafer.d.ts +2 -0
  150. package/dist/types/utils/oauth/xai-oauth.d.ts +60 -0
  151. package/dist/types/utils/oauth/xiaomi.d.ts +25 -0
  152. package/dist/types/utils/oauth/zai.d.ts +18 -0
  153. package/dist/types/utils/oauth/zenmux.d.ts +1 -0
  154. package/dist/types/utils/oauth/zhipu.d.ts +18 -0
  155. package/dist/types/utils/overflow.d.ts +54 -0
  156. package/dist/types/utils/parse-bind.d.ts +23 -0
  157. package/dist/types/utils/provider-response.d.ts +3 -0
  158. package/dist/types/utils/request-debug.d.ts +29 -0
  159. package/dist/types/utils/retry-after.d.ts +3 -0
  160. package/dist/types/utils/retry.d.ts +26 -0
  161. package/dist/types/utils/schema/adapt.d.ts +24 -0
  162. package/dist/types/utils/schema/compatibility.d.ts +30 -0
  163. package/dist/types/utils/schema/dereference.d.ts +11 -0
  164. package/dist/types/utils/schema/draft.d.ts +10 -0
  165. package/dist/types/utils/schema/equality.d.ts +4 -0
  166. package/dist/types/utils/schema/fields.d.ts +49 -0
  167. package/dist/types/utils/schema/index.d.ts +13 -0
  168. package/dist/types/utils/schema/json-schema-validator.d.ts +12 -0
  169. package/dist/types/utils/schema/meta-validator.d.ts +2 -0
  170. package/dist/types/utils/schema/normalize.d.ts +93 -0
  171. package/dist/types/utils/schema/spill.d.ts +8 -0
  172. package/dist/types/utils/schema/stamps.d.ts +25 -0
  173. package/dist/types/utils/schema/types.d.ts +4 -0
  174. package/dist/types/utils/schema/wire.d.ts +53 -0
  175. package/dist/types/utils/schema/zod-decontaminate.d.ts +31 -0
  176. package/dist/types/utils/sdk-stream-timeout.d.ts +33 -0
  177. package/dist/types/utils/sse-debug.d.ts +10 -0
  178. package/dist/types/utils/stream-markup-healing.d.ts +80 -0
  179. package/dist/types/utils/tool-choice.d.ts +50 -0
  180. package/dist/types/utils/validation.d.ts +17 -0
  181. package/dist/types/utils.d.ts +28 -0
  182. package/package.json +142 -0
  183. package/src/api-registry.ts +96 -0
  184. package/src/auth-broker/client.ts +358 -0
  185. package/src/auth-broker/index.ts +6 -0
  186. package/src/auth-broker/refresher.ts +117 -0
  187. package/src/auth-broker/remote-store.ts +637 -0
  188. package/src/auth-broker/server.ts +644 -0
  189. package/src/auth-broker/snapshot-cache.ts +174 -0
  190. package/src/auth-broker/types.ts +130 -0
  191. package/src/auth-broker/wire-schemas.ts +200 -0
  192. package/src/auth-gateway/http.ts +194 -0
  193. package/src/auth-gateway/index.ts +3 -0
  194. package/src/auth-gateway/server.ts +822 -0
  195. package/src/auth-gateway/types.ts +143 -0
  196. package/src/auth-storage.ts +4608 -0
  197. package/src/index.ts +54 -0
  198. package/src/model-cache.ts +129 -0
  199. package/src/model-manager.ts +469 -0
  200. package/src/model-thinking.ts +756 -0
  201. package/src/models.json +60287 -0
  202. package/src/models.json.d.ts +9 -0
  203. package/src/models.ts +56 -0
  204. package/src/prompts/turn-aborted-guidance.md +4 -0
  205. package/src/provider-details.ts +90 -0
  206. package/src/provider-models/bundled-references.ts +38 -0
  207. package/src/provider-models/descriptors.ts +364 -0
  208. package/src/provider-models/google.ts +88 -0
  209. package/src/provider-models/index.ts +5 -0
  210. package/src/provider-models/ollama.ts +153 -0
  211. package/src/provider-models/openai-compat.ts +2904 -0
  212. package/src/provider-models/special.ts +67 -0
  213. package/src/providers/amazon-bedrock.ts +873 -0
  214. package/src/providers/anthropic-client.ts +318 -0
  215. package/src/providers/anthropic-messages-server-schema.ts +243 -0
  216. package/src/providers/anthropic-messages-server.ts +681 -0
  217. package/src/providers/anthropic-wire.ts +268 -0
  218. package/src/providers/anthropic.ts +3106 -0
  219. package/src/providers/aws-credentials.ts +501 -0
  220. package/src/providers/aws-eventstream.ts +185 -0
  221. package/src/providers/aws-sigv4.ts +218 -0
  222. package/src/providers/azure-openai-responses.ts +361 -0
  223. package/src/providers/cursor/gen/agent_pb.ts +15274 -0
  224. package/src/providers/cursor/proto/agent.proto +3526 -0
  225. package/src/providers/cursor/proto/buf.gen.yaml +6 -0
  226. package/src/providers/cursor/proto/buf.yaml +17 -0
  227. package/src/providers/cursor.ts +2621 -0
  228. package/src/providers/error-message.ts +21 -0
  229. package/src/providers/github-copilot-headers.ts +140 -0
  230. package/src/providers/gitlab-duo.ts +372 -0
  231. package/src/providers/google-auth.ts +252 -0
  232. package/src/providers/google-gemini-cli.ts +809 -0
  233. package/src/providers/google-gemini-headers.ts +41 -0
  234. package/src/providers/google-shared.ts +917 -0
  235. package/src/providers/google-types.ts +167 -0
  236. package/src/providers/google-vertex.ts +91 -0
  237. package/src/providers/google.ts +41 -0
  238. package/src/providers/grammar.ts +70 -0
  239. package/src/providers/kimi.ts +52 -0
  240. package/src/providers/mock.ts +496 -0
  241. package/src/providers/ollama.ts +644 -0
  242. package/src/providers/openai-anthropic-shim.ts +138 -0
  243. package/src/providers/openai-chat-server-schema.ts +252 -0
  244. package/src/providers/openai-chat-server.ts +647 -0
  245. package/src/providers/openai-codex/constants.ts +43 -0
  246. package/src/providers/openai-codex/request-transformer.ts +161 -0
  247. package/src/providers/openai-codex/response-handler.ts +81 -0
  248. package/src/providers/openai-codex-responses.ts +3027 -0
  249. package/src/providers/openai-completions-compat.ts +320 -0
  250. package/src/providers/openai-completions.ts +2002 -0
  251. package/src/providers/openai-responses-server-schema.ts +290 -0
  252. package/src/providers/openai-responses-server.ts +1183 -0
  253. package/src/providers/openai-responses-shared.ts +956 -0
  254. package/src/providers/openai-responses.ts +679 -0
  255. package/src/providers/prometheus-native-client.ts +228 -0
  256. package/src/providers/prometheus-native-server.ts +212 -0
  257. package/src/providers/register-builtins.ts +457 -0
  258. package/src/providers/synthetic.ts +50 -0
  259. package/src/providers/transform-messages.ts +382 -0
  260. package/src/providers/vision-guard.ts +52 -0
  261. package/src/providers/xai-responses.ts +82 -0
  262. package/src/rate-limit-utils.ts +91 -0
  263. package/src/stream.ts +1068 -0
  264. package/src/types.ts +965 -0
  265. package/src/usage/claude.ts +482 -0
  266. package/src/usage/gemini.ts +250 -0
  267. package/src/usage/github-copilot.ts +421 -0
  268. package/src/usage/google-antigravity.ts +201 -0
  269. package/src/usage/kimi.ts +271 -0
  270. package/src/usage/minimax-code.ts +31 -0
  271. package/src/usage/openai-codex.ts +503 -0
  272. package/src/usage/shared.ts +10 -0
  273. package/src/usage/zai.ts +247 -0
  274. package/src/usage.ts +185 -0
  275. package/src/utils/abort.ts +51 -0
  276. package/src/utils/abortable-iterator.ts +69 -0
  277. package/src/utils/anthropic-auth.ts +93 -0
  278. package/src/utils/discovery/antigravity.ts +261 -0
  279. package/src/utils/discovery/codex.ts +371 -0
  280. package/src/utils/discovery/cursor.ts +306 -0
  281. package/src/utils/discovery/gemini.ts +248 -0
  282. package/src/utils/discovery/index.ts +4 -0
  283. package/src/utils/discovery/openai-compatible.ts +224 -0
  284. package/src/utils/event-stream.ts +142 -0
  285. package/src/utils/fireworks-model-id.ts +30 -0
  286. package/src/utils/foundry.ts +8 -0
  287. package/src/utils/http-inspector.ts +176 -0
  288. package/src/utils/idle-iterator.ts +273 -0
  289. package/src/utils/json-parse.ts +182 -0
  290. package/src/utils/oauth/__tests__/xai-oauth.test.ts +107 -0
  291. package/src/utils/oauth/alibaba-coding-plan.ts +59 -0
  292. package/src/utils/oauth/anthropic.ts +273 -0
  293. package/src/utils/oauth/api-key-login.ts +87 -0
  294. package/src/utils/oauth/api-key-validation.ts +92 -0
  295. package/src/utils/oauth/callback-server.ts +276 -0
  296. package/src/utils/oauth/cerebras.ts +16 -0
  297. package/src/utils/oauth/cloudflare-ai-gateway.ts +48 -0
  298. package/src/utils/oauth/cursor.ts +157 -0
  299. package/src/utils/oauth/deepseek.ts +53 -0
  300. package/src/utils/oauth/firepass.ts +24 -0
  301. package/src/utils/oauth/fireworks.ts +15 -0
  302. package/src/utils/oauth/github-copilot.ts +362 -0
  303. package/src/utils/oauth/gitlab-duo.ts +123 -0
  304. package/src/utils/oauth/google-antigravity.ts +200 -0
  305. package/src/utils/oauth/google-gemini-cli.ts +256 -0
  306. package/src/utils/oauth/google-oauth-shared.ts +110 -0
  307. package/src/utils/oauth/huggingface.ts +62 -0
  308. package/src/utils/oauth/index.ts +502 -0
  309. package/src/utils/oauth/kagi.ts +47 -0
  310. package/src/utils/oauth/kilo.ts +87 -0
  311. package/src/utils/oauth/kimi.ts +254 -0
  312. package/src/utils/oauth/litellm.ts +47 -0
  313. package/src/utils/oauth/lm-studio.ts +38 -0
  314. package/src/utils/oauth/minimax-code.ts +80 -0
  315. package/src/utils/oauth/moonshot.ts +23 -0
  316. package/src/utils/oauth/nanogpt.ts +15 -0
  317. package/src/utils/oauth/nvidia.ts +70 -0
  318. package/src/utils/oauth/oauth.html +199 -0
  319. package/src/utils/oauth/ollama-cloud.ts +28 -0
  320. package/src/utils/oauth/ollama.ts +47 -0
  321. package/src/utils/oauth/openai-codex.ts +299 -0
  322. package/src/utils/oauth/opencode.ts +49 -0
  323. package/src/utils/oauth/openrouter.ts +20 -0
  324. package/src/utils/oauth/parallel.ts +46 -0
  325. package/src/utils/oauth/perplexity.ts +206 -0
  326. package/src/utils/oauth/pkce.ts +18 -0
  327. package/src/utils/oauth/qianfan.ts +58 -0
  328. package/src/utils/oauth/qwen-portal.ts +60 -0
  329. package/src/utils/oauth/synthetic.ts +15 -0
  330. package/src/utils/oauth/tavily.ts +46 -0
  331. package/src/utils/oauth/together.ts +16 -0
  332. package/src/utils/oauth/types.ts +102 -0
  333. package/src/utils/oauth/venice.ts +59 -0
  334. package/src/utils/oauth/vercel-ai-gateway.ts +47 -0
  335. package/src/utils/oauth/vllm.ts +40 -0
  336. package/src/utils/oauth/wafer.ts +50 -0
  337. package/src/utils/oauth/xai-oauth.ts +342 -0
  338. package/src/utils/oauth/xiaomi.ts +194 -0
  339. package/src/utils/oauth/zai.ts +60 -0
  340. package/src/utils/oauth/zenmux.ts +15 -0
  341. package/src/utils/oauth/zhipu.ts +60 -0
  342. package/src/utils/overflow.ts +137 -0
  343. package/src/utils/parse-bind.ts +54 -0
  344. package/src/utils/provider-response.ts +30 -0
  345. package/src/utils/request-debug.ts +336 -0
  346. package/src/utils/retry-after.ts +110 -0
  347. package/src/utils/retry.ts +54 -0
  348. package/src/utils/schema/CONSTRAINTS.md +164 -0
  349. package/src/utils/schema/adapt.ts +36 -0
  350. package/src/utils/schema/compatibility.ts +435 -0
  351. package/src/utils/schema/dereference.ts +98 -0
  352. package/src/utils/schema/draft.ts +341 -0
  353. package/src/utils/schema/equality.ts +97 -0
  354. package/src/utils/schema/fields.ts +191 -0
  355. package/src/utils/schema/index.ts +13 -0
  356. package/src/utils/schema/json-schema-validator.ts +577 -0
  357. package/src/utils/schema/meta-validator.ts +167 -0
  358. package/src/utils/schema/normalize.ts +1588 -0
  359. package/src/utils/schema/spill.ts +43 -0
  360. package/src/utils/schema/stamps.ts +97 -0
  361. package/src/utils/schema/types.ts +10 -0
  362. package/src/utils/schema/wire.ts +293 -0
  363. package/src/utils/schema/zod-decontaminate.ts +331 -0
  364. package/src/utils/sdk-stream-timeout.ts +43 -0
  365. package/src/utils/sse-debug.ts +289 -0
  366. package/src/utils/stream-markup-healing.ts +612 -0
  367. package/src/utils/tool-choice.ts +99 -0
  368. package/src/utils/validation.ts +1024 -0
  369. package/src/utils.ts +166 -0
@@ -0,0 +1,762 @@
1
+ /**
2
+ * Credential storage for API keys and OAuth tokens.
3
+ * Handles loading, saving, refreshing credentials, and usage tracking.
4
+ *
5
+ * This module defines:
6
+ * - `AuthCredentialStore` interface: persistence abstraction (SQLite, remote vault, …)
7
+ * - `AuthStorage` class: credential management with round-robin, usage limits, OAuth refresh
8
+ * - `SqliteAuthCredentialStore`: concrete SQLite-backed implementation
9
+ */
10
+ import { Database } from "bun:sqlite";
11
+ import type { Provider } from "./types";
12
+ import type { CredentialRankingStrategy, UsageLogger, UsageProvider, UsageReport } from "./usage";
13
+ import type { OAuthController, OAuthCredentials, OAuthProviderId } from "./utils/oauth/types";
14
+ export type ApiKeyCredential = {
15
+ type: "api_key";
16
+ key: string;
17
+ };
18
+ export type OAuthCredential = {
19
+ type: "oauth";
20
+ } & OAuthCredentials;
21
+ export type AuthCredential = ApiKeyCredential | OAuthCredential;
22
+ export type AuthCredentialEntry = AuthCredential | AuthCredential[];
23
+ export type AuthStorageData = Record<string, AuthCredentialEntry>;
24
+ /**
25
+ * Serialized representation of AuthStorage for passing to subagent workers.
26
+ * Contains only the essential credential data, not runtime state.
27
+ */
28
+ export interface SerializedAuthStorage {
29
+ credentials: Record<string, Array<{
30
+ id: number;
31
+ type: "api_key" | "oauth";
32
+ data: Record<string, unknown>;
33
+ }>>;
34
+ runtimeOverrides?: Record<string, string>;
35
+ dbPath?: string;
36
+ }
37
+ /**
38
+ * Auth credential with database row ID for updates/deletes.
39
+ * Wraps AuthCredential with storage metadata.
40
+ */
41
+ export interface StoredAuthCredential {
42
+ id: number;
43
+ provider: string;
44
+ credential: AuthCredential;
45
+ disabledCause: string | null;
46
+ }
47
+ /**
48
+ * Per-credential health record returned by {@link AuthStorage.checkCredentials}.
49
+ *
50
+ * Use this to identify which credential in a multi-account pool is causing
51
+ * auth errors. `ok` is tri-state:
52
+ *
53
+ * - `true` — credential authenticated against the provider's auth-verifying
54
+ * probe (today: the usage endpoint). For OAuth this also exercises refresh
55
+ * when the access token was expired.
56
+ * - `false` — the probe rejected the credential (401/403/refresh failure/etc).
57
+ * `reason` carries the upstream error string.
58
+ * - `null` — no probe is configured for this provider (or the configured
59
+ * probe doesn't support this credential type). The credential's auth
60
+ * status is unverifiable from here.
61
+ */
62
+ export interface CredentialHealthResult {
63
+ /** Database row id (matches {@link StoredAuthCredential.id}). */
64
+ id: number;
65
+ provider: string;
66
+ type: AuthCredential["type"];
67
+ /** OAuth email if known on the stored credential or surfaced by the probe. */
68
+ email?: string;
69
+ /** OAuth account id / org id if known. */
70
+ accountId?: string;
71
+ /** `true` when the refresh token lives on a remote broker (sentinel was present). */
72
+ remoteRefresh?: true;
73
+ ok: boolean | null;
74
+ /** Failure / unverifiable reason; absent when `ok === true`. */
75
+ reason?: string;
76
+ /** Probe usage report (raw payload stripped) when `ok === true`. */
77
+ report?: Omit<UsageReport, "raw">;
78
+ /**
79
+ * Result of the optional end-to-end completion probe (see
80
+ * {@link CheckCredentialsOptions.completionProbe}). Absent when no probe was
81
+ * supplied. The completion probe exercises the provider's chat-completion
82
+ * endpoint with the credential's bearer bytes, which is a stricter signal
83
+ * than the usage endpoint (some providers happily 200 a `/usage` call while
84
+ * the chat endpoint 401s the same bearer).
85
+ */
86
+ completion?: CredentialCompletionResult;
87
+ }
88
+ /**
89
+ * Outcome of the end-to-end completion probe. `null` means the probe was
90
+ * skipped (no bearer bytes were available — e.g. OAuth refresh failed
91
+ * upstream of the probe).
92
+ */
93
+ export interface CredentialCompletionResult {
94
+ ok: boolean | null;
95
+ /** Failure / unverifiable reason; absent when `ok === true`. */
96
+ reason?: string;
97
+ /** Probe model id used (carried back from the caller for display). */
98
+ modelId?: string;
99
+ /** Round-trip latency in milliseconds. */
100
+ latencyMs?: number;
101
+ }
102
+ /**
103
+ * Credential payload handed to {@link CompletionProbe}. For API-key
104
+ * credentials only the bytes are exposed; for OAuth, every identity field
105
+ * carried by the refreshed credential is included so the probe can compose
106
+ * provider-specific apiKey shapes (e.g. GitHub Copilot / Google Gemini CLI
107
+ * expect a JSON blob with `token` + `projectId`, not the raw access token).
108
+ *
109
+ * `refreshToken` may be {@link REMOTE_REFRESH_SENTINEL} when the credential
110
+ * lives behind a broker; the chat endpoint never reads it, so the probe can
111
+ * forward it verbatim into the structured shape without harm.
112
+ */
113
+ export type CompletionProbeCredential = {
114
+ type: "api_key";
115
+ apiKey: string;
116
+ } | {
117
+ type: "oauth";
118
+ accessToken: string;
119
+ refreshToken?: string;
120
+ expiresAt?: number;
121
+ accountId?: string;
122
+ projectId?: string;
123
+ email?: string;
124
+ enterpriseUrl?: string;
125
+ };
126
+ /**
127
+ * Caller-supplied bearer probe. Receives the post-refresh credential for a
128
+ * single row and reports whether a real chat-completion round-trip succeeds.
129
+ * The check-credentials pipeline calls this AFTER any OAuth refresh so the
130
+ * bytes match what a live request would send.
131
+ */
132
+ export interface CompletionProbeInput {
133
+ provider: Provider;
134
+ credentialId: number;
135
+ credential: CompletionProbeCredential;
136
+ signal: AbortSignal;
137
+ }
138
+ export type CompletionProbe = (input: CompletionProbeInput) => Promise<CredentialCompletionResult>;
139
+ export interface CheckCredentialsOptions {
140
+ signal?: AbortSignal;
141
+ /** Per-credential probe timeout (ms). Defaults to the configured usage request timeout. */
142
+ timeoutMs?: number;
143
+ /** Provider → base URL override, same shape as {@link AuthStorage.fetchUsageReports}. */
144
+ baseUrlResolver?: (provider: Provider) => string | undefined;
145
+ /**
146
+ * Optional end-to-end probe. When provided, `checkCredentials` invokes it
147
+ * for every credential where a usable bearer is available (API key, or
148
+ * OAuth access token after refresh-on-expiry succeeded). The result lands
149
+ * on {@link CredentialHealthResult.completion}.
150
+ *
151
+ * The probe runs INDEPENDENTLY of whether a {@link UsageProvider} is
152
+ * configured: providers without a usage endpoint still benefit from the
153
+ * extra signal. The probe is NOT invoked when OAuth refresh fails — the
154
+ * bytes would be stale anyway and the upstream failure is already captured
155
+ * on `reason`.
156
+ */
157
+ completionProbe?: CompletionProbe;
158
+ /** Per-credential completion probe timeout (ms). Defaults to `timeoutMs`. */
159
+ completionTimeoutMs?: number;
160
+ }
161
+ /**
162
+ * Sentinel value placed in OAuth `refresh` fields when a credential is shared
163
+ * via {@link AuthStorage.exportSnapshot}. Refresh tokens never leave the broker;
164
+ * clients must call back to refresh.
165
+ */
166
+ export declare const REMOTE_REFRESH_SENTINEL: "__remote__";
167
+ export type RemoteRefreshSentinel = typeof REMOTE_REFRESH_SENTINEL;
168
+ /** OAuth credential with refresh token replaced by the broker sentinel. */
169
+ export type RemoteOAuthCredential = Omit<OAuthCredential, "refresh"> & {
170
+ refresh: RemoteRefreshSentinel;
171
+ };
172
+ /** Discriminated credential payload as published by the broker. */
173
+ export type SnapshotCredential = ApiKeyCredential | RemoteOAuthCredential;
174
+ export interface AuthCredentialSnapshotEntry {
175
+ id: number;
176
+ provider: string;
177
+ credential: SnapshotCredential;
178
+ identityKey: string | null;
179
+ }
180
+ /**
181
+ * Wire-shaped snapshot exported by {@link AuthStorage.exportSnapshot} and
182
+ * served by the auth-broker server on `GET /v1/snapshot`.
183
+ */
184
+ export interface AuthCredentialSnapshot {
185
+ generation: number;
186
+ generatedAt: number;
187
+ credentials: AuthCredentialSnapshotEntry[];
188
+ }
189
+ /**
190
+ * Persistence abstraction consumed by {@link AuthStorage}.
191
+ *
192
+ * Concrete implementations:
193
+ * - {@link SqliteAuthCredentialStore} — local SQLite-backed store (default).
194
+ * - `RemoteAuthCredentialStore` from `./auth-broker` — client-side snapshot of
195
+ * a remote broker; mutating methods (`replace*`, `upsert*`, `delete*ForProvider`)
196
+ * throw because login flows route through the broker, not the client.
197
+ */
198
+ export interface AuthCredentialStore {
199
+ close(): void;
200
+ listAuthCredentials(provider?: string): StoredAuthCredential[];
201
+ updateAuthCredential(id: number, credential: AuthCredential): void;
202
+ deleteAuthCredential(id: number, disabledCause: string): void;
203
+ tryDisableAuthCredentialIfMatches(id: number, expectedData: string, disabledCause: string): boolean;
204
+ replaceAuthCredentialsForProvider(provider: string, credentials: AuthCredential[]): StoredAuthCredential[];
205
+ upsertAuthCredentialForProvider(provider: string, credential: AuthCredential): StoredAuthCredential[];
206
+ deleteAuthCredentialsForProvider(provider: string, disabledCause: string): void;
207
+ getCache(key: string, options?: {
208
+ includeExpired?: boolean;
209
+ }): string | null;
210
+ setCache(key: string, value: string, expiresAtSec: number): void;
211
+ cleanExpiredCache(): void;
212
+ /**
213
+ * Optional store-supplied OAuth refresh. When present, `AuthStorage` uses
214
+ * it before the per-provider local refresh path. `RemoteAuthCredentialStore`
215
+ * implements this against the broker; SQLite stores leave it undefined.
216
+ *
217
+ * Precedence: `AuthStorageOptions.refreshOAuthCredential` > this hook > local.
218
+ *
219
+ * `signal` propagates the agent's cancel (ESC, request abort, …) all the
220
+ * way to the broker fetch so a hung connection can't strand the caller
221
+ * for `timeoutMs * (maxRetries + 1)`.
222
+ */
223
+ refreshOAuthCredential?(provider: Provider, credentialId: number, credential: OAuthCredential, signal?: AbortSignal): Promise<OAuthCredentials>;
224
+ /**
225
+ * Optional async pre-read hook invoked after AuthStorage selects a stored
226
+ * credential but before it returns that credential for an outbound request.
227
+ * Remote broker stores use this to wait out imminent rotations and refresh
228
+ * their local snapshot before the caller sees a stale access token.
229
+ */
230
+ prepareForRequest?(credentialId: number, opts?: {
231
+ signal?: AbortSignal;
232
+ }): Promise<boolean | undefined>;
233
+ /**
234
+ * Optional store-supplied aggregate usage fetch. When present, `AuthStorage`
235
+ * routes `fetchUsageReports()` here instead of fanning out per-credential.
236
+ * `RemoteAuthCredentialStore` proxies to the broker (whose datacenter IP
237
+ * isn't rate-limited like a heavy residential client).
238
+ *
239
+ * Precedence: `AuthStorageOptions.fetchUsageReports` > this hook > local fan-out.
240
+ *
241
+ * `signal` propagates the agent's cancel down to the broker fetch.
242
+ */
243
+ fetchUsageReports?(signal?: AbortSignal): Promise<UsageReport[] | null>;
244
+ /**
245
+ * Optional store-supplied per-credential usage report lookup. When present,
246
+ * `AuthStorage` consults this before its own per-credential upstream fetch
247
+ * (`#getUsageReport`). `RemoteAuthCredentialStore` implements this against
248
+ * the broker's aggregate `/v1/usage` (one coalesced round-trip shared across
249
+ * all callers) so multi-credential ranking on the client never hits the
250
+ * upstream provider's rate-limited usage endpoint from the laptop IP.
251
+ *
252
+ * Returning `null` is authoritative — `AuthStorage` does NOT fall back to
253
+ * the local fetch path. The store hook owns the decision, since falling
254
+ * back would re-introduce the per-IP rate-limit problem the broker exists
255
+ * to avoid.
256
+ *
257
+ * `signal` propagates the agent's cancel down to the broker fetch.
258
+ */
259
+ getUsageReport?(provider: Provider, credential: OAuthCredential, signal?: AbortSignal): Promise<UsageReport | null>;
260
+ /**
261
+ * Optional store hook to invalidate a specific credential after the upstream
262
+ * provider returned 401 on a supposedly-fresh key. Remote stores force the
263
+ * broker to re-issue the row; local stores can leave it undefined and let
264
+ * {@link AuthStorage.invalidateCredentialMatching} fall back to `reload()`.
265
+ */
266
+ markCredentialSuspect?(credentialId: number, opts?: {
267
+ signal?: AbortSignal;
268
+ }): Promise<void>;
269
+ /**
270
+ * Optional async write hook for upserting a single credential. When present,
271
+ * `AuthStorage.#upsertOAuthCredential` routes through this instead of the
272
+ * sync `upsertAuthCredentialForProvider`. `RemoteAuthCredentialStore` uses
273
+ * it to send the upsert to the broker via `POST /v1/credential`.
274
+ *
275
+ * Implementations MUST update the in-memory snapshot before returning so the
276
+ * post-write read path is consistent.
277
+ */
278
+ upsertAuthCredentialRemote?(provider: string, credential: AuthCredential): Promise<StoredAuthCredential[]>;
279
+ /**
280
+ * Optional async write hook for replace-all semantics (e.g. API-key login
281
+ * overwriting any previous keys for the same provider). When present,
282
+ * `AuthStorage.set` routes through this instead of the sync
283
+ * `replaceAuthCredentialsForProvider`.
284
+ */
285
+ replaceAuthCredentialsRemote?(provider: string, credentials: AuthCredential[]): Promise<StoredAuthCredential[]>;
286
+ /**
287
+ * Optional async write hook for clearing every credential for a provider
288
+ * (logout). When present, `AuthStorage.remove` routes through this instead
289
+ * of the sync `deleteAuthCredentialsForProvider`.
290
+ */
291
+ deleteAuthCredentialsRemote?(provider: string, disabledCause: string): Promise<void>;
292
+ }
293
+ /**
294
+ * Event payload describing a credential that was just soft-disabled.
295
+ *
296
+ * Today the only call site is OAuth refresh failures with a definitive cause
297
+ * (`invalid_grant`, `401/403` not from a network blip, etc.) — the
298
+ * disabled_cause string is the verbatim error captured for forensics.
299
+ *
300
+ * Subscribers can use this to surface a notification, banner, or auto-launch
301
+ * a re-login flow instead of letting the credential silently disappear.
302
+ */
303
+ export interface CredentialDisabledEvent {
304
+ provider: string;
305
+ disabledCause: string;
306
+ }
307
+ export type AuthStorageOptions = {
308
+ usageProviderResolver?: (provider: Provider) => UsageProvider | undefined;
309
+ rankingStrategyResolver?: (provider: Provider) => CredentialRankingStrategy | undefined;
310
+ usageFetch?: typeof fetch;
311
+ usageRequestTimeoutMs?: number;
312
+ usageLogger?: UsageLogger;
313
+ /**
314
+ * Resolve a config value (API key, header value, etc.) to an actual value.
315
+ * - coding-agent injects its resolveConfigValue (supports "!command" syntax via prometheus-natives)
316
+ * - Default: checks environment variable first, then treats as literal
317
+ */
318
+ configValueResolver?: (config: string) => Promise<string | undefined>;
319
+ /**
320
+ * Optional callback fired when AuthStorage automatically disables a
321
+ * credential because something detected it as no longer usable — today
322
+ * that's the OAuth refresh-failure path in `getApiKey`. NOT fired for
323
+ * user-initiated `remove()` (the user already knows) or dedup of
324
+ * duplicate credentials (uninteresting hygiene).
325
+ */
326
+ onCredentialDisabled?: (event: CredentialDisabledEvent) => void | Promise<void>;
327
+ /**
328
+ * Override OAuth refresh. When set, `AuthStorage` calls this instead of the
329
+ * per-provider local refresh function. Receives the credential id so the
330
+ * implementation can address remote credentials.
331
+ *
332
+ * Must return updated {@link OAuthCredentials} with at least `access` and
333
+ * `expires`. `refresh` may be an opaque sentinel (e.g. `"__remote__"`) when
334
+ * the actual refresh token never leaves the broker.
335
+ */
336
+ refreshOAuthCredential?: (provider: Provider, credentialId: number, credential: OAuthCredential, signal?: AbortSignal) => Promise<OAuthCredentials>;
337
+ /**
338
+ * Human-readable description of the credential store backing this
339
+ * AuthStorage instance. Surfaced through {@link AuthStorage.describeCredentialSource}
340
+ * so the TUI can show where a token came from (broker URL or local SQLite path).
341
+ *
342
+ * Examples:
343
+ * - `"local ~/.prometheus/agent/agent.db"`
344
+ * - `"broker http://can.internal:8765"`
345
+ */
346
+ sourceLabel?: string;
347
+ /**
348
+ * Override `fetchUsageReports`. When set, `AuthStorage.fetchUsageReports`
349
+ * calls this instead of fanning out per-credential. The primary use case is
350
+ * routing through a broker that egresses from a less-throttled IP — e.g. a
351
+ * residential laptop trips Anthropic's per-IP rate limit on the usage
352
+ * endpoint and drops 2-of-5 credentials, while the VPS broker gets all 5.
353
+ *
354
+ * Implementations may return null when no usage data is available; the
355
+ * AuthStorage caller surfaces that to its own consumer unchanged.
356
+ */
357
+ fetchUsageReports?: (signal?: AbortSignal) => Promise<UsageReport[] | null>;
358
+ };
359
+ export declare function isDefinitiveOAuthFailure(errorMsg: string): boolean;
360
+ type AuthApiKeyOptions = {
361
+ baseUrl?: string;
362
+ modelId?: string;
363
+ /**
364
+ * Caller's cancel signal. Threaded into any broker-bound OAuth refresh so
365
+ * `ESC` / request abort actually kills a hung broker fetch instead of
366
+ * stranding the caller for `timeoutMs * (maxRetries + 1)`.
367
+ */
368
+ signal?: AbortSignal;
369
+ };
370
+ /**
371
+ * Refreshed OAuth access plus identity metadata returned by
372
+ * {@link AuthStorage.getOAuthAccess}. Callers that authenticate via a bearer
373
+ * AND need the credential's identity (Codex `chatgpt-account-id`, Google
374
+ * `projectId`, GitHub `enterpriseUrl`) consume this shape directly; the
375
+ * refresh slot is deliberately omitted because rotating refresh tokens never
376
+ * leave {@link AuthStorage}.
377
+ */
378
+ export interface OAuthAccess {
379
+ accessToken: string;
380
+ credentialId?: number;
381
+ accountId?: string;
382
+ email?: string;
383
+ projectId?: string;
384
+ enterpriseUrl?: string;
385
+ }
386
+ export interface OAuthAccessFailure {
387
+ credentialId?: number;
388
+ accountId?: string;
389
+ email?: string;
390
+ projectId?: string;
391
+ enterpriseUrl?: string;
392
+ error: string;
393
+ }
394
+ export type OAuthAccessResolution = ({
395
+ ok: true;
396
+ } & OAuthAccess) | ({
397
+ ok: false;
398
+ } & OAuthAccessFailure);
399
+ export interface InvalidateCredentialMatchingOptions {
400
+ signal?: AbortSignal;
401
+ sessionId?: string;
402
+ }
403
+ /**
404
+ * Credential storage backed by an AuthCredentialStore.
405
+ * Reads from storage on reload(), manages round-robin credential selection,
406
+ * usage limit tracking, and OAuth token refresh.
407
+ */
408
+ export declare class AuthStorage {
409
+ #private;
410
+ constructor(store: AuthCredentialStore, options?: AuthStorageOptions);
411
+ /**
412
+ * Create an AuthStorage instance backed by a AuthCredentialStore.
413
+ * Convenience factory for standalone use (e.g., @prometheus-ai/ai CLI).
414
+ * @param dbPath - Path to SQLite database
415
+ */
416
+ static create(dbPath: string, options?: AuthStorageOptions): Promise<AuthStorage>;
417
+ /**
418
+ * Close the underlying credential store.
419
+ *
420
+ * After calling this, the instance must not be reused.
421
+ */
422
+ close(): void;
423
+ getGeneration(): number;
424
+ onGenerationChanged(listener: (generation: number) => void): () => void;
425
+ offGenerationChanged(listener: (generation: number) => void): void;
426
+ /**
427
+ * Subscribe to {@link CredentialDisabledEvent}s. Multiple subscribers are supported and
428
+ * each fires for every disable event; subscribers are invoked in registration order with
429
+ * exceptions and async rejections isolated per-listener so a misbehaving subscriber
430
+ * cannot break the disable path or starve the rest of the chain.
431
+ *
432
+ * If `credential_disabled` events were emitted while no listener was subscribed, they are
433
+ * replayed (in insertion order) to the listener that triggers the empty→non-empty
434
+ * transition. The drain is one-shot — listeners that subscribe after that no longer see
435
+ * past events.
436
+ *
437
+ * Returns an unsubscribe function. The function is idempotent: calling it more than once
438
+ * is a no-op. After every subscriber has unsubscribed, subsequent disable events buffer
439
+ * again until the next subscribe.
440
+ *
441
+ * @param listener Callback invoked with each disable event. May be sync or async.
442
+ * @returns A function that removes this listener from the subscriber set.
443
+ */
444
+ onCredentialDisabled(listener: (event: CredentialDisabledEvent) => void | Promise<void>): () => void;
445
+ /**
446
+ * Set a runtime API key override (not persisted to disk).
447
+ * Used for CLI --api-key flag.
448
+ */
449
+ setRuntimeApiKey(provider: string, apiKey: string): void;
450
+ /**
451
+ * Remove a runtime API key override.
452
+ */
453
+ removeRuntimeApiKey(provider: string): void;
454
+ /**
455
+ * Register a per-provider API key sourced from user configuration
456
+ * (e.g. `models.yml` `providers.<name>.apiKey`). Higher priority than
457
+ * stored credentials and OAuth tokens — when the user pins a key in
458
+ * config, that key is what authenticates outbound requests, regardless
459
+ * of whatever the broker happens to have loaded for that provider.
460
+ *
461
+ * Lower priority than {@link setRuntimeApiKey} so a CLI `--api-key`
462
+ * still wins for the duration of a single invocation.
463
+ */
464
+ setConfigApiKey(provider: string, apiKey: string): void;
465
+ /**
466
+ * Remove a single config-sourced API key override.
467
+ */
468
+ removeConfigApiKey(provider: string): void;
469
+ /**
470
+ * Drop every config-sourced API key. Called by `ModelRegistry` before
471
+ * re-parsing `models.yml` so removed entries actually disappear.
472
+ */
473
+ clearConfigApiKeys(): void;
474
+ /**
475
+ * Set a fallback resolver for API keys not found in storage or env vars.
476
+ * Used for custom provider keys from models.json.
477
+ */
478
+ setFallbackResolver(resolver: (provider: string) => string | undefined): void;
479
+ /**
480
+ * Reload credentials from storage.
481
+ */
482
+ reload(): Promise<void>;
483
+ /**
484
+ * Get credential for a provider (first entry if multiple).
485
+ */
486
+ get(provider: string): AuthCredential | undefined;
487
+ /**
488
+ * Set credential for a provider.
489
+ */
490
+ set(provider: string, credential: AuthCredentialEntry): Promise<void>;
491
+ /**
492
+ * Remove credential for a provider.
493
+ */
494
+ remove(provider: string): Promise<void>;
495
+ /**
496
+ * List all providers with credentials.
497
+ */
498
+ list(): string[];
499
+ /**
500
+ * Check if credentials exist for a provider in storage.
501
+ */
502
+ has(provider: string): boolean;
503
+ /**
504
+ * Check if any form of auth is configured for a provider.
505
+ * Unlike getApiKey(), this doesn't refresh OAuth tokens.
506
+ */
507
+ hasAuth(provider: string): boolean;
508
+ /**
509
+ * True iff a dedicated, non-env credential source is configured for this
510
+ * provider — i.e. anything in the cascade EXCEPT `getEnvApiKey(provider)`.
511
+ *
512
+ * Mirrors `hasAuth` minus the env-fallback leg. Useful for callers that
513
+ * need to distinguish "the user explicitly configured this provider"
514
+ * from "an env var happens to alias this provider via the cross-provider
515
+ * fallback map" (see e.g. `xai-oauth → XAI_OAUTH_TOKEN || XAI_API_KEY` in
516
+ * `stream.ts`). Without that distinction, an `XAI_API_KEY`-only setup
517
+ * silently satisfies xai-oauth and routes around `providers.xai.baseUrl`.
518
+ */
519
+ hasNonEnvCredential(provider: string): boolean;
520
+ /**
521
+ * Check if OAuth credentials are configured for a provider.
522
+ */
523
+ hasOAuth(provider: string): boolean;
524
+ /**
525
+ * Get OAuth credentials for a provider.
526
+ */
527
+ getOAuthCredential(provider: string): OAuthCredential | undefined;
528
+ /**
529
+ * Get the OAuth `accountId` for a provider, preferring the credential that is
530
+ * session-sticky for `sessionId` when multiple OAuth credentials are configured.
531
+ * Falls back to the first OAuth credential when no session preference exists (e.g.
532
+ * first call before any `getApiKey` has been issued, or single-credential setups).
533
+ * Returns `undefined` when no OAuth credential carries an `accountId`.
534
+ */
535
+ getOAuthAccountId(provider: string, sessionId?: string): string | undefined;
536
+ /**
537
+ * Get all credentials.
538
+ */
539
+ getAll(): AuthStorageData;
540
+ /**
541
+ * Login to an OAuth provider.
542
+ */
543
+ login(provider: OAuthProviderId, ctrl: OAuthController & {
544
+ /** onAuth is required by auth-storage but optional in OAuthController */
545
+ onAuth: (info: {
546
+ url: string;
547
+ instructions?: string;
548
+ }) => void;
549
+ /** onPrompt is required for some providers (github-copilot, openai-codex) */
550
+ onPrompt: (prompt: {
551
+ message: string;
552
+ placeholder?: string;
553
+ }) => Promise<string>;
554
+ }): Promise<void>;
555
+ /**
556
+ * Logout from a provider.
557
+ */
558
+ logout(provider: string): Promise<void>;
559
+ ingestUsageHeaders(provider: Provider, headers: Record<string, string>, options?: {
560
+ sessionId?: string;
561
+ baseUrl?: string;
562
+ }): boolean;
563
+ fetchUsageReports(options?: {
564
+ baseUrlResolver?: (provider: Provider) => string | undefined;
565
+ /** Caller's cancel signal; only rejects this caller, never the shared upstream fetch. */
566
+ signal?: AbortSignal;
567
+ }): Promise<UsageReport[] | null>;
568
+ /**
569
+ * Probe each stored credential against its provider's auth-verifying usage
570
+ * endpoint and report per-credential auth health.
571
+ *
572
+ * Surfaces the identity of failing credentials so callers running a
573
+ * multi-account pool (e.g. a broker-backed auth-gateway) can tell which
574
+ * row is producing 401s. The probe mirrors the per-credential fan-out
575
+ * inside {@link AuthStorage.fetchUsageReports} (OAuth refresh-on-expiry,
576
+ * then `UsageProvider.fetchUsage`) but does NOT swallow errors — every
577
+ * credential gets either `ok: true`, `ok: false` with `reason`, or
578
+ * `ok: null` when no probe is configured for the provider.
579
+ *
580
+ * Iterates sequentially to avoid synchronized N-account fan-out that
581
+ * upstream `/usage` rate limiters (per source IP) treat as a burst.
582
+ *
583
+ * Only inspects active rows from {@link AuthCredentialStore.listAuthCredentials};
584
+ * soft-disabled rows are already known-bad and don't need a network probe.
585
+ * Environment-variable API keys are not enumerated — the caller's intent
586
+ * here is "which of my stored credentials is broken".
587
+ *
588
+ * Pass {@link CheckCredentialsOptions.completionProbe} to additionally
589
+ * exercise each credential against the provider's chat-completion endpoint
590
+ * (strict mode). The result lands on
591
+ * {@link CredentialHealthResult.completion}; the usage `ok` field is
592
+ * unchanged so callers can tell the two signals apart.
593
+ */
594
+ checkCredentials(options?: CheckCredentialsOptions): Promise<CredentialHealthResult[]>;
595
+ /**
596
+ * Marks the current session's credential as temporarily blocked due to usage limits.
597
+ * Uses usage reports to determine accurate reset time when available.
598
+ * Returns true if a credential was blocked, enabling automatic fallback to the next credential.
599
+ */
600
+ markUsageLimitReached(provider: string, sessionId: string | undefined, options?: {
601
+ retryAfterMs?: number;
602
+ baseUrl?: string;
603
+ signal?: AbortSignal;
604
+ }): Promise<boolean>;
605
+ /**
606
+ * Peek at API key for a provider without refreshing OAuth tokens.
607
+ * Used for model discovery where we only need to know if credentials exist
608
+ * and get a best-effort token. For GitHub Copilot we preserve enterprise
609
+ * routing metadata so discovery can hit the correct host.
610
+ */
611
+ peekApiKey(provider: string): Promise<string | undefined>;
612
+ /**
613
+ * Get API key for a provider.
614
+ * Priority:
615
+ * 1. Runtime override (CLI --api-key)
616
+ * 2. Config override (models.yml `providers.<name>.apiKey`)
617
+ * 3. API key from storage
618
+ * 4. OAuth token from storage (auto-refreshed)
619
+ * 5. Environment variable
620
+ * 6. Fallback resolver (models.yml custom providers, last-resort)
621
+ */
622
+ getApiKey(provider: string, sessionId?: string, options?: AuthApiKeyOptions): Promise<string | undefined>;
623
+ /**
624
+ * Resolve the OAuth credential for `provider`, refreshing through the same
625
+ * pipeline as {@link AuthStorage.getApiKey} but returning the refreshed
626
+ * {@link OAuthAccess} (raw access token + identity metadata) instead of
627
+ * the API-key bytes.
628
+ *
629
+ * Use this when the caller needs to inject identity headers alongside the
630
+ * bearer (Codex `chatgpt-account-id`, Google `project`, GitHub
631
+ * `enterpriseUrl`). For pure "give me the bytes for `Authorization`"
632
+ * scenarios, prefer {@link AuthStorage.getApiKey}.
633
+ *
634
+ * Returns `undefined` when no OAuth credential is available, the
635
+ * credential fails to refresh, or runtime/config overrides have replaced
636
+ * OAuth with an explicit API key.
637
+ */
638
+ getOAuthAccess(provider: string, sessionId?: string, options?: AuthApiKeyOptions): Promise<OAuthAccess | undefined>;
639
+ /**
640
+ * Resolve every stored OAuth credential for `provider` independently.
641
+ *
642
+ * Refreshes credentials through the same broker/local path as
643
+ * {@link AuthStorage.getOAuthAccess}, but does not rank, round-robin, or
644
+ * stop after the first usable account. Intended for diagnostics that must
645
+ * exercise each stored account exactly once.
646
+ */
647
+ getOAuthAccesses(provider: string, options?: AuthApiKeyOptions): Promise<OAuthAccessResolution[]>;
648
+ invalidateCredentialMatching(provider: string, apiKey: string, options?: InvalidateCredentialMatchingOptions): Promise<boolean>;
649
+ invalidateCredentialMatching(provider: string, apiKey: string, signal?: AbortSignal): Promise<boolean>;
650
+ /**
651
+ * Build a redacted snapshot of all loaded credentials for the auth-broker
652
+ * wire. OAuth refresh tokens are replaced with {@link REMOTE_REFRESH_SENTINEL}
653
+ * so clients never see the actual refresh token.
654
+ *
655
+ * Callers must {@link AuthStorage.reload} first when serving a stale snapshot
656
+ * (the broker server's HTTP handler does this).
657
+ */
658
+ exportSnapshot(): AuthCredentialSnapshot;
659
+ /**
660
+ * Refresh the OAuth credential with the given id through a per-credential
661
+ * single-flight. Concurrent callers for the same row await the same upstream
662
+ * refresh attempt, which is required for providers that rotate refresh tokens
663
+ * on every successful refresh.
664
+ */
665
+ refreshCredentialById(id: number, signal?: AbortSignal): Promise<AuthCredentialSnapshotEntry>;
666
+ /**
667
+ * Force-refresh the OAuth credential with the given id, bypassing the
668
+ * not-yet-expired guard. Used by the auth-broker server to honour
669
+ * `POST /v1/credential/:id/refresh`.
670
+ *
671
+ * Returns the redacted snapshot entry for the refreshed row.
672
+ * Throws when no OAuth credential with that id is loaded.
673
+ */
674
+ forceRefreshCredentialById(id: number, signal?: AbortSignal): Promise<AuthCredentialSnapshotEntry>;
675
+ /**
676
+ * Disable the credential with the given id and emit a
677
+ * {@link CredentialDisabledEvent}. Used by the auth-broker server to honour
678
+ * `POST /v1/credential/:id/disable`. Returns `false` when no such row exists.
679
+ */
680
+ disableCredentialById(id: number, disabledCause: string): boolean;
681
+ /**
682
+ * Upsert a credential into the underlying store, refresh the in-memory
683
+ * snapshot, and return the redacted snapshot entries for the provider.
684
+ *
685
+ * Used by the auth-broker server to honour `POST /v1/credential`. The
686
+ * persistence layer (`SqliteAuthCredentialStore.upsertAuthCredentialForProvider`)
687
+ * does identity-key matching, so re-uploading the same email/account replaces
688
+ * the existing row instead of inserting a duplicate.
689
+ */
690
+ upsertCredential(provider: string, credential: AuthCredential): AuthCredentialSnapshotEntry[];
691
+ /**
692
+ * Describe where the active credential for a provider came from.
693
+ *
694
+ * Surfaces four layers, highest precedence first:
695
+ * 1. Runtime override (`--api-key`).
696
+ * 2. Config override (`models.yml` `providers.<name>.apiKey`).
697
+ * 3. Stored credential (the one this session is currently sticky to, or the
698
+ * one round-robin would pick next when no session id is supplied).
699
+ * 4. Env var / fallback resolver — when no stored credential exists.
700
+ *
701
+ * The string is purely informational; consumers must not parse it.
702
+ */
703
+ describeCredentialSource(provider: string, sessionId?: string): string | undefined;
704
+ }
705
+ /**
706
+ * Default SQLite-backed implementation of {@link AuthCredentialStore}.
707
+ *
708
+ * Used by the @prometheus-ai/ai CLI and as the default store for `AuthStorage.create()`.
709
+ * Also exposes convenience methods (`saveOAuth`, `getOAuth`, `saveApiKey`,
710
+ * `getApiKey`, `listProviders`, `deleteProvider`) that callers can use directly
711
+ * without going through `AuthStorage`.
712
+ */
713
+ export declare class SqliteAuthCredentialStore implements AuthCredentialStore {
714
+ #private;
715
+ constructor(db: Database);
716
+ static open(dbPath?: string): Promise<SqliteAuthCredentialStore>;
717
+ listAuthCredentials(provider?: string): StoredAuthCredential[];
718
+ replaceAuthCredentialsForProvider(provider: string, credentials: AuthCredential[]): StoredAuthCredential[];
719
+ upsertAuthCredentialForProvider(provider: string, credential: AuthCredential): StoredAuthCredential[];
720
+ updateAuthCredential(id: number, credential: AuthCredential): void;
721
+ deleteAuthCredential(id: number, disabledCause: string): void;
722
+ /**
723
+ * CAS-style disable: only soft-deletes the row when its `data` column still
724
+ * matches `expectedData` and the row has not already been disabled. Used by
725
+ * the OAuth refresh-failure path to avoid clobbering a peer that rotated the
726
+ * row between our pre-check and the disable.
727
+ */
728
+ tryDisableAuthCredentialIfMatches(id: number, expectedData: string, disabledCause: string): boolean;
729
+ deleteAuthCredentialsForProvider(provider: string, disabledCause: string): void;
730
+ getCache(key: string, options?: {
731
+ includeExpired?: boolean;
732
+ }): string | null;
733
+ setCache(key: string, value: string, expiresAtSec: number): void;
734
+ cleanExpiredCache(): void;
735
+ /**
736
+ * Save OAuth credentials for a provider.
737
+ * Preserves unrelated identities and replaces only the matching credential.
738
+ */
739
+ saveOAuth(provider: string, credentials: OAuthCredentials): void;
740
+ /**
741
+ * Get OAuth credentials for a provider.
742
+ */
743
+ getOAuth(provider: string): OAuthCredentials | null;
744
+ /**
745
+ * Save API key for a provider (replaces existing).
746
+ */
747
+ saveApiKey(provider: string, apiKey: string): void;
748
+ /**
749
+ * Get API key for a provider.
750
+ */
751
+ getApiKey(provider: string): string | null;
752
+ /**
753
+ * List all providers with credentials.
754
+ */
755
+ listProviders(): string[];
756
+ /**
757
+ * Delete all credentials for a provider.
758
+ */
759
+ deleteProvider(provider: string): void;
760
+ close(): void;
761
+ }
762
+ export {};