5 Commits

Author SHA1 Message Date
956ca21f54 docs(claude-code): mark subscription mode broken on macOS
PR #12 shipped oauthToken support with optimistic framing, but testing
revealed @anthropic-ai/claude-agent-sdk has a macOS-specific bug: the
SDK isolates CLAUDE_CONFIG_DIR per invocation and tries to copy
~/.claude/.credentials.json, which doesn't exist on macOS (creds live
in the Keychain). The OAuth token gets misclassified as an API key
and requests are billed against API credits instead of the
subscription — subscribers without API credits see "Credit balance is
too low" errors.

Direct `claude -p` works fine, so the upstream SDK is the broken layer.

Changes:
- README Claude Code section: lead with the macOS status note and
  point subscribers at ClaudeProvider + ANTHROPIC_API_KEY as the
  current workaround.
- JSDoc on ClaudeCodeProvider: same status note.
- billing_error message: explain the bug and recommend the workaround
  so users can self-diagnose from the error alone.

No code removal — when the SDK fix lands upstream, oauthToken users
will Just Work without any code changes here.
2026-05-21 22:05:48 +02:00
b99707c629 Merge pull request 'fix(claude-code): add oauthToken config for subscription SDK auth' (#12) from fix/claude-code-oauth-token into main
Reviewed-on: #12
2026-05-21 12:56:28 +00:00
b935209142 fix(claude-code): add oauthToken config for subscription SDK auth
`claude login` alone does not authorize SDK / non-interactive
(`claude -p`) usage. Anthropic gates that behind a separate long-lived
token minted by `claude setup-token`. Without it, subscription users
hit `billing_error` from the SDK even though interactive Claude Code
works fine.

Changes:

- ClaudeCodeConfig.oauthToken: new optional field. When set, the
  provider exports it as CLAUDE_CODE_OAUTH_TOKEN before invoking the
  SDK, which makes the SDK bill against the user's Pro/Max
  subscription instead of API credits.
- apiKey continues to work for users with a console API key. If both
  are present, oauthToken takes precedence.
- billing_error / authentication_failed messages now point users at
  `claude setup-token` so the fix is obvious from the error alone.
- README: rewrite the Claude Code section to document both modes and
  the `claude setup-token` step explicitly.
- examples/claude-code.ts: read CLAUDE_CODE_OAUTH_TOKEN from env so
  the smoke test actually works for subscribers.
2026-05-21 14:56:14 +02:00
d4871d22fe chore(release): 2.1.0
Minor release adding the ClaudeCodeProvider for Claude Pro/Max
subscription users (PR #10) along with a rewritten README (PR #10)
and a repaired test suite (PR #11).

New public exports:
- ClaudeCodeProvider, ClaudeCodeConfig
- createClaudeCodeProvider
- 'claude-code' added to PROVIDER_REGISTRY and SUPPORTED_PROVIDERS

No breaking changes.
2026-05-21 14:39:36 +02:00
9cb7ee8f9e Merge pull request 'test: repair test suite after R1-R3 refactors and add Claude Code tests' (#11) from chore/test-suite-repair into main
Reviewed-on: #11
2026-05-21 12:38:05 +00:00
5 changed files with 149 additions and 27 deletions

View File

@@ -52,27 +52,45 @@ const claude = new ClaudeProvider({
});
```
### Claude Code (subscription via local CLI)
### Claude Code (via local CLI)
`ClaudeCodeProvider` wraps `@anthropic-ai/claude-agent-sdk`, which authenticates through the local `claude` CLI. This is the supported path for **Claude Pro / Max subscribers** who don't have a console API key.
`ClaudeCodeProvider` wraps `@anthropic-ai/claude-agent-sdk`, which spawns the local `claude` CLI under the hood.
**Setup:** install the CLI and run `claude login` once. No API key required.
> **Honest status:** the **API-key mode** (Mode B below) works reliably. The **subscription mode** (Mode A) is currently broken on macOS due to an upstream bug in `@anthropic-ai/claude-agent-sdk` — the SDK isolates `CLAUDE_CONFIG_DIR` per invocation and tries to copy a `~/.claude/.credentials.json` file that doesn't exist on macOS (Claude Code stores credentials in the Keychain). The provider misidentifies the OAuth token and bills it as an API key, so subscribers without API credits hit "Credit balance is too low".
>
> If you have a Pro/Max subscription and need programmatic access today: use the direct [`ClaudeProvider`](#claude-anthropic-api) with an API key, or wait for the upstream SDK fix. The `oauthToken` field is wired up and ready for when Anthropic fixes the Keychain handling.
**Mode A — Claude Pro/Max subscription** *(see status note above)*:
```bash
# One-time: install the CLI, then mint a long-lived OAuth token.
# `claude login` alone is NOT enough — Anthropic gates SDK use behind setup-token.
claude setup-token
```
```typescript
import { ClaudeCodeProvider } from 'simple-ai-provider';
const claude = new ClaudeCodeProvider({}); // uses local credentials
await claude.initialize();
const response = await claude.complete({
messages: [{ role: 'user', content: 'Hello!' }]
const claude = new ClaudeCodeProvider({
oauthToken: process.env.CLAUDE_CODE_OAUTH_TOKEN // from `claude setup-token`
});
```
You can still pass `apiKey` to override (it's set as `ANTHROPIC_API_KEY` for the SDK). Optional config:
**Mode B — Console API key** (billed per-token, works reliably):
```typescript
const claude = new ClaudeCodeProvider({
apiKey: process.env.ANTHROPIC_API_KEY
});
```
If both are present, `oauthToken` wins. Either field can also be picked up from the environment (`CLAUDE_CODE_OAUTH_TOKEN` / `ANTHROPIC_API_KEY`) without being passed explicitly.
Optional config:
```typescript
new ClaudeCodeProvider({
apiKey: '...',
defaultModel: 'sonnet', // 'sonnet' | 'opus' | 'haiku' | 'inherit' | full model ID
maxTurns: 1, // 1 for plain completion; raise for agent/tool loops
allowedTools: [], // tool names to enable (default: none)
@@ -80,10 +98,11 @@ new ClaudeCodeProvider({
});
```
**Trade-offs to know:**
**Trade-offs even when it works:**
- Requires `claude` CLI installed on the host. Not ideal for typical server deployments.
- Higher latency than the direct API (spawns a CLI process per request).
- Streaming yields text as the SDK emits successive assistant messages, not token-by-token deltas.
- If you have an API key, [`ClaudeProvider`](#claude-anthropic-api) is the simpler, faster choice.
### OpenAI

61
examples/claude-code.ts Normal file
View File

@@ -0,0 +1,61 @@
/**
* Quick smoke test for ClaudeCodeProvider.
*
* Prereqs (pick one):
* - Claude Pro/Max subscription: run `claude setup-token`, then set
* CLAUDE_CODE_OAUTH_TOKEN=<the token it prints> in your environment
* (or pass it as `oauthToken` below).
* - Console API key: set ANTHROPIC_API_KEY (or pass `apiKey`).
*
* `claude login` alone is NOT sufficient for SDK use — Anthropic gates
* non-interactive (`claude -p`) invocations behind a separate token.
*
* Run: bun run examples/claude-code.ts
*/
import { ClaudeCodeProvider } from '../src/index.js';
async function main() {
const claude = new ClaudeCodeProvider({
defaultModel: 'sonnet',
oauthToken: process.env.CLAUDE_CODE_OAUTH_TOKEN
});
console.log('Initializing…');
await claude.initialize();
console.log('Ready.\n');
// ── Non-streaming completion ───────────────────────────────────────────
console.log('--- complete() ---');
const response = await claude.complete({
messages: [
{ role: 'system', content: 'You are concise. Reply in one sentence.' },
{ role: 'user', content: 'What is the capital of Japan?' }
],
maxTokens: 100
});
console.log(response.content);
console.log(`tokens: ${response.usage.totalTokens} | cost: $${response.metadata?.costUsd ?? '?'}\n`);
// ── Streaming ──────────────────────────────────────────────────────────
console.log('--- stream() ---');
let totalTokens = 0;
for await (const chunk of claude.stream({
messages: [{ role: 'user', content: 'Count from 1 to 5, one per line.' }],
maxTokens: 100
})) {
if (!chunk.isComplete) {
process.stdout.write(chunk.content);
} else {
totalTokens = chunk.usage?.totalTokens ?? 0;
}
}
console.log(`\ntokens: ${totalTokens}`);
}
main().catch(err => {
console.error('Failed:', err.message);
if (err.type) console.error('Type:', err.type);
process.exit(1);
});

View File

@@ -1,6 +1,6 @@
{
"name": "simple-ai-provider",
"version": "2.0.1",
"version": "2.1.0",
"description": "A simple and extensible AI provider package for easy integration of multiple AI services",
"main": "dist/index.js",
"module": "dist/index.mjs",

View File

@@ -60,4 +60,4 @@ export const SUPPORTED_PROVIDERS = ['claude', 'openai', 'gemini', 'openwebui', '
/**
* Package version
*/
export const VERSION = '2.0.1';
export const VERSION = '2.1.0';

View File

@@ -2,17 +2,27 @@
* Claude Code Provider
*
* Wraps `@anthropic-ai/claude-agent-sdk` so consumers can use a local
* Claude Code installation — including Claude Pro/Max subscription
* accounts authenticated via `claude login` — through the same
* BaseAIProvider interface as the other providers.
* Claude Code installation through the same BaseAIProvider interface
* as the other providers.
*
* Tradeoffs vs the direct Anthropic API provider:
* - Authenticates via the local CLI, so subscription users (no API key)
* can use it.
* - Requires `claude` to be installed and logged in on the host.
* - Higher latency (shells out to a CLI process per request).
* Status:
* - apiKey mode: works reliably.
* - oauthToken (subscription) mode: currently broken on macOS due to an
* upstream SDK bug. The SDK isolates CLAUDE_CONFIG_DIR per invocation
* and tries to copy ~/.claude/.credentials.json, which doesn't exist on
* macOS (creds live in the Keychain). The OAuth token gets misclassified
* as an API key by the SDK, so requests are billed against API credits
* instead of the subscription. Subscribers without API credits will see
* "Credit balance is too low" errors. The oauthToken field is wired up
* and will work as expected once the SDK fix lands upstream.
*
* Other tradeoffs:
* - Requires `claude` CLI installed on the host.
* - Higher latency (spawns a CLI process per request).
* - Designed for agent workflows, but used here in single-turn mode
* (maxTurns: 1, no tools) for plain text completion.
* - If you have an API key, prefer the direct ClaudeProvider for simpler,
* lower-latency access.
*
* @see https://docs.anthropic.com/claude/docs/claude-code-sdk
*/
@@ -31,12 +41,31 @@ import { AIProviderError, AIErrorType } from '../types/index.js';
import { DEFAULT_MODELS } from '../constants.js';
/**
* Configuration for the Claude Code provider. `apiKey` is optional —
* when omitted the SDK falls back to ANTHROPIC_API_KEY in the
* environment or the local subscription credentials managed by the
* `claude` CLI.
* Configuration for the Claude Code provider.
*
* Authentication options, in priority order:
*
* 1. `oauthToken` — long-lived subscription token generated by
* `claude setup-token` (recommended for Pro/Max subscribers using
* the SDK programmatically). Billed against the subscription.
* 2. `apiKey` — standard Anthropic console API key. Billed per-token.
* 3. Environment fallback — the SDK reads `CLAUDE_CODE_OAUTH_TOKEN` or
* `ANTHROPIC_API_KEY` from the environment if neither field is set.
*
* Note: `claude login` alone is not sufficient for SDK use because
* Anthropic gates non-interactive (`claude -p`) invocations behind a
* separate token. Run `claude setup-token` once to mint one.
*/
export interface ClaudeCodeConfig extends Omit<AIProviderConfig, 'apiKey'> {
/**
* Long-lived OAuth token from `claude setup-token`. This is the
* supported path for Claude Pro/Max subscribers to use the SDK
* programmatically — billed against the subscription instead of
* per-token.
*/
oauthToken?: string;
/** Standard Anthropic console API key. Mutually compatible with oauthToken (oauthToken takes precedence). */
apiKey?: string;
/**
@@ -71,6 +100,7 @@ export class ClaudeCodeProvider extends BaseAIProvider {
private readonly cwd: string | undefined;
private readonly allowedTools: string[];
private readonly maxTurns: number;
private readonly oauthToken: string | undefined;
constructor(config: ClaudeCodeConfig) {
super(config as AIProviderConfig);
@@ -79,6 +109,7 @@ export class ClaudeCodeProvider extends BaseAIProvider {
this.cwd = config.cwd;
this.allowedTools = config.allowedTools ?? [];
this.maxTurns = config.maxTurns ?? 1;
this.oauthToken = config.oauthToken;
}
/**
@@ -95,13 +126,21 @@ export class ClaudeCodeProvider extends BaseAIProvider {
);
}
if (config.apiKey !== undefined && (typeof config.apiKey !== 'string')) {
if (config.apiKey !== undefined && typeof config.apiKey !== 'string') {
throw new AIProviderError(
'API key, when provided, must be a string',
AIErrorType.INVALID_REQUEST
);
}
const oauthToken = (config as ClaudeCodeConfig).oauthToken;
if (oauthToken !== undefined && typeof oauthToken !== 'string') {
throw new AIProviderError(
'oauthToken, when provided, must be a string',
AIErrorType.INVALID_REQUEST
);
}
return {
...config,
apiKey: config.apiKey ?? '',
@@ -114,6 +153,9 @@ export class ClaudeCodeProvider extends BaseAIProvider {
// The SDK lazy-spawns the CLI per query; no connection probe is
// necessary (and a probe would cost a real billed turn). We defer
// auth/model errors to the first complete()/stream() call.
if (this.oauthToken) {
process.env.CLAUDE_CODE_OAUTH_TOKEN = this.oauthToken;
}
if (this.config.apiKey) {
process.env.ANTHROPIC_API_KEY = this.config.apiKey;
}
@@ -322,7 +364,7 @@ export class ClaudeCodeProvider extends BaseAIProvider {
const map: Record<string, { type: AIErrorType; message: string }> = {
authentication_failed: {
type: AIErrorType.AUTHENTICATION,
message: 'Claude Code authentication failed. Run `claude login` or set ANTHROPIC_API_KEY.'
message: 'Claude Code authentication failed. For subscriptions run `claude setup-token` and pass the result as `oauthToken`. Otherwise set ANTHROPIC_API_KEY or pass `apiKey`.'
},
oauth_org_not_allowed: {
type: AIErrorType.AUTHENTICATION,
@@ -330,7 +372,7 @@ export class ClaudeCodeProvider extends BaseAIProvider {
},
billing_error: {
type: AIErrorType.AUTHENTICATION,
message: 'Billing error. Check your Anthropic account billing or subscription status.'
message: 'Billing error from Claude Code SDK. If you are a Pro/Max subscriber on macOS, this is likely the upstream @anthropic-ai/claude-agent-sdk Keychain bug — the SDK misclassifies the OAuth token as an API key. Workaround: use ClaudeProvider directly with an ANTHROPIC_API_KEY instead.'
},
rate_limit: {
type: AIErrorType.RATE_LIMIT,