mirror of
https://github.com/foxhui/WebAI2API.git
synced 2026-06-16 21:03:59 +08:00
354 lines
15 KiB
JavaScript
354 lines
15 KiB
JavaScript
import http from 'http';
|
|
import fs from 'fs';
|
|
import path from 'path';
|
|
import sharp from 'sharp';
|
|
import { getBackend } from './lib/backend/index.js';
|
|
import { getModelsForBackend, resolveModelId } from './lib/backend/models.js';
|
|
import { logger } from './lib/logger.js';
|
|
import crypto from 'crypto';
|
|
|
|
// 使用统一后端获取配置和函数
|
|
const { config, name, initBrowser, generateImage, TEMP_DIR } = getBackend();
|
|
|
|
|
|
const PORT = config.server.port || 3000;
|
|
const AUTH_TOKEN = config.server.auth;
|
|
const SERVER_MODE = config.server.type || 'openai'; // 'openai' 或 'queue'
|
|
|
|
// --- 全局状态 ---
|
|
let browserContext = null; // 浏览器上下文 {browser, page, client, width, height}
|
|
const queue = []; // 请求队列
|
|
let processingCount = 0; // 当前正在处理的任务数
|
|
const MAX_CONCURRENT = config.queue?.maxConcurrent || 1; // 从配置读取
|
|
const MAX_QUEUE_SIZE = config.queue?.maxQueueSize || 2; // 从配置读取
|
|
const IMAGE_LIMIT = config.queue?.imageLimit || 5; // 图片数量上限
|
|
|
|
/**
|
|
* 处理队列中的任务
|
|
*/
|
|
async function processQueue() {
|
|
// 如果正在处理的任务已满,或队列为空,则停止
|
|
if (processingCount >= MAX_CONCURRENT || queue.length === 0) return;
|
|
|
|
// 取出下一个任务
|
|
const task = queue.shift();
|
|
processingCount++;
|
|
|
|
// 如果是 Queue 模式,通知客户端状态变更
|
|
if (SERVER_MODE === 'queue' && task.sse) {
|
|
task.sse.send('status', { status: 'processing' });
|
|
}
|
|
|
|
try {
|
|
const { req, res, prompt, imagePaths, modelId, modelName, id, sse } = task;
|
|
logger.info('服务器', '[队列] 开始处理任务', { id, remaining: queue.length });
|
|
|
|
// 确保浏览器已初始化
|
|
if (!browserContext) {
|
|
browserContext = await initBrowser(config);
|
|
}
|
|
|
|
// 调用核心生图逻辑
|
|
const result = await generateImage(browserContext, prompt, imagePaths, modelId, { id });
|
|
|
|
// 清理临时图片
|
|
for (const p of imagePaths) {
|
|
try { fs.unlinkSync(p); } catch (e) { }
|
|
}
|
|
|
|
// 处理结果
|
|
let finalContent = '';
|
|
let queueResult = {};
|
|
|
|
if (result.error) {
|
|
// 特殊错误处理:reCAPTCHA
|
|
if (result.error === 'recaptcha validation failed') {
|
|
if (SERVER_MODE === 'openai') {
|
|
res.writeHead(500, { 'Content-Type': 'application/json' });
|
|
res.end(JSON.stringify({ error: 'recaptcha validation failed' }));
|
|
} else {
|
|
task.sse.send('result', { status: 'error', image: null, msg: 'recaptcha validation failed' });
|
|
task.sse.send('done', '[DONE]');
|
|
task.sse.end();
|
|
}
|
|
return;
|
|
}
|
|
finalContent = `[生成错误] ${result.error}`;
|
|
queueResult = { status: 'error', image: null, msg: result.error };
|
|
} else if (result.image) {
|
|
try {
|
|
// result.image 已经是 "data:image/png;base64,..." 格式
|
|
// 提取纯 Base64 部分用于 b64_json
|
|
const base64Data = result.image.split(',')[1];
|
|
|
|
// 构造 Markdown 图片展示 (Data URI)
|
|
finalContent = ``;
|
|
|
|
queueResult = { status: 'completed', image: base64Data, msg: '' };
|
|
queueResult = { status: 'completed', image: base64Data, msg: '' };
|
|
logger.info('服务器', '图片已准备就绪 (Base64)', { id });
|
|
} catch (e) {
|
|
logger.error('服务器', '图片处理失败', { id, error: e.message });
|
|
finalContent = `[图片处理失败] ${e.message}`;
|
|
queueResult = { status: 'error', image: null, msg: `Processing failed: ${e.message}` };
|
|
}
|
|
} else {
|
|
finalContent = result.text || '生成失败';
|
|
queueResult = { status: 'completed', image: null, msg: result.text };
|
|
}
|
|
|
|
// 发送响应
|
|
if (SERVER_MODE === 'openai') {
|
|
const response = {
|
|
id: 'chatcmpl-' + Date.now(),
|
|
object: 'chat.completion',
|
|
created: Math.floor(Date.now() / 1000),
|
|
model: modelName || 'default-model',
|
|
choices: [{
|
|
index: 0,
|
|
message: {
|
|
role: 'assistant',
|
|
content: finalContent
|
|
},
|
|
finish_reason: 'stop'
|
|
}]
|
|
};
|
|
res.writeHead(200, { 'Content-Type': 'application/json' });
|
|
res.end(JSON.stringify(response));
|
|
} else {
|
|
// Queue Mode
|
|
task.sse.send('result', queueResult);
|
|
task.sse.send('done', '[DONE]');
|
|
task.sse.end();
|
|
}
|
|
|
|
} catch (err) {
|
|
logger.error('服务器', '任务处理失败', { id: task.id, error: err.message });
|
|
if (SERVER_MODE === 'openai') {
|
|
if (!task.res.writableEnded) {
|
|
task.res.writeHead(500, { 'Content-Type': 'application/json' });
|
|
task.res.end(JSON.stringify({ error: err.message }));
|
|
}
|
|
} else {
|
|
task.sse.send('result', { status: 'error', image: null, msg: err.message });
|
|
task.sse.send('done', '[DONE]');
|
|
task.sse.end();
|
|
}
|
|
} finally {
|
|
processingCount--;
|
|
// 递归处理下一个任务
|
|
processQueue();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* 启动 HTTP 服务器
|
|
*/
|
|
async function startServer() {
|
|
// 预先启动浏览器
|
|
try {
|
|
browserContext = await initBrowser(config);
|
|
} catch (err) {
|
|
logger.error('服务器', '浏览器初始化失败', { error: err.message });
|
|
process.exit(1);
|
|
}
|
|
|
|
const server = http.createServer(async (req, res) => {
|
|
// 为每个请求生成唯一 ID
|
|
const id = crypto.randomUUID().slice(0, 8);
|
|
|
|
// --- 鉴权中间件 ---
|
|
const authHeader = req.headers['authorization'];
|
|
if (!authHeader || authHeader !== `Bearer ${AUTH_TOKEN}`) {
|
|
res.writeHead(401, { 'Content-Type': 'application/json' });
|
|
res.end(JSON.stringify({ error: 'Unauthorized' }));
|
|
return;
|
|
}
|
|
|
|
// --- 路由分发 ---
|
|
const isQueueMode = SERVER_MODE === 'queue';
|
|
const targetPath = isQueueMode ? '/v1/queue/join' : '/v1/chat/completions';
|
|
|
|
// 1. 模型列表接口 (OpenAI & Queue 模式通用)
|
|
if (req.method === 'GET' && req.url === '/v1/models') {
|
|
const models = getModelsForBackend(name);
|
|
res.writeHead(200, { 'Content-Type': 'application/json' });
|
|
res.end(JSON.stringify(models));
|
|
return;
|
|
}
|
|
|
|
if (req.method === 'POST' && req.url.startsWith(targetPath)) {
|
|
// --- SSE 设置 (仅 Queue 模式) ---
|
|
let sseHelper = null;
|
|
if (isQueueMode) {
|
|
res.writeHead(200, {
|
|
'Content-Type': 'text/event-stream',
|
|
'Cache-Control': 'no-cache',
|
|
'Connection': 'keep-alive'
|
|
});
|
|
|
|
sseHelper = {
|
|
send: (event, data) => {
|
|
res.write(`event: ${event}\n`);
|
|
res.write(`data: ${typeof data === 'object' ? JSON.stringify(data) : data}\n\n`);
|
|
},
|
|
end: () => res.end()
|
|
};
|
|
|
|
// 启动心跳
|
|
const heartbeat = setInterval(() => {
|
|
if (res.writableEnded) {
|
|
clearInterval(heartbeat);
|
|
return;
|
|
}
|
|
sseHelper.send('heartbeat', Date.now());
|
|
}, 3000);
|
|
}
|
|
|
|
const chunks = [];
|
|
req.on('data', chunk => chunks.push(chunk));
|
|
req.on('end', async () => {
|
|
try {
|
|
// --- 限流检查 ---
|
|
if (!isQueueMode && processingCount + queue.length >= MAX_CONCURRENT + MAX_QUEUE_SIZE) {
|
|
logger.warn('服务器', '请求过多,已拒绝 (最大队列限制)', { id });
|
|
if (isQueueMode) {
|
|
sseHelper.send('error', { msg: 'Too Many Requests' });
|
|
sseHelper.end();
|
|
} else {
|
|
res.writeHead(429, { 'Content-Type': 'application/json' });
|
|
res.end(JSON.stringify({ error: 'Too Many Requests. Server is busy.' }));
|
|
}
|
|
return;
|
|
}
|
|
|
|
const body = Buffer.concat(chunks).toString();
|
|
const data = JSON.parse(body);
|
|
const messages = data.messages;
|
|
|
|
if (!messages || messages.length === 0) {
|
|
if (isQueueMode) { sseHelper.send('error', { msg: 'No messages' }); sseHelper.end(); }
|
|
else { res.writeHead(400); res.end(JSON.stringify({ error: 'No messages' })); }
|
|
return;
|
|
}
|
|
|
|
// 筛选用户消息
|
|
const userMessages = messages.filter(m => m.role === 'user');
|
|
if (userMessages.length === 0) {
|
|
if (isQueueMode) { sseHelper.send('error', { msg: 'No user messages' }); sseHelper.end(); }
|
|
else { res.writeHead(400); res.end(JSON.stringify({ error: 'No user messages' })); }
|
|
return;
|
|
}
|
|
const lastMessage = userMessages[userMessages.length - 1];
|
|
|
|
let prompt = '';
|
|
const imagePaths = [];
|
|
let imageCount = 0;
|
|
|
|
// 解析内容 (拼接文本 + 处理图片)
|
|
if (Array.isArray(lastMessage.content)) {
|
|
for (const item of lastMessage.content) {
|
|
if (item.type === 'text') {
|
|
prompt += item.text + ' ';
|
|
} else if (item.type === 'image_url' && item.image_url && item.image_url.url) {
|
|
imageCount++;
|
|
|
|
// 逻辑:
|
|
// 1. 如果配置限制 <= 10 (浏览器硬限制), 则严格执行, 超过报错
|
|
// 2. 如果配置限制 > 10, 则视为用户想"尽力而为", 自动截断到 10 张, 忽略多余的
|
|
|
|
if (IMAGE_LIMIT <= 10) {
|
|
if (imageCount > IMAGE_LIMIT) {
|
|
const errorMsg = `Too many images. Maximum ${IMAGE_LIMIT} images allowed.`;
|
|
logger.warn('server', errorMsg, { id });
|
|
if (isQueueMode) { sseHelper.send('error', { msg: errorMsg }); sseHelper.end(); }
|
|
else { res.writeHead(400); res.end(JSON.stringify({ error: errorMsg })); }
|
|
return;
|
|
}
|
|
} else {
|
|
// IMAGE_LIMIT > 10
|
|
if (imageCount > 10) {
|
|
// 超过浏览器硬限制, 忽略该图片
|
|
continue;
|
|
}
|
|
}
|
|
|
|
const url = item.image_url.url;
|
|
if (url.startsWith('data:image')) {
|
|
const matches = url.match(/^data:([A-Za-z-+\/]+);base64,(.+)$/);
|
|
if (matches && matches.length === 3) {
|
|
const buffer = Buffer.from(matches[2], 'base64');
|
|
// 压缩图片
|
|
const processedBuffer = await sharp(buffer)
|
|
.jpeg({ quality: 90 })
|
|
.toBuffer();
|
|
|
|
const filename = `img_${Date.now()}_${Math.random().toString(36).substring(7)}.jpg`;
|
|
const filePath = path.join(TEMP_DIR, filename);
|
|
fs.writeFileSync(filePath, processedBuffer);
|
|
imagePaths.push(filePath);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
prompt = lastMessage.content; // 回落保留
|
|
}
|
|
|
|
prompt = prompt.trim();
|
|
|
|
// 解析模型参数
|
|
let modelId = null;
|
|
if (data.model) {
|
|
modelId = resolveModelId(name, data.model);
|
|
if (modelId) {
|
|
logger.info('服务器', `触发模型: ${data.model} (${modelId})`, { id });
|
|
} else {
|
|
const errorMsg = `Invalid model for backend ${name}: ${data.model}`;
|
|
logger.warn('服务器', errorMsg, { id });
|
|
if (isQueueMode) { sseHelper.send('error', { msg: errorMsg }); sseHelper.end(); }
|
|
else { res.writeHead(400); res.end(JSON.stringify({ error: errorMsg })); }
|
|
return;
|
|
}
|
|
} else {
|
|
logger.info('服务器', '未指定模型,使用网页默认', { id });
|
|
}
|
|
|
|
logger.info('服务器', `[队列] 请求入队: ${prompt.slice(0, 10)}...`, { id, images: imagePaths.length });
|
|
|
|
if (isQueueMode) {
|
|
sseHelper.send('status', { status: 'queued', position: queue.length + 1 });
|
|
}
|
|
|
|
// 将任务加入队列
|
|
queue.push({ req, res, prompt, imagePaths, sse: sseHelper, modelId, modelName: data.model || null, id });
|
|
|
|
// 触发队列处理
|
|
processQueue();
|
|
|
|
} catch (err) {
|
|
logger.error('服务器', '服务器处理失败', { id, error: err.message });
|
|
if (isQueueMode && sseHelper) {
|
|
sseHelper.send('error', { msg: err.message });
|
|
sseHelper.end();
|
|
} else if (!res.writableEnded) {
|
|
res.writeHead(500, { 'Content-Type': 'application/json' });
|
|
res.end(JSON.stringify({ error: err.message }));
|
|
}
|
|
}
|
|
});
|
|
} else {
|
|
res.writeHead(404);
|
|
res.end();
|
|
}
|
|
});
|
|
|
|
server.listen(PORT, () => {
|
|
logger.info('服务器', `HTTP 服务器启动成功,监听端口 ${PORT}`);
|
|
logger.info('服务器', `运行模式: ${SERVER_MODE === 'openai' ? 'OpenAI 兼容模式' : 'Queue 队列模式'}`);
|
|
logger.info('服务器', `最大队列: ${MAX_QUEUE_SIZE},最大图片数量: ${IMAGE_LIMIT}`);
|
|
});
|
|
}
|
|
|
|
startServer();
|