mirror of
https://github.com/decolua/9router.git
synced 2026-05-08 12:01:28 +00:00
- Add claudeHeaderCache.js to intercept and cache live Claude Code client headers - Forward cached headers dynamically to api.anthropic.com via default.js - Strip first-party identity headers (x-app, claude-code-* beta) for non-Anthropic upstreams - Validate and sanitize tool call IDs to match Anthropic pattern (^[a-zA-Z0-9_-]+$) - Skip thinking blocks when applying cache_control; fix max_tokens buffer (+1024) - Strip cache_control from thinking blocks in openai-to-claude translator - Comment out thoughtSignature in Gemini translator (kept for reference) - Expand .gitignore to match all deploy*.sh variants Co-authored-by: kwanLeeFrmVi <quanle96@outlook.com> Closes #433 Made-with: Cursor
418 lines
15 KiB
JavaScript
418 lines
15 KiB
JavaScript
/**
|
|
* Unit tests for Anthropic header caching + forwarding pipeline
|
|
*
|
|
* Tests cover:
|
|
* - claudeHeaderCache: detection, capture, and retrieval of Claude Code headers
|
|
* - default.js buildHeaders(): live header overlay for "claude" provider
|
|
* - default.js buildHeaders(): cold-start fallback when cache is empty
|
|
* - default.js buildHeaders(): anthropic-compatible non-Anthropic host stripping
|
|
* - default.js buildHeaders(): anthropic-compatible official host keeps headers
|
|
* - proxyFetch.js: api.anthropic.com routes through anthropicFetch path
|
|
*/
|
|
|
|
import { describe, it, expect, vi, beforeEach, afterEach } from "vitest";
|
|
|
|
// ─── claudeHeaderCache ────────────────────────────────────────────────────────
|
|
|
|
describe("claudeHeaderCache", () => {
|
|
let cacheModule;
|
|
|
|
beforeEach(async () => {
|
|
// Re-import fresh module each time to reset singleton state
|
|
vi.resetModules();
|
|
cacheModule = await import("open-sse/utils/claudeHeaderCache.js");
|
|
});
|
|
|
|
it("returns null before any headers are cached (cold start)", () => {
|
|
expect(cacheModule.getCachedClaudeHeaders()).toBeNull();
|
|
});
|
|
|
|
it("caches headers when user-agent contains 'claude-code'", () => {
|
|
cacheModule.cacheClaudeHeaders({
|
|
"user-agent": "claude-code/2.1.63 node/24.3.0",
|
|
"anthropic-beta": "claude-code-20250219,oauth-2025-04-20",
|
|
"anthropic-version": "2023-06-01",
|
|
"x-app": "cli",
|
|
"x-stainless-os": "MacOS",
|
|
"x-stainless-arch": "arm64",
|
|
"x-stainless-lang": "js",
|
|
"x-stainless-runtime": "node",
|
|
"x-stainless-runtime-version": "v24.3.0",
|
|
"x-stainless-package-version": "0.74.0",
|
|
"x-stainless-helper-method": "stream",
|
|
"x-stainless-retry-count": "0",
|
|
"x-stainless-timeout": "600",
|
|
"anthropic-dangerous-direct-browser-access": "true",
|
|
// Non-identity header — should NOT be captured
|
|
"content-type": "application/json",
|
|
});
|
|
|
|
const cached = cacheModule.getCachedClaudeHeaders();
|
|
expect(cached).not.toBeNull();
|
|
expect(cached["user-agent"]).toBe("claude-code/2.1.63 node/24.3.0");
|
|
expect(cached["anthropic-beta"]).toBe("claude-code-20250219,oauth-2025-04-20");
|
|
expect(cached["x-app"]).toBe("cli");
|
|
expect(cached["x-stainless-os"]).toBe("MacOS");
|
|
// Non-identity header must not leak in
|
|
expect(cached["content-type"]).toBeUndefined();
|
|
});
|
|
|
|
it("caches headers when user-agent contains 'claude-cli'", () => {
|
|
cacheModule.cacheClaudeHeaders({
|
|
"user-agent": "claude-cli/1.0.0",
|
|
"anthropic-version": "2023-06-01",
|
|
});
|
|
expect(cacheModule.getCachedClaudeHeaders()).not.toBeNull();
|
|
expect(cacheModule.getCachedClaudeHeaders()["user-agent"]).toBe("claude-cli/1.0.0");
|
|
});
|
|
|
|
it("caches headers when x-app is 'cli' (regardless of user-agent)", () => {
|
|
cacheModule.cacheClaudeHeaders({
|
|
"user-agent": "axios/1.7.0",
|
|
"x-app": "cli",
|
|
"anthropic-version": "2023-06-01",
|
|
});
|
|
expect(cacheModule.getCachedClaudeHeaders()).not.toBeNull();
|
|
});
|
|
|
|
it("does NOT cache headers for non-Claude clients", () => {
|
|
cacheModule.cacheClaudeHeaders({
|
|
"user-agent": "PostmanRuntime/7.43.0",
|
|
"anthropic-version": "2023-06-01",
|
|
});
|
|
expect(cacheModule.getCachedClaudeHeaders()).toBeNull();
|
|
});
|
|
|
|
it("refreshes cache on each matching request", () => {
|
|
cacheModule.cacheClaudeHeaders({
|
|
"user-agent": "claude-code/2.0.0",
|
|
"x-stainless-package-version": "0.70.0",
|
|
});
|
|
cacheModule.cacheClaudeHeaders({
|
|
"user-agent": "claude-code/2.1.63",
|
|
"x-stainless-package-version": "0.74.0",
|
|
});
|
|
const cached = cacheModule.getCachedClaudeHeaders();
|
|
expect(cached["user-agent"]).toBe("claude-code/2.1.63");
|
|
expect(cached["x-stainless-package-version"]).toBe("0.74.0");
|
|
});
|
|
|
|
it("ignores calls with null or non-object headers", () => {
|
|
cacheModule.cacheClaudeHeaders(null);
|
|
cacheModule.cacheClaudeHeaders(undefined);
|
|
cacheModule.cacheClaudeHeaders("string");
|
|
expect(cacheModule.getCachedClaudeHeaders()).toBeNull();
|
|
});
|
|
|
|
it("only stores keys that are actually present in the headers object", () => {
|
|
cacheModule.cacheClaudeHeaders({
|
|
"user-agent": "claude-code/2.1.63",
|
|
// Most stainless headers absent
|
|
});
|
|
const cached = cacheModule.getCachedClaudeHeaders();
|
|
expect(cached["x-stainless-os"]).toBeUndefined();
|
|
expect(cached["user-agent"]).toBe("claude-code/2.1.63");
|
|
});
|
|
});
|
|
|
|
// ─── DefaultExecutor.buildHeaders() ──────────────────────────────────────────
|
|
|
|
describe("DefaultExecutor.buildHeaders() — claude provider", () => {
|
|
let DefaultExecutor;
|
|
|
|
beforeEach(async () => {
|
|
vi.resetModules();
|
|
// Prime the cache with live client headers before importing executor
|
|
const cache = await import("open-sse/utils/claudeHeaderCache.js");
|
|
cache.cacheClaudeHeaders({
|
|
"user-agent": "claude-code/2.1.63 node/24.3.0",
|
|
"anthropic-beta": "claude-code-20250219,oauth-2025-04-20,interleaved-thinking-2025-05-14",
|
|
"anthropic-version": "2023-06-01",
|
|
"anthropic-dangerous-direct-browser-access": "true",
|
|
"x-app": "cli",
|
|
"x-stainless-os": "MacOS",
|
|
"x-stainless-arch": "arm64",
|
|
"x-stainless-lang": "js",
|
|
"x-stainless-runtime": "node",
|
|
"x-stainless-runtime-version": "v24.3.0",
|
|
"x-stainless-package-version": "0.74.0",
|
|
"x-stainless-helper-method": "stream",
|
|
"x-stainless-retry-count": "0",
|
|
"x-stainless-timeout": "600",
|
|
});
|
|
const mod = await import("open-sse/executors/default.js");
|
|
DefaultExecutor = mod.DefaultExecutor || mod.default;
|
|
});
|
|
|
|
it("overlays live cached headers over static provider defaults", () => {
|
|
const executor = new DefaultExecutor("claude");
|
|
const headers = executor.buildHeaders({ apiKey: "sk-test" }, true);
|
|
|
|
// Live values should win over static providers.js values
|
|
expect(headers["user-agent"]).toBe("claude-code/2.1.63 node/24.3.0");
|
|
// Beta flags are MERGED (static + cached) to preserve required flags like oauth
|
|
const betaFlags = headers["anthropic-beta"].split(",").map(s => s.trim());
|
|
expect(betaFlags).toContain("claude-code-20250219");
|
|
expect(betaFlags).toContain("oauth-2025-04-20");
|
|
expect(betaFlags).toContain("interleaved-thinking-2025-05-14");
|
|
expect(headers["x-stainless-package-version"]).toBe("0.74.0");
|
|
expect(headers["x-stainless-os"]).toBe("MacOS");
|
|
});
|
|
|
|
it("removes conflicting Title-Case static keys when cached lowercase keys exist", () => {
|
|
const executor = new DefaultExecutor("claude");
|
|
const headers = executor.buildHeaders({ apiKey: "sk-test" }, true);
|
|
|
|
// Title-Case variants from providers.js must be gone
|
|
expect(headers["Anthropic-Version"]).toBeUndefined();
|
|
expect(headers["Anthropic-Beta"]).toBeUndefined();
|
|
expect(headers["User-Agent"]).toBeUndefined();
|
|
expect(headers["X-App"]).toBeUndefined();
|
|
// Lowercase variants must be present
|
|
expect(headers["anthropic-version"]).toBe("2023-06-01");
|
|
expect(headers["x-app"]).toBe("cli");
|
|
});
|
|
|
|
it("sets x-api-key auth when apiKey is provided", () => {
|
|
const executor = new DefaultExecutor("claude");
|
|
const headers = executor.buildHeaders({ apiKey: "sk-live-key" }, true);
|
|
expect(headers["x-api-key"]).toBe("sk-live-key");
|
|
expect(headers["Authorization"]).toBeUndefined();
|
|
});
|
|
|
|
it("sets Bearer Authorization when only accessToken is provided", () => {
|
|
const executor = new DefaultExecutor("claude");
|
|
const headers = executor.buildHeaders({ accessToken: "tok-abc" }, true);
|
|
expect(headers["Authorization"]).toBe("Bearer tok-abc");
|
|
expect(headers["x-api-key"]).toBeUndefined();
|
|
});
|
|
|
|
it("includes Accept: text/event-stream when stream=true", () => {
|
|
const executor = new DefaultExecutor("claude");
|
|
const headers = executor.buildHeaders({ apiKey: "k" }, true);
|
|
expect(headers["Accept"]).toBe("text/event-stream");
|
|
});
|
|
|
|
it("omits Accept: text/event-stream when stream=false", () => {
|
|
const executor = new DefaultExecutor("claude");
|
|
const headers = executor.buildHeaders({ apiKey: "k" }, false);
|
|
expect(headers["Accept"]).toBeUndefined();
|
|
});
|
|
});
|
|
|
|
describe("DefaultExecutor.buildHeaders() — claude provider cold start (no cache)", () => {
|
|
let DefaultExecutor;
|
|
|
|
beforeEach(async () => {
|
|
vi.resetModules();
|
|
// Do NOT prime cache — simulate cold start
|
|
const mod = await import("open-sse/executors/default.js");
|
|
DefaultExecutor = mod.DefaultExecutor || mod.default;
|
|
});
|
|
|
|
it("falls back to static provider headers when cache is empty", () => {
|
|
const executor = new DefaultExecutor("claude");
|
|
const headers = executor.buildHeaders({ apiKey: "sk-test" }, true);
|
|
|
|
// Static fallback values from providers.js must still be present
|
|
// They may be Title-Case since no cache to conflict with them
|
|
const hasVersion =
|
|
headers["Anthropic-Version"] === "2023-06-01" ||
|
|
headers["anthropic-version"] === "2023-06-01";
|
|
expect(hasVersion).toBe(true);
|
|
});
|
|
|
|
it("does not throw when cache returns null", () => {
|
|
const executor = new DefaultExecutor("claude");
|
|
expect(() => executor.buildHeaders({ apiKey: "sk" }, false)).not.toThrow();
|
|
});
|
|
});
|
|
|
|
// ─── anthropic-compatible header stripping ────────────────────────────────────
|
|
|
|
describe("DefaultExecutor.buildHeaders() — anthropic-compatible stripping", () => {
|
|
let DefaultExecutor;
|
|
|
|
beforeEach(async () => {
|
|
vi.resetModules();
|
|
const mod = await import("open-sse/executors/default.js");
|
|
DefaultExecutor = mod.DefaultExecutor || mod.default;
|
|
});
|
|
|
|
it("strips x-app and anthropic-dangerous-direct-browser-access for non-Anthropic host", () => {
|
|
const executor = new DefaultExecutor("anthropic-compatible-custom");
|
|
const headers = executor.buildHeaders(
|
|
{
|
|
apiKey: "key",
|
|
providerSpecificData: { baseUrl: "https://myproxy.example.com/v1" },
|
|
},
|
|
true
|
|
);
|
|
|
|
expect(headers["x-app"]).toBeUndefined();
|
|
expect(headers["X-App"]).toBeUndefined();
|
|
expect(headers["anthropic-dangerous-direct-browser-access"]).toBeUndefined();
|
|
expect(headers["Anthropic-Dangerous-Direct-Browser-Access"]).toBeUndefined();
|
|
});
|
|
|
|
it("removes claude-code-20250219 from anthropic-beta for non-Anthropic host", () => {
|
|
const executor = new DefaultExecutor("anthropic-compatible-custom");
|
|
const headers = executor.buildHeaders(
|
|
{
|
|
apiKey: "key",
|
|
providerSpecificData: { baseUrl: "https://myproxy.example.com/v1" },
|
|
},
|
|
true
|
|
);
|
|
|
|
const betaVal = headers["anthropic-beta"] || headers["Anthropic-Beta"] || "";
|
|
expect(betaVal).not.toContain("claude-code-20250219");
|
|
});
|
|
|
|
it("keeps other beta flags intact after stripping", () => {
|
|
const executor = new DefaultExecutor("anthropic-compatible-custom");
|
|
// The static CLAUDE_API_HEADERS used by anthropic-compatible providers include
|
|
// 'interleaved-thinking-2025-05-14' — check it survives stripping
|
|
const headers = executor.buildHeaders(
|
|
{
|
|
apiKey: "key",
|
|
providerSpecificData: { baseUrl: "https://myproxy.example.com/v1" },
|
|
},
|
|
false
|
|
);
|
|
|
|
const betaVal = headers["anthropic-beta"] || headers["Anthropic-Beta"] || "";
|
|
// If any beta value remains it should not be empty and should not have the stripped value
|
|
if (betaVal) {
|
|
expect(betaVal).not.toContain("claude-code-20250219");
|
|
}
|
|
});
|
|
|
|
it("does NOT strip headers when baseUrl is api.anthropic.com", () => {
|
|
const executor = new DefaultExecutor("anthropic-compatible-official");
|
|
const headers = executor.buildHeaders(
|
|
{
|
|
apiKey: "key",
|
|
providerSpecificData: { baseUrl: "https://api.anthropic.com/v1" },
|
|
},
|
|
true
|
|
);
|
|
|
|
// No stripping — anthropic-version should survive
|
|
const hasVersion =
|
|
headers["Anthropic-Version"] || headers["anthropic-version"];
|
|
expect(hasVersion).toBeDefined();
|
|
});
|
|
|
|
it("does NOT strip headers when baseUrl is empty (defaults to Anthropic)", () => {
|
|
const executor = new DefaultExecutor("anthropic-compatible-official");
|
|
const headers = executor.buildHeaders(
|
|
{
|
|
apiKey: "key",
|
|
providerSpecificData: {},
|
|
},
|
|
true
|
|
);
|
|
|
|
const hasVersion =
|
|
headers["Anthropic-Version"] || headers["anthropic-version"];
|
|
expect(hasVersion).toBeDefined();
|
|
});
|
|
});
|
|
|
|
// ─── proxyFetch anthropicFetch routing ────────────────────────────────────────
|
|
|
|
describe("proxyAwareFetch — api.anthropic.com routing", () => {
|
|
afterEach(() => {
|
|
vi.restoreAllMocks();
|
|
});
|
|
|
|
it("routes api.anthropic.com to gotScraping (non-streaming) and returns ok response", async () => {
|
|
// Mock got-scraping before module load
|
|
vi.doMock("got-scraping", () => {
|
|
const mockGotScraping = vi.fn().mockResolvedValue({
|
|
statusCode: 200,
|
|
statusMessage: "OK",
|
|
headers: { "content-type": "application/json" },
|
|
rawBody: Buffer.from(JSON.stringify({ id: "msg_test" })),
|
|
});
|
|
mockGotScraping.stream = vi.fn();
|
|
return { gotScraping: mockGotScraping };
|
|
});
|
|
|
|
vi.resetModules();
|
|
const { proxyAwareFetch } = await import("open-sse/utils/proxyFetch.js");
|
|
const { gotScraping } = await import("got-scraping");
|
|
|
|
const res = await proxyAwareFetch("https://api.anthropic.com/v1/messages", {
|
|
method: "POST",
|
|
// No Accept: text/event-stream → non-streaming path
|
|
headers: { "Content-Type": "application/json" },
|
|
body: JSON.stringify({ model: "claude-3-5-sonnet-20241022", messages: [] }),
|
|
});
|
|
|
|
expect(gotScraping).toHaveBeenCalledOnce();
|
|
expect(res.ok).toBe(true);
|
|
expect(res.status).toBe(200);
|
|
const data = await res.json();
|
|
expect(data.id).toBe("msg_test");
|
|
});
|
|
|
|
it("falls back gracefully when got-scraping throws on non-streaming path", async () => {
|
|
vi.doMock("got-scraping", () => {
|
|
const fn = vi.fn().mockRejectedValue(new Error("TLS error"));
|
|
fn.stream = vi.fn();
|
|
return { gotScraping: fn };
|
|
});
|
|
|
|
const originalFetch = globalThis.fetch;
|
|
globalThis.fetch = vi.fn().mockResolvedValue({
|
|
ok: true,
|
|
status: 200,
|
|
statusText: "OK",
|
|
headers: new Headers(),
|
|
body: null,
|
|
text: async () => "{}",
|
|
json: async () => ({}),
|
|
});
|
|
|
|
vi.resetModules();
|
|
const { proxyAwareFetch } = await import("open-sse/utils/proxyFetch.js");
|
|
|
|
const res = await proxyAwareFetch("https://api.anthropic.com/v1/messages", {
|
|
method: "POST",
|
|
headers: { "Content-Type": "application/json" },
|
|
body: "{}",
|
|
});
|
|
|
|
expect(res.ok).toBe(true);
|
|
globalThis.fetch = originalFetch;
|
|
});
|
|
|
|
it("does NOT route non-Anthropic hosts through gotScraping", async () => {
|
|
const gotScrapingMock = vi.fn();
|
|
vi.doMock("got-scraping", () => ({ gotScraping: gotScrapingMock }));
|
|
|
|
globalThis.fetch = vi.fn().mockResolvedValue({
|
|
ok: true,
|
|
status: 200,
|
|
statusText: "OK",
|
|
headers: new Headers(),
|
|
body: null,
|
|
text: async () => "{}",
|
|
json: async () => ({}),
|
|
});
|
|
|
|
vi.resetModules();
|
|
const { proxyAwareFetch } = await import("open-sse/utils/proxyFetch.js");
|
|
|
|
await proxyAwareFetch("https://openrouter.ai/api/v1/chat/completions", {
|
|
method: "POST",
|
|
headers: { "Content-Type": "application/json" },
|
|
body: "{}",
|
|
});
|
|
|
|
expect(gotScrapingMock).not.toHaveBeenCalled();
|
|
});
|
|
});
|