@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.
- package/CHANGELOG.md +2644 -0
- package/README.md +1181 -0
- package/dist/types/api-registry.d.ts +30 -0
- package/dist/types/auth-broker/client.d.ts +66 -0
- package/dist/types/auth-broker/index.d.ts +5 -0
- package/dist/types/auth-broker/refresher.d.ts +25 -0
- package/dist/types/auth-broker/remote-store.d.ts +96 -0
- package/dist/types/auth-broker/server.d.ts +32 -0
- package/dist/types/auth-broker/types.d.ts +105 -0
- package/dist/types/auth-broker/wire-schemas.d.ts +412 -0
- package/dist/types/auth-gateway/http.d.ts +39 -0
- package/dist/types/auth-gateway/index.d.ts +3 -0
- package/dist/types/auth-gateway/server.d.ts +17 -0
- package/dist/types/auth-gateway/types.d.ts +115 -0
- package/dist/types/auth-storage.d.ts +641 -0
- package/dist/types/cli.d.ts +2 -0
- package/dist/types/index.d.ts +49 -0
- package/dist/types/model-cache.d.ts +17 -0
- package/dist/types/model-manager.d.ts +62 -0
- package/dist/types/model-thinking.d.ts +71 -0
- package/dist/types/models.d.ts +12 -0
- package/dist/types/provider-details.d.ts +24 -0
- package/dist/types/provider-models/bundled-references.d.ts +4 -0
- package/dist/types/provider-models/descriptors.d.ts +48 -0
- package/dist/types/provider-models/google.d.ts +20 -0
- package/dist/types/provider-models/index.d.ts +5 -0
- package/dist/types/provider-models/ollama.d.ts +7 -0
- package/dist/types/provider-models/openai-compat.d.ts +237 -0
- package/dist/types/provider-models/special.d.ts +16 -0
- package/dist/types/providers/amazon-bedrock.d.ts +36 -0
- package/dist/types/providers/anthropic-messages-server-schema.d.ts +450 -0
- package/dist/types/providers/anthropic-messages-server.d.ts +17 -0
- package/dist/types/providers/anthropic.d.ts +188 -0
- package/dist/types/providers/aws-credentials.d.ts +43 -0
- package/dist/types/providers/aws-eventstream.d.ts +38 -0
- package/dist/types/providers/aws-sigv4.d.ts +55 -0
- package/dist/types/providers/azure-openai-responses.d.ts +15 -0
- package/dist/types/providers/cursor/gen/agent_pb.d.ts +13022 -0
- package/dist/types/providers/cursor.d.ts +42 -0
- package/dist/types/providers/error-message.d.ts +27 -0
- package/dist/types/providers/github-copilot-headers.d.ts +40 -0
- package/dist/types/providers/gitlab-duo.d.ts +27 -0
- package/dist/types/providers/google-auth.d.ts +24 -0
- package/dist/types/providers/google-gemini-cli.d.ts +72 -0
- package/dist/types/providers/google-gemini-headers.d.ts +18 -0
- package/dist/types/providers/google-shared.d.ts +163 -0
- package/dist/types/providers/google-types.d.ts +138 -0
- package/dist/types/providers/google-vertex.d.ts +7 -0
- package/dist/types/providers/google.d.ts +4 -0
- package/dist/types/providers/grammar.d.ts +1 -0
- package/dist/types/providers/kimi.d.ts +27 -0
- package/dist/types/providers/mock.d.ts +175 -0
- package/dist/types/providers/ollama.d.ts +6 -0
- package/dist/types/providers/openai-anthropic-shim.d.ts +31 -0
- package/dist/types/providers/openai-chat-server-schema.d.ts +814 -0
- package/dist/types/providers/openai-chat-server.d.ts +16 -0
- package/dist/types/providers/openai-codex/constants.d.ts +26 -0
- package/dist/types/providers/openai-codex/request-transformer.d.ts +49 -0
- package/dist/types/providers/openai-codex/response-handler.d.ts +17 -0
- package/dist/types/providers/openai-codex-responses.d.ts +67 -0
- package/dist/types/providers/openai-completions-compat.d.ts +25 -0
- package/dist/types/providers/openai-completions.d.ts +33 -0
- package/dist/types/providers/openai-responses-server-schema.d.ts +392 -0
- package/dist/types/providers/openai-responses-server.d.ts +17 -0
- package/dist/types/providers/openai-responses-shared.d.ts +89 -0
- package/dist/types/providers/openai-responses.d.ts +32 -0
- package/dist/types/providers/pi-native-client.d.ts +13 -0
- package/dist/types/providers/pi-native-server.d.ts +68 -0
- package/dist/types/providers/register-builtins.d.ts +31 -0
- package/dist/types/providers/synthetic.d.ts +26 -0
- package/dist/types/providers/transform-messages.d.ts +12 -0
- package/dist/types/providers/vision-guard.d.ts +8 -0
- package/dist/types/rate-limit-utils.d.ts +19 -0
- package/dist/types/stream.d.ts +24 -0
- package/dist/types/types.d.ts +746 -0
- package/dist/types/usage/claude.d.ts +3 -0
- package/dist/types/usage/gemini.d.ts +2 -0
- package/dist/types/usage/github-copilot.d.ts +7 -0
- package/dist/types/usage/google-antigravity.d.ts +2 -0
- package/dist/types/usage/kimi.d.ts +2 -0
- package/dist/types/usage/minimax-code.d.ts +2 -0
- package/dist/types/usage/openai-codex.d.ts +3 -0
- package/dist/types/usage/shared.d.ts +1 -0
- package/dist/types/usage/zai.d.ts +2 -0
- package/dist/types/usage.d.ts +258 -0
- package/dist/types/utils/abort.d.ts +19 -0
- package/dist/types/utils/anthropic-auth.d.ts +31 -0
- package/dist/types/utils/discovery/antigravity.d.ts +61 -0
- package/dist/types/utils/discovery/codex.d.ts +38 -0
- package/dist/types/utils/discovery/cursor.d.ts +23 -0
- package/dist/types/utils/discovery/gemini.d.ts +25 -0
- package/dist/types/utils/discovery/index.d.ts +4 -0
- package/dist/types/utils/discovery/openai-compatible.d.ts +72 -0
- package/dist/types/utils/event-stream.d.ts +28 -0
- package/dist/types/utils/fireworks-model-id.d.ts +10 -0
- package/dist/types/utils/foundry.d.ts +1 -0
- package/dist/types/utils/h2-fetch.d.ts +22 -0
- package/dist/types/utils/http-inspector.d.ts +31 -0
- package/dist/types/utils/idle-iterator.d.ts +67 -0
- package/dist/types/utils/json-parse.d.ts +10 -0
- package/dist/types/utils/oauth/alibaba-coding-plan.d.ts +18 -0
- package/dist/types/utils/oauth/anthropic.d.ts +22 -0
- package/dist/types/utils/oauth/api-key-login.d.ts +35 -0
- package/dist/types/utils/oauth/api-key-validation.d.ts +27 -0
- package/dist/types/utils/oauth/callback-server.d.ts +57 -0
- package/dist/types/utils/oauth/cerebras.d.ts +1 -0
- package/dist/types/utils/oauth/cloudflare-ai-gateway.d.ts +18 -0
- package/dist/types/utils/oauth/cursor.d.ts +15 -0
- package/dist/types/utils/oauth/deepseek.d.ts +10 -0
- package/dist/types/utils/oauth/firepass.d.ts +1 -0
- package/dist/types/utils/oauth/fireworks.d.ts +1 -0
- package/dist/types/utils/oauth/github-copilot.d.ts +38 -0
- package/dist/types/utils/oauth/gitlab-duo.d.ts +3 -0
- package/dist/types/utils/oauth/google-antigravity.d.ts +11 -0
- package/dist/types/utils/oauth/google-gemini-cli.d.ts +10 -0
- package/dist/types/utils/oauth/google-oauth-shared.d.ts +28 -0
- package/dist/types/utils/oauth/huggingface.d.ts +19 -0
- package/dist/types/utils/oauth/index.d.ts +38 -0
- package/dist/types/utils/oauth/kagi.d.ts +17 -0
- package/dist/types/utils/oauth/kilo.d.ts +5 -0
- package/dist/types/utils/oauth/kimi.d.ts +21 -0
- package/dist/types/utils/oauth/litellm.d.ts +18 -0
- package/dist/types/utils/oauth/lm-studio.d.ts +17 -0
- package/dist/types/utils/oauth/minimax-code.d.ts +28 -0
- package/dist/types/utils/oauth/moonshot.d.ts +1 -0
- package/dist/types/utils/oauth/nanogpt.d.ts +1 -0
- package/dist/types/utils/oauth/nvidia.d.ts +18 -0
- package/dist/types/utils/oauth/ollama-cloud.d.ts +2 -0
- package/dist/types/utils/oauth/ollama.d.ts +18 -0
- package/dist/types/utils/oauth/openai-codex.d.ts +21 -0
- package/dist/types/utils/oauth/opencode.d.ts +18 -0
- package/dist/types/utils/oauth/parallel.d.ts +17 -0
- package/dist/types/utils/oauth/perplexity.d.ts +9 -0
- package/dist/types/utils/oauth/pkce.d.ts +8 -0
- package/dist/types/utils/oauth/qianfan.d.ts +17 -0
- package/dist/types/utils/oauth/qwen-portal.d.ts +19 -0
- package/dist/types/utils/oauth/synthetic.d.ts +1 -0
- package/dist/types/utils/oauth/tavily.d.ts +17 -0
- package/dist/types/utils/oauth/together.d.ts +1 -0
- package/dist/types/utils/oauth/types.d.ts +44 -0
- package/dist/types/utils/oauth/venice.d.ts +18 -0
- package/dist/types/utils/oauth/vercel-ai-gateway.d.ts +18 -0
- package/dist/types/utils/oauth/vllm.d.ts +16 -0
- package/dist/types/utils/oauth/xiaomi.d.ts +19 -0
- package/dist/types/utils/oauth/zai.d.ts +18 -0
- package/dist/types/utils/oauth/zenmux.d.ts +1 -0
- package/dist/types/utils/overflow.d.ts +54 -0
- package/dist/types/utils/parse-bind.d.ts +23 -0
- package/dist/types/utils/provider-response.d.ts +3 -0
- package/dist/types/utils/retry-after.d.ts +3 -0
- package/dist/types/utils/retry.d.ts +26 -0
- package/dist/types/utils/schema/adapt.d.ts +24 -0
- package/dist/types/utils/schema/compatibility.d.ts +30 -0
- package/dist/types/utils/schema/dereference.d.ts +11 -0
- package/dist/types/utils/schema/draft.d.ts +10 -0
- package/dist/types/utils/schema/equality.d.ts +4 -0
- package/dist/types/utils/schema/fields.d.ts +49 -0
- package/dist/types/utils/schema/index.d.ts +13 -0
- package/dist/types/utils/schema/json-schema-validator.d.ts +12 -0
- package/dist/types/utils/schema/meta-validator.d.ts +2 -0
- package/dist/types/utils/schema/normalize.d.ts +93 -0
- package/dist/types/utils/schema/spill.d.ts +8 -0
- package/dist/types/utils/schema/stamps.d.ts +25 -0
- package/dist/types/utils/schema/types.d.ts +4 -0
- package/dist/types/utils/schema/wire.d.ts +54 -0
- package/dist/types/utils/schema/zod-decontaminate.d.ts +31 -0
- package/dist/types/utils/sse-debug.d.ts +10 -0
- package/dist/types/utils/tool-call-healing.d.ts +71 -0
- package/dist/types/utils/tool-choice.d.ts +50 -0
- package/dist/types/utils/validation.d.ts +17 -0
- package/dist/types/utils.d.ts +28 -0
- package/package.json +146 -0
- package/src/api-registry.ts +96 -0
- package/src/auth-broker/client.ts +358 -0
- package/src/auth-broker/index.ts +5 -0
- package/src/auth-broker/refresher.ts +127 -0
- package/src/auth-broker/remote-store.ts +623 -0
- package/src/auth-broker/server.ts +644 -0
- package/src/auth-broker/types.ts +127 -0
- package/src/auth-broker/wire-schemas.ts +200 -0
- package/src/auth-gateway/http.ts +194 -0
- package/src/auth-gateway/index.ts +3 -0
- package/src/auth-gateway/server.ts +717 -0
- package/src/auth-gateway/types.ts +134 -0
- package/src/auth-storage.ts +4104 -0
- package/src/cli.ts +262 -0
- package/src/index.ts +54 -0
- package/src/model-cache.ts +129 -0
- package/src/model-manager.ts +450 -0
- package/src/model-thinking.ts +691 -0
- package/src/models.json +73853 -0
- package/src/models.json.d.ts +9 -0
- package/src/models.ts +56 -0
- package/src/prompts/turn-aborted-guidance.md +4 -0
- package/src/provider-details.ts +90 -0
- package/src/provider-models/bundled-references.ts +38 -0
- package/src/provider-models/descriptors.ts +308 -0
- package/src/provider-models/google.ts +91 -0
- package/src/provider-models/index.ts +5 -0
- package/src/provider-models/ollama.ts +153 -0
- package/src/provider-models/openai-compat.ts +2275 -0
- package/src/provider-models/special.ts +67 -0
- package/src/providers/amazon-bedrock.ts +849 -0
- package/src/providers/anthropic-messages-server-schema.ts +229 -0
- package/src/providers/anthropic-messages-server.ts +677 -0
- package/src/providers/anthropic.ts +2696 -0
- package/src/providers/aws-credentials.ts +501 -0
- package/src/providers/aws-eventstream.ts +185 -0
- package/src/providers/aws-sigv4.ts +218 -0
- package/src/providers/azure-openai-responses.ts +337 -0
- package/src/providers/cursor/gen/agent_pb.ts +15274 -0
- package/src/providers/cursor/proto/agent.proto +3526 -0
- package/src/providers/cursor/proto/buf.gen.yaml +6 -0
- package/src/providers/cursor/proto/buf.yaml +17 -0
- package/src/providers/cursor.ts +2561 -0
- package/src/providers/error-message.ts +21 -0
- package/src/providers/github-copilot-headers.ts +140 -0
- package/src/providers/gitlab-duo.ts +372 -0
- package/src/providers/google-auth.ts +252 -0
- package/src/providers/google-gemini-cli.ts +795 -0
- package/src/providers/google-gemini-headers.ts +41 -0
- package/src/providers/google-shared.ts +902 -0
- package/src/providers/google-types.ts +167 -0
- package/src/providers/google-vertex.ts +88 -0
- package/src/providers/google.ts +41 -0
- package/src/providers/grammar.ts +70 -0
- package/src/providers/kimi.ts +52 -0
- package/src/providers/mock.ts +500 -0
- package/src/providers/ollama.ts +544 -0
- package/src/providers/openai-anthropic-shim.ts +138 -0
- package/src/providers/openai-chat-server-schema.ts +243 -0
- package/src/providers/openai-chat-server.ts +628 -0
- package/src/providers/openai-codex/constants.ts +43 -0
- package/src/providers/openai-codex/request-transformer.ts +161 -0
- package/src/providers/openai-codex/response-handler.ts +81 -0
- package/src/providers/openai-codex-responses.ts +2598 -0
- package/src/providers/openai-completions-compat.ts +279 -0
- package/src/providers/openai-completions.ts +1853 -0
- package/src/providers/openai-responses-server-schema.ts +290 -0
- package/src/providers/openai-responses-server.ts +1183 -0
- package/src/providers/openai-responses-shared.ts +800 -0
- package/src/providers/openai-responses.ts +621 -0
- package/src/providers/pi-native-client.ts +228 -0
- package/src/providers/pi-native-server.ts +210 -0
- package/src/providers/register-builtins.ts +412 -0
- package/src/providers/synthetic.ts +50 -0
- package/src/providers/transform-messages.ts +309 -0
- package/src/providers/vision-guard.ts +31 -0
- package/src/rate-limit-utils.ts +84 -0
- package/src/stream.ts +895 -0
- package/src/types.ts +884 -0
- package/src/usage/claude.ts +431 -0
- package/src/usage/gemini.ts +250 -0
- package/src/usage/github-copilot.ts +421 -0
- package/src/usage/google-antigravity.ts +201 -0
- package/src/usage/kimi.ts +271 -0
- package/src/usage/minimax-code.ts +31 -0
- package/src/usage/openai-codex.ts +503 -0
- package/src/usage/shared.ts +10 -0
- package/src/usage/zai.ts +247 -0
- package/src/usage.ts +183 -0
- package/src/utils/abort.ts +51 -0
- package/src/utils/anthropic-auth.ts +87 -0
- package/src/utils/discovery/antigravity.ts +261 -0
- package/src/utils/discovery/codex.ts +371 -0
- package/src/utils/discovery/cursor.ts +306 -0
- package/src/utils/discovery/gemini.ts +248 -0
- package/src/utils/discovery/index.ts +4 -0
- package/src/utils/discovery/openai-compatible.ts +224 -0
- package/src/utils/event-stream.ts +142 -0
- package/src/utils/fireworks-model-id.ts +30 -0
- package/src/utils/foundry.ts +8 -0
- package/src/utils/h2-fetch.ts +60 -0
- package/src/utils/http-inspector.ts +176 -0
- package/src/utils/idle-iterator.ts +250 -0
- package/src/utils/json-parse.ts +148 -0
- package/src/utils/oauth/alibaba-coding-plan.ts +59 -0
- package/src/utils/oauth/anthropic.ts +200 -0
- package/src/utils/oauth/api-key-login.ts +87 -0
- package/src/utils/oauth/api-key-validation.ts +92 -0
- package/src/utils/oauth/callback-server.ts +276 -0
- package/src/utils/oauth/cerebras.ts +16 -0
- package/src/utils/oauth/cloudflare-ai-gateway.ts +48 -0
- package/src/utils/oauth/cursor.ts +157 -0
- package/src/utils/oauth/deepseek.ts +53 -0
- package/src/utils/oauth/firepass.ts +24 -0
- package/src/utils/oauth/fireworks.ts +15 -0
- package/src/utils/oauth/github-copilot.ts +362 -0
- package/src/utils/oauth/gitlab-duo.ts +123 -0
- package/src/utils/oauth/google-antigravity.ts +200 -0
- package/src/utils/oauth/google-gemini-cli.ts +256 -0
- package/src/utils/oauth/google-oauth-shared.ts +110 -0
- package/src/utils/oauth/huggingface.ts +62 -0
- package/src/utils/oauth/index.ts +444 -0
- package/src/utils/oauth/kagi.ts +47 -0
- package/src/utils/oauth/kilo.ts +87 -0
- package/src/utils/oauth/kimi.ts +254 -0
- package/src/utils/oauth/litellm.ts +47 -0
- package/src/utils/oauth/lm-studio.ts +38 -0
- package/src/utils/oauth/minimax-code.ts +78 -0
- package/src/utils/oauth/moonshot.ts +16 -0
- package/src/utils/oauth/nanogpt.ts +15 -0
- package/src/utils/oauth/nvidia.ts +70 -0
- package/src/utils/oauth/oauth.html +199 -0
- package/src/utils/oauth/ollama-cloud.ts +28 -0
- package/src/utils/oauth/ollama.ts +47 -0
- package/src/utils/oauth/openai-codex.ts +299 -0
- package/src/utils/oauth/opencode.ts +49 -0
- package/src/utils/oauth/parallel.ts +46 -0
- package/src/utils/oauth/perplexity.ts +206 -0
- package/src/utils/oauth/pkce.ts +18 -0
- package/src/utils/oauth/qianfan.ts +58 -0
- package/src/utils/oauth/qwen-portal.ts +60 -0
- package/src/utils/oauth/synthetic.ts +16 -0
- package/src/utils/oauth/tavily.ts +46 -0
- package/src/utils/oauth/together.ts +16 -0
- package/src/utils/oauth/types.ts +94 -0
- package/src/utils/oauth/venice.ts +59 -0
- package/src/utils/oauth/vercel-ai-gateway.ts +47 -0
- package/src/utils/oauth/vllm.ts +40 -0
- package/src/utils/oauth/xiaomi.ts +137 -0
- package/src/utils/oauth/zai.ts +60 -0
- package/src/utils/oauth/zenmux.ts +15 -0
- package/src/utils/overflow.ts +137 -0
- package/src/utils/parse-bind.ts +54 -0
- package/src/utils/provider-response.ts +30 -0
- package/src/utils/retry-after.ts +110 -0
- package/src/utils/retry.ts +54 -0
- package/src/utils/schema/CONSTRAINTS.md +164 -0
- package/src/utils/schema/adapt.ts +36 -0
- package/src/utils/schema/compatibility.ts +435 -0
- package/src/utils/schema/dereference.ts +98 -0
- package/src/utils/schema/draft.ts +341 -0
- package/src/utils/schema/equality.ts +97 -0
- package/src/utils/schema/fields.ts +190 -0
- package/src/utils/schema/index.ts +13 -0
- package/src/utils/schema/json-schema-validator.ts +577 -0
- package/src/utils/schema/meta-validator.ts +167 -0
- package/src/utils/schema/normalize.ts +1588 -0
- package/src/utils/schema/spill.ts +43 -0
- package/src/utils/schema/stamps.ts +97 -0
- package/src/utils/schema/types.ts +11 -0
- package/src/utils/schema/wire.ts +213 -0
- package/src/utils/schema/zod-decontaminate.ts +331 -0
- package/src/utils/sse-debug.ts +289 -0
- package/src/utils/tool-call-healing.ts +271 -0
- package/src/utils/tool-choice.ts +99 -0
- package/src/utils/validation.ts +1019 -0
- 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 {};
|