|
import process from 'node:process'; |
|
import util from 'node:util'; |
|
import express from 'express'; |
|
import fetch from 'node-fetch'; |
|
|
|
import { |
|
CHAT_COMPLETION_SOURCES, |
|
GEMINI_SAFETY, |
|
OPENROUTER_HEADERS, |
|
} from '../../constants.js'; |
|
import { |
|
forwardFetchResponse, |
|
getConfigValue, |
|
tryParse, |
|
uuidv4, |
|
mergeObjectWithYaml, |
|
excludeKeysByYaml, |
|
color, |
|
} from '../../util.js'; |
|
import { |
|
convertClaudeMessages, |
|
convertGooglePrompt, |
|
convertTextCompletionPrompt, |
|
convertCohereMessages, |
|
convertMistralMessages, |
|
convertAI21Messages, |
|
convertXAIMessages, |
|
mergeMessages, |
|
cachingAtDepthForOpenRouterClaude, |
|
cachingAtDepthForClaude, |
|
getPromptNames, |
|
calculateClaudeBudgetTokens, |
|
calculateGoogleBudgetTokens, |
|
} from '../../prompt-converters.js'; |
|
|
|
import { readSecret, SECRET_KEYS } from '../secrets.js'; |
|
import { |
|
getTokenizerModel, |
|
getSentencepiceTokenizer, |
|
getTiktokenTokenizer, |
|
sentencepieceTokenizers, |
|
TEXT_COMPLETION_MODELS, |
|
webTokenizers, |
|
getWebTokenizer, |
|
} from '../tokenizers.js'; |
|
|
|
const API_OPENAI = 'https://api.openai.com/v1'; |
|
const API_CLAUDE = 'https://api.anthropic.com/v1'; |
|
const API_MISTRAL = 'https://api.mistral.ai/v1'; |
|
const API_COHERE_V1 = 'https://api.cohere.ai/v1'; |
|
const API_COHERE_V2 = 'https://api.cohere.ai/v2'; |
|
const API_PERPLEXITY = 'https://api.perplexity.ai'; |
|
const API_GROQ = 'https://api.groq.com/openai/v1'; |
|
const API_MAKERSUITE = 'https://generativelanguage.googleapis.com'; |
|
const API_VERTEX_AI = 'https://us-central1-aiplatform.googleapis.com'; |
|
const API_01AI = 'https://api.lingyiwanwu.com/v1'; |
|
const API_AI21 = 'https://api.ai21.com/studio/v1'; |
|
const API_NANOGPT = 'https://nano-gpt.com/api/v1'; |
|
const API_DEEPSEEK = 'https://api.deepseek.com/beta'; |
|
const API_XAI = 'https://api.x.ai/v1'; |
|
const API_POLLINATIONS = 'https://text.pollinations.ai/openai'; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function postProcessPrompt(messages, type, names) { |
|
const addAssistantPrefix = x => x.length && (x[x.length - 1].role !== 'assistant' || (x[x.length - 1].prefix = true)) ? x : x; |
|
switch (type) { |
|
case 'merge': |
|
case 'claude': |
|
return mergeMessages(messages, names, { strict: false, placeholders: false, single: false }); |
|
case 'semi': |
|
return mergeMessages(messages, names, { strict: true, placeholders: false, single: false }); |
|
case 'strict': |
|
return mergeMessages(messages, names, { strict: true, placeholders: true, single: false }); |
|
case 'deepseek': |
|
return addAssistantPrefix(mergeMessages(messages, names, { strict: true, placeholders: false, single: false })); |
|
case 'deepseek-reasoner': |
|
return addAssistantPrefix(mergeMessages(messages, names, { strict: true, placeholders: true, single: false })); |
|
case 'single': |
|
return mergeMessages(messages, names, { strict: true, placeholders: false, single: true }); |
|
default: |
|
return messages; |
|
} |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
function getOpenRouterTransforms(request) { |
|
switch (request.body.middleout) { |
|
case 'on': |
|
return ['middle-out']; |
|
case 'off': |
|
return []; |
|
case 'auto': |
|
return undefined; |
|
} |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
function getOpenRouterPlugins(request) { |
|
const plugins = []; |
|
|
|
if (request.body.enable_web_search) { |
|
plugins.push({ 'id': 'web' }); |
|
} |
|
|
|
return plugins; |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
async function sendClaudeRequest(request, response) { |
|
const apiUrl = new URL(request.body.reverse_proxy || API_CLAUDE).toString(); |
|
const apiKey = request.body.reverse_proxy ? request.body.proxy_password : readSecret(request.user.directories, SECRET_KEYS.CLAUDE); |
|
const divider = '-'.repeat(process.stdout.columns); |
|
const isClaude3or4 = /^claude-(3|opus-4|sonnet-4)/.test(request.body.model); |
|
const enableSystemPromptCache = getConfigValue('claude.enableSystemPromptCache', false, 'boolean') && isClaude3or4; |
|
let cachingAtDepth = getConfigValue('claude.cachingAtDepth', -1, 'number'); |
|
|
|
if (!Number.isInteger(cachingAtDepth) || cachingAtDepth < 0 || !isClaude3or4) { |
|
cachingAtDepth = -1; |
|
} |
|
|
|
if (!apiKey) { |
|
console.warn(color.red(`Claude API key is missing.\n${divider}`)); |
|
return response.status(400).send({ error: true }); |
|
} |
|
|
|
try { |
|
const controller = new AbortController(); |
|
request.socket.removeAllListeners('close'); |
|
request.socket.on('close', function () { |
|
controller.abort(); |
|
}); |
|
const additionalHeaders = {}; |
|
const betaHeaders = ['output-128k-2025-02-19']; |
|
const useTools = isClaude3or4 && Array.isArray(request.body.tools) && request.body.tools.length > 0; |
|
const useSystemPrompt = Boolean(request.body.claude_use_sysprompt); |
|
const convertedPrompt = convertClaudeMessages(request.body.messages, request.body.assistant_prefill, useSystemPrompt, useTools, getPromptNames(request)); |
|
const useThinking = /^claude-(3-7|opus-4|sonnet-4)/.test(request.body.model); |
|
const useWebSearch = /^claude-(3-5|3-7|opus-4|sonnet-4)/.test(request.body.model) && Boolean(request.body.enable_web_search); |
|
const cacheTTL = getConfigValue('claude.extendedTTL', false, 'boolean') ? '1h' : '5m'; |
|
let fixThinkingPrefill = false; |
|
|
|
const stopSequences = []; |
|
if (Array.isArray(request.body.stop)) { |
|
stopSequences.push(...request.body.stop); |
|
} |
|
|
|
const requestBody = { |
|
system: [], |
|
messages: convertedPrompt.messages, |
|
model: request.body.model, |
|
max_tokens: request.body.max_tokens, |
|
stop_sequences: stopSequences, |
|
temperature: request.body.temperature, |
|
top_p: request.body.top_p, |
|
top_k: request.body.top_k, |
|
stream: request.body.stream, |
|
}; |
|
if (useSystemPrompt) { |
|
if (enableSystemPromptCache && Array.isArray(convertedPrompt.systemPrompt) && convertedPrompt.systemPrompt.length) { |
|
convertedPrompt.systemPrompt[convertedPrompt.systemPrompt.length - 1]['cache_control'] = { type: 'ephemeral', ttl: cacheTTL }; |
|
} |
|
|
|
requestBody.system = convertedPrompt.systemPrompt; |
|
} else { |
|
delete requestBody.system; |
|
} |
|
if (useTools) { |
|
betaHeaders.push('tools-2024-05-16'); |
|
requestBody.tool_choice = { type: request.body.tool_choice }; |
|
requestBody.tools = request.body.tools |
|
.filter(tool => tool.type === 'function') |
|
.map(tool => tool.function) |
|
.map(fn => ({ name: fn.name, description: fn.description, input_schema: fn.parameters })); |
|
|
|
if (enableSystemPromptCache && requestBody.tools.length) { |
|
requestBody.tools[requestBody.tools.length - 1]['cache_control'] = { type: 'ephemeral', ttl: cacheTTL }; |
|
} |
|
} |
|
|
|
if (useWebSearch) { |
|
const webSearchTool = [{ |
|
'type': 'web_search_20250305', |
|
'name': 'web_search', |
|
}]; |
|
requestBody.tools = [...webSearchTool, ...(requestBody.tools || [])]; |
|
} |
|
|
|
if (cachingAtDepth !== -1) { |
|
cachingAtDepthForClaude(convertedPrompt.messages, cachingAtDepth, cacheTTL); |
|
} |
|
|
|
if (enableSystemPromptCache || cachingAtDepth !== -1) { |
|
betaHeaders.push('prompt-caching-2024-07-31'); |
|
betaHeaders.push('extended-cache-ttl-2025-04-11'); |
|
} |
|
|
|
const reasoningEffort = request.body.reasoning_effort; |
|
const budgetTokens = calculateClaudeBudgetTokens(requestBody.max_tokens, reasoningEffort, requestBody.stream); |
|
|
|
if (useThinking && Number.isInteger(budgetTokens)) { |
|
|
|
fixThinkingPrefill = true; |
|
const minThinkTokens = 1024; |
|
if (requestBody.max_tokens <= minThinkTokens) { |
|
const newValue = requestBody.max_tokens + minThinkTokens; |
|
console.warn(color.yellow(`Claude thinking requires a minimum of ${minThinkTokens} response tokens.`)); |
|
console.info(color.blue(`Increasing response length to ${newValue}.`)); |
|
requestBody.max_tokens = newValue; |
|
} |
|
requestBody.thinking = { |
|
type: 'enabled', |
|
budget_tokens: budgetTokens, |
|
}; |
|
|
|
|
|
delete requestBody.temperature; |
|
delete requestBody.top_p; |
|
delete requestBody.top_k; |
|
} |
|
|
|
if (fixThinkingPrefill && convertedPrompt.messages.length && convertedPrompt.messages[convertedPrompt.messages.length - 1].role === 'assistant') { |
|
convertedPrompt.messages[convertedPrompt.messages.length - 1].role = 'user'; |
|
} |
|
|
|
if (betaHeaders.length) { |
|
additionalHeaders['anthropic-beta'] = betaHeaders.join(','); |
|
} |
|
|
|
console.debug('Claude request:', requestBody); |
|
|
|
const generateResponse = await fetch(apiUrl + '/messages', { |
|
method: 'POST', |
|
signal: controller.signal, |
|
body: JSON.stringify(requestBody), |
|
headers: { |
|
'Content-Type': 'application/json', |
|
'anthropic-version': '2023-06-01', |
|
'x-api-key': apiKey, |
|
...additionalHeaders, |
|
}, |
|
}); |
|
|
|
if (request.body.stream) { |
|
|
|
forwardFetchResponse(generateResponse, response); |
|
} else { |
|
if (!generateResponse.ok) { |
|
const generateResponseText = await generateResponse.text(); |
|
console.warn(color.red(`Claude API returned error: ${generateResponse.status} ${generateResponse.statusText}\n${generateResponseText}\n${divider}`)); |
|
return response.status(500).send({ error: true }); |
|
} |
|
|
|
|
|
const generateResponseJson = await generateResponse.json(); |
|
const responseText = generateResponseJson?.content?.[0]?.text || ''; |
|
console.debug('Claude response:', generateResponseJson); |
|
|
|
|
|
const reply = { choices: [{ 'message': { 'content': responseText } }], content: generateResponseJson.content }; |
|
return response.send(reply); |
|
} |
|
} catch (error) { |
|
console.error(color.red(`Error communicating with Claude: ${error}\n${divider}`)); |
|
if (!response.headersSent) { |
|
return response.status(500).send({ error: true }); |
|
} |
|
} |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
async function sendScaleRequest(request, response) { |
|
const apiUrl = new URL(request.body.api_url_scale).toString(); |
|
const apiKey = readSecret(request.user.directories, SECRET_KEYS.SCALE); |
|
|
|
if (!apiKey) { |
|
console.warn('Scale API key is missing.'); |
|
return response.status(400).send({ error: true }); |
|
} |
|
|
|
const requestPrompt = convertTextCompletionPrompt(request.body.messages); |
|
console.debug('Scale request:', requestPrompt); |
|
|
|
try { |
|
const controller = new AbortController(); |
|
request.socket.removeAllListeners('close'); |
|
request.socket.on('close', function () { |
|
controller.abort(); |
|
}); |
|
|
|
const generateResponse = await fetch(apiUrl, { |
|
method: 'POST', |
|
body: JSON.stringify({ input: { input: requestPrompt } }), |
|
headers: { |
|
'Content-Type': 'application/json', |
|
'Authorization': `Basic ${apiKey}`, |
|
}, |
|
}); |
|
|
|
if (!generateResponse.ok) { |
|
console.warn(`Scale API returned error: ${generateResponse.status} ${generateResponse.statusText} ${await generateResponse.text()}`); |
|
return response.status(500).send({ error: true }); |
|
} |
|
|
|
|
|
const generateResponseJson = await generateResponse.json(); |
|
console.debug('Scale response:', generateResponseJson); |
|
|
|
const reply = { choices: [{ 'message': { 'content': generateResponseJson.output } }] }; |
|
return response.send(reply); |
|
} catch (error) { |
|
console.error(error); |
|
if (!response.headersSent) { |
|
return response.status(500).send({ error: true }); |
|
} |
|
} |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
async function sendMakerSuiteRequest(request, response) { |
|
const useVertexAi = request.body.chat_completion_source === CHAT_COMPLETION_SOURCES.VERTEXAI; |
|
const apiName = useVertexAi ? 'Google Vertex AI' : 'Google AI Studio'; |
|
let apiUrl; |
|
let apiKey; |
|
|
|
if (useVertexAi) { |
|
apiUrl = new URL(request.body.reverse_proxy || API_VERTEX_AI); |
|
apiKey = request.body.reverse_proxy ? request.body.proxy_password : readSecret(request.user.directories, SECRET_KEYS.VERTEXAI); |
|
|
|
if (!request.body.reverse_proxy && !apiKey) { |
|
console.warn(`${apiName} API key is missing.`); |
|
return response.status(400).send({ error: true }); |
|
} |
|
} else { |
|
apiUrl = new URL(request.body.reverse_proxy || API_MAKERSUITE); |
|
apiKey = request.body.reverse_proxy ? request.body.proxy_password : readSecret(request.user.directories, SECRET_KEYS.MAKERSUITE); |
|
|
|
if (!request.body.reverse_proxy && !apiKey) { |
|
console.warn(`${apiName} API key is missing.`); |
|
return response.status(400).send({ error: true }); |
|
} |
|
} |
|
|
|
const model = String(request.body.model); |
|
const stream = Boolean(request.body.stream); |
|
const enableWebSearch = Boolean(request.body.enable_web_search); |
|
const requestImages = Boolean(request.body.request_images); |
|
const reasoningEffort = String(request.body.reasoning_effort); |
|
const includeReasoning = Boolean(request.body.include_reasoning); |
|
const isGemma = model.includes('gemma'); |
|
const isLearnLM = model.includes('learnlm'); |
|
|
|
const generationConfig = { |
|
stopSequences: request.body.stop, |
|
candidateCount: 1, |
|
maxOutputTokens: request.body.max_tokens, |
|
temperature: request.body.temperature, |
|
topP: request.body.top_p, |
|
topK: request.body.top_k || undefined, |
|
responseMimeType: request.body.responseMimeType, |
|
responseSchema: request.body.responseSchema, |
|
}; |
|
|
|
function getGeminiBody() { |
|
|
|
const imageGenerationModels = [ |
|
'gemini-2.0-flash-exp', |
|
'gemini-2.0-flash-exp-image-generation', |
|
]; |
|
|
|
|
|
const blockNoneModels = [ |
|
'gemini-1.5-pro-001', |
|
'gemini-1.5-flash-001', |
|
'gemini-1.5-flash-8b-exp-0827', |
|
'gemini-1.5-flash-8b-exp-0924', |
|
]; |
|
|
|
const isThinkingConfigModel = m => /^gemini-2.5-(flash|pro)/.test(m); |
|
const isThinkingBudgetModel = m => /^gemini-2.5-flash/.test(m); |
|
|
|
const noSearchModels = [ |
|
'gemini-2.0-flash-lite', |
|
'gemini-2.0-flash-lite-001', |
|
'gemini-2.0-flash-lite-preview-02-05', |
|
'gemini-1.5-flash-8b-exp-0924', |
|
'gemini-1.5-flash-8b-exp-0827', |
|
]; |
|
|
|
|
|
if (!Array.isArray(generationConfig.stopSequences) || !generationConfig.stopSequences.length) { |
|
delete generationConfig.stopSequences; |
|
} |
|
|
|
const enableImageModality = requestImages && imageGenerationModels.includes(model); |
|
if (enableImageModality) { |
|
generationConfig.responseModalities = ['text', 'image']; |
|
} |
|
|
|
const useSystemPrompt = !enableImageModality && !isGemma && request.body.use_makersuite_sysprompt; |
|
|
|
const tools = []; |
|
const prompt = convertGooglePrompt(request.body.messages, model, useSystemPrompt, getPromptNames(request)); |
|
let safetySettings = GEMINI_SAFETY; |
|
|
|
if (blockNoneModels.includes(model)) { |
|
safetySettings = GEMINI_SAFETY.map(setting => ({ ...setting, threshold: 'BLOCK_NONE' })); |
|
} |
|
|
|
if (enableWebSearch && !enableImageModality && !isGemma && !isLearnLM && !noSearchModels.includes(model)) { |
|
const searchTool = model.includes('1.5') |
|
? ({ google_search_retrieval: {} }) |
|
: ({ google_search: {} }); |
|
tools.push(searchTool); |
|
} |
|
|
|
if (Array.isArray(request.body.tools) && request.body.tools.length > 0 && !enableImageModality && !isGemma) { |
|
const functionDeclarations = []; |
|
for (const tool of request.body.tools) { |
|
if (tool.type === 'function') { |
|
if (tool.function.parameters?.$schema) { |
|
delete tool.function.parameters.$schema; |
|
} |
|
if (tool.function.parameters?.properties && Object.keys(tool.function.parameters.properties).length === 0) { |
|
delete tool.function.parameters; |
|
} |
|
functionDeclarations.push(tool.function); |
|
} |
|
} |
|
tools.push({ function_declarations: functionDeclarations }); |
|
} |
|
|
|
if (isThinkingConfigModel(model)) { |
|
const thinkingConfig = { includeThoughts: includeReasoning }; |
|
|
|
if (isThinkingBudgetModel(model)) { |
|
const thinkingBudget = calculateGoogleBudgetTokens(generationConfig.maxOutputTokens, reasoningEffort); |
|
if (Number.isInteger(thinkingBudget)) { |
|
thinkingConfig.thinkingBudget = thinkingBudget; |
|
} |
|
} |
|
|
|
generationConfig.thinkingConfig = thinkingConfig; |
|
} |
|
|
|
let body = { |
|
contents: prompt.contents, |
|
safetySettings: safetySettings, |
|
generationConfig: generationConfig, |
|
}; |
|
|
|
if (useSystemPrompt && Array.isArray(prompt.system_instruction.parts) && prompt.system_instruction.parts.length) { |
|
body.systemInstruction = prompt.system_instruction; |
|
} |
|
|
|
if (tools.length) { |
|
body.tools = tools; |
|
} |
|
|
|
return body; |
|
} |
|
|
|
const body = getGeminiBody(); |
|
console.debug(`${apiName} request:`, body); |
|
|
|
try { |
|
const controller = new AbortController(); |
|
request.socket.removeAllListeners('close'); |
|
request.socket.on('close', function () { |
|
controller.abort(); |
|
}); |
|
|
|
const apiVersion = getConfigValue('gemini.apiVersion', 'v1beta'); |
|
const responseType = (stream ? 'streamGenerateContent' : 'generateContent'); |
|
|
|
let url; |
|
if (useVertexAi) { |
|
url = `${apiUrl.toString().replace(/\/$/, '')}/v1/publishers/google/models/${model}:${responseType}?key=${apiKey}${stream ? '&alt=sse' : ''}`; |
|
} else { |
|
url = `${apiUrl.toString().replace(/\/$/, '')}/${apiVersion}/models/${model}:${responseType}?key=${apiKey}${stream ? '&alt=sse' : ''}`; |
|
} |
|
const generateResponse = await fetch(url, { |
|
body: JSON.stringify(body), |
|
method: 'POST', |
|
headers: { |
|
'Content-Type': 'application/json', |
|
}, |
|
signal: controller.signal, |
|
}); |
|
|
|
if (stream) { |
|
try { |
|
|
|
forwardFetchResponse(generateResponse, response); |
|
} catch (error) { |
|
console.error('Error forwarding streaming response:', error); |
|
if (!response.headersSent) { |
|
return response.status(500).send({ error: true }); |
|
} |
|
} |
|
} else { |
|
if (!generateResponse.ok) { |
|
console.warn(`${apiName} API returned error: ${generateResponse.status} ${generateResponse.statusText} ${await generateResponse.text()}`); |
|
return response.status(500).send({ error: true }); |
|
} |
|
|
|
|
|
const generateResponseJson = await generateResponse.json(); |
|
|
|
const candidates = generateResponseJson?.candidates; |
|
if (!candidates || candidates.length === 0) { |
|
let message = `${apiName} API returned no candidate`; |
|
console.warn(message, generateResponseJson); |
|
if (generateResponseJson?.promptFeedback?.blockReason) { |
|
message += `\nPrompt was blocked due to : ${generateResponseJson.promptFeedback.blockReason}`; |
|
} |
|
return response.send({ error: { message } }); |
|
} |
|
|
|
const responseContent = candidates[0].content ?? candidates[0].output; |
|
const functionCall = (candidates?.[0]?.content?.parts ?? []).some(part => part.functionCall); |
|
const inlineData = (candidates?.[0]?.content?.parts ?? []).some(part => part.inlineData); |
|
console.debug(`${apiName} response:`, util.inspect(generateResponseJson, { depth: 5, colors: true })); |
|
|
|
const responseText = typeof responseContent === 'string' ? responseContent : responseContent?.parts?.filter(part => !part.thought)?.map(part => part.text)?.join('\n\n'); |
|
if (!responseText && !functionCall && !inlineData) { |
|
let message = `${apiName} Candidate text empty`; |
|
console.warn(message, generateResponseJson); |
|
return response.send({ error: { message } }); |
|
} |
|
|
|
|
|
const reply = { choices: [{ 'message': { 'content': responseText } }], responseContent }; |
|
return response.send(reply); |
|
} |
|
} catch (error) { |
|
console.error(`Error communicating with ${apiName} API:`, error); |
|
if (!response.headersSent) { |
|
return response.status(500).send({ error: true }); |
|
} |
|
} |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
async function sendAI21Request(request, response) { |
|
if (!request.body) return response.sendStatus(400); |
|
|
|
const apiKey = readSecret(request.user.directories, SECRET_KEYS.AI21); |
|
if (!apiKey) { |
|
console.warn('AI21 API key is missing.'); |
|
return response.status(400).send({ error: true }); |
|
} |
|
|
|
const controller = new AbortController(); |
|
console.debug(request.body.messages); |
|
request.socket.removeAllListeners('close'); |
|
request.socket.on('close', function () { |
|
controller.abort(); |
|
}); |
|
const convertedPrompt = convertAI21Messages(request.body.messages, getPromptNames(request)); |
|
const body = { |
|
messages: convertedPrompt, |
|
model: request.body.model, |
|
max_tokens: request.body.max_tokens, |
|
temperature: request.body.temperature, |
|
top_p: request.body.top_p, |
|
stop: request.body.stop, |
|
stream: request.body.stream, |
|
tools: request.body.tools, |
|
}; |
|
const options = { |
|
method: 'POST', |
|
headers: { |
|
accept: 'application/json', |
|
'content-type': 'application/json', |
|
Authorization: `Bearer ${apiKey}`, |
|
}, |
|
body: JSON.stringify(body), |
|
signal: controller.signal, |
|
}; |
|
|
|
console.debug('AI21 request:', body); |
|
|
|
try { |
|
const generateResponse = await fetch(API_AI21 + '/chat/completions', options); |
|
if (request.body.stream) { |
|
forwardFetchResponse(generateResponse, response); |
|
} else { |
|
if (!generateResponse.ok) { |
|
const errorText = await generateResponse.text(); |
|
console.warn(`AI21 API returned error: ${generateResponse.status} ${generateResponse.statusText} ${errorText}`); |
|
const errorJson = tryParse(errorText) ?? { error: true }; |
|
return response.status(500).send(errorJson); |
|
} |
|
const generateResponseJson = await generateResponse.json(); |
|
console.debug('AI21 response:', generateResponseJson); |
|
return response.send(generateResponseJson); |
|
} |
|
} catch (error) { |
|
console.error('Error communicating with AI21 API: ', error); |
|
if (!response.headersSent) { |
|
response.send({ error: true }); |
|
} else { |
|
response.end(); |
|
} |
|
} |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
async function sendMistralAIRequest(request, response) { |
|
const apiUrl = new URL(request.body.reverse_proxy || API_MISTRAL).toString(); |
|
const apiKey = request.body.reverse_proxy ? request.body.proxy_password : readSecret(request.user.directories, SECRET_KEYS.MISTRALAI); |
|
|
|
if (!apiKey) { |
|
console.warn('MistralAI API key is missing.'); |
|
return response.status(400).send({ error: true }); |
|
} |
|
|
|
try { |
|
const messages = convertMistralMessages(request.body.messages, getPromptNames(request)); |
|
const controller = new AbortController(); |
|
request.socket.removeAllListeners('close'); |
|
request.socket.on('close', function () { |
|
controller.abort(); |
|
}); |
|
|
|
const requestBody = { |
|
'model': request.body.model, |
|
'messages': messages, |
|
'temperature': request.body.temperature, |
|
'top_p': request.body.top_p, |
|
'frequency_penalty': request.body.frequency_penalty, |
|
'presence_penalty': request.body.presence_penalty, |
|
'max_tokens': request.body.max_tokens, |
|
'stream': request.body.stream, |
|
'safe_prompt': request.body.safe_prompt, |
|
'random_seed': request.body.seed === -1 ? undefined : request.body.seed, |
|
'stop': Array.isArray(request.body.stop) && request.body.stop.length > 0 ? request.body.stop : undefined, |
|
}; |
|
|
|
if (Array.isArray(request.body.tools) && request.body.tools.length > 0) { |
|
requestBody['tools'] = request.body.tools; |
|
requestBody['tool_choice'] = request.body.tool_choice; |
|
} |
|
|
|
const config = { |
|
method: 'POST', |
|
headers: { |
|
'Content-Type': 'application/json', |
|
'Authorization': 'Bearer ' + apiKey, |
|
}, |
|
body: JSON.stringify(requestBody), |
|
signal: controller.signal, |
|
timeout: 0, |
|
}; |
|
|
|
console.debug('MisralAI request:', requestBody); |
|
|
|
const generateResponse = await fetch(apiUrl + '/chat/completions', config); |
|
if (request.body.stream) { |
|
forwardFetchResponse(generateResponse, response); |
|
} else { |
|
if (!generateResponse.ok) { |
|
const errorText = await generateResponse.text(); |
|
console.warn(`MistralAI API returned error: ${generateResponse.status} ${generateResponse.statusText} ${errorText}`); |
|
const errorJson = tryParse(errorText) ?? { error: true }; |
|
return response.status(500).send(errorJson); |
|
} |
|
const generateResponseJson = await generateResponse.json(); |
|
console.debug('MistralAI response:', generateResponseJson); |
|
return response.send(generateResponseJson); |
|
} |
|
} catch (error) { |
|
console.error('Error communicating with MistralAI API: ', error); |
|
if (!response.headersSent) { |
|
response.send({ error: true }); |
|
} else { |
|
response.end(); |
|
} |
|
} |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
async function sendCohereRequest(request, response) { |
|
const apiKey = readSecret(request.user.directories, SECRET_KEYS.COHERE); |
|
const controller = new AbortController(); |
|
request.socket.removeAllListeners('close'); |
|
request.socket.on('close', function () { |
|
controller.abort(); |
|
}); |
|
|
|
if (!apiKey) { |
|
console.warn('Cohere API key is missing.'); |
|
return response.status(400).send({ error: true }); |
|
} |
|
|
|
try { |
|
const convertedHistory = convertCohereMessages(request.body.messages, getPromptNames(request)); |
|
const tools = []; |
|
|
|
if (Array.isArray(request.body.tools) && request.body.tools.length > 0) { |
|
tools.push(...request.body.tools); |
|
tools.forEach(tool => { |
|
if (tool?.function?.parameters?.$schema) { |
|
delete tool.function.parameters.$schema; |
|
} |
|
}); |
|
} |
|
|
|
|
|
const requestBody = { |
|
stream: Boolean(request.body.stream), |
|
model: request.body.model, |
|
messages: convertedHistory.chatHistory, |
|
temperature: request.body.temperature, |
|
max_tokens: request.body.max_tokens, |
|
k: request.body.top_k, |
|
p: request.body.top_p, |
|
seed: request.body.seed, |
|
stop_sequences: request.body.stop, |
|
frequency_penalty: request.body.frequency_penalty, |
|
presence_penalty: request.body.presence_penalty, |
|
documents: [], |
|
tools: tools, |
|
}; |
|
|
|
const canDoSafetyMode = String(request.body.model).endsWith('08-2024'); |
|
if (canDoSafetyMode) { |
|
requestBody.safety_mode = 'OFF'; |
|
} |
|
|
|
console.debug('Cohere request:', requestBody); |
|
|
|
const config = { |
|
method: 'POST', |
|
headers: { |
|
'Content-Type': 'application/json', |
|
'Authorization': 'Bearer ' + apiKey, |
|
}, |
|
body: JSON.stringify(requestBody), |
|
signal: controller.signal, |
|
timeout: 0, |
|
}; |
|
|
|
const apiUrl = API_COHERE_V2 + '/chat'; |
|
|
|
if (request.body.stream) { |
|
const stream = await fetch(apiUrl, config); |
|
forwardFetchResponse(stream, response); |
|
} else { |
|
const generateResponse = await fetch(apiUrl, config); |
|
if (!generateResponse.ok) { |
|
const errorText = await generateResponse.text(); |
|
console.warn(`Cohere API returned error: ${generateResponse.status} ${generateResponse.statusText} ${errorText}`); |
|
const errorJson = tryParse(errorText) ?? { error: true }; |
|
return response.status(500).send(errorJson); |
|
} |
|
const generateResponseJson = await generateResponse.json(); |
|
console.debug('Cohere response:', generateResponseJson); |
|
return response.send(generateResponseJson); |
|
} |
|
} catch (error) { |
|
console.error('Error communicating with Cohere API: ', error); |
|
if (!response.headersSent) { |
|
response.send({ error: true }); |
|
} else { |
|
response.end(); |
|
} |
|
} |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
async function sendDeepSeekRequest(request, response) { |
|
const apiUrl = new URL(request.body.reverse_proxy || API_DEEPSEEK).toString(); |
|
const apiKey = request.body.reverse_proxy ? request.body.proxy_password : readSecret(request.user.directories, SECRET_KEYS.DEEPSEEK); |
|
|
|
if (!apiKey && !request.body.reverse_proxy) { |
|
console.warn('DeepSeek API key is missing.'); |
|
return response.status(400).send({ error: true }); |
|
} |
|
|
|
const controller = new AbortController(); |
|
request.socket.removeAllListeners('close'); |
|
request.socket.on('close', function () { |
|
controller.abort(); |
|
}); |
|
|
|
try { |
|
let bodyParams = {}; |
|
|
|
if (request.body.logprobs > 0) { |
|
bodyParams['top_logprobs'] = request.body.logprobs; |
|
bodyParams['logprobs'] = true; |
|
} |
|
|
|
if (Array.isArray(request.body.tools) && request.body.tools.length > 0) { |
|
bodyParams['tools'] = request.body.tools; |
|
bodyParams['tool_choice'] = request.body.tool_choice; |
|
|
|
|
|
bodyParams.tools.forEach(tool => { |
|
const required = tool?.function?.parameters?.required; |
|
if (Array.isArray(required) && required.length === 0) { |
|
delete tool.function.parameters.required; |
|
} |
|
}); |
|
} |
|
|
|
const postProcessType = String(request.body.model).endsWith('-reasoner') ? 'deepseek-reasoner' : 'deepseek'; |
|
const processedMessages = postProcessPrompt(request.body.messages, postProcessType, getPromptNames(request)); |
|
|
|
const requestBody = { |
|
'messages': processedMessages, |
|
'model': request.body.model, |
|
'temperature': request.body.temperature, |
|
'max_tokens': request.body.max_tokens, |
|
'stream': request.body.stream, |
|
'presence_penalty': request.body.presence_penalty, |
|
'frequency_penalty': request.body.frequency_penalty, |
|
'top_p': request.body.top_p, |
|
'stop': request.body.stop, |
|
'seed': request.body.seed, |
|
...bodyParams, |
|
}; |
|
|
|
const config = { |
|
method: 'POST', |
|
headers: { |
|
'Content-Type': 'application/json', |
|
'Authorization': 'Bearer ' + apiKey, |
|
}, |
|
body: JSON.stringify(requestBody), |
|
signal: controller.signal, |
|
}; |
|
|
|
console.debug('DeepSeek request:', requestBody); |
|
|
|
const generateResponse = await fetch(apiUrl + '/chat/completions', config); |
|
|
|
if (request.body.stream) { |
|
forwardFetchResponse(generateResponse, response); |
|
} else { |
|
if (!generateResponse.ok) { |
|
const errorText = await generateResponse.text(); |
|
console.warn(`DeepSeek API returned error: ${generateResponse.status} ${generateResponse.statusText} ${errorText}`); |
|
const errorJson = tryParse(errorText) ?? { error: true }; |
|
return response.status(500).send(errorJson); |
|
} |
|
const generateResponseJson = await generateResponse.json(); |
|
console.debug('DeepSeek response:', generateResponseJson); |
|
return response.send(generateResponseJson); |
|
} |
|
} catch (error) { |
|
console.error('Error communicating with DeepSeek API: ', error); |
|
if (!response.headersSent) { |
|
response.send({ error: true }); |
|
} else { |
|
response.end(); |
|
} |
|
} |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
async function sendXaiRequest(request, response) { |
|
const apiUrl = new URL(request.body.reverse_proxy || API_XAI).toString(); |
|
const apiKey = request.body.reverse_proxy ? request.body.proxy_password : readSecret(request.user.directories, SECRET_KEYS.XAI); |
|
|
|
if (!apiKey && !request.body.reverse_proxy) { |
|
console.warn('xAI API key is missing.'); |
|
return response.status(400).send({ error: true }); |
|
} |
|
|
|
const controller = new AbortController(); |
|
request.socket.removeAllListeners('close'); |
|
request.socket.on('close', function () { |
|
controller.abort(); |
|
}); |
|
|
|
try { |
|
let bodyParams = {}; |
|
|
|
if (request.body.logprobs > 0) { |
|
bodyParams['top_logprobs'] = request.body.logprobs; |
|
bodyParams['logprobs'] = true; |
|
} |
|
|
|
if (Array.isArray(request.body.tools) && request.body.tools.length > 0) { |
|
bodyParams['tools'] = request.body.tools; |
|
bodyParams['tool_choice'] = request.body.tool_choice; |
|
} |
|
|
|
if (Array.isArray(request.body.stop) && request.body.stop.length > 0) { |
|
bodyParams['stop'] = request.body.stop; |
|
} |
|
|
|
if (request.body.reasoning_effort && ['grok-3-mini-beta', 'grok-3-mini-fast-beta'].includes(request.body.model)) { |
|
bodyParams['reasoning_effort'] = request.body.reasoning_effort === 'high' ? 'high' : 'low'; |
|
} |
|
|
|
const processedMessages = request.body.messages = convertXAIMessages(request.body.messages, getPromptNames(request)); |
|
|
|
const requestBody = { |
|
'messages': processedMessages, |
|
'model': request.body.model, |
|
'temperature': request.body.temperature, |
|
'max_tokens': request.body.max_tokens, |
|
'max_completion_tokens': request.body.max_completion_tokens, |
|
'stream': request.body.stream, |
|
'presence_penalty': request.body.presence_penalty, |
|
'frequency_penalty': request.body.frequency_penalty, |
|
'top_p': request.body.top_p, |
|
'seed': request.body.seed, |
|
'n': request.body.n, |
|
...bodyParams, |
|
}; |
|
|
|
const config = { |
|
method: 'POST', |
|
headers: { |
|
'Content-Type': 'application/json', |
|
'Authorization': 'Bearer ' + apiKey, |
|
}, |
|
body: JSON.stringify(requestBody), |
|
signal: controller.signal, |
|
}; |
|
|
|
console.debug('xAI request:', requestBody); |
|
|
|
const generateResponse = await fetch(apiUrl + '/chat/completions', config); |
|
|
|
if (request.body.stream) { |
|
forwardFetchResponse(generateResponse, response); |
|
} else { |
|
if (!generateResponse.ok) { |
|
const errorText = await generateResponse.text(); |
|
console.warn(`xAI API returned error: ${generateResponse.status} ${generateResponse.statusText} ${errorText}`); |
|
const errorJson = tryParse(errorText) ?? { error: true }; |
|
return response.status(500).send(errorJson); |
|
} |
|
const generateResponseJson = await generateResponse.json(); |
|
console.debug('xAI response:', generateResponseJson); |
|
return response.send(generateResponseJson); |
|
} |
|
} catch (error) { |
|
console.error('Error communicating with xAI API: ', error); |
|
if (!response.headersSent) { |
|
response.send({ error: true }); |
|
} else { |
|
response.end(); |
|
} |
|
} |
|
} |
|
|
|
export const router = express.Router(); |
|
|
|
router.post('/status', async function (request, response_getstatus_openai) { |
|
if (!request.body) return response_getstatus_openai.sendStatus(400); |
|
|
|
let api_url; |
|
let api_key_openai; |
|
let headers; |
|
|
|
if (request.body.chat_completion_source === CHAT_COMPLETION_SOURCES.OPENAI) { |
|
api_url = new URL(request.body.reverse_proxy || API_OPENAI).toString(); |
|
api_key_openai = request.body.reverse_proxy ? request.body.proxy_password : readSecret(request.user.directories, SECRET_KEYS.OPENAI); |
|
headers = {}; |
|
} else if (request.body.chat_completion_source === CHAT_COMPLETION_SOURCES.OPENROUTER) { |
|
api_url = 'https://openrouter.ai/api/v1'; |
|
api_key_openai = readSecret(request.user.directories, SECRET_KEYS.OPENROUTER); |
|
|
|
headers = { ...OPENROUTER_HEADERS }; |
|
} else if (request.body.chat_completion_source === CHAT_COMPLETION_SOURCES.MISTRALAI) { |
|
api_url = new URL(request.body.reverse_proxy || API_MISTRAL).toString(); |
|
api_key_openai = request.body.reverse_proxy ? request.body.proxy_password : readSecret(request.user.directories, SECRET_KEYS.MISTRALAI); |
|
headers = {}; |
|
} else if (request.body.chat_completion_source === CHAT_COMPLETION_SOURCES.CUSTOM) { |
|
api_url = request.body.custom_url; |
|
api_key_openai = readSecret(request.user.directories, SECRET_KEYS.CUSTOM); |
|
headers = {}; |
|
mergeObjectWithYaml(headers, request.body.custom_include_headers); |
|
} else if (request.body.chat_completion_source === CHAT_COMPLETION_SOURCES.COHERE) { |
|
api_url = API_COHERE_V1; |
|
api_key_openai = readSecret(request.user.directories, SECRET_KEYS.COHERE); |
|
headers = {}; |
|
} else if (request.body.chat_completion_source === CHAT_COMPLETION_SOURCES.ZEROONEAI) { |
|
api_url = API_01AI; |
|
api_key_openai = readSecret(request.user.directories, SECRET_KEYS.ZEROONEAI); |
|
headers = {}; |
|
} else if (request.body.chat_completion_source === CHAT_COMPLETION_SOURCES.NANOGPT) { |
|
api_url = API_NANOGPT; |
|
api_key_openai = readSecret(request.user.directories, SECRET_KEYS.NANOGPT); |
|
headers = {}; |
|
} else if (request.body.chat_completion_source === CHAT_COMPLETION_SOURCES.DEEPSEEK) { |
|
api_url = new URL(request.body.reverse_proxy || API_DEEPSEEK.replace('/beta', '')); |
|
api_key_openai = request.body.reverse_proxy ? request.body.proxy_password : readSecret(request.user.directories, SECRET_KEYS.DEEPSEEK); |
|
headers = {}; |
|
} else if (request.body.chat_completion_source === CHAT_COMPLETION_SOURCES.XAI) { |
|
api_url = new URL(request.body.reverse_proxy || API_XAI); |
|
api_key_openai = request.body.reverse_proxy ? request.body.proxy_password : readSecret(request.user.directories, SECRET_KEYS.XAI); |
|
headers = {}; |
|
} else if (request.body.chat_completion_source === CHAT_COMPLETION_SOURCES.POLLINATIONS) { |
|
api_url = 'https://text.pollinations.ai'; |
|
api_key_openai = 'NONE'; |
|
headers = {}; |
|
} else { |
|
console.warn('This chat completion source is not supported yet.'); |
|
return response_getstatus_openai.status(400).send({ error: true }); |
|
} |
|
|
|
if (!api_key_openai && !request.body.reverse_proxy && request.body.chat_completion_source !== CHAT_COMPLETION_SOURCES.CUSTOM) { |
|
console.warn('Chat Completion API key is missing.'); |
|
return response_getstatus_openai.status(400).send({ error: true }); |
|
} |
|
|
|
try { |
|
const response = await fetch(api_url + '/models', { |
|
method: 'GET', |
|
headers: { |
|
'Authorization': 'Bearer ' + api_key_openai, |
|
...headers, |
|
}, |
|
}); |
|
|
|
if (response.ok) { |
|
|
|
let data = await response.json(); |
|
|
|
if (request.body.chat_completion_source === CHAT_COMPLETION_SOURCES.POLLINATIONS && Array.isArray(data)) { |
|
data = { data: data.map(model => ({ id: model.name, ...model })) }; |
|
} |
|
|
|
response_getstatus_openai.send(data); |
|
|
|
if (request.body.chat_completion_source === CHAT_COMPLETION_SOURCES.COHERE && Array.isArray(data?.models)) { |
|
data.data = data.models.map(model => ({ id: model.name, ...model })); |
|
} |
|
|
|
if (request.body.chat_completion_source === CHAT_COMPLETION_SOURCES.OPENROUTER && Array.isArray(data?.data)) { |
|
let models = []; |
|
|
|
data.data.forEach(model => { |
|
const context_length = model.context_length; |
|
const tokens_dollar = Number(1 / (1000 * model.pricing?.prompt)); |
|
const tokens_rounded = (Math.round(tokens_dollar * 1000) / 1000).toFixed(0); |
|
models[model.id] = { |
|
tokens_per_dollar: tokens_rounded + 'k', |
|
context_length: context_length, |
|
}; |
|
}); |
|
|
|
console.info('Available OpenRouter models:', models); |
|
} else if (request.body.chat_completion_source === CHAT_COMPLETION_SOURCES.MISTRALAI) { |
|
const models = data?.data; |
|
console.info(models); |
|
} else { |
|
const models = data?.data; |
|
|
|
if (Array.isArray(models)) { |
|
const modelIds = models.filter(x => x && typeof x === 'object').map(x => x.id).sort(); |
|
console.info('Available models:', modelIds); |
|
} else { |
|
console.warn('Chat Completion endpoint did not return a list of models.'); |
|
} |
|
} |
|
} |
|
else { |
|
console.error('Chat Completion status check failed. Either Access Token is incorrect or API endpoint is down.'); |
|
response_getstatus_openai.send({ error: true, can_bypass: true, data: { data: [] } }); |
|
} |
|
} catch (e) { |
|
console.error(e); |
|
|
|
if (!response_getstatus_openai.headersSent) { |
|
response_getstatus_openai.send({ error: true }); |
|
} else { |
|
response_getstatus_openai.end(); |
|
} |
|
} |
|
}); |
|
|
|
router.post('/bias', async function (request, response) { |
|
if (!request.body || !Array.isArray(request.body)) |
|
return response.sendStatus(400); |
|
|
|
try { |
|
const result = {}; |
|
const model = getTokenizerModel(String(request.query.model || '')); |
|
|
|
|
|
if (model == 'claude') { |
|
return response.send(result); |
|
} |
|
|
|
let encodeFunction; |
|
|
|
if (sentencepieceTokenizers.includes(model)) { |
|
const tokenizer = getSentencepiceTokenizer(model); |
|
const instance = await tokenizer?.get(); |
|
if (!instance) { |
|
console.error('Tokenizer not initialized:', model); |
|
return response.send({}); |
|
} |
|
encodeFunction = (text) => new Uint32Array(instance.encodeIds(text)); |
|
} else if (webTokenizers.includes(model)) { |
|
const tokenizer = getWebTokenizer(model); |
|
const instance = await tokenizer?.get(); |
|
if (!instance) { |
|
console.warn('Tokenizer not initialized:', model); |
|
return response.send({}); |
|
} |
|
encodeFunction = (text) => new Uint32Array(instance.encode(text)); |
|
} else { |
|
const tokenizer = getTiktokenTokenizer(model); |
|
encodeFunction = (tokenizer.encode.bind(tokenizer)); |
|
} |
|
|
|
for (const entry of request.body) { |
|
if (!entry || !entry.text) { |
|
continue; |
|
} |
|
|
|
try { |
|
const tokens = getEntryTokens(entry.text, encodeFunction); |
|
|
|
for (const token of tokens) { |
|
result[token] = entry.value; |
|
} |
|
} catch { |
|
console.warn('Tokenizer failed to encode:', entry.text); |
|
} |
|
} |
|
|
|
|
|
|
|
return response.send(result); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function getEntryTokens(text, encode) { |
|
|
|
if (text.trim().startsWith('[') && text.trim().endsWith(']')) { |
|
try { |
|
const json = JSON.parse(text); |
|
if (Array.isArray(json) && json.every(x => typeof x === 'number')) { |
|
return new Uint32Array(json); |
|
} |
|
} catch { |
|
|
|
} |
|
} |
|
|
|
|
|
return encode(text); |
|
} |
|
} catch (error) { |
|
console.error(error); |
|
return response.send({}); |
|
} |
|
}); |
|
|
|
|
|
router.post('/generate', function (request, response) { |
|
if (!request.body) return response.status(400).send({ error: true }); |
|
|
|
const postProcessingType = request.body.custom_prompt_post_processing; |
|
if (Array.isArray(request.body.messages) && postProcessingType) { |
|
console.info('Applying custom prompt post-processing of type', postProcessingType); |
|
request.body.messages = postProcessPrompt( |
|
request.body.messages, |
|
postProcessingType, |
|
getPromptNames(request)); |
|
} |
|
|
|
switch (request.body.chat_completion_source) { |
|
case CHAT_COMPLETION_SOURCES.CLAUDE: return sendClaudeRequest(request, response); |
|
case CHAT_COMPLETION_SOURCES.SCALE: return sendScaleRequest(request, response); |
|
case CHAT_COMPLETION_SOURCES.AI21: return sendAI21Request(request, response); |
|
case CHAT_COMPLETION_SOURCES.MAKERSUITE: return sendMakerSuiteRequest(request, response); |
|
case CHAT_COMPLETION_SOURCES.VERTEXAI: return sendMakerSuiteRequest(request, response); |
|
case CHAT_COMPLETION_SOURCES.MISTRALAI: return sendMistralAIRequest(request, response); |
|
case CHAT_COMPLETION_SOURCES.COHERE: return sendCohereRequest(request, response); |
|
case CHAT_COMPLETION_SOURCES.DEEPSEEK: return sendDeepSeekRequest(request, response); |
|
case CHAT_COMPLETION_SOURCES.XAI: return sendXaiRequest(request, response); |
|
} |
|
|
|
let apiUrl; |
|
let apiKey; |
|
let headers; |
|
let bodyParams; |
|
const isTextCompletion = Boolean(request.body.model && TEXT_COMPLETION_MODELS.includes(request.body.model)) || typeof request.body.messages === 'string'; |
|
|
|
if (request.body.chat_completion_source === CHAT_COMPLETION_SOURCES.OPENAI) { |
|
apiUrl = new URL(request.body.reverse_proxy || API_OPENAI).toString(); |
|
apiKey = request.body.reverse_proxy ? request.body.proxy_password : readSecret(request.user.directories, SECRET_KEYS.OPENAI); |
|
headers = {}; |
|
bodyParams = { |
|
logprobs: request.body.logprobs, |
|
top_logprobs: undefined, |
|
}; |
|
|
|
|
|
if (!isTextCompletion && bodyParams.logprobs > 0) { |
|
bodyParams.top_logprobs = bodyParams.logprobs; |
|
bodyParams.logprobs = true; |
|
} |
|
|
|
if (getConfigValue('openai.randomizeUserId', false, 'boolean')) { |
|
bodyParams['user'] = uuidv4(); |
|
} |
|
} else if (request.body.chat_completion_source === CHAT_COMPLETION_SOURCES.OPENROUTER) { |
|
apiUrl = 'https://openrouter.ai/api/v1'; |
|
apiKey = readSecret(request.user.directories, SECRET_KEYS.OPENROUTER); |
|
|
|
headers = { ...OPENROUTER_HEADERS }; |
|
bodyParams = { |
|
'transforms': getOpenRouterTransforms(request), |
|
'plugins': getOpenRouterPlugins(request), |
|
'include_reasoning': Boolean(request.body.include_reasoning), |
|
}; |
|
|
|
if (request.body.min_p !== undefined) { |
|
bodyParams['min_p'] = request.body.min_p; |
|
} |
|
|
|
if (request.body.top_a !== undefined) { |
|
bodyParams['top_a'] = request.body.top_a; |
|
} |
|
|
|
if (request.body.repetition_penalty !== undefined) { |
|
bodyParams['repetition_penalty'] = request.body.repetition_penalty; |
|
} |
|
|
|
if (Array.isArray(request.body.provider) && request.body.provider.length > 0) { |
|
bodyParams['provider'] = { |
|
allow_fallbacks: request.body.allow_fallbacks ?? true, |
|
order: request.body.provider ?? [], |
|
}; |
|
} |
|
|
|
if (request.body.use_fallback) { |
|
bodyParams['route'] = 'fallback'; |
|
} |
|
|
|
if (request.body.reasoning_effort) { |
|
bodyParams['reasoning'] = { effort: request.body.reasoning_effort }; |
|
} |
|
|
|
let cachingAtDepth = getConfigValue('claude.cachingAtDepth', -1, 'number'); |
|
const isClaude3or4 = /anthropic\/claude-(3|opus-4|sonnet-4)/.test(request.body.model); |
|
if (Number.isInteger(cachingAtDepth) && cachingAtDepth >= 0 && isClaude3or4) { |
|
cachingAtDepthForOpenRouterClaude(request.body.messages, cachingAtDepth); |
|
} |
|
} else if (request.body.chat_completion_source === CHAT_COMPLETION_SOURCES.CUSTOM) { |
|
apiUrl = request.body.custom_url; |
|
apiKey = readSecret(request.user.directories, SECRET_KEYS.CUSTOM); |
|
headers = {}; |
|
bodyParams = { |
|
logprobs: request.body.logprobs, |
|
top_logprobs: undefined, |
|
}; |
|
|
|
|
|
if (!isTextCompletion && bodyParams.logprobs > 0) { |
|
bodyParams.top_logprobs = bodyParams.logprobs; |
|
bodyParams.logprobs = true; |
|
} |
|
|
|
mergeObjectWithYaml(bodyParams, request.body.custom_include_body); |
|
mergeObjectWithYaml(headers, request.body.custom_include_headers); |
|
} else if (request.body.chat_completion_source === CHAT_COMPLETION_SOURCES.PERPLEXITY) { |
|
apiUrl = API_PERPLEXITY; |
|
apiKey = readSecret(request.user.directories, SECRET_KEYS.PERPLEXITY); |
|
headers = {}; |
|
bodyParams = {}; |
|
request.body.messages = postProcessPrompt(request.body.messages, 'strict', getPromptNames(request)); |
|
} else if (request.body.chat_completion_source === CHAT_COMPLETION_SOURCES.GROQ) { |
|
apiUrl = API_GROQ; |
|
apiKey = readSecret(request.user.directories, SECRET_KEYS.GROQ); |
|
headers = {}; |
|
bodyParams = {}; |
|
} else if (request.body.chat_completion_source === CHAT_COMPLETION_SOURCES.NANOGPT) { |
|
apiUrl = API_NANOGPT; |
|
apiKey = readSecret(request.user.directories, SECRET_KEYS.NANOGPT); |
|
headers = {}; |
|
bodyParams = {}; |
|
} else if (request.body.chat_completion_source === CHAT_COMPLETION_SOURCES.ZEROONEAI) { |
|
apiUrl = API_01AI; |
|
apiKey = readSecret(request.user.directories, SECRET_KEYS.ZEROONEAI); |
|
headers = {}; |
|
bodyParams = {}; |
|
} else if (request.body.chat_completion_source === CHAT_COMPLETION_SOURCES.POLLINATIONS) { |
|
apiUrl = API_POLLINATIONS; |
|
apiKey = 'NONE'; |
|
headers = {}; |
|
bodyParams = { |
|
reasoning_effort: request.body.reasoning_effort, |
|
private: true, |
|
}; |
|
} else { |
|
console.warn('This chat completion source is not supported yet.'); |
|
return response.status(400).send({ error: true }); |
|
} |
|
|
|
|
|
if (request.body.reasoning_effort && [CHAT_COMPLETION_SOURCES.CUSTOM, CHAT_COMPLETION_SOURCES.OPENAI].includes(request.body.chat_completion_source)) { |
|
if (['o1', 'o3-mini', 'o3-mini-2025-01-31', 'o4-mini', 'o4-mini-2025-04-16', 'o3', 'o3-2025-04-16'].includes(request.body.model)) { |
|
bodyParams['reasoning_effort'] = request.body.reasoning_effort; |
|
} |
|
} |
|
|
|
if (!apiKey && !request.body.reverse_proxy && request.body.chat_completion_source !== CHAT_COMPLETION_SOURCES.CUSTOM) { |
|
console.warn('OpenAI API key is missing.'); |
|
return response.status(400).send({ error: true }); |
|
} |
|
|
|
|
|
if (Array.isArray(request.body.stop) && request.body.stop.length > 0) { |
|
bodyParams['stop'] = request.body.stop; |
|
} |
|
|
|
const textPrompt = isTextCompletion ? convertTextCompletionPrompt(request.body.messages) : ''; |
|
const endpointUrl = isTextCompletion && request.body.chat_completion_source !== CHAT_COMPLETION_SOURCES.OPENROUTER ? |
|
`${apiUrl}/completions` : |
|
`${apiUrl}/chat/completions`; |
|
|
|
const controller = new AbortController(); |
|
request.socket.removeAllListeners('close'); |
|
request.socket.on('close', function () { |
|
controller.abort(); |
|
}); |
|
|
|
if (!isTextCompletion && Array.isArray(request.body.tools) && request.body.tools.length > 0) { |
|
bodyParams['tools'] = request.body.tools; |
|
bodyParams['tool_choice'] = request.body.tool_choice; |
|
} |
|
|
|
const requestBody = { |
|
'messages': isTextCompletion === false ? request.body.messages : undefined, |
|
'prompt': isTextCompletion === true ? textPrompt : undefined, |
|
'model': request.body.model, |
|
'temperature': request.body.temperature, |
|
'max_tokens': request.body.max_tokens, |
|
'max_completion_tokens': request.body.max_completion_tokens, |
|
'stream': request.body.stream, |
|
'presence_penalty': request.body.presence_penalty, |
|
'frequency_penalty': request.body.frequency_penalty, |
|
'top_p': request.body.top_p, |
|
'top_k': request.body.top_k, |
|
'stop': isTextCompletion === false ? request.body.stop : undefined, |
|
'logit_bias': request.body.logit_bias, |
|
'seed': request.body.seed, |
|
'n': request.body.n, |
|
...bodyParams, |
|
}; |
|
|
|
if (request.body.chat_completion_source === CHAT_COMPLETION_SOURCES.CUSTOM) { |
|
excludeKeysByYaml(requestBody, request.body.custom_exclude_body); |
|
} |
|
|
|
|
|
const config = { |
|
method: 'post', |
|
headers: { |
|
'Content-Type': 'application/json', |
|
'Authorization': 'Bearer ' + apiKey, |
|
...headers, |
|
}, |
|
body: JSON.stringify(requestBody), |
|
signal: controller.signal, |
|
}; |
|
|
|
console.debug(requestBody); |
|
|
|
makeRequest(config, response, request); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
async function makeRequest(config, response, request, retries = 5, timeout = 5000) { |
|
try { |
|
controller.signal.throwIfAborted(); |
|
const fetchResponse = await fetch(endpointUrl, config); |
|
|
|
if (request.body.stream) { |
|
console.info('Streaming request in progress'); |
|
forwardFetchResponse(fetchResponse, response); |
|
return; |
|
} |
|
|
|
if (fetchResponse.ok) { |
|
|
|
let json = await fetchResponse.json(); |
|
response.send(json); |
|
console.debug(json); |
|
console.debug(json?.choices?.[0]?.message); |
|
} else if (fetchResponse.status === 429 && retries > 0) { |
|
console.warn(`Out of quota, retrying in ${Math.round(timeout / 1000)}s`); |
|
setTimeout(() => { |
|
timeout *= 2; |
|
makeRequest(config, response, request, retries - 1, timeout); |
|
}, timeout); |
|
} else { |
|
await handleErrorResponse(fetchResponse); |
|
} |
|
} catch (error) { |
|
console.error('Generation failed', error); |
|
const message = error.code === 'ECONNREFUSED' |
|
? `Connection refused: ${error.message}` |
|
: error.message || 'Unknown error occurred'; |
|
|
|
if (!response.headersSent) { |
|
response.status(502).send({ error: { message, ...error } }); |
|
} else { |
|
response.end(); |
|
} |
|
} |
|
} |
|
|
|
|
|
|
|
|
|
async function handleErrorResponse(errorResponse) { |
|
const responseText = await errorResponse.text(); |
|
const errorData = tryParse(responseText); |
|
|
|
const message = errorResponse.statusText || 'Unknown error occurred'; |
|
const quota_error = errorResponse.status === 429 && errorData?.error?.type === 'insufficient_quota'; |
|
console.error('Chat completion request error: ', message, responseText); |
|
|
|
if (!response.headersSent) { |
|
response.send({ error: { message }, quota_error: quota_error }); |
|
} else if (!response.writableEnded) { |
|
response.write(errorResponse); |
|
} else { |
|
response.end(); |
|
} |
|
} |
|
}); |
|
|