mirror of
https://github.com/ikechan8370/chatgpt-plugin.git
synced 2025-12-16 13:27:08 +00:00
* 修复后台API反代地址未能正确显示的问题 * 更新渲染页面配置 * 添加个人聊天模式配置 * 将用户数据获取改到common中 * 修复错误的渲染页面参数 * 修复bug * 添加Live2D * 修复渲染页面错误 * 修复渲染传入值 * 更新渲染 * 修复图表渲染bug * 调整live2d模型大小 * 修复live2d无法关闭问题 * 修复错误的传值 * 修复ai命名 * 更新渲染 * 添加用户独立设定 * 更新渲染配置适配个人设置 * 修复合并导致的渲染文件异常删除 * 修复用户数据缺失问题 * 修复旧版本数据缺失问题 * 修复bing参数不存在问题,兼容miao的截图 * 修复受限token重试时不被排除的问题 * 修复个人模式下结束对话的模式错误 * 更新渲染页面,将预览版转为正式版 * 修复传统渲染无法调用截图功能的问题 * 文字模式也进行一次缓存 * 更新README * Update README.md * 更新渲染 * 更新渲染页面 * 添加版本信息 * 遗漏参数 * 丢失引用 * 补充路由 * 添加云转码功能 * 判断node-silk是否正常合成 * 云转码提示 * 修复图片渲染出错 * 云转码支持发送Buffer * 添加云转码模式支持 * 更新描述 * 更新后台渲染页面 * 更新配置 * 更新渲染页面 * 添加云渲染 * 修复错误的接口调用 * 修复遗漏的数据转换 * 修复获取的图片数据异常问题 * 更新后台配置 * 更新渲染页面 * 修复云渲染访问地址错误 * 更新渲染页面 * 修复遗漏的模型文件 * 修复live2d问题 * 更新live2d以及相关配置 * 修复遗漏的数据参数 * 修复新live2d情况下云渲染错误的问题 * 适配云渲染1.1.2等待参数 * 添加云服务api检查 * 更新渲染页面 * 添加live2d加载检测 * 修复错误的属性判断 * 添加云渲染DPR * 更新sydney支持内容生成 * 修改文件模式语音转码接收模式 * 添加云转码时recordUrl检查 * 更新后台配置项 * 修复错误的文本描述 * 更新后台页面 * 添加全局对话模式设置,更新后台面板 * 添加第三方渲染服务适配 * 修复第三方服务器live2d加载导致的渲染失败问题 * 修复后台地址无法实时保存的问题 * 添加live2d模型透明度设置 * 合并更新 * 更新渲染页面 * 更新渲染页面 * 使dpr对本地渲染也生效 * 更新渲染页面 * 添加网页截图功能 * 添加后台配置项 * 添加配置导出和导入功能 * 运行通过参数传递用户token * 登录时将token作为参数返回 * 修复错误 * 添加密码修改和用户删除接口 * 修正密码比对 * 修复user错误 * 优化数据保存时的返回值 * 添加系统额外服务检查api * 添加AccessToken配置 * 修复错误的导入提示 * 添加ws连接 * 添加ws用户信息获取 * 修复错误的循环 * 修正ws参数 * 添加群消息获取权限 * 添加用户多端登录支持 * 修复错误的路径引用 * 修复错误的路径引用 * 修复错误 * 修复页面数据获取失败问题 * 修复异常的中断 * 添加配置视图 * 更新配置面板 * 添加用户版本信息 * 更新配置视图 * 修复错误的视图绑定 * 修改视图文件位置,添加mediaLink相关代码 * 修复错误的视图配置绑定 * 更新依赖,添加qq消息组件初始化信息获取 * 修复异常的群名称无法获取问题 * 修改注释 * 撤销对management的错误合并 * 添加Sydney图片识别功能 * 更新配置文件和后台页面 * 修改view配置 --------- Co-authored-by: ikechan8370 <geyinchibuaa@gmail.com>
616 lines
21 KiB
JavaScript
616 lines
21 KiB
JavaScript
import fastify from 'fastify'
|
||
import fastifyCookie from '@fastify/cookie'
|
||
import cors from '@fastify/cors'
|
||
import fstatic from '@fastify/static'
|
||
import websocket from '@fastify/websocket'
|
||
|
||
import fs from 'fs'
|
||
import path from 'path'
|
||
import os from 'os'
|
||
import schedule from 'node-schedule'
|
||
import websocketclient from 'ws'
|
||
|
||
import { Config } from '../utils/config.js'
|
||
import { UserInfo, GetUser } from './modules/user_data.js'
|
||
import { getPublicIP, getUserData, getMasterQQ, randomString } from '../utils/common.js'
|
||
|
||
import webRoute from './modules/web_route.js'
|
||
import webUser from './modules/user.js'
|
||
import SettingView from './modules/setting_view.js'
|
||
|
||
const __dirname = path.resolve()
|
||
const server = fastify({
|
||
logger: Config.debug
|
||
})
|
||
|
||
let Statistics = {
|
||
SystemAccess: {
|
||
count: 0,
|
||
oldCount: 0
|
||
},
|
||
CacheFile: {
|
||
count: 0,
|
||
oldCount: 0
|
||
},
|
||
WebAccess: {
|
||
count: 0,
|
||
oldCount: 0
|
||
},
|
||
SystemLoad: {
|
||
count: 0,
|
||
oldCount: 0
|
||
}
|
||
}
|
||
|
||
async function getLoad() {
|
||
// 获取当前操作系统平台
|
||
const platform = os.platform()
|
||
// 判断平台是Linux还是Windows
|
||
if (platform === 'linux') {
|
||
// 如果是Linux,使用os.loadavg()方法获取负载平均值
|
||
const loadAvg = os.loadavg()
|
||
return loadAvg[0] * 100
|
||
} else if (platform === 'win32') {
|
||
// 如果是Windows不获取性能
|
||
return 0
|
||
} else {
|
||
return 0
|
||
}
|
||
}
|
||
|
||
async function setUserData(qq, data) {
|
||
const dir = 'resources/ChatGPTCache/user'
|
||
const filename = `${qq}.json`
|
||
const filepath = path.join(dir, filename)
|
||
fs.mkdirSync(dir, { recursive: true })
|
||
fs.writeFileSync(filepath, JSON.stringify(data))
|
||
}
|
||
|
||
await server.register(cors, {
|
||
origin: '*'
|
||
})
|
||
await server.register(fstatic, {
|
||
root: path.join(__dirname, 'plugins/chatgpt-plugin/server/static/')
|
||
})
|
||
await server.register(websocket, {
|
||
cors: true,
|
||
options: {
|
||
maxPayload: 1048576
|
||
}
|
||
})
|
||
await server.register(fastifyCookie)
|
||
await server.register(webRoute)
|
||
await server.register(webUser)
|
||
await server.register(SettingView)
|
||
|
||
// 无法访问端口的情况下创建与media的通讯
|
||
async function mediaLink() {
|
||
const ip = await getPublicIP()
|
||
const testServer = await fetch(`${Config.cloudTranscode}/check`,
|
||
{
|
||
method: 'POST',
|
||
headers: {
|
||
'Content-Type': 'application/json'
|
||
},
|
||
body: JSON.stringify({
|
||
url: `http://${ip}:${Config.serverPort || 3321}/`
|
||
})
|
||
})
|
||
if (testServer.ok) {
|
||
const checkCloudData = await testServer.json()
|
||
if (checkCloudData.state != 'error') {
|
||
console.log('本地服务无法访问,开启media服务代理')
|
||
const serverurl = new URL(Config.cloudTranscode)
|
||
const ws = new websocketclient(`ws://${serverurl.hostname}${serverurl.port ? ':' + serverurl.port : ''}/ws`)
|
||
ws.on('open', () => {
|
||
ws.send(JSON.stringify({
|
||
command: 'register',
|
||
region: Bot.uin,
|
||
type: 'server',
|
||
}))
|
||
})
|
||
ws.on('message', async (message) => {
|
||
try {
|
||
const data = JSON.parse(message)
|
||
switch (data.command) {
|
||
case 'register':
|
||
if (data.state) {
|
||
let master = (await getMasterQQ())[0]
|
||
Bot.sendPrivateMsg(master, `当前chatgpt插件服务无法被外网访问,已启用代理链接,访问代码:${data.token}`, false)
|
||
} else {
|
||
console.log('注册区域失败')
|
||
}
|
||
break
|
||
case 'login':
|
||
if (data.token) {
|
||
const user = UserInfo(data.token)
|
||
if (user) {
|
||
ws.login = true
|
||
ws.send(JSON.stringify({ command: data.command, state: true, region: Bot.uin, type: 'server' }))
|
||
} else {
|
||
ws.send(JSON.stringify({ command: data.command, state: false, error: '权限验证失败', region: Bot.uin, type: 'server' }))
|
||
}
|
||
}
|
||
break
|
||
case 'post_login':
|
||
if (data.qq && data.passwd) {
|
||
const token = randomString(32)
|
||
if (data.qq == Bot.uin && await redis.get('CHATGPT:ADMIN_PASSWD') == data.passwd) {
|
||
AddUser({ user: data.qq, token: token, autho: 'admin' })
|
||
ws.send(JSON.stringify({ command: data.command, state: true, autho: 'admin', token: token, region: Bot.uin, type: 'server' }))
|
||
|
||
} else {
|
||
const user = await getUserData(data.qq)
|
||
if (user.passwd != '' && user.passwd === data.passwd) {
|
||
AddUser({ user: data.qq, token: token, autho: 'user' })
|
||
ws.send(JSON.stringify({ command: data.command, state: true, autho: 'user', token: token, region: Bot.uin, type: 'server' }))
|
||
} else {
|
||
ws.send(JSON.stringify({ command: data.command, state: false, error: `用户名密码错误,如果忘记密码请私聊机器人输入 ${data.qq == Bot.uin ? '#修改管理密码' : '#修改用户密码'} 进行修改`, region: Bot.uin, type: 'server' }))
|
||
}
|
||
}
|
||
} else {
|
||
ws.send(JSON.stringify({ command: data.command, state: false, error: '未输入用户名或密码', region: Bot.uin, type: 'server' }))
|
||
}
|
||
break
|
||
}
|
||
} catch (error) {
|
||
console.log(error)
|
||
}
|
||
})
|
||
|
||
} else {
|
||
console.log('本地服务网络正常,无需开启通讯')
|
||
}
|
||
} else {
|
||
console.log('media服务器未响应')
|
||
}
|
||
}
|
||
// 未完工,暂不开启这个功能
|
||
// mediaLink()
|
||
|
||
export async function createServer() {
|
||
// 页面数据获取
|
||
server.post('/page', async (request, reply) => {
|
||
const body = request.body || {}
|
||
if (body.code) {
|
||
const dir = 'resources/ChatGPTCache/page'
|
||
const filename = body.code + '.json'
|
||
const filepath = path.join(dir, filename)
|
||
let data = fs.readFileSync(filepath, 'utf8')
|
||
reply.send(data)
|
||
}
|
||
return reply
|
||
})
|
||
// 帮助内容获取
|
||
server.post('/help', async (request, reply) => {
|
||
const body = request.body || {}
|
||
if (body.use) {
|
||
const dir = 'plugins/chatgpt-plugin/resources'
|
||
const filename = 'help.json'
|
||
const filepath = path.join(dir, filename)
|
||
let data = fs.readFileSync(filepath, 'utf8')
|
||
data = JSON.parse(data)
|
||
reply.send(data[body.use])
|
||
}
|
||
return reply
|
||
})
|
||
// 创建页面缓存内容
|
||
server.post('/cache', async (request, reply) => {
|
||
const body = request.body || {}
|
||
if (body.content) {
|
||
const dir = 'resources/ChatGPTCache/page'
|
||
const filename = body.entry + '.json'
|
||
const filepath = path.join(dir, filename)
|
||
const regexUrl = /\b((?:https?|ftp|file):\/\/[-a-zA-Z0-9+&@#\/%?=~_|!:,.;]*[-a-zA-Z0-9+&@#\/%=~_|])/g
|
||
const ip = await getPublicIP()
|
||
let botName = ''
|
||
switch (body.model) {
|
||
case 'bing':
|
||
botName = 'Bing'
|
||
break
|
||
case 'api':
|
||
botName = 'ChatGPT'
|
||
break
|
||
case 'api3':
|
||
botName = 'ChatGPT'
|
||
break
|
||
case 'browser':
|
||
botName = 'ChatGPT'
|
||
break
|
||
case 'chatglm':
|
||
botName = 'ChatGLM'
|
||
break
|
||
case 'claude':
|
||
botName = 'Claude'
|
||
break
|
||
default:
|
||
botName = body.model
|
||
break
|
||
}
|
||
try {
|
||
fs.mkdirSync(dir, { recursive: true })
|
||
const data = {
|
||
user: body.content.senderName,
|
||
bot: Config.chatViewBotName || botName,
|
||
userImg: body.userImg || '',
|
||
botImg: body.botImg || '',
|
||
question: body.content.prompt,
|
||
message: body.content.content,
|
||
group: body.content.group,
|
||
herf: `http://${body.cacheHost || (ip + ':' + Config.serverPort || 3321)}/page/${body.entry}`,
|
||
quote: body.content.quote,
|
||
images: body.content.images || [],
|
||
suggest: body.content.suggest || [],
|
||
model: body.model,
|
||
mood: body.content.mood || 'blandness',
|
||
live2d: Config.live2d,
|
||
live2dModel: Config.live2dModel,
|
||
live2dOption: {
|
||
scale: Config.live2dOption_scale,
|
||
position: {
|
||
x: Config.live2dOption_positionX,
|
||
y: Config.live2dOption_positionY
|
||
},
|
||
rotation: Config.live2dOption_rotation,
|
||
alpha: Config.live2dOption_alpha,
|
||
dpr: Config.cloudDPR
|
||
},
|
||
time: new Date()
|
||
}
|
||
fs.writeFileSync(filepath, JSON.stringify(data))
|
||
const user = await getUserData(body.qq)
|
||
user.chat.push({
|
||
user: data.user,
|
||
bot: data.bot,
|
||
group: data.group,
|
||
herf: data.herf,
|
||
model: data.model,
|
||
time: data.time
|
||
})
|
||
await setUserData(body.qq, user)
|
||
Statistics.CacheFile.count += 1
|
||
reply.send({ file: body.entry, cacheUrl: `http://${ip}:${Config.serverPort || 3321}/page/${body.entry}` })
|
||
} catch (err) {
|
||
server.log.error(`用户生成缓存${body.entry}时发生错误: ${err}`)
|
||
reply.send({ file: body.entry, cacheUrl: `http://${ip}:${Config.serverPort || 3321}/page/${body.entry}`, error: body.entry + '生成失败' })
|
||
}
|
||
}
|
||
return reply
|
||
})
|
||
// 获取系统状态
|
||
server.post('/system-statistics', async (request, reply) => {
|
||
Statistics.SystemLoad.count = await getLoad()
|
||
reply.send(Statistics)
|
||
return reply
|
||
})
|
||
|
||
// 清除缓存数据
|
||
server.post('/cleanCache', async (request, reply) => {
|
||
const token = request.cookies.token || request.body?.token || 'unknown'
|
||
let user = UserInfo(token)
|
||
if (!user) user = { user: '' }
|
||
const userData = await getUserData(user.user)
|
||
const dir = 'resources/ChatGPTCache/page'
|
||
userData.chat.forEach(function (item, index) {
|
||
const filename = item.herf.substring(item.herf.lastIndexOf('/') + 1) + '.json'
|
||
const filepath = path.join(dir, filename)
|
||
fs.unlinkSync(filepath)
|
||
})
|
||
userData.chat = []
|
||
await setUserData(user.user, userData)
|
||
reply.send({ state: true })
|
||
return reply
|
||
})
|
||
let clients = []
|
||
// 获取消息
|
||
const wsFn = async (connection, request) => {
|
||
connection.socket.on('open', message => {
|
||
// 开始连接
|
||
console.log(`Received message: ${message}`)
|
||
const response = { data: 'hello, client' }
|
||
connection.socket.send(JSON.stringify(response))
|
||
})
|
||
connection.socket.on('message', async (message) => {
|
||
try {
|
||
const data = JSON.parse(message)
|
||
|
||
switch (data.command) {
|
||
case 'sendMsg': // 代理消息发送
|
||
if (!connection.login) {
|
||
await connection.socket.send(JSON.stringify({ command: data.command, state: false, error: '请先登录账号' }))
|
||
return
|
||
}
|
||
if (data.id && data.message) {
|
||
if (data.group) {
|
||
Bot.sendGroupMsg(parseInt(data.id), data.message, data.quotable)
|
||
} else {
|
||
Bot.sendPrivateMsg(parseInt(data.id), data.message, data.quotable)
|
||
}
|
||
await connection.socket.send(JSON.stringify({ command: data.command, state: true, }))
|
||
} else {
|
||
await connection.socket.send(JSON.stringify({ command: data.command, state: false, error: '参数不足' }))
|
||
}
|
||
break
|
||
case 'userInfo': // 获取用户信息
|
||
if (!connection.login) {
|
||
await connection.socket.send(JSON.stringify({ command: data.command, state: false, error: '请先登录账号' }))
|
||
} else {
|
||
await connection.socket.send(JSON.stringify({ command: data.command, state: true, user: { user: user.user, autho: user.autho } }))
|
||
}
|
||
break
|
||
case 'login': // 登录
|
||
const user = UserInfo(data.token)
|
||
if (user) {
|
||
clients[user.user] = connection.socket
|
||
connection.login = true
|
||
await connection.socket.send(JSON.stringify({ command: data.command, state: true }))
|
||
} else {
|
||
await connection.socket.send(JSON.stringify({ command: data.command, state: false, error: '权限验证失败' }))
|
||
}
|
||
break
|
||
case 'initQQMessageInfo': // qq消息模块初始化信息
|
||
if (!connection.login) {
|
||
await connection.socket.send(JSON.stringify({ command: data.command, state: false, error: '请先登录账号' }))
|
||
return
|
||
}
|
||
const groupList = Bot.getGroupList()
|
||
groupList.forEach(async (item) => {
|
||
const group = Bot.pickGroup(item.group_id)
|
||
const groupMessages = await group.getChatHistory()
|
||
groupMessages.forEach(async (e) => {
|
||
const messageData = {
|
||
notice: 'clientMessage',
|
||
message: e.message,
|
||
sender: e.sender,
|
||
group: {
|
||
isGroup: true,
|
||
group_id: e.group_id,
|
||
group_name: e.group_name || item.group_name
|
||
},
|
||
quotable: {
|
||
user_id: e.user_id,
|
||
time: e.time,
|
||
seq: e.seq,
|
||
rand: e.rand,
|
||
message: e.message,
|
||
user_name: e.sender.nickname,
|
||
}
|
||
}
|
||
await connection.socket.send(JSON.stringify(messageData))
|
||
})
|
||
})
|
||
break
|
||
default:
|
||
await connection.socket.send(JSON.stringify({ "data": data }))
|
||
break
|
||
}
|
||
} catch (error) {
|
||
await connection.socket.send(JSON.stringify({ "error": error.message }))
|
||
}
|
||
})
|
||
connection.socket.on('close', () => {
|
||
// 监听连接关闭事件
|
||
const response = { code: 403, data: 'Client disconnected', message: 'Client disconnected' }
|
||
connection.socket.send(JSON.stringify(response))
|
||
})
|
||
return request
|
||
}
|
||
Bot.on("message", e => {
|
||
const messageData = {
|
||
notice: 'clientMessage',
|
||
message: e.message,
|
||
sender: e.sender,
|
||
group: {
|
||
isGroup: e.isGroup,
|
||
group_id: e.group_id,
|
||
group_name: e.group_name
|
||
},
|
||
quotable: {
|
||
user_id: e.user_id,
|
||
time: e.time,
|
||
seq: e.seq,
|
||
rand: e.rand,
|
||
message: e.message,
|
||
user_name: e.sender.nickname,
|
||
}
|
||
}
|
||
if (clients) {
|
||
for (const index in clients) {
|
||
const user = GetUser(index)
|
||
if (user.autho == 'admin' || user.user == e.user_id) {
|
||
clients[index].send(JSON.stringify(messageData))
|
||
}
|
||
}
|
||
}
|
||
})
|
||
server.get('/ws', {
|
||
websocket: true
|
||
}, wsFn)
|
||
|
||
// 获取系统参数
|
||
server.post('/sysconfig', async (request, reply) => {
|
||
const token = request.cookies.token || request.body?.token || 'unknown'
|
||
const user = UserInfo(token)
|
||
if (!user) {
|
||
reply.send({ err: '未登录' })
|
||
} else if (user.autho === 'admin') {
|
||
let redisConfig = {}
|
||
if (await redis.exists('CHATGPT:BING_TOKENS') != 0) {
|
||
let bingTokens = await redis.get('CHATGPT:BING_TOKENS')
|
||
if (bingTokens) { bingTokens = JSON.parse(bingTokens) } else bingTokens = []
|
||
redisConfig.bingTokens = bingTokens
|
||
} else {
|
||
redisConfig.bingTokens = []
|
||
}
|
||
if (await redis.exists('CHATGPT:CONFIRM') != 0) {
|
||
redisConfig.turnConfirm = await redis.get('CHATGPT:CONFIRM') === 'on'
|
||
}
|
||
if (await redis.exists('CHATGPT:USE') != 0) {
|
||
redisConfig.useMode = await redis.get('CHATGPT:USE')
|
||
}
|
||
if (await redis.exists('CHATGPT:?') != 0) {
|
||
redisConfig.openAiPlatformAccessToken = await redis.get('CHATGPT:TOKEN')
|
||
}
|
||
reply.send({
|
||
chatConfig: Config,
|
||
redisConfig
|
||
})
|
||
} else {
|
||
let userSetting = await redis.get(`CHATGPT:USER:${user.user}`)
|
||
if (!userSetting) {
|
||
userSetting = {
|
||
usePicture: Config.defaultUsePicture,
|
||
useTTS: Config.defaultUseTTS,
|
||
ttsRole: Config.defaultTTSRole
|
||
}
|
||
} else {
|
||
userSetting = JSON.parse(userSetting)
|
||
}
|
||
reply.send({
|
||
userSetting
|
||
})
|
||
}
|
||
return reply
|
||
})
|
||
|
||
// 设置系统参数
|
||
server.post('/saveconfig', async (request, reply) => {
|
||
const token = request.cookies.token || request.body?.token || 'unknown'
|
||
const user = UserInfo(token)
|
||
const body = request.body || {}
|
||
let changeConfig = []
|
||
if (!user) {
|
||
reply.send({ state: false, error: '未登录' })
|
||
} else if (user.autho === 'admin') {
|
||
const chatdata = body.chatConfig || {}
|
||
for (let [keyPath, value] of Object.entries(chatdata)) {
|
||
if (keyPath === 'blockWords' || keyPath === 'promptBlockWords' || keyPath === 'initiativeChatGroups') { value = value.toString().split(/[,,;;\|]/) }
|
||
if (Config[keyPath] != value) {
|
||
//检查云服务api
|
||
if (keyPath === 'cloudTranscode') {
|
||
const referer = request.headers.referer;
|
||
const origin = referer.match(/(https?:\/\/[^/]+)/)[1];
|
||
const checkCloud = await fetch(`${value}/check`,
|
||
{
|
||
method: 'POST',
|
||
headers: {
|
||
'Content-Type': 'application/json'
|
||
},
|
||
body: JSON.stringify({
|
||
url: origin
|
||
})
|
||
})
|
||
if (checkCloud.ok) {
|
||
const checkCloudData = await checkCloud.json()
|
||
if (checkCloudData.state != 'ok') {
|
||
value = ''
|
||
}
|
||
} else value = ''
|
||
}
|
||
changeConfig.push({
|
||
item: keyPath,
|
||
old: Config[keyPath],
|
||
new: value
|
||
})
|
||
Config[keyPath] = value
|
||
}
|
||
}
|
||
const redisConfig = body.redisConfig || {}
|
||
if (redisConfig.bingTokens != null) {
|
||
await redis.set('CHATGPT:BING_TOKENS', JSON.stringify(redisConfig.bingTokens))
|
||
}
|
||
if (redisConfig.turnConfirm != null) {
|
||
await redis.set('CHATGPT:CONFIRM', redisConfig.turnConfirm ? 'on' : 'off')
|
||
}
|
||
if (redisConfig.useMode != null) {
|
||
await redis.set('CHATGPT:USE', redisConfig.useMode)
|
||
}
|
||
if (redisConfig.openAiPlatformAccessToken != null) {
|
||
await redis.set('CHATGPT:TOKEN', redisConfig.openAiPlatformAccessToken)
|
||
}
|
||
reply.send({ change: changeConfig, state: true })
|
||
// 通知所有WS客户端刷新数据
|
||
if (clients) {
|
||
for (const index in clients) {
|
||
const user = GetUser(index)
|
||
if (user.autho == 'admin') {
|
||
clients[index].send(JSON.stringify({
|
||
notice: 'updateConfig'
|
||
}))
|
||
}
|
||
}
|
||
}
|
||
} else {
|
||
if (body.userSetting) {
|
||
await redis.set(`CHATGPT:USER:${user.user}`, JSON.stringify(body.userSetting))
|
||
}
|
||
if (body.userConfig) {
|
||
let temp_userData = await getUserData(user.user)
|
||
if (body.userConfig.mode) {
|
||
temp_userData.mode = body.userConfig.mode
|
||
}
|
||
if (body.userConfig.cast) {
|
||
temp_userData.cast = body.userConfig.cast
|
||
}
|
||
await setUserData(user.user, temp_userData)
|
||
}
|
||
reply.send({ state: true })
|
||
}
|
||
return reply
|
||
})
|
||
|
||
// 系统服务测试
|
||
server.post('/serverTest', async (request, reply) => {
|
||
let serverState = {
|
||
cache: false,
|
||
cloud: false
|
||
}
|
||
if (Config.cacheUrl) {
|
||
const checkCacheUrl = await fetch(Config.cacheUrl, { method: 'GET' })
|
||
if (checkCacheUrl.ok) {
|
||
serverState.cache = true
|
||
}
|
||
}
|
||
if (Config.cloudTranscode) {
|
||
const checkCheckCloud = await fetch(Config.cloudTranscode, { method: 'GET' })
|
||
if (checkCheckCloud.ok) {
|
||
serverState.cloud = true
|
||
}
|
||
}
|
||
reply.send(serverState)
|
||
return reply
|
||
})
|
||
|
||
server.addHook('onRequest', (request, reply, done) => {
|
||
if (request.method == 'POST') { Statistics.SystemAccess.count += 1 }
|
||
if (request.method == 'GET') { Statistics.WebAccess.count += 1 }
|
||
done()
|
||
})
|
||
// 定时任务
|
||
let rule = new schedule.RecurrenceRule()
|
||
rule.hour = 0
|
||
rule.minute = 0
|
||
let job_Statistics = schedule.scheduleJob(rule, function () {
|
||
Statistics.SystemAccess.oldCount = Statistics.SystemAccess.count
|
||
Statistics.CacheFile.oldCount = Statistics.CacheFile.count
|
||
Statistics.WebAccess.oldCount = Statistics.WebAccess.count
|
||
Statistics.SystemAccess.count = 0
|
||
Statistics.CacheFile.count = 0
|
||
Statistics.WebAccess.count = 0
|
||
})
|
||
let job_Statistics_SystemLoad = schedule.scheduleJob('0 * * * *', async function () {
|
||
Statistics.SystemLoad.count = await getLoad()
|
||
Statistics.SystemLoad.oldCount = Statistics.SystemLoad.count
|
||
})
|
||
|
||
server.listen({
|
||
port: Config.serverPort || 3321,
|
||
host: '::'
|
||
}, (error) => {
|
||
if (error) {
|
||
server.log.error(`服务启动失败: ${error}`)
|
||
} else {
|
||
server.log.info(`server listening on ${server.server.address().port}`)
|
||
}
|
||
})
|
||
}
|