mirror of
https://github.com/ikechan8370/chatgpt-plugin.git
synced 2025-12-16 13:27:08 +00:00
* 修复引用转发,默认bing模式并发 * 开启stream增加稳定性 * fix: remove queue element only in non-bing mode * 使用chatgpt-api自带的超时逻辑,文字过多时启动切换到图片输出防止被吞 * Update chat.js * 添加Bing专用的图片输出样式 * 添加chatgpt的新图片模式,临时处理切换api导致的对话异常 * 修改bing样式表 * 为图片添加外部页面缓存 * 为图片模式添加MathJax * feat: add switch for qrcode * 防止script攻击 * 修复网页模板错误 * 修复bing页面引用错误 * 缓存服务器异常时处理 * 添加默认配置加载 * 修复配置文件路径错误 * 删除重复的模板文件,修复二维码地址错误 * 修正图片渲染错误 * 修复引用渲染错误 * 二维码网址统一改为使用本地配置 * 添加关闭思考提示的配置项 * 修复在Windows上无法载入配置文件的问题 * 修复关闭qr的情况下渲染错误 * 改为使用base64传递返回数据 * 当异常过多时使用图片输出 * 添加锅巴面板配置支持 * 补充遗漏的默认配置 * 修复qr模式下引用未被传递的问题 * 修复未将引用数据传输给缓存服务器的问题 * 删除无用的bingTimeoutMs配置项 * 添加消息队列超时弹出 * 优化图片模式处理,解决对话队列卡住的问题 * 添加对图片ocr的支持 * 添加图片识别配置项 * 添加黑名单配置项 * 修复一些bug * 修改锅巴配置格式和描述 * 传入数据也使用markdown * 图片识别换行改为marked兼容 * 添加绘图CD配置项 * 独立render模块,添加图片回复引用 * 添加必应风格 * 修复上下文,修改bing样式 * 修复上下文 * 添加Sydney上下文支持 * 调整不同模式下的bing渲染颜色 * 修复样式 * 修复无法结束会话的问题 * fix: 更新版本号 * 修复无法结束对话的问题 * 向缓存服务器传送样式 * 为网址格式的配置添加验证 * 去除重复的Keyv删除,取消锅巴配置格式检查 * 闭合中断的代码块 * 试添加Sydney图片模式的情感显示 * 修复at不兼容 * 处理意外的markdown包裹和结构解析修复 * 修复markdown处理的顺序错误 * 兼容json换行 * 重写completeJSON和使用 * 修复换行格式异常 * 均衡BingToken使用 * 修复删除token的数组处理错误 * 修改token文字描述 * 创建本地缓存服务 * 修复首次使用无法添加bingtoken的问题 * 修复意外的删除格式问题,添加查看token功能 * 修复路由错误,暂时固定ip测试 * 恢复引用功能 * 更新渲染页面 * 更换缓存目录 * 清除调试用消息 * 调整屏幕分辨率 * 使用服务器生成的访问地址 * 改为使用api获取公网ip * 修复引用显示 * 添加依赖需求 * 更新渲染页面和渲染api * 修复渲染页面错误 * 修复建议字符串切割,添加帮助路由 * 添加内容中图片数据获取功能 * 试修复suggestbug * 修复图片导致服务器卡死的问题 * 暂时禁用图片 * 尝试恢复图片 * 添加链接图片识别 * 替换掉request * 修复可能的responseUrls空值 * 优化格式 * 更新渲染页面 * 尝试新的引用索引 * 取消渲染时旧的策略 * 更新帮助页面 * 修复帮助路由 * 修复渲染页面错误 * 修复错误的正则 * 修改系统api服务 * 添加配置项 * 将新渲染方式加入配置并还原原渲染方式,进行并存 * 暂时取消端口设置功能 * 重新开启端口设置 * 修复旧渲染引用 * 更新帮助样式 * 更新帮助,增强功能 * 有cacheHost的情况下不再附带端口号 * 添加渲染图片的宽度设置 * 添加渲染页面宽度调整,修bug * 修复二维码不显示 * 添加第三方渲染支持 * 修复一些渲染页面问题 * 更新渲染页面 * 修正错误的变量调用 * 添加新渲染模式bot命名 * 修复空消息问题 * 撤销之前的修复,使用新方法修复 * 修复返回空页面问题 * 尝试不依赖网络获取外网地址 * 修bug,初步创建管理系统 * 依赖名写错了 * 修复错误的异步 * 修正错误的配置调用 * 放弃本机设置的获取方案,对服务器获取多半失效 * 添加配置页面接口 * 更新渲染页面 * 添加依赖 * 修复bug * 移除windows性能显示,更换依赖 * 添加依赖 * 修复图片异常时不反回文字而是直接报错的问题 * 修改必应token记录和均衡方法,更新渲染页面 * 修复错误 * 修复bug,更新渲染页面 * 更新渲染 * 修复ip错误 * 完善配置页面 * 渲染页面错误修复 * 更新版本号 * 只获取一次有效ip * 修复渲染页面bug --------- Co-authored-by: ikechan8370 <geyinchibuaa@gmail.com> Co-authored-by: Err0rCM <68117733+Err0rCM@users.noreply.github.com>
284 lines
9.5 KiB
JavaScript
284 lines
9.5 KiB
JavaScript
import plugin from '../../../lib/plugins/plugin.js'
|
||
import { createImage, editImage, imageVariation } from '../utils/dalle.js'
|
||
import { makeForwardMsg } from '../utils/common.js'
|
||
import _ from 'lodash'
|
||
import { Config } from '../utils/config.js'
|
||
import BingDrawClient from '../utils/BingDraw.js'
|
||
|
||
export class dalle extends plugin {
|
||
constructor (e) {
|
||
super({
|
||
name: 'ChatGPT-Plugin Dalle 绘图',
|
||
dsc: 'ChatGPT-Plugin基于OpenAI Dalle的绘图插件',
|
||
event: 'message',
|
||
priority: 500,
|
||
rule: [
|
||
{
|
||
reg: '^#(chatgpt|ChatGPT|dalle|Dalle)(绘图|画图)',
|
||
fnc: 'draw'
|
||
},
|
||
{
|
||
reg: '^#(chatgpt|ChatGPT|dalle|Dalle)(修图|图片变形|改图)$',
|
||
fnc: 'variation'
|
||
},
|
||
{
|
||
reg: '^#(搞|改)(她|他)头像',
|
||
fnc: 'avatarVariation'
|
||
},
|
||
{
|
||
reg: '^#(chatgpt|dalle)编辑图片',
|
||
fnc: 'edit'
|
||
},
|
||
{
|
||
reg: '^#bing(画图|绘图)',
|
||
fnc: 'bingDraw'
|
||
}
|
||
]
|
||
})
|
||
}
|
||
|
||
async draw (e) {
|
||
if (!Config.enableDraw) {
|
||
this.reply('画图功能未开启')
|
||
return false
|
||
}
|
||
let ttl = await redis.ttl(`CHATGPT:DRAW:${e.sender.user_id}`)
|
||
if (ttl > 0 && !e.isMaster) {
|
||
this.reply(`冷却中,请${ttl}秒后再试`)
|
||
return false
|
||
}
|
||
let splits = _.split(e.msg, '图', 2)
|
||
if (splits.length < 2) {
|
||
this.reply('请带上绘图要求')
|
||
return false
|
||
}
|
||
let rules = _.split(splits[1], '/')
|
||
let [prompt = '', num = '1', size = '512x512'] = rules.slice(0, 3)
|
||
if (['256x256', '512x512', '1024x1024'].indexOf(size) === -1) {
|
||
this.reply('大小不符合要求,必须是256x256/512x512/1024x1024中的一个')
|
||
return false
|
||
}
|
||
await redis.set(`CHATGPT:DRAW:${e.sender.user_id}`, 'c', { EX: 30 })
|
||
let priceMap = {
|
||
'1024x1024': 0.02,
|
||
'512x512': 0.018,
|
||
'256x256': 0.016
|
||
}
|
||
num = parseInt(num, 10)
|
||
if (num > 5) {
|
||
this.reply('太多啦!你要花光我的余额吗!')
|
||
return false
|
||
}
|
||
await this.reply(`正在为您绘制大小为${size}的${num}张图片,预计消耗${priceMap[size] * num}美元余额,请稍候……`)
|
||
try {
|
||
let images = (await createImage(prompt, num, size)).map(image => segment.image(`base64://${image}`))
|
||
if (images.length > 1) {
|
||
this.reply(await makeForwardMsg(e, images, prompt))
|
||
} else {
|
||
this.reply(images[0], true)
|
||
}
|
||
} catch (err) {
|
||
logger.error(err.response?.data?.error?.message)
|
||
this.reply(`绘图失败: ${err.response?.data?.error?.message}`, true)
|
||
await redis.del(`CHATGPT:DRAW:${e.sender.user_id}`)
|
||
}
|
||
}
|
||
|
||
async variation (e) {
|
||
if (!Config.enableDraw) {
|
||
this.reply('画图功能未开启')
|
||
return false
|
||
}
|
||
let ttl = await redis.ttl(`CHATGPT:VARIATION:${e.sender.user_id}`)
|
||
if (ttl > 0 && !e.isMaster) {
|
||
this.reply(`冷却中,请${ttl}秒后再试`)
|
||
return false
|
||
}
|
||
let imgUrl
|
||
if (e.source) {
|
||
let reply
|
||
if (e.isGroup) {
|
||
reply = (await e.group.getChatHistory(e.source.seq, 1)).pop()?.message
|
||
} else {
|
||
reply = (await e.friend.getChatHistory(e.source.time, 1)).pop()?.message
|
||
}
|
||
if (reply) {
|
||
for (let val of reply) {
|
||
if (val.type === 'image') {
|
||
console.log(val)
|
||
imgUrl = val.url
|
||
break
|
||
}
|
||
}
|
||
}
|
||
} else if (e.img) {
|
||
console.log(e.img)
|
||
imgUrl = e.img[0]
|
||
}
|
||
if (!imgUrl) {
|
||
this.reply('图呢?')
|
||
return false
|
||
}
|
||
await redis.set(`CHATGPT:VARIATION:${e.sender.user_id}`, 'c', { EX: 30 })
|
||
await this.reply('正在为您生成图片变形,请稍候……')
|
||
try {
|
||
let images = (await imageVariation(imgUrl)).map(image => segment.image(`base64://${image}`))
|
||
if (images.length > 1) {
|
||
this.reply(await makeForwardMsg(e, images))
|
||
} else {
|
||
this.reply(images[0], true)
|
||
}
|
||
} catch (err) {
|
||
console.log(err.response?.data?.error?.message || err.message || JSON.stringify(err.response || {}))
|
||
this.reply(`绘图失败: ${err.response?.data?.error?.message || err.message || JSON.stringify(err.response || {})}`, true)
|
||
await redis.del(`CHATGPT:VARIATION:${e.sender.user_id}`)
|
||
}
|
||
}
|
||
|
||
async avatarVariation (e) {
|
||
if (!Config.enableDraw) {
|
||
this.reply('画图功能未开启')
|
||
return false
|
||
}
|
||
let ats = e.message.filter(m => m.type === 'at').filter(at => at.qq !== e.self_id)
|
||
if (ats.length > 0) {
|
||
for (let i = 0; i < ats.length; i++) {
|
||
let qq = ats[i].qq
|
||
let imgUrl = `https://q1.qlogo.cn/g?b=qq&s=0&nk=${qq}`
|
||
try {
|
||
let images = (await imageVariation(imgUrl)).map(image => segment.image(`base64://${image}`))
|
||
if (images.length > 1) {
|
||
this.reply(await makeForwardMsg(e, images))
|
||
} else {
|
||
this.reply(images[0], true)
|
||
}
|
||
} catch (err) {
|
||
console.log(err.response?.data?.error?.message || err.message || JSON.stringify(err.response || {}))
|
||
this.reply(`搞失败了: ${err.response?.data?.error?.message || err.message || JSON.stringify(err.response || {})}`, true)
|
||
await redis.del(`CHATGPT:VARIATION:${e.sender.user_id}`)
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
async edit (e) {
|
||
if (!Config.enableDraw) {
|
||
this.reply('画图功能未开启')
|
||
return false
|
||
}
|
||
let ttl = await redis.ttl(`CHATGPT:EDIT:${e.sender.user_id}`)
|
||
if (ttl > 0 && !e.isMaster) {
|
||
this.reply(`冷却中,请${ttl}秒后再试`)
|
||
return false
|
||
}
|
||
let imgUrl
|
||
if (e.source) {
|
||
let reply
|
||
if (e.isGroup) {
|
||
reply = (await e.group.getChatHistory(e.source.seq, 1)).pop()?.message
|
||
} else {
|
||
reply = (await e.friend.getChatHistory(e.source.time, 1)).pop()?.message
|
||
}
|
||
if (reply) {
|
||
for (let val of reply) {
|
||
if (val.type === 'image') {
|
||
console.log(val)
|
||
imgUrl = val.url
|
||
break
|
||
}
|
||
}
|
||
}
|
||
} else if (e.img) {
|
||
console.log(e.img)
|
||
imgUrl = e.img[0]
|
||
}
|
||
if (!imgUrl) {
|
||
this.reply('图呢?')
|
||
return false
|
||
}
|
||
await redis.set(`CHATGPT:EDIT:${e.sender.user_id}`, 'c', { EX: 30 })
|
||
await this.reply('正在为您编辑图片,请稍候……')
|
||
|
||
let command = _.trimStart(e.msg, '#chatgpt编辑图片')
|
||
command = _.trimStart(command, '#dalle编辑图片')
|
||
// command = 'A bird on it/100,100,300,200/2/512x512'
|
||
let args = command.split('/')
|
||
let [prompt = '', position = '', num = '1', size = '512x512'] = args.slice(0, 4)
|
||
if (!prompt || !position) {
|
||
this.reply('编辑图片必须填写prompt和涂抹位置.参考格式:A bird on it/100,100,300,200/2/512x512')
|
||
return false
|
||
}
|
||
num = parseInt(num, 10)
|
||
if (num > 5) {
|
||
this.reply('太多啦!你要花光我的余额吗!')
|
||
return false
|
||
}
|
||
try {
|
||
let images = (await editImage(imgUrl, position.split(',').map(p => parseInt(p, 10)), prompt, num, size))
|
||
.map(image => segment.image(`base64://${image}`))
|
||
if (images.length > 1) {
|
||
this.reply(await makeForwardMsg(e, images, prompt))
|
||
} else {
|
||
this.reply(images[0], true)
|
||
}
|
||
} catch (err) {
|
||
logger.error(err.response?.data?.error?.message || err.message || JSON.stringify(err.response || {}))
|
||
this.reply(`图片编辑失败: ${err.response?.data?.error?.message || err.message || JSON.stringify(err.response || {})}`, true)
|
||
await redis.del(`CHATGPT:EDIT:${e.sender.user_id}`)
|
||
}
|
||
}
|
||
|
||
async bingDraw (e) {
|
||
let ttl = await redis.ttl(`CHATGPT:DRAW:${e.sender.user_id}`)
|
||
if (ttl > 0 && !e.isMaster) {
|
||
this.reply(`冷却中,请${ttl}秒后再试`)
|
||
return false
|
||
}
|
||
let prompt = e.msg.replace(/^#bing(画图|绘图)/, '')
|
||
if (!prompt) {
|
||
this.reply('请提供绘图prompt')
|
||
return false
|
||
}
|
||
|
||
let bingToken = ''
|
||
if (await redis.exists('CHATGPT:BING_TOKENS') != 0) {
|
||
let bingTokens = JSON.parse(await redis.get('CHATGPT:BING_TOKENS'))
|
||
const normal = bingTokens.filter(element => element.State === '正常')
|
||
const restricted = bingTokens.filter(element => element.State === '受限')
|
||
if (normal.length > 0) {
|
||
const minElement = normal.reduce((min, current) => {
|
||
return current.Usage < min.Usage ? current : min
|
||
})
|
||
bingToken = minElement.Token
|
||
} else if (restricted.length > 0) {
|
||
allThrottled = true
|
||
const minElement = restricted.reduce((min, current) => {
|
||
return current.Usage < min.Usage ? current : min
|
||
})
|
||
bingToken = minElement.Token
|
||
} else {
|
||
throw new Error('全部Token均已失效,暂时无法使用')
|
||
}
|
||
}
|
||
if (!bingToken) {
|
||
throw new Error('未绑定Bing Cookie,请使用#chatgpt设置必应token命令绑定Bing Cookie')
|
||
}
|
||
// 记录token使用
|
||
let bingTokens = JSON.parse(await redis.get('CHATGPT:BING_TOKENS'))
|
||
const index = bingTokens.findIndex(element => element.Token === bingToken)
|
||
bingTokens[index].Usage += 1
|
||
await redis.set('CHATGPT:BING_TOKENS', JSON.stringify(bingTokens))
|
||
|
||
let client = new BingDrawClient({
|
||
baseUrl: Config.sydneyReverseProxy,
|
||
userToken: bingToken
|
||
})
|
||
await redis.set(`CHATGPT:DRAW:${e.sender.user_id}`, 'c', { EX: 30 })
|
||
try {
|
||
await client.getImages(prompt, e)
|
||
} catch (err) {
|
||
await redis.del(`CHATGPT:DRAW:${e.sender.user_id}`)
|
||
await e.reply('绘图失败:' + err)
|
||
}
|
||
}
|
||
}
|