// Tool call helper functions for translator // Anthropic tool_use.id must match: ^[a-zA-Z0-9_-]+$ const TOOL_ID_PATTERN = /^[a-zA-Z0-9_-]+$/; // Generate deterministic tool call ID from position + tool name (cache-friendly) export function generateToolCallId(msgIndex = 0, tcIndex = 0, toolName = "") { const name = toolName ? `_${toolName.replace(/[^a-zA-Z0-9_-]/g, "")}` : ""; return `call_msg${msgIndex}_tc${tcIndex}${name}`; } // Sanitize ID to match Anthropic pattern: keep only alphanumeric, underscore, hyphen function sanitizeToolId(id) { if (!id || typeof id !== "string") return null; const sanitized = id.replace(/[^a-zA-Z0-9_-]/g, ""); return sanitized.length > 0 ? sanitized : null; } // Ensure all tool_calls have valid id field and arguments is string (some providers require it) export function ensureToolCallIds(body) { if (!body.messages || !Array.isArray(body.messages)) return body; for (let i = 0; i < body.messages.length; i++) { const msg = body.messages[i]; if (msg.role === "assistant" && msg.tool_calls && Array.isArray(msg.tool_calls)) { for (let j = 0; j < msg.tool_calls.length; j++) { const tc = msg.tool_calls[j]; // Validate or regenerate ID for Anthropic compatibility if (!tc.id || !TOOL_ID_PATTERN.test(tc.id)) { const sanitized = sanitizeToolId(tc.id); tc.id = sanitized || generateToolCallId(i, j, tc.function?.name); } if (!tc.type) { tc.type = "function"; } // Ensure arguments is JSON string, not object if (tc.function?.arguments && typeof tc.function.arguments !== "string") { tc.function.arguments = JSON.stringify(tc.function.arguments); } } } // Validate tool_call_id in tool messages (role: "tool") if (msg.role === "tool" && msg.tool_call_id && !TOOL_ID_PATTERN.test(msg.tool_call_id)) { const sanitized = sanitizeToolId(msg.tool_call_id); msg.tool_call_id = sanitized || generateToolCallId(i, 0); } // Also validate tool_use blocks in content (Claude format) if (Array.isArray(msg.content)) { for (let k = 0; k < msg.content.length; k++) { const block = msg.content[k]; if (block.type === "tool_use" && block.id && !TOOL_ID_PATTERN.test(block.id)) { const sanitized = sanitizeToolId(block.id); block.id = sanitized || generateToolCallId(i, k, block.name); } // Validate tool_use_id in tool_result blocks if (block.type === "tool_result" && block.tool_use_id && !TOOL_ID_PATTERN.test(block.tool_use_id)) { const sanitized = sanitizeToolId(block.tool_use_id); block.tool_use_id = sanitized || generateToolCallId(i, k); } } } } return body; } // Get tool_call ids from assistant message (OpenAI format: tool_calls, Claude format: tool_use in content) export function getToolCallIds(msg) { if (msg.role !== "assistant") return []; const ids = []; // OpenAI format: tool_calls array if (msg.tool_calls && Array.isArray(msg.tool_calls)) { for (const tc of msg.tool_calls) { if (tc.id) ids.push(tc.id); } } // Claude format: tool_use blocks in content if (Array.isArray(msg.content)) { for (const block of msg.content) { if (block.type === "tool_use" && block.id) { ids.push(block.id); } } } return ids; } // Check if user message has tool_result for given ids (OpenAI format: role=tool, Claude format: tool_result in content) export function hasToolResults(msg, toolCallIds) { if (!msg || !toolCallIds.length) return false; // OpenAI format: role = "tool" with tool_call_id if (msg.role === "tool" && msg.tool_call_id) { return toolCallIds.includes(msg.tool_call_id); } // Claude format: tool_result blocks in user message content if (msg.role === "user" && Array.isArray(msg.content)) { for (const block of msg.content) { if (block.type === "tool_result" && toolCallIds.includes(block.tool_use_id)) { return true; } } } return false; } // Fix missing tool responses - insert empty tool_result if assistant has tool_use but next message has no tool_result export function fixMissingToolResponses(body) { if (!body.messages || !Array.isArray(body.messages)) return body; const newMessages = []; for (let i = 0; i < body.messages.length; i++) { const msg = body.messages[i]; const nextMsg = body.messages[i + 1]; newMessages.push(msg); // Check if this is assistant with tool_calls/tool_use const toolCallIds = getToolCallIds(msg); if (toolCallIds.length === 0) continue; // Check if next message has tool_result if (nextMsg && !hasToolResults(nextMsg, toolCallIds)) { // Insert tool responses for each tool_call for (const id of toolCallIds) { // OpenAI format: role = "tool" newMessages.push({ role: "tool", tool_call_id: id, content: "" }); } } } body.messages = newMessages; return body; }