Fix : Updated Anthropic-Beta header.

This commit is contained in:
decolua 2026-04-05 07:46:26 +07:00
parent d076c7d7f5
commit 67e0db77da
7 changed files with 143 additions and 27 deletions

View file

@ -65,6 +65,34 @@ export const INTERNAL_REQUEST_HEADER = { name: "x-request-source", value: "local
// Suffix added to client tools when forwarding to Antigravity provider (anti-ban cloaking)
export const AG_TOOL_SUFFIX = "_ide";
// Suffix added to client tools when forwarding to Claude provider (anti-ban cloaking)
export const CLAUDE_TOOL_SUFFIX = "_ide";
// CC native default tools — these are Claude Code's own tools, kept as decoys
// Client tools matching these names are skipped (not renamed), others get _cc suffix
export const CC_DEFAULT_TOOLS = new Set([
"Task",
"TaskOutput",
"TaskStop",
"TaskCreate",
"TaskGet",
"TaskUpdate",
"TaskList",
"Bash",
"Glob",
"Grep",
"Read",
"Edit",
"Write",
"NotebookEdit",
"WebFetch",
"WebSearch",
"AskUserQuestion",
"Skill",
"EnterPlanMode",
"ExitPlanMode",
]);
// AG native default tools — kept as decoys with neutral description/properties
// These names must match exactly what AG sends in the real request log
export const AG_DEFAULT_TOOLS = new Set([
@ -115,7 +143,7 @@ export const LOAD_CODE_ASSIST_METADATA = {
};
// System prompts
export const CLAUDE_SYSTEM_PROMPT = "You are a Claude agent, built on Anthropic's Claude Agent SDK.";
export const CLAUDE_SYSTEM_PROMPT = "You are Claude Code, Anthropic's official CLI for Claude.";
export const ANTIGRAVITY_DEFAULT_SYSTEM = "You are Antigravity, a powerful agentic AI coding assistant designed by the Google Deepmind team working on Advanced Agentic Coding.You are pair programming with a USER to solve their coding task. The task may require creating a new codebase, modifying or debugging an existing codebase, or simply answering a question.**Absolute paths only****Proactiveness**";
// OAuth endpoints

View file

@ -36,14 +36,14 @@ export const PROVIDERS = {
retry: { 429: 0 },
headers: {
"Anthropic-Version": "2023-06-01",
"Anthropic-Beta": "claude-code-20250219,oauth-2025-04-20,interleaved-thinking-2025-05-14,fine-grained-tool-streaming-2025-05-14,context-management-2025-06-27,prompt-caching-scope-2026-01-05",
"Anthropic-Beta": "claude-code-20250219,oauth-2025-04-20,interleaved-thinking-2025-05-14,context-management-2025-06-27,prompt-caching-scope-2026-01-05,advanced-tool-use-2025-11-20,effort-2025-11-24,structured-outputs-2025-12-15,fast-mode-2026-02-01,redact-thinking-2026-02-12,token-efficient-tools-2026-03-28",
"Anthropic-Dangerous-Direct-Browser-Access": "true",
"User-Agent": "claude-cli/2.1.63 (external, cli)",
"User-Agent": "claude-cli/2.1.92 (external, sdk-cli)",
"X-App": "cli",
"X-Stainless-Helper-Method": "stream",
"X-Stainless-Retry-Count": "0",
"X-Stainless-Runtime-Version": "v24.3.0",
"X-Stainless-Package-Version": "0.74.0",
"X-Stainless-Runtime-Version": "v24.14.0",
"X-Stainless-Package-Version": "0.80.0",
"X-Stainless-Runtime": "node",
"X-Stainless-Lang": "js",
"X-Stainless-Arch": mapStainlessArch(),

View file

@ -68,7 +68,7 @@ export async function handleChatCore({ body, modelInfo, credentials, log, onCred
log?.debug?.("PASSTHROUGH", `${clientTool}${provider} | native lossless`);
translatedBody = { ...body, model };
} else {
translatedBody = translateRequest(sourceFormat, targetFormat, model, body, stream, credentials, provider, reqLogger, modelCaps);
translatedBody = translateRequest(sourceFormat, targetFormat, model, body, stream, credentials, provider, reqLogger, modelCaps, connectionId);
if (!translatedBody) {
trackPendingRequest(model, provider, connectionId, false, true);
return createErrorResult(HTTP_STATUS.BAD_REQUEST, `Failed to translate request for ${sourceFormat}${targetFormat}`);

View file

@ -2,6 +2,7 @@
import { DEFAULT_THINKING_CLAUDE_SIGNATURE } from "../../config/defaultThinkingSignature.js";
import { adjustMaxTokens } from "./maxTokensHelper.js";
import { applyCloaking } from "../../utils/claudeCloaking.js";
import { deriveSessionId } from "../../utils/sessionManager.js";
// Check if message has valid non-empty content
export function hasValidContent(msg) {
@ -81,7 +82,7 @@ export function fixToolUseOrdering(messages) {
// - Add thinking block for Anthropic endpoint (provider === "claude")
// - Fix tool_use/tool_result ordering
// - Apply cloaking (billing header + fake user ID) for OAuth tokens
export function prepareClaudeRequest(body, provider = null, apiKey = null) {
export function prepareClaudeRequest(body, provider = null, apiKey = null, connectionId = null) {
// 1. System: remove all cache_control, add only to last block with ttl 1h
if (body.system && Array.isArray(body.system)) {
body.system = body.system.map((block, i) => {
@ -196,8 +197,10 @@ export function prepareClaudeRequest(body, provider = null, apiKey = null) {
}
// Apply cloaking for OAuth tokens (billing header + fake user ID)
// session_id in user_id must match X-Claude-Code-Session-Id for fingerprint consistency
if ((provider === "claude" || provider?.startsWith("anthropic-compatible")) && apiKey) {
body = applyCloaking(body, apiKey);
const sessionId = connectionId ? deriveSessionId(connectionId) : null;
body = applyCloaking(body, apiKey, sessionId);
}
return body;

View file

@ -1,6 +1,7 @@
import { FORMATS } from "./formats.js";
import { ensureToolCallIds, fixMissingToolResponses } from "./helpers/toolCallHelper.js";
import { prepareClaudeRequest } from "./helpers/claudeHelper.js";
import { cloakClaudeTools } from "../utils/claudeCloaking.js";
import { filterToOpenAIFormat } from "./helpers/openaiHelper.js";
import { normalizeThinkingConfig } from "../services/provider.js";
import { AntigravityExecutor } from "../executors/antigravity.js";
@ -67,7 +68,7 @@ function stripUnsupportedMultimodal(body, multimodal = {}) {
}
// Translate request: source -> openai -> target
export function translateRequest(sourceFormat, targetFormat, model, body, stream = true, credentials = null, provider = null, reqLogger = null, caps = null) {
export function translateRequest(sourceFormat, targetFormat, model, body, stream = true, credentials = null, provider = null, reqLogger = null, caps = null, connectionId = null) {
ensureInitialized();
let result = body;
@ -122,7 +123,20 @@ export function translateRequest(sourceFormat, targetFormat, model, body, stream
// Final step: prepare request for Claude format endpoints
if (targetFormat === FORMATS.CLAUDE) {
const apiKey = credentials?.accessToken || credentials?.apiKey || null;
result = prepareClaudeRequest(result, provider, apiKey);
result = prepareClaudeRequest(result, provider, apiKey, connectionId);
}
// Claude cloaking: rename client tools with _cc suffix (anti-ban)
// Only for claude provider (not anthropic-compatible-*) with OAuth token
if (provider === "claude") {
const apiKey = credentials?.accessToken || credentials?.apiKey || null;
if (apiKey?.includes("sk-ant-oat")) {
const { body: cloakedBody, toolNameMap } = cloakClaudeTools(result);
result = cloakedBody;
if (toolNameMap?.size > 0) {
result._toolNameMap = toolNameMap;
}
}
}
// Antigravity cloaking: rename client tools + inject decoys (anti-ban)

View file

@ -1,36 +1,106 @@
import { createHash, randomBytes } from "crypto";
import { createHash, randomBytes, randomUUID } from "crypto";
import { CLAUDE_TOOL_SUFFIX, CC_DEFAULT_TOOLS } from "../config/appConstants.js";
const CLAUDE_VERSION = "2.1.63";
const CLAUDE_VERSION = "2.1.92";
const CC_ENTRYPOINT = "sdk-cli";
// Generate billing header matching real Claude Code format:
// x-anthropic-billing-header: cc_version=<ver>.<build>; cc_entrypoint=cli; cch=<hash>;
// Generate billing header matching real Claude Code 2.1.92+ format:
// x-anthropic-billing-header: cc_version=<ver>.<build>; cc_entrypoint=sdk-cli; cch=<hash>;
function generateBillingHeader(payload) {
const content = JSON.stringify(payload);
const cch = createHash("sha256").update(content).digest("hex").slice(0, 5);
const buildHash = randomBytes(2).toString("hex").slice(0, 3);
return `x-anthropic-billing-header: cc_version=${CLAUDE_VERSION}.${buildHash}; cc_entrypoint=cli; cch=${cch};`;
return `x-anthropic-billing-header: cc_version=${CLAUDE_VERSION}.${buildHash}; cc_entrypoint=${CC_ENTRYPOINT}; cch=${cch};`;
}
// Generate a random UUID v4
function generateFakeUserID() {
const bytes = randomBytes(16);
bytes[6] = (bytes[6] & 0x0f) | 0x40;
bytes[8] = (bytes[8] & 0x3f) | 0x80;
const hex = bytes.toString("hex");
return `${hex.slice(0,8)}-${hex.slice(8,12)}-${hex.slice(12,16)}-${hex.slice(16,20)}-${hex.slice(20)}`;
// Generate fake user ID in Claude Code 2.1.92+ JSON format:
// {"device_id":"<64hex>","account_uuid":"<uuid>","session_id":"<uuid>"}
function generateFakeUserID(sessionId) {
const deviceId = randomBytes(32).toString("hex");
const accountUuid = randomUUID();
const sessionUuid = sessionId || randomUUID();
return `{"device_id":"${deviceId}","account_uuid":"${accountUuid}","session_id":"${sessionUuid}"}`;
}
/**
* Cloak tools before sending to Claude provider (anti-ban):
* - Rename non-CC client tools with _cc suffix in tools[] and messages[]
* - Skip tools that are already CC default names (they become decoys as-is)
* - Inject CC_DECOY_TOOLS after client tools
* Returns { body, toolNameMap } where toolNameMap maps suffixed original
* @param {object} body - Claude API request body
* @returns {{ body: object, toolNameMap: Map|null }}
*/
export function cloakClaudeTools(body) {
const tools = body.tools;
if (!tools || tools.length === 0) return { body, toolNameMap: null };
const toolNameMap = new Map();
const clientDeclarations = [];
// All client tools get renamed with suffix
for (const tool of tools) {
const suffixed = `${tool.name}${CLAUDE_TOOL_SUFFIX}`;
toolNameMap.set(suffixed, tool.name);
clientDeclarations.push({ ...tool, name: suffixed });
}
// Client tools first, then CC decoy tools (no overlap: client tools all have _cc suffix)
const allTools = [...clientDeclarations, ...CC_DECOY_TOOLS];
// Rename tool_use in message history (all client tools get suffix)
const renamedMessages = body.messages?.map(msg => {
if (!Array.isArray(msg.content)) return msg;
const renamedContent = msg.content.map(block => {
if (block.type === "tool_use") {
return { ...block, name: `${block.name}${CLAUDE_TOOL_SUFFIX}` };
}
return block;
});
return { ...msg, content: renamedContent };
});
return {
body: { ...body, tools: allTools, messages: renamedMessages || body.messages },
toolNameMap: toolNameMap.size > 0 ? toolNameMap : null
};
}
// CC decoy tools — Claude Code native tool names, marked unavailable
const CC_DECOY_TOOLS = [
{ name: "Task", description: "This tool is currently unavailable.", input_schema: { type: "object", properties: {} } },
{ name: "TaskOutput", description: "This tool is currently unavailable.", input_schema: { type: "object", properties: {} } },
{ name: "TaskStop", description: "This tool is currently unavailable.", input_schema: { type: "object", properties: {} } },
{ name: "TaskCreate", description: "This tool is currently unavailable.", input_schema: { type: "object", properties: {} } },
{ name: "TaskGet", description: "This tool is currently unavailable.", input_schema: { type: "object", properties: {} } },
{ name: "TaskUpdate", description: "This tool is currently unavailable.", input_schema: { type: "object", properties: {} } },
{ name: "TaskList", description: "This tool is currently unavailable.", input_schema: { type: "object", properties: {} } },
{ name: "Bash", description: "This tool is currently unavailable.", input_schema: { type: "object", properties: {} } },
{ name: "Glob", description: "This tool is currently unavailable.", input_schema: { type: "object", properties: {} } },
{ name: "Grep", description: "This tool is currently unavailable.", input_schema: { type: "object", properties: {} } },
{ name: "Read", description: "This tool is currently unavailable.", input_schema: { type: "object", properties: {} } },
{ name: "Edit", description: "This tool is currently unavailable.", input_schema: { type: "object", properties: {} } },
{ name: "Write", description: "This tool is currently unavailable.", input_schema: { type: "object", properties: {} } },
{ name: "NotebookEdit", description: "This tool is currently unavailable.", input_schema: { type: "object", properties: {} } },
{ name: "WebFetch", description: "This tool is currently unavailable.", input_schema: { type: "object", properties: {} } },
{ name: "WebSearch", description: "This tool is currently unavailable.", input_schema: { type: "object", properties: {} } },
{ name: "AskUserQuestion", description: "This tool is currently unavailable.", input_schema: { type: "object", properties: {} } },
{ name: "Skill", description: "This tool is currently unavailable.", input_schema: { type: "object", properties: {} } },
{ name: "EnterPlanMode", description: "This tool is currently unavailable.", input_schema: { type: "object", properties: {} } },
{ name: "ExitPlanMode", description: "This tool is currently unavailable.", input_schema: { type: "object", properties: {} } },
];
/**
* Apply Claude cloaking to request body:
* 1. Inject billing header as first system block
* 2. Inject fake user ID into metadata
*
* 2. Inject fake user ID into metadata (JSON format, session_id aligned with X-Claude-Code-Session-Id)
* Only applies when using OAuth token (sk-ant-oat).
* @param {object} body - Claude API request body
* @param {string} apiKey - API key or OAuth token
* @param {string} [sessionId] - Session ID to align with X-Claude-Code-Session-Id header
* @returns {object} Modified body
*/
export function applyCloaking(body, apiKey) {
export function applyCloaking(body, apiKey, sessionId) {
if (!apiKey || !apiKey.includes("sk-ant-oat")) return body;
const result = { ...body };
@ -50,10 +120,10 @@ export function applyCloaking(body, apiKey) {
result.system = [billingBlock];
}
// Inject fake user ID into metadata
// Inject fake user ID into metadata (session_id must match X-Claude-Code-Session-Id)
const existingUserId = result.metadata?.user_id;
if (!existingUserId) {
result.metadata = { ...result.metadata, user_id: generateFakeUserID() };
result.metadata = { ...result.metadata, user_id: generateFakeUserID(sessionId) };
}
return result;

View file

@ -19,6 +19,7 @@ const CLAUDE_IDENTITY_HEADERS = [
"x-stainless-arch",
"x-stainless-os",
"x-stainless-timeout",
"x-claude-code-session-id",
"package-version",
"runtime-version",
"os",