@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,641 @@
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
+ export interface CheckCredentialsOptions {
80
+ signal?: AbortSignal;
81
+ /** Per-credential probe timeout (ms). Defaults to the configured usage request timeout. */
82
+ timeoutMs?: number;
83
+ /** Provider → base URL override, same shape as {@link AuthStorage.fetchUsageReports}. */
84
+ baseUrlResolver?: (provider: Provider) => string | undefined;
85
+ }
86
+ /**
87
+ * Sentinel value placed in OAuth `refresh` fields when a credential is shared
88
+ * via {@link AuthStorage.exportSnapshot}. Refresh tokens never leave the broker;
89
+ * clients must call back to refresh.
90
+ */
91
+ export declare const REMOTE_REFRESH_SENTINEL: "__remote__";
92
+ export type RemoteRefreshSentinel = typeof REMOTE_REFRESH_SENTINEL;
93
+ /** OAuth credential with refresh token replaced by the broker sentinel. */
94
+ export type RemoteOAuthCredential = Omit<OAuthCredential, "refresh"> & {
95
+ refresh: RemoteRefreshSentinel;
96
+ };
97
+ /** Discriminated credential payload as published by the broker. */
98
+ export type SnapshotCredential = ApiKeyCredential | RemoteOAuthCredential;
99
+ export interface AuthCredentialSnapshotEntry {
100
+ id: number;
101
+ provider: string;
102
+ credential: SnapshotCredential;
103
+ identityKey: string | null;
104
+ }
105
+ /**
106
+ * Wire-shaped snapshot exported by {@link AuthStorage.exportSnapshot} and
107
+ * served by the auth-broker server on `GET /v1/snapshot`.
108
+ */
109
+ export interface AuthCredentialSnapshot {
110
+ generation: number;
111
+ generatedAt: number;
112
+ credentials: AuthCredentialSnapshotEntry[];
113
+ }
114
+ /**
115
+ * Persistence abstraction consumed by {@link AuthStorage}.
116
+ *
117
+ * Concrete implementations:
118
+ * - {@link SqliteAuthCredentialStore} — local SQLite-backed store (default).
119
+ * - `RemoteAuthCredentialStore` from `./auth-broker` — client-side snapshot of
120
+ * a remote broker; mutating methods (`replace*`, `upsert*`, `delete*ForProvider`)
121
+ * throw because login flows route through the broker, not the client.
122
+ */
123
+ export interface AuthCredentialStore {
124
+ close(): void;
125
+ listAuthCredentials(provider?: string): StoredAuthCredential[];
126
+ updateAuthCredential(id: number, credential: AuthCredential): void;
127
+ deleteAuthCredential(id: number, disabledCause: string): void;
128
+ tryDisableAuthCredentialIfMatches(id: number, expectedData: string, disabledCause: string): boolean;
129
+ replaceAuthCredentialsForProvider(provider: string, credentials: AuthCredential[]): StoredAuthCredential[];
130
+ upsertAuthCredentialForProvider(provider: string, credential: AuthCredential): StoredAuthCredential[];
131
+ deleteAuthCredentialsForProvider(provider: string, disabledCause: string): void;
132
+ getCache(key: string, options?: {
133
+ includeExpired?: boolean;
134
+ }): string | null;
135
+ setCache(key: string, value: string, expiresAtSec: number): void;
136
+ cleanExpiredCache(): void;
137
+ /**
138
+ * Optional store-supplied OAuth refresh. When present, `AuthStorage` uses
139
+ * it before the per-provider local refresh path. `RemoteAuthCredentialStore`
140
+ * implements this against the broker; SQLite stores leave it undefined.
141
+ *
142
+ * Precedence: `AuthStorageOptions.refreshOAuthCredential` > this hook > local.
143
+ *
144
+ * `signal` propagates the agent's cancel (ESC, request abort, …) all the
145
+ * way to the broker fetch so a hung connection can't strand the caller
146
+ * for `timeoutMs * (maxRetries + 1)`.
147
+ */
148
+ refreshOAuthCredential?(provider: Provider, credentialId: number, credential: OAuthCredential, signal?: AbortSignal): Promise<OAuthCredentials>;
149
+ /**
150
+ * Optional async pre-read hook invoked after AuthStorage selects a stored
151
+ * credential but before it returns that credential for an outbound request.
152
+ * Remote broker stores use this to wait out imminent rotations and refresh
153
+ * their local snapshot before the caller sees a stale access token.
154
+ */
155
+ prepareForRequest?(credentialId: number, opts?: {
156
+ signal?: AbortSignal;
157
+ }): Promise<boolean | undefined>;
158
+ /**
159
+ * Optional store-supplied aggregate usage fetch. When present, `AuthStorage`
160
+ * routes `fetchUsageReports()` here instead of fanning out per-credential.
161
+ * `RemoteAuthCredentialStore` proxies to the broker (whose datacenter IP
162
+ * isn't rate-limited like a heavy residential client).
163
+ *
164
+ * Precedence: `AuthStorageOptions.fetchUsageReports` > this hook > local fan-out.
165
+ *
166
+ * `signal` propagates the agent's cancel down to the broker fetch.
167
+ */
168
+ fetchUsageReports?(signal?: AbortSignal): Promise<UsageReport[] | null>;
169
+ /**
170
+ * Optional store-supplied per-credential usage report lookup. When present,
171
+ * `AuthStorage` consults this before its own per-credential upstream fetch
172
+ * (`#getUsageReport`). `RemoteAuthCredentialStore` implements this against
173
+ * the broker's aggregate `/v1/usage` (one coalesced round-trip shared across
174
+ * all callers) so multi-credential ranking on the client never hits the
175
+ * upstream provider's rate-limited usage endpoint from the laptop IP.
176
+ *
177
+ * Returning `null` is authoritative — `AuthStorage` does NOT fall back to
178
+ * the local fetch path. The store hook owns the decision, since falling
179
+ * back would re-introduce the per-IP rate-limit problem the broker exists
180
+ * to avoid.
181
+ *
182
+ * `signal` propagates the agent's cancel down to the broker fetch.
183
+ */
184
+ getUsageReport?(provider: Provider, credential: OAuthCredential, signal?: AbortSignal): Promise<UsageReport | null>;
185
+ /**
186
+ * Optional store hook to invalidate a specific credential after the upstream
187
+ * provider returned 401 on a supposedly-fresh key. Remote stores force the
188
+ * broker to re-issue the row; local stores can leave it undefined and let
189
+ * {@link AuthStorage.invalidateCredentialMatching} fall back to `reload()`.
190
+ */
191
+ markCredentialSuspect?(credentialId: number, opts?: {
192
+ signal?: AbortSignal;
193
+ }): Promise<void>;
194
+ /**
195
+ * Optional async write hook for upserting a single credential. When present,
196
+ * `AuthStorage.#upsertOAuthCredential` routes through this instead of the
197
+ * sync `upsertAuthCredentialForProvider`. `RemoteAuthCredentialStore` uses
198
+ * it to send the upsert to the broker via `POST /v1/credential`.
199
+ *
200
+ * Implementations MUST update the in-memory snapshot before returning so the
201
+ * post-write read path is consistent.
202
+ */
203
+ upsertAuthCredentialRemote?(provider: string, credential: AuthCredential): Promise<StoredAuthCredential[]>;
204
+ /**
205
+ * Optional async write hook for replace-all semantics (e.g. API-key login
206
+ * overwriting any previous keys for the same provider). When present,
207
+ * `AuthStorage.set` routes through this instead of the sync
208
+ * `replaceAuthCredentialsForProvider`.
209
+ */
210
+ replaceAuthCredentialsRemote?(provider: string, credentials: AuthCredential[]): Promise<StoredAuthCredential[]>;
211
+ /**
212
+ * Optional async write hook for clearing every credential for a provider
213
+ * (logout). When present, `AuthStorage.remove` routes through this instead
214
+ * of the sync `deleteAuthCredentialsForProvider`.
215
+ */
216
+ deleteAuthCredentialsRemote?(provider: string, disabledCause: string): Promise<void>;
217
+ }
218
+ /**
219
+ * Event payload describing a credential that was just soft-disabled.
220
+ *
221
+ * Today the only call site is OAuth refresh failures with a definitive cause
222
+ * (`invalid_grant`, `401/403` not from a network blip, etc.) — the
223
+ * disabled_cause string is the verbatim error captured for forensics.
224
+ *
225
+ * Subscribers can use this to surface a notification, banner, or auto-launch
226
+ * a re-login flow instead of letting the credential silently disappear.
227
+ */
228
+ export interface CredentialDisabledEvent {
229
+ provider: string;
230
+ disabledCause: string;
231
+ }
232
+ export type AuthStorageOptions = {
233
+ usageProviderResolver?: (provider: Provider) => UsageProvider | undefined;
234
+ rankingStrategyResolver?: (provider: Provider) => CredentialRankingStrategy | undefined;
235
+ usageFetch?: typeof fetch;
236
+ usageRequestTimeoutMs?: number;
237
+ usageLogger?: UsageLogger;
238
+ /**
239
+ * Resolve a config value (API key, header value, etc.) to an actual value.
240
+ * - coding-agent injects its resolveConfigValue (supports "!command" syntax via pi-natives)
241
+ * - Default: checks environment variable first, then treats as literal
242
+ */
243
+ configValueResolver?: (config: string) => Promise<string | undefined>;
244
+ /**
245
+ * Optional callback fired when AuthStorage automatically disables a
246
+ * credential because something detected it as no longer usable — today
247
+ * that's the OAuth refresh-failure path in `getApiKey`. NOT fired for
248
+ * user-initiated `remove()` (the user already knows) or dedup of
249
+ * duplicate credentials (uninteresting hygiene).
250
+ */
251
+ onCredentialDisabled?: (event: CredentialDisabledEvent) => void | Promise<void>;
252
+ /**
253
+ * Override OAuth refresh. When set, `AuthStorage` calls this instead of the
254
+ * per-provider local refresh function. Receives the credential id so the
255
+ * implementation can address remote credentials.
256
+ *
257
+ * Must return updated {@link OAuthCredentials} with at least `access` and
258
+ * `expires`. `refresh` may be an opaque sentinel (e.g. `"__remote__"`) when
259
+ * the actual refresh token never leaves the broker.
260
+ */
261
+ refreshOAuthCredential?: (provider: Provider, credentialId: number, credential: OAuthCredential, signal?: AbortSignal) => Promise<OAuthCredentials>;
262
+ /**
263
+ * Human-readable description of the credential store backing this
264
+ * AuthStorage instance. Surfaced through {@link AuthStorage.describeCredentialSource}
265
+ * so the TUI can show where a token came from (broker URL or local SQLite path).
266
+ *
267
+ * Examples:
268
+ * - `"local ~/.gjc/agent/agent.db"`
269
+ * - `"broker http://can.internal:8765"`
270
+ */
271
+ sourceLabel?: string;
272
+ /**
273
+ * Override `fetchUsageReports`. When set, `AuthStorage.fetchUsageReports`
274
+ * calls this instead of fanning out per-credential. The primary use case is
275
+ * routing through a broker that egresses from a less-throttled IP — e.g. a
276
+ * residential laptop trips Anthropic's per-IP rate limit on the usage
277
+ * endpoint and drops 2-of-5 credentials, while the VPS broker gets all 5.
278
+ *
279
+ * Implementations may return null when no usage data is available; the
280
+ * AuthStorage caller surfaces that to its own consumer unchanged.
281
+ */
282
+ fetchUsageReports?: (signal?: AbortSignal) => Promise<UsageReport[] | null>;
283
+ };
284
+ type AuthApiKeyOptions = {
285
+ baseUrl?: string;
286
+ modelId?: string;
287
+ /**
288
+ * Caller's cancel signal. Threaded into any broker-bound OAuth refresh so
289
+ * `ESC` / request abort actually kills a hung broker fetch instead of
290
+ * stranding the caller for `timeoutMs * (maxRetries + 1)`.
291
+ */
292
+ signal?: AbortSignal;
293
+ };
294
+ /**
295
+ * Refreshed OAuth access plus identity metadata returned by
296
+ * {@link AuthStorage.getOAuthAccess}. Callers that authenticate via a bearer
297
+ * AND need the credential's identity (OpenAI code backend `chatgpt-account-id`, Google
298
+ * `projectId`, GitHub `enterpriseUrl`) consume this shape directly; the
299
+ * refresh slot is deliberately omitted because rotating refresh tokens never
300
+ * leave {@link AuthStorage}.
301
+ */
302
+ export interface OAuthAccess {
303
+ accessToken: string;
304
+ accountId?: string;
305
+ email?: string;
306
+ projectId?: string;
307
+ enterpriseUrl?: string;
308
+ }
309
+ export interface InvalidateCredentialMatchingOptions {
310
+ signal?: AbortSignal;
311
+ sessionId?: string;
312
+ }
313
+ /**
314
+ * Credential storage backed by an AuthCredentialStore.
315
+ * Reads from storage on reload(), manages round-robin credential selection,
316
+ * usage limit tracking, and OAuth token refresh.
317
+ */
318
+ export declare class AuthStorage {
319
+ #private;
320
+ constructor(store: AuthCredentialStore, options?: AuthStorageOptions);
321
+ /**
322
+ * Create an AuthStorage instance backed by a AuthCredentialStore.
323
+ * Convenience factory for standalone use (e.g., pi-ai CLI).
324
+ * @param dbPath - Path to SQLite database
325
+ */
326
+ static create(dbPath: string, options?: AuthStorageOptions): Promise<AuthStorage>;
327
+ /**
328
+ * Close the underlying credential store.
329
+ *
330
+ * After calling this, the instance must not be reused.
331
+ */
332
+ close(): void;
333
+ getGeneration(): number;
334
+ onGenerationChanged(listener: (generation: number) => void): () => void;
335
+ offGenerationChanged(listener: (generation: number) => void): void;
336
+ /**
337
+ * Subscribe to {@link CredentialDisabledEvent}s. Multiple subscribers are supported and
338
+ * each fires for every disable event; subscribers are invoked in registration order with
339
+ * exceptions and async rejections isolated per-listener so a misbehaving subscriber
340
+ * cannot break the disable path or starve the rest of the chain.
341
+ *
342
+ * If `credential_disabled` events were emitted while no listener was subscribed, they are
343
+ * replayed (in insertion order) to the listener that triggers the empty→non-empty
344
+ * transition. The drain is one-shot — listeners that subscribe after that no longer see
345
+ * past events.
346
+ *
347
+ * Returns an unsubscribe function. The function is idempotent: calling it more than once
348
+ * is a no-op. After every subscriber has unsubscribed, subsequent disable events buffer
349
+ * again until the next subscribe.
350
+ *
351
+ * @param listener Callback invoked with each disable event. May be sync or async.
352
+ * @returns A function that removes this listener from the subscriber set.
353
+ */
354
+ onCredentialDisabled(listener: (event: CredentialDisabledEvent) => void | Promise<void>): () => void;
355
+ /**
356
+ * Set a runtime API key override (not persisted to disk).
357
+ * Used for CLI --api-key flag.
358
+ */
359
+ setRuntimeApiKey(provider: string, apiKey: string): void;
360
+ /**
361
+ * Remove a runtime API key override.
362
+ */
363
+ removeRuntimeApiKey(provider: string): void;
364
+ /**
365
+ * Register a per-provider API key sourced from user configuration
366
+ * (e.g. `models.yml` `providers.<name>.apiKey`). Higher priority than
367
+ * stored credentials and OAuth tokens — when the user pins a key in
368
+ * config, that key is what authenticates outbound requests, regardless
369
+ * of whatever the broker happens to have loaded for that provider.
370
+ *
371
+ * Lower priority than {@link setRuntimeApiKey} so a CLI `--api-key`
372
+ * still wins for the duration of a single invocation.
373
+ */
374
+ setConfigApiKey(provider: string, apiKey: string): void;
375
+ /**
376
+ * Remove a single config-sourced API key override.
377
+ */
378
+ removeConfigApiKey(provider: string): void;
379
+ /**
380
+ * Drop every config-sourced API key. Called by `ModelRegistry` before
381
+ * re-parsing `models.yml` so removed entries actually disappear.
382
+ */
383
+ clearConfigApiKeys(): void;
384
+ /**
385
+ * Set a fallback resolver for API keys not found in storage or env vars.
386
+ * Used for custom provider keys from models.json.
387
+ */
388
+ setFallbackResolver(resolver: (provider: string) => string | undefined): void;
389
+ /**
390
+ * Reload credentials from storage.
391
+ */
392
+ reload(): Promise<void>;
393
+ /**
394
+ * Get credential for a provider (first entry if multiple).
395
+ */
396
+ get(provider: string): AuthCredential | undefined;
397
+ /**
398
+ * Set credential for a provider.
399
+ */
400
+ set(provider: string, credential: AuthCredentialEntry): Promise<void>;
401
+ /**
402
+ * Remove credential for a provider.
403
+ */
404
+ remove(provider: string): Promise<void>;
405
+ /**
406
+ * List all providers with credentials.
407
+ */
408
+ list(): string[];
409
+ /**
410
+ * Check if credentials exist for a provider in storage.
411
+ */
412
+ has(provider: string): boolean;
413
+ /**
414
+ * Check if any form of auth is configured for a provider.
415
+ * Unlike getApiKey(), this doesn't refresh OAuth tokens.
416
+ */
417
+ hasAuth(provider: string): boolean;
418
+ /**
419
+ * Check if OAuth credentials are configured for a provider.
420
+ */
421
+ hasOAuth(provider: string): boolean;
422
+ /**
423
+ * Get OAuth credentials for a provider.
424
+ */
425
+ getOAuthCredential(provider: string): OAuthCredential | undefined;
426
+ /**
427
+ * Get the OAuth `accountId` for a provider, preferring the credential that is
428
+ * session-sticky for `sessionId` when multiple OAuth credentials are configured.
429
+ * Falls back to the first OAuth credential when no session preference exists (e.g.
430
+ * first call before any `getApiKey` has been issued, or single-credential setups).
431
+ * Returns `undefined` when no OAuth credential carries an `accountId`.
432
+ */
433
+ getOAuthAccountId(provider: string, sessionId?: string): string | undefined;
434
+ /**
435
+ * Get all credentials.
436
+ */
437
+ getAll(): AuthStorageData;
438
+ /**
439
+ * Login to an OAuth provider.
440
+ */
441
+ login(provider: OAuthProviderId, ctrl: OAuthController & {
442
+ /** onAuth is required by auth-storage but optional in OAuthController */
443
+ onAuth: (info: {
444
+ url: string;
445
+ instructions?: string;
446
+ }) => void;
447
+ /** onPrompt is required for some providers (github-copilot, OpenAI code provider) */
448
+ onPrompt: (prompt: {
449
+ message: string;
450
+ placeholder?: string;
451
+ }) => Promise<string>;
452
+ }): Promise<void>;
453
+ /**
454
+ * Logout from a provider.
455
+ */
456
+ logout(provider: string): Promise<void>;
457
+ fetchUsageReports(options?: {
458
+ baseUrlResolver?: (provider: Provider) => string | undefined;
459
+ /** Caller's cancel signal; only rejects this caller, never the shared upstream fetch. */
460
+ signal?: AbortSignal;
461
+ }): Promise<UsageReport[] | null>;
462
+ /**
463
+ * Probe each stored credential against its provider's auth-verifying usage
464
+ * endpoint and report per-credential auth health.
465
+ *
466
+ * Surfaces the identity of failing credentials so callers running a
467
+ * multi-account pool (e.g. a broker-backed auth-gateway) can tell which
468
+ * row is producing 401s. The probe mirrors the per-credential fan-out
469
+ * inside {@link AuthStorage.fetchUsageReports} (OAuth refresh-on-expiry,
470
+ * then `UsageProvider.fetchUsage`) but does NOT swallow errors — every
471
+ * credential gets either `ok: true`, `ok: false` with `reason`, or
472
+ * `ok: null` when no probe is configured for the provider.
473
+ *
474
+ * Iterates sequentially to avoid synchronized N-account fan-out that
475
+ * upstream `/usage` rate limiters (per source IP) treat as a burst.
476
+ *
477
+ * Only inspects active rows from {@link AuthCredentialStore.listAuthCredentials};
478
+ * soft-disabled rows are already known-bad and don't need a network probe.
479
+ * Environment-variable API keys are not enumerated — the caller's intent
480
+ * here is "which of my stored credentials is broken".
481
+ */
482
+ checkCredentials(options?: CheckCredentialsOptions): Promise<CredentialHealthResult[]>;
483
+ /**
484
+ * Marks the current session's credential as temporarily blocked due to usage limits.
485
+ * Uses usage reports to determine accurate reset time when available.
486
+ * Returns true if a credential was blocked, enabling automatic fallback to the next credential.
487
+ */
488
+ markUsageLimitReached(provider: string, sessionId: string | undefined, options?: {
489
+ retryAfterMs?: number;
490
+ baseUrl?: string;
491
+ signal?: AbortSignal;
492
+ }): Promise<boolean>;
493
+ /**
494
+ * Peek at API key for a provider without refreshing OAuth tokens.
495
+ * Used for model discovery where we only need to know if credentials exist
496
+ * and get a best-effort token. For GitHub Copilot we preserve enterprise
497
+ * routing metadata so discovery can hit the correct host.
498
+ */
499
+ peekApiKey(provider: string): Promise<string | undefined>;
500
+ /**
501
+ * Get API key for a provider.
502
+ * Priority:
503
+ * 1. Runtime override (CLI --api-key)
504
+ * 2. Config override (models.yml `providers.<name>.apiKey`)
505
+ * 3. API key from storage
506
+ * 4. OAuth token from storage (auto-refreshed)
507
+ * 5. Environment variable
508
+ * 6. Fallback resolver (models.yml custom providers, last-resort)
509
+ */
510
+ getApiKey(provider: string, sessionId?: string, options?: AuthApiKeyOptions): Promise<string | undefined>;
511
+ /**
512
+ * Resolve the OAuth credential for `provider`, refreshing through the same
513
+ * pipeline as {@link AuthStorage.getApiKey} but returning the refreshed
514
+ * {@link OAuthAccess} (raw access token + identity metadata) instead of
515
+ * the API-key bytes.
516
+ *
517
+ * Use this when the caller needs to inject identity headers alongside the
518
+ * bearer (OpenAI code backend `chatgpt-account-id`, Google `project`, GitHub
519
+ * `enterpriseUrl`). For pure "give me the bytes for `Authorization`"
520
+ * scenarios, prefer {@link AuthStorage.getApiKey}.
521
+ *
522
+ * Returns `undefined` when no OAuth credential is available, the
523
+ * credential fails to refresh, or runtime/config overrides have replaced
524
+ * OAuth with an explicit API key.
525
+ */
526
+ getOAuthAccess(provider: string, sessionId?: string, options?: AuthApiKeyOptions): Promise<OAuthAccess | undefined>;
527
+ invalidateCredentialMatching(provider: string, apiKey: string, options?: InvalidateCredentialMatchingOptions): Promise<boolean>;
528
+ invalidateCredentialMatching(provider: string, apiKey: string, signal?: AbortSignal): Promise<boolean>;
529
+ /**
530
+ * Build a redacted snapshot of all loaded credentials for the auth-broker
531
+ * wire. OAuth refresh tokens are replaced with {@link REMOTE_REFRESH_SENTINEL}
532
+ * so clients never see the actual refresh token.
533
+ *
534
+ * Callers must {@link AuthStorage.reload} first when serving a stale snapshot
535
+ * (the broker server's HTTP handler does this).
536
+ */
537
+ exportSnapshot(): AuthCredentialSnapshot;
538
+ /**
539
+ * Refresh the OAuth credential with the given id through a per-credential
540
+ * single-flight. Concurrent callers for the same row await the same upstream
541
+ * refresh attempt, which is required for providers that rotate refresh tokens
542
+ * on every successful refresh.
543
+ */
544
+ refreshCredentialById(id: number, signal?: AbortSignal): Promise<AuthCredentialSnapshotEntry>;
545
+ /**
546
+ * Force-refresh the OAuth credential with the given id, bypassing the
547
+ * not-yet-expired guard. Used by the auth-broker server to honour
548
+ * `POST /v1/credential/:id/refresh`.
549
+ *
550
+ * Returns the redacted snapshot entry for the refreshed row.
551
+ * Throws when no OAuth credential with that id is loaded.
552
+ */
553
+ forceRefreshCredentialById(id: number, signal?: AbortSignal): Promise<AuthCredentialSnapshotEntry>;
554
+ /**
555
+ * Disable the credential with the given id and emit a
556
+ * {@link CredentialDisabledEvent}. Used by the auth-broker server to honour
557
+ * `POST /v1/credential/:id/disable`. Returns `false` when no such row exists.
558
+ */
559
+ disableCredentialById(id: number, disabledCause: string): boolean;
560
+ /**
561
+ * Upsert a credential into the underlying store, refresh the in-memory
562
+ * snapshot, and return the redacted snapshot entries for the provider.
563
+ *
564
+ * Used by the auth-broker server to honour `POST /v1/credential`. The
565
+ * persistence layer (`SqliteAuthCredentialStore.upsertAuthCredentialForProvider`)
566
+ * does identity-key matching, so re-uploading the same email/account replaces
567
+ * the existing row instead of inserting a duplicate.
568
+ */
569
+ upsertCredential(provider: string, credential: AuthCredential): AuthCredentialSnapshotEntry[];
570
+ /**
571
+ * Describe where the active credential for a provider came from.
572
+ *
573
+ * Surfaces four layers, highest precedence first:
574
+ * 1. Runtime override (`--api-key`).
575
+ * 2. Config override (`models.yml` `providers.<name>.apiKey`).
576
+ * 3. Stored credential (the one this session is currently sticky to, or the
577
+ * one round-robin would pick next when no session id is supplied).
578
+ * 4. Env var / fallback resolver — when no stored credential exists.
579
+ *
580
+ * The string is purely informational; consumers must not parse it.
581
+ */
582
+ describeCredentialSource(provider: string, sessionId?: string): string | undefined;
583
+ }
584
+ /**
585
+ * Default SQLite-backed implementation of {@link AuthCredentialStore}.
586
+ *
587
+ * Used by the pi-ai CLI and as the default store for `AuthStorage.create()`.
588
+ * Also exposes convenience methods (`saveOAuth`, `getOAuth`, `saveApiKey`,
589
+ * `getApiKey`, `listProviders`, `deleteProvider`) that callers can use directly
590
+ * without going through `AuthStorage`.
591
+ */
592
+ export declare class SqliteAuthCredentialStore implements AuthCredentialStore {
593
+ #private;
594
+ constructor(db: Database);
595
+ static open(dbPath?: string): Promise<SqliteAuthCredentialStore>;
596
+ listAuthCredentials(provider?: string): StoredAuthCredential[];
597
+ replaceAuthCredentialsForProvider(provider: string, credentials: AuthCredential[]): StoredAuthCredential[];
598
+ upsertAuthCredentialForProvider(provider: string, credential: AuthCredential): StoredAuthCredential[];
599
+ updateAuthCredential(id: number, credential: AuthCredential): void;
600
+ deleteAuthCredential(id: number, disabledCause: string): void;
601
+ /**
602
+ * CAS-style disable: only soft-deletes the row when its `data` column still
603
+ * matches `expectedData` and the row has not already been disabled. Used by
604
+ * the OAuth refresh-failure path to avoid clobbering a peer that rotated the
605
+ * row between our pre-check and the disable.
606
+ */
607
+ tryDisableAuthCredentialIfMatches(id: number, expectedData: string, disabledCause: string): boolean;
608
+ deleteAuthCredentialsForProvider(provider: string, disabledCause: string): void;
609
+ getCache(key: string, options?: {
610
+ includeExpired?: boolean;
611
+ }): string | null;
612
+ setCache(key: string, value: string, expiresAtSec: number): void;
613
+ cleanExpiredCache(): void;
614
+ /**
615
+ * Save OAuth credentials for a provider.
616
+ * Preserves unrelated identities and replaces only the matching credential.
617
+ */
618
+ saveOAuth(provider: string, credentials: OAuthCredentials): void;
619
+ /**
620
+ * Get OAuth credentials for a provider.
621
+ */
622
+ getOAuth(provider: string): OAuthCredentials | null;
623
+ /**
624
+ * Save API key for a provider (replaces existing).
625
+ */
626
+ saveApiKey(provider: string, apiKey: string): void;
627
+ /**
628
+ * Get API key for a provider.
629
+ */
630
+ getApiKey(provider: string): string | null;
631
+ /**
632
+ * List all providers with credentials.
633
+ */
634
+ listProviders(): string[];
635
+ /**
636
+ * Delete all credentials for a provider.
637
+ */
638
+ deleteProvider(provider: string): void;
639
+ close(): void;
640
+ }
641
+ export {};
@@ -0,0 +1,2 @@
1
+ #!/usr/bin/env bun
2
+ export {};