From 084e8be811150e513d95715a501f48a397387373 Mon Sep 17 00:00:00 2001 From: Wx-2025 <351320169@qq.com> Date: Tue, 6 Jan 2026 10:43:50 +0800 Subject: [PATCH] Create ConcurrentApi.js --- core/api/ConcurrentApi.js | 160 ++++++++++++++++++++++++++++++++++++++ 1 file changed, 160 insertions(+) create mode 100644 core/api/ConcurrentApi.js diff --git a/core/api/ConcurrentApi.js b/core/api/ConcurrentApi.js new file mode 100644 index 0000000..f36864d --- /dev/null +++ b/core/api/ConcurrentApi.js @@ -0,0 +1,160 @@ +import { extension_settings, getContext } from "/scripts/extensions.js"; +import { getRequestHeaders } from "/script.js"; +import { extensionName } from "../../utils/settings.js"; + +function getConcurrentApiSettings() { + const settings = extension_settings[extensionName] || {}; + return { + apiProvider: settings.plotOpt_concurrentApiProvider || 'openai', + apiUrl: settings.plotOpt_concurrentApiUrl?.trim() || '', + apiKey: settings.plotOpt_concurrentApiKey?.trim() || '', + model: settings.plotOpt_concurrentModel || '', + maxTokens: settings.plotOpt_max_tokens || 20000, + temperature: settings.plotOpt_temperature || 1, + }; +} + +export async function callConcurrentAI(messages, options = {}) { + if (window.AMILY2_SYSTEM_PARALYZED === true) { + console.error("[Amily2-Concurrent制裁] 系统完整性已受损,所有外交活动被无限期中止。"); + return null; + } + + const apiSettings = getConcurrentApiSettings(); + + const finalOptions = { + ...apiSettings, + ...options + }; + + if (!finalOptions.apiUrl || !finalOptions.model || !finalOptions.apiKey) { + console.warn("[Amily2-Concurrent外交部] API配置不完整,无法调用AI"); + toastr.error("并发API配置不完整,请检查URL、Key和模型配置。", "Concurrent-外交部"); + return null; + } + + console.groupCollapsed(`[Amily2号-Concurrent统一API调用] ${new Date().toLocaleTimeString()}`); + console.log("【请求参数】:", { + provider: finalOptions.apiProvider, + model: finalOptions.model, + maxTokens: finalOptions.maxTokens, + temperature: finalOptions.temperature, + messagesCount: messages.length + }); + console.log("【消息内容】:", messages); + console.groupEnd(); + + try { + let responseContent; + + // For now, we only support openai_test like provider. + // More can be added here following the structure of JqyhApi.js + switch (finalOptions.apiProvider) { + case 'openai': + case 'openai_test': + responseContent = await callConcurrentOpenAITest(messages, finalOptions); + break; + default: + console.error(`[Amily2-Concurrent外交部] 未支持的API模式: ${finalOptions.apiProvider}`); + toastr.error(`并发API模式 "${finalOptions.apiProvider}" 不被支持。`, "Concurrent-外交部"); + return null; + } + + if (!responseContent) { + console.warn('[Amily2-Concurrent外交部] 未能获取AI响应内容'); + return null; + } + + console.groupCollapsed("[Amily2号-Concurrent AI回复]"); + console.log(responseContent); + console.groupEnd(); + + return responseContent; + + } catch (error) { + console.error(`[Amily2-Concurrent外交部] API调用发生错误:`, error); + toastr.error(`并发API调用失败: ${error.message}`, "Concurrent API调用失败"); + return null; + } +} + +async function callConcurrentOpenAITest(messages, options) { + const body = { + chat_completion_source: 'openai', + messages: messages, + model: options.model, + reverse_proxy: options.apiUrl, + proxy_password: options.apiKey, + stream: false, + max_tokens: options.maxTokens || 20000, + temperature: options.temperature || 0.7, + top_p: options.top_p || 0.95, + }; + + const response = await fetch('/api/backends/chat-completions/generate', { + method: 'POST', + headers: { ...getRequestHeaders(), 'Content-Type': 'application/json' }, + body: JSON.stringify(body) + }); + + if (!response.ok) { + const errorText = await response.text(); + throw new Error(`Concurrent全兼容API请求失败: ${response.status} - ${errorText}`); + } + + const responseData = await response.json(); + return responseData?.choices?.[0]?.message?.content; +} + +export async function testConcurrentApiConnection() { + const apiSettings = getConcurrentApiSettings(); + if (!apiSettings.apiUrl || !apiSettings.apiKey) { + toastr.error("并发API的URL或API Key未设置。", "测试连接失败"); + return; + } + + const modelsUrl = new URL('/v1/models', apiSettings.apiUrl).toString(); + + try { + const response = await fetch(modelsUrl, { + method: 'GET', + headers: { + 'Authorization': `Bearer ${apiSettings.apiKey}`, + }, + }); + + if (response.ok) { + toastr.success("并发API连接成功!", "测试连接"); + } else { + const errorText = await response.text(); + toastr.error(`连接失败: ${response.status}. ${errorText}`, "测试连接失败"); + } + } catch (error) { + console.error("[Amily2-Concurrent] 测试连接时出错:", error); + toastr.error(`网络错误: ${error.message}`, "测试连接失败"); + } +} + +export async function fetchConcurrentModels() { + const apiSettings = getConcurrentApiSettings(); + if (!apiSettings.apiUrl || !apiSettings.apiKey) { + throw new Error("并发API的URL或API Key未设置。"); + } + + const modelsUrl = new URL('/v1/models', apiSettings.apiUrl).toString(); + + const response = await fetch(modelsUrl, { + method: 'GET', + headers: { + 'Authorization': `Bearer ${apiSettings.apiKey}`, + }, + }); + + if (!response.ok) { + const errorText = await response.text(); + throw new Error(`获取模型列表失败: ${response.status} - ${errorText}`); + } + + const data = await response.json(); + return data.data.map(model => ({ id: model.id, name: model.id })); // Return in the same format as other fetchers +}