mirror of
https://github.com/yangjian102621/geekai.git
synced 2025-11-05 16:53:46 +08:00
Compare commits
85 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
4e65a5b1a1 | ||
|
|
b09d23f97f | ||
|
|
3529649ba9 | ||
|
|
fdd659f393 | ||
|
|
9eb8da2789 | ||
|
|
ffb1ef0470 | ||
|
|
862c6aea43 | ||
|
|
54fe4b7588 | ||
|
|
c6062ee70e | ||
|
|
bed184dc1f | ||
|
|
29094ba3b3 | ||
|
|
a18188876c | ||
|
|
4faee3e48e | ||
|
|
1a6afcd266 | ||
|
|
f567831d92 | ||
|
|
cf36ca4285 | ||
|
|
0e4ae01498 | ||
|
|
7b90f8cb13 | ||
|
|
c33215529a | ||
|
|
c5be114db2 | ||
|
|
cab955c292 | ||
|
|
ca8c8e6490 | ||
|
|
253951b4b3 | ||
|
|
4d6444ebf3 | ||
|
|
94d8d8a9d4 | ||
|
|
e02badf7bb | ||
|
|
dd88622c64 | ||
|
|
c4d7126c4d | ||
|
|
86bc063941 | ||
|
|
dce85eb519 | ||
|
|
4ab879d697 | ||
|
|
681e52df50 | ||
|
|
fb554c0315 | ||
|
|
accf8eeb77 | ||
|
|
3e41edd3b5 | ||
|
|
9126cfff20 | ||
|
|
9806d5ff4c | ||
|
|
d1d13a72e4 | ||
|
|
00c520d066 | ||
|
|
797ff66474 | ||
|
|
9d51a478b9 | ||
|
|
1d4179df75 | ||
|
|
917b6012e8 | ||
|
|
da14632794 | ||
|
|
a868a8a8b7 | ||
|
|
5037df744f | ||
|
|
da88a501ad | ||
|
|
b9885e8de4 | ||
|
|
22efe81080 | ||
|
|
2926717aef | ||
|
|
a49d54d66c | ||
|
|
ce0267e25b | ||
|
|
9088d22a66 | ||
|
|
1ff32d5d0a | ||
|
|
adf6916598 | ||
|
|
31c14bf748 | ||
|
|
5395385d1e | ||
|
|
0035da548b | ||
|
|
9bceaade05 | ||
|
|
3194becdad | ||
|
|
6174b17c24 | ||
|
|
53fa4a20e9 | ||
|
|
43c1de51f5 | ||
|
|
7eb8c5ec35 | ||
|
|
296bf63196 | ||
|
|
6c65a21692 | ||
|
|
daf83cfc84 | ||
|
|
871f5d39e4 | ||
|
|
3f91f37aff | ||
|
|
a08981f876 | ||
|
|
5187a43543 | ||
|
|
6a733de556 | ||
|
|
b9e9eae93f | ||
|
|
811f12135a | ||
|
|
2c172c0851 | ||
|
|
399a16fa28 | ||
|
|
d971e95900 | ||
|
|
0b6940b121 | ||
|
|
ad0f96fcb1 | ||
|
|
063b5655f7 | ||
|
|
d03ed6570b | ||
|
|
1795a891ce | ||
|
|
05bdd81646 | ||
|
|
cba54be913 | ||
|
|
da0acfe851 |
68
CHANGELOG.md
Normal file
68
CHANGELOG.md
Normal file
@@ -0,0 +1,68 @@
|
|||||||
|
# 更新日志
|
||||||
|
|
||||||
|
## v3.0.6
|
||||||
|
1. 管理后台:新增用户名和手机号码搜索功能
|
||||||
|
2. 管理后台:新增重置用户密码功能
|
||||||
|
3. 管理后台:支持关闭注册功能,新增添加用户功能,适用于内部使用场景
|
||||||
|
4. 管理后台:新增仪表盘页面,统计当天的新增用户,新增会话数据,以及 Token 消耗
|
||||||
|
5. Bug修复:修复注册页面验证码不显示 Bug
|
||||||
|
6. Bug修复:优化上下文 Token 计算算法,修复聊天上下文超出限制时循环发送消息的 Bug
|
||||||
|
7. 功能修正:允许用户使用手机号码登录
|
||||||
|
8. 功能优化:更新系统配置后同步更新服务端内存变量数据
|
||||||
|
9. 功能优化:优化打包脚本,减少容器镜像大小
|
||||||
|
|
||||||
|
## v3.0.5
|
||||||
|
|
||||||
|
重磅功能更新!!! 新增函数插件支持,可以轻松地接入你的第三方插件服务,ChatGPT 自动帮您调用对应的函数完成任务。
|
||||||
|
|
||||||
|
1. 新增函数功能支持,全球早报,今日头条和微博热搜等插件服务,您也可以接入自己的第三方服务。
|
||||||
|
2. 集成微信机器人模块,可以通过微信个人收款码来完成充值,无需接入微信支付功能也可以完成收款功能。
|
||||||
|
3. 用户注册添加短信验证码功能,引入交互安全认证服务,有效防刷短信。
|
||||||
|
4. 支持配置聊天上下文深度,精确统计每轮对话所消耗的总 TOKEN 数量。
|
||||||
|
5. 修复已知的 Bug。
|
||||||
|
|
||||||
|
## v3.0.4
|
||||||
|
|
||||||
|
1. 调整项目目录结构,移除其他语言 API 目录
|
||||||
|
2. 修复 nodejs apple M1 跨平台打包,运行报错 exec format error
|
||||||
|
3. 增加用户 token 消耗统计功能
|
||||||
|
|
||||||
|
## v3.0.3
|
||||||
|
|
||||||
|
1. 优化启动参数接收处理,支持环境变量传参
|
||||||
|
2. 修复 PC 端聊天界面出现滚动条的 Bug
|
||||||
|
3. 修正前端 user_init_call 字段错误和用户注册初始化头像路径问题
|
||||||
|
4. 更改 docker 构建镜像的基础镜像,改用作者的阿里云镜像,这样打包更快一些。
|
||||||
|
|
||||||
|
## v3.0.2
|
||||||
|
|
||||||
|
1. Feat:新增移动端的聊天和用户设置功能
|
||||||
|
2. Fix: 修复 markdown 换行符解析的 Bug
|
||||||
|
3. Feat: 新增头像上传功能
|
||||||
|
4. Docs: 增加容器部署支持,支持 docker-compose 一键部署
|
||||||
|
5. Fix: 增加全局错误处理 handler,修复业务处理异常导致服务退出的 Bug
|
||||||
|
|
||||||
|
## v3.0.1
|
||||||
|
|
||||||
|
1. 紧急修复前端 Home 组件路由被后台管理 Home 组件路由覆盖的 Bug。
|
||||||
|
2. 增加 docker-compose 部署脚本
|
||||||
|
|
||||||
|
## v3.0.0
|
||||||
|
|
||||||
|
全新的重构版本!!!
|
||||||
|
新版的系统前后端都进行大改动的重构,后端还是用的 Gin Web 框架,但是作者整合了 fx 自动注入框架,整个后端应用结构非常简洁,特别适合二次开发。
|
||||||
|
另外,数据存储用 MySQL 替换了 leveldb, 因为要对 C 端,后期会涉及到很多业务数据查询统计,leveldb 已经完全不够用了。
|
||||||
|
前后台技术架构还是基于 `Vue3 + Element-Plus` ,但是页面风格已经全部变了,几乎所有页面样式代码都重写了,希望会你是希望的风格!
|
||||||
|
|
||||||
|
此次重构改版主要是为了后面功能的扩展准备了。
|
||||||
|
|
||||||
|
新版本已经实现的功能如下:
|
||||||
|
|
||||||
|
1. 引入用户体系,新增用户注册和登录功能。
|
||||||
|
2. 聊天页面改版,实现了跟 ChatGPT 官方版本一致的聊天体验。
|
||||||
|
3. 创建会话的时候可以选择聊天角色和模型。
|
||||||
|
4. 新增聊天设置功能,用户可以导入自己的 API KEY
|
||||||
|
5. 保存聊天记录,支持聊天上下文。
|
||||||
|
6. 重构后台管理模块,更友好,扩展性更好的后台管理系统。
|
||||||
|
7. 引入 ip2region 组件,记录用户的登录IP和地址。
|
||||||
|
8. 支持会话搜索过滤。
|
||||||
11
Dockerfile
11
Dockerfile
@@ -1,11 +0,0 @@
|
|||||||
# FROM 表示设置要制作的镜像基于哪个镜像,FROM指令必须是整个Dockerfile的第一个指令,如果指定的镜像不存在默认会自动从Docker Hub上下载。
|
|
||||||
FROM centos:7
|
|
||||||
|
|
||||||
WORKDIR /usr/src/app
|
|
||||||
COPY src/bin/wechatGPT-amd64-linux /usr/src/app
|
|
||||||
|
|
||||||
# 容器对外暴露的端口号,这里和配置文件保持一致就可以
|
|
||||||
EXPOSE 5678
|
|
||||||
|
|
||||||
# 容器启动时执行的命令
|
|
||||||
CMD ["./wechatGPT-amd64-linux"]
|
|
||||||
201
README.md
201
README.md
@@ -6,37 +6,49 @@
|
|||||||
* 聊天体验跟 ChatGPT 官方版本完全一致。
|
* 聊天体验跟 ChatGPT 官方版本完全一致。
|
||||||
* 内置了各种预训练好的角色,比如小红书写手,英语翻译大师,苏格拉底,孔子,乔布斯,周报助手等。轻松满足你的各种聊天和应用需求。
|
* 内置了各种预训练好的角色,比如小红书写手,英语翻译大师,苏格拉底,孔子,乔布斯,周报助手等。轻松满足你的各种聊天和应用需求。
|
||||||
|
|
||||||
**本项目基于 MIT 协议,免费开放全部源代码,可以作为个人学习使用或者商用。如需商用建议联系作者登记,仅做统计使用,优秀项目我们将在项目首页为您展示。
|
|
||||||
**
|
|
||||||
|
|
||||||
## 功能截图
|
## 功能截图
|
||||||
|
|
||||||
### 1.PC 端聊天界面
|
### PC 端聊天界面
|
||||||
|
|
||||||

|

|
||||||
|
|
||||||
### 2. 新版聊天界面
|
### 新版聊天界面
|
||||||
|
|
||||||

|

|
||||||
|
|
||||||
### 3. 用户设置
|
### 自动调用函数插件
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
### 用户设置
|
||||||
|
|
||||||

|

|
||||||
|
|
||||||
### 4. 登录页面
|
### 登录页面
|
||||||
|
|
||||||

|

|
||||||
|
|
||||||
### 5. 管理后台
|
### 管理后台
|
||||||
|
|
||||||

|

|
||||||
|
|
||||||

|

|
||||||
|
|
||||||
### 6. 体验地址
|
### 移动端 Web 页面
|
||||||
|
|
||||||
> 体验地址:[https://www.chat-plus.net/chat](https://www.chat-plus.net/chat) <br/>
|

|
||||||
> 涉及到数据隐私问题,没有提供共享账号,大家自己快速注册一个账号就可以免费体验
|

|
||||||
|

|
||||||
|
|
||||||
|
### 7. 体验地址
|
||||||
|
|
||||||
|
> 免费体验地址:[https://www.chat-plus.net/chat](https://www.chat-plus.net/chat) <br/>
|
||||||
|
> **注意:请合法使用,禁止输出任何敏感、不友好或违规的内容!!!**
|
||||||
|
|
||||||
|
## 使用须知
|
||||||
|
|
||||||
|
1. 本项目基于 MIT 协议,免费开放全部源代码,可以作为个人学习使用或者商用。
|
||||||
|
2. 如需商用必须保留版权信息,请自觉遵守。确保合法合规使用,在运营过程中产生的一切任何后果自负,与作者无关。
|
||||||
|
|
||||||
## 项目介绍
|
## 项目介绍
|
||||||
|
|
||||||
@@ -67,9 +79,10 @@ ChatGPT 的服务。
|
|||||||
3. 创建会话的时候可以选择聊天角色和模型。
|
3. 创建会话的时候可以选择聊天角色和模型。
|
||||||
4. 新增聊天设置功能,用户可以导入自己的 API KEY
|
4. 新增聊天设置功能,用户可以导入自己的 API KEY
|
||||||
5. 保存聊天记录,支持聊天上下文。
|
5. 保存聊天记录,支持聊天上下文。
|
||||||
7. 重构后台管理模块,更友好,扩展性更好的后台管理系统。
|
6. 重构后台管理模块,更友好,扩展性更好的后台管理系统。
|
||||||
8. 引入 ip2region 组件,记录用户的登录IP和地址。
|
7. 引入 ip2region 组件,记录用户的登录IP和地址。
|
||||||
9. 支持会话搜索过滤。
|
8. 支持会话搜索过滤。
|
||||||
|
9. 支持微信支付充值
|
||||||
|
|
||||||
## 项目地址
|
## 项目地址
|
||||||
|
|
||||||
@@ -80,35 +93,50 @@ ChatGPT 的服务。
|
|||||||
|
|
||||||
* [ ] 整合 Midjourney AI 绘画 API
|
* [ ] 整合 Midjourney AI 绘画 API
|
||||||
* [ ] 开发移动端聊天页面
|
* [ ] 开发移动端聊天页面
|
||||||
* [ ] 接入微信支付功能
|
* [x] 接入微信支付功能
|
||||||
|
* [x] 支持 ChatGPT 函数功能,通过函数实现插件
|
||||||
* [ ] 接入语音和 TTS API,支持语音聊天
|
* [ ] 接入语音和 TTS API,支持语音聊天
|
||||||
* [ ] 开发手机 App 客户端
|
* [ ] 开发手机 App 客户端
|
||||||
|
|
||||||
## 安装部署
|
## Docker 快速部署
|
||||||
|
|
||||||
由于本项目采用的是前后端分离的开发方式,所以部署也需要前后端分开部署。我这里以 linux 系统为例,演示一下部署过程:
|
> 鉴于最新不少网友反馈在部署的时候遇到一些问题,大部分问题都是相同的,所以我这边做了一个视频教程 [五分钟部署自己的 ChatGPT 服务](https://www.bilibili.com/video/BV1H14y1B7Qw/)。
|
||||||
|
> 习惯看视频教程的朋友可以去看视频教程,视频的语速比较慢,建议 2 倍速观看。
|
||||||
|
|
||||||
|
V3.0.0 版本以后已经支持使用容器部署了,跳过所有的繁琐的环境准备,一条命令就可以轻松部署上线。
|
||||||
|
|
||||||
### 1. 导入数据库
|
### 1. 导入数据库
|
||||||
|
|
||||||
|
首先我们需要创建一个 MySQL 容器,并导入初始数据库。
|
||||||
|
|
||||||
|
```shell
|
||||||
|
cd docker/mysql
|
||||||
|
# 创建 mysql 容器
|
||||||
|
docker-compose up -d
|
||||||
|
# 导入数据库
|
||||||
|
docker exec -i chatgpt-plus-mysql sh -c 'exec mysql -uroot -p12345678' < ../../database/chatgpt_plus.sql
|
||||||
|
```
|
||||||
|
|
||||||
|
如果你本地已经安装了 MySQL 服务,那么你只需手动导入数据库即可。
|
||||||
|
|
||||||
```shell
|
```shell
|
||||||
# 下载数据库
|
|
||||||
wget wget https://github.com/yangjian102621/chatgpt-plus/releases/download/v3.0.0/chatgpt_plus.sql
|
|
||||||
# 连接数据库
|
# 连接数据库
|
||||||
mysql -u username -p password
|
mysql -u username -p password
|
||||||
# 导入数据库
|
# 导入数据库
|
||||||
source chatgpt_plus.sql
|
source database/chatgpt_plus.sql
|
||||||
```
|
```
|
||||||
|
|
||||||
### 2. 修改配置文档
|
### 2. 修改配置文档
|
||||||
|
|
||||||
先拷贝项目中的 `api/go/config.sample.toml` 配置文档,修改代理地址和管理员密码:
|
修改配置文档 `docker/conf/config.toml` 配置文档,修改代理地址和管理员密码:
|
||||||
|
|
||||||
```toml
|
```toml
|
||||||
Listen = "0.0.0.0:5678"
|
Listen = "0.0.0.0:5678"
|
||||||
ProxyURL = ["YOUR_PROXY_URL"] # 替换成你本地代理,如:http://127.0.0.1:7777
|
ProxyURL = ["YOUR_PROXY_URL"] # 替换成你本地代理,如:http://127.0.0.1:7777
|
||||||
#ProxyURL = "http://127.0.0.1:7777"
|
|
||||||
#ProxyURL = "" 如果你的服务器本身就在墙外,那么你直接留空就好了
|
#ProxyURL = "" 如果你的服务器本身就在墙外,那么你直接留空就好了
|
||||||
MysqlDns = "mysql_user:mysql_pass@tcp(localhost:3306)/chatgpt_plus?charset=utf8&parseTime=True&loc=Local"
|
MysqlDns = "root:12345678@tcp(172.22.11.200:3307)/chatgpt_plus?charset=utf8&parseTime=True&loc=Local"
|
||||||
|
StartWechatBot = false # 是否启动微信机器人,默认关闭,如果设置为 TRUE 则启动服务的时候需要微信扫码登录
|
||||||
|
EnabledMsgService = false # 注册时是否开启短信验证功能,该功能需要配合短信服务一起使用
|
||||||
|
|
||||||
[Session]
|
[Session]
|
||||||
SecretKey = "azyehq3ivunjhbntz78isj00i4hz2mt9xtddysfucxakadq4qbfrt0b7q3lnvg80"
|
SecretKey = "azyehq3ivunjhbntz78isj00i4hz2mt9xtddysfucxakadq4qbfrt0b7q3lnvg80"
|
||||||
@@ -123,93 +151,56 @@ MysqlDns = "mysql_user:mysql_pass@tcp(localhost:3306)/chatgpt_plus?charset=utf8&
|
|||||||
[Manager]
|
[Manager]
|
||||||
Username = "admin"
|
Username = "admin"
|
||||||
Password = "admin123" # 如果是生产环境的话,这里管理员的密码记得修改
|
Password = "admin123" # 如果是生产环境的话,这里管理员的密码记得修改
|
||||||
|
|
||||||
|
[ApiConfig] # 插件 API 服务配置,此为第三方插件服务,如需使用请联系作者开通
|
||||||
|
ApiURL = "{URL}"
|
||||||
|
AppId = "{APP_ID}"
|
||||||
|
Token = "{TOKEN}"
|
||||||
|
|
||||||
|
[SmsConfig] # 阿里云短信服务配置
|
||||||
|
AccessKey = "{YOUR_ACCESS_KEY}"
|
||||||
|
AccessSecret = "{YOUR_SECRET_KEY}"
|
||||||
|
Product = "Dysmsapi"
|
||||||
|
Domain = "dysmsapi.aliyuncs.com"
|
||||||
|
|
||||||
```
|
```
|
||||||
|
|
||||||
### 3. 运行后端程序
|
修改 nginx 配置文档 `docker/conf/nginx/conf.d/chatgpt-plus.conf`,把后端转发的地址改成当前主机的内网 IP 地址。
|
||||||
|
|
||||||
你可以自己编译或者直接下载我打包好的后端程序运行。
|
|
||||||
|
|
||||||
```shell
|
```shell
|
||||||
# 1. 下载程序,你也可以自己编译
|
# 这里配置后端 API 的转发
|
||||||
wget https://github.com/yangjian102621/chatgpt-plus/releases/download/v3.0.0/chatgpt-v3-amd64-linux
|
location /api/ {
|
||||||
# 2. 添加执行权限
|
proxy_http_version 1.1;
|
||||||
chmod +x chatgpt-v3-amd64-linux
|
proxy_connect_timeout 300s;
|
||||||
# 3. 运行程序,如果配置文档不在当前目录,注意指定配置文档
|
proxy_read_timeout 300s;
|
||||||
./chatgpt-v3-amd64-linux
|
proxy_send_timeout 12s;
|
||||||
```
|
proxy_set_header Host $host;
|
||||||
|
proxy_set_header X-Real-IP $remote_addr;
|
||||||
### 4. 前端部署
|
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
|
||||||
|
proxy_set_header Upgrade $http_upgrade;
|
||||||
前端是 Vue 项目编译好静态资源文件,同样你也可以直接下载我编译好的文件解压。
|
proxy_set_header Connection $connection_upgrade;
|
||||||
|
proxy_pass http://172.28.173.76:6789; # 这里改成后端服务的内网 IP 地址
|
||||||
```shell
|
|
||||||
# 1. 下载程序
|
|
||||||
wget https://github.com/yangjian102621/chatgpt-plus/releases/download/v3.0.0/dist.tar.gz
|
|
||||||
# 2. 解压
|
|
||||||
tar -xf dist.tar.gz
|
|
||||||
```
|
|
||||||
|
|
||||||
### 5. 配置 Nginx 服务
|
|
||||||
|
|
||||||
前端程序需要搭载 Web 服务器才可以运行,这里我们选择 Nginx,先安装:
|
|
||||||
|
|
||||||
```shell
|
|
||||||
sudo apt install nginx -y
|
|
||||||
```
|
|
||||||
|
|
||||||
建立 Nginx 配置文件:
|
|
||||||
|
|
||||||
```conf
|
|
||||||
server {
|
|
||||||
listen 443 ssl;
|
|
||||||
server_name www.chatgpt.com; #替换成你自己的域名
|
|
||||||
|
|
||||||
ssl_certificate xxx.pem; # 替换成自己的 SSL 证书
|
|
||||||
ssl_certificate_key xxx.key;
|
|
||||||
ssl_session_timeout 5m;
|
|
||||||
ssl_ciphers ECDHE-RSA-AES128-GCM-SHA256:ECDHE:ECDH:AES:HIGH:!NULL:!aNULL:!MD5:!ADH:!RC4;
|
|
||||||
ssl_protocols TLSv1 TLSv1.1 TLSv1.2;
|
|
||||||
ssl_prefer_server_ciphers on;
|
|
||||||
|
|
||||||
# 日志地址
|
|
||||||
access_log /var/log/chatgpt/access.log;
|
|
||||||
error_log /var/log/chatgpt/error.log;
|
|
||||||
|
|
||||||
index index.html;
|
|
||||||
root /var/www/chatgpt/dist; # 这里改成前端静态页面的地址
|
|
||||||
|
|
||||||
location / {
|
|
||||||
try_files $uri $uri/ /index.html;
|
|
||||||
|
|
||||||
# 这里配置后端 API 的转发
|
|
||||||
location /api/ {
|
|
||||||
proxy_http_version 1.1;
|
|
||||||
proxy_connect_timeout 300s;
|
|
||||||
proxy_read_timeout 300s;
|
|
||||||
proxy_send_timeout 12s;
|
|
||||||
proxy_set_header Host $host;
|
|
||||||
proxy_set_header X-Real-IP $remote_addr;
|
|
||||||
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
|
|
||||||
proxy_set_header Upgrade $http_upgrade;
|
|
||||||
proxy_set_header Connection $connection_upgrade;
|
|
||||||
proxy_pass http://localhost:5678;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
# 关闭静态资源的日志
|
|
||||||
location ~ .*\.(gif|jpg|jpeg|png|bmp|swf|js|css)$ {
|
|
||||||
access_log off;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
配置好之后重启 Nginx,然后访问后台管理系统 [http://www.chatgpt.com/admin](http://www.chatgpt.com/admin),
|
### 3. 启动应用
|
||||||
输入你前面配置文档中设置的管理员用户名和密码登录。
|
|
||||||
然后进入 `API KEY 管理` 菜单,添加一个 OpenAI 的 API KEY 即可。
|
```shell
|
||||||
|
cd docker
|
||||||
|
docker-compose up -d
|
||||||
|
```
|
||||||
|
|
||||||
|
* 前端访问地址:http://localhost:8080/chat
|
||||||
|
* 后台管理地址:http://localhost:8080/admin
|
||||||
|
* 移动端地址:http://localhost:8080/mobile
|
||||||
|
|
||||||
|
> 注意:你得访问后台管理系统 http://localhost:8080/admin
|
||||||
|
> 输入你前面配置文档中设置的管理员用户名和密码登录。
|
||||||
|
> 然后进入 `API KEY 管理` 菜单,添加一个 OpenAI 的 API KEY 才可以正常开启 AI 对话。
|
||||||
|
|
||||||

|

|
||||||
|
|
||||||
最后登录前端聊天页面 [http://www.chatgpt.com/admin](http://www.chatgpt.com/admin)
|
最后登录前端聊天页面 [http://localhost:8080/chat](http://localhost:8080/chat)
|
||||||
你可以注册新用户,也可以使用系统默认有个账号:`geekmaster/12345678` 登录聊天。
|
你可以注册新用户,也可以使用系统默认有个账号:`geekmaster/12345678` 登录聊天。
|
||||||
|
|
||||||
祝你使用愉快!!!
|
祝你使用愉快!!!
|
||||||
@@ -225,7 +216,7 @@ server {
|
|||||||
3. 运行后端程序:
|
3. 运行后端程序:
|
||||||
|
|
||||||
```shell
|
```shell
|
||||||
cd api/go
|
cd api
|
||||||
# 1. 先下载依赖
|
# 1. 先下载依赖
|
||||||
go mod tidy
|
go mod tidy
|
||||||
# 2. 运行程序
|
# 2. 运行程序
|
||||||
@@ -278,11 +269,11 @@ npm run build
|
|||||||
你可以根据个人需求将项目打包成 windows/linux/darwin 平台项目。
|
你可以根据个人需求将项目打包成 windows/linux/darwin 平台项目。
|
||||||
|
|
||||||
```shell
|
```shell
|
||||||
cd api/go
|
cd api
|
||||||
# for all platforms
|
# for all platforms
|
||||||
make all
|
make clean all
|
||||||
# for linux only
|
# for linux only
|
||||||
make linux
|
make clean linux
|
||||||
```
|
```
|
||||||
|
|
||||||
打包后的可执行文件在 `bin` 目录下。
|
打包后的可执行文件在 `bin` 目录下。
|
||||||
@@ -292,7 +283,7 @@ make linux
|
|||||||
个人的力量始终有限,任何形式的贡献都是欢迎的,包括但不限于贡献代码,优化文档,提交 issue 和 PR 等。
|
个人的力量始终有限,任何形式的贡献都是欢迎的,包括但不限于贡献代码,优化文档,提交 issue 和 PR 等。
|
||||||
**尤其是新版本的开发计划比较大,包括各种语言的后端 API 实现,本人精力有限,希望借助社区的力量来完成这些 API 的开发。**
|
**尤其是新版本的开发计划比较大,包括各种语言的后端 API 实现,本人精力有限,希望借助社区的力量来完成这些 API 的开发。**
|
||||||
|
|
||||||
如果有兴趣的话,也可以加微信进入微信讨论群。
|
如果有兴趣的话,也可以加微信进入微信讨论群(**添加好友时请注明来自Github!!!**)。
|
||||||
|
|
||||||

|

|
||||||
|
|
||||||
|
|||||||
27
Startup.sh
27
Startup.sh
@@ -1,27 +0,0 @@
|
|||||||
# 前端
|
|
||||||
|
|
||||||
if ! command -v node > /dev/null; then
|
|
||||||
printf 'node is not installed.\n'
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
cd web
|
|
||||||
npm install
|
|
||||||
npm run build
|
|
||||||
cd ..
|
|
||||||
|
|
||||||
# 后端
|
|
||||||
if ! command -v go > /dev/null; then
|
|
||||||
printf 'go is not installed.\n'
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
cd src
|
|
||||||
go mod tidy
|
|
||||||
make linux
|
|
||||||
cd ..
|
|
||||||
|
|
||||||
# Docker
|
|
||||||
if ! command -v docker > /dev/null; then
|
|
||||||
printf 'docker is not installed.\n'
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
docker compose up -d
|
|
||||||
2
api/go/.gitignore → api/.gitignore
vendored
2
api/go/.gitignore → api/.gitignore
vendored
@@ -16,3 +16,5 @@ tmp
|
|||||||
bin
|
bin
|
||||||
data
|
data
|
||||||
config.toml
|
config.toml
|
||||||
|
static/upload
|
||||||
|
storage.json
|
||||||
41
api/config.sample.toml
Normal file
41
api/config.sample.toml
Normal file
@@ -0,0 +1,41 @@
|
|||||||
|
Listen = "0.0.0.0:5678"
|
||||||
|
ProxyURL = "http://172.22.11.200:7777"
|
||||||
|
MysqlDns = "root:mysql_pass@tcp(localhost:3306)/chatgpt_plus?charset=utf8mb4&collation=utf8mb4_unicode_ci&parseTime=True&loc=Local"
|
||||||
|
StaticDir = "./static"
|
||||||
|
StaticUrl = "http://localhost:5678/static"
|
||||||
|
AesEncryptKey = "{YOUR_AES_KEY}"
|
||||||
|
StartWechatBot = false
|
||||||
|
EnabledMsgService = false
|
||||||
|
|
||||||
|
[Session]
|
||||||
|
Driver = "cookie"
|
||||||
|
SecretKey = "m0cjm3gsuw9jk73np1ni7r42koilybjcndlycjdmq7za3pbqn7w12fyok5pqh6q5"
|
||||||
|
Name = "CHAT_SESSION_ID"
|
||||||
|
Path = "/"
|
||||||
|
Domain = "localhost"
|
||||||
|
MaxAge = 86400
|
||||||
|
Secure = false
|
||||||
|
HttpOnly = false
|
||||||
|
SameSite = 2
|
||||||
|
|
||||||
|
[Manager]
|
||||||
|
Username = "admin"
|
||||||
|
Password = "admin123"
|
||||||
|
|
||||||
|
[Redis]
|
||||||
|
Host = "localhost"
|
||||||
|
Port = 6379
|
||||||
|
Password = ""
|
||||||
|
|
||||||
|
[ApiConfig]
|
||||||
|
ApiURL = "{URL}"
|
||||||
|
AppId = "{APP_ID}"
|
||||||
|
Token = "{TOKEN}"
|
||||||
|
|
||||||
|
[SmsConfig]
|
||||||
|
AccessKey = "{YOUR_ACCESS_KEY}"
|
||||||
|
AccessSecret = "{YOUR_SECRET_KEY}"
|
||||||
|
Product = "Dysmsapi"
|
||||||
|
Domain = "dysmsapi.aliyuncs.com"
|
||||||
|
|
||||||
|
|
||||||
@@ -2,12 +2,15 @@ package core
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"chatplus/core/types"
|
"chatplus/core/types"
|
||||||
|
"chatplus/service/function"
|
||||||
"chatplus/store/model"
|
"chatplus/store/model"
|
||||||
"chatplus/utils"
|
"chatplus/utils"
|
||||||
"chatplus/utils/resp"
|
"chatplus/utils/resp"
|
||||||
"context"
|
"context"
|
||||||
"github.com/gin-contrib/sessions"
|
"github.com/gin-contrib/sessions"
|
||||||
"github.com/gin-contrib/sessions/cookie"
|
"github.com/gin-contrib/sessions/cookie"
|
||||||
|
"github.com/gin-contrib/sessions/memstore"
|
||||||
|
"github.com/gin-contrib/sessions/redis"
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
"gorm.io/gorm"
|
"gorm.io/gorm"
|
||||||
"io"
|
"io"
|
||||||
@@ -17,62 +20,88 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
type AppServer struct {
|
type AppServer struct {
|
||||||
AppConfig *types.AppConfig
|
Debug bool
|
||||||
|
Config *types.AppConfig
|
||||||
Engine *gin.Engine
|
Engine *gin.Engine
|
||||||
ChatContexts *types.LMap[string, []types.Message] // 聊天上下文 Map [chatId] => []Message
|
ChatContexts *types.LMap[string, []interface{}] // 聊天上下文 Map [chatId] => []Message
|
||||||
ChatConfig *types.ChatConfig // 聊天配置
|
|
||||||
|
ChatConfig *types.ChatConfig // chat config cache
|
||||||
|
SysConfig *types.SystemConfig // system config cache
|
||||||
|
|
||||||
// 保存 Websocket 会话 UserId, 每个 UserId 只能连接一次
|
// 保存 Websocket 会话 UserId, 每个 UserId 只能连接一次
|
||||||
// 防止第三方直接连接 socket 调用 OpenAI API
|
// 防止第三方直接连接 socket 调用 OpenAI API
|
||||||
ChatSession *types.LMap[string, types.ChatSession] //map[sessionId]UserId
|
ChatSession *types.LMap[string, types.ChatSession] //map[sessionId]UserId
|
||||||
ChatClients *types.LMap[string, *types.WsClient] // Websocket 连接集合
|
ChatClients *types.LMap[string, *types.WsClient] // map[sessionId]Websocket 连接集合
|
||||||
ReqCancelFunc *types.LMap[string, context.CancelFunc] // HttpClient 请求取消 handle function
|
ReqCancelFunc *types.LMap[string, context.CancelFunc] // HttpClient 请求取消 handle function
|
||||||
|
Functions map[string]function.Function
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewServer(appConfig *types.AppConfig) *AppServer {
|
func NewServer(
|
||||||
|
appConfig *types.AppConfig,
|
||||||
|
funZaoBao function.FuncZaoBao,
|
||||||
|
funZhiHu function.FuncHeadlines,
|
||||||
|
funWeibo function.FuncWeiboHot) *AppServer {
|
||||||
gin.SetMode(gin.ReleaseMode)
|
gin.SetMode(gin.ReleaseMode)
|
||||||
gin.DefaultWriter = io.Discard
|
gin.DefaultWriter = io.Discard
|
||||||
return &AppServer{
|
return &AppServer{
|
||||||
AppConfig: appConfig,
|
Debug: false,
|
||||||
|
Config: appConfig,
|
||||||
Engine: gin.Default(),
|
Engine: gin.Default(),
|
||||||
ChatContexts: types.NewLMap[string, []types.Message](),
|
ChatContexts: types.NewLMap[string, []interface{}](),
|
||||||
ChatSession: types.NewLMap[string, types.ChatSession](),
|
ChatSession: types.NewLMap[string, types.ChatSession](),
|
||||||
ChatClients: types.NewLMap[string, *types.WsClient](),
|
ChatClients: types.NewLMap[string, *types.WsClient](),
|
||||||
ReqCancelFunc: types.NewLMap[string, context.CancelFunc](),
|
ReqCancelFunc: types.NewLMap[string, context.CancelFunc](),
|
||||||
|
Functions: map[string]function.Function{
|
||||||
|
types.FuncZaoBao: funZaoBao,
|
||||||
|
types.FuncWeibo: funWeibo,
|
||||||
|
types.FuncHeadLine: funZhiHu,
|
||||||
|
},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *AppServer) Init(debug bool) {
|
func (s *AppServer) Init(debug bool) {
|
||||||
if debug { // 调试模式允许跨域请求 API
|
if debug { // 调试模式允许跨域请求 API
|
||||||
|
s.Debug = debug
|
||||||
logger.Info("Enabled debug mode")
|
logger.Info("Enabled debug mode")
|
||||||
s.Engine.Use(corsMiddleware())
|
|
||||||
}
|
}
|
||||||
s.Engine.Use(sessionMiddleware(s.AppConfig))
|
s.Engine.Use(corsMiddleware())
|
||||||
|
s.Engine.Use(sessionMiddleware(s.Config))
|
||||||
s.Engine.Use(authorizeMiddleware(s))
|
s.Engine.Use(authorizeMiddleware(s))
|
||||||
s.Engine.Use(errorHandler)
|
s.Engine.Use(errorHandler)
|
||||||
//gob.Register(model.User{})
|
// 添加静态资源访问
|
||||||
|
s.Engine.Static("/static", s.Config.StaticDir)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *AppServer) Run(db *gorm.DB) error {
|
func (s *AppServer) Run(db *gorm.DB) error {
|
||||||
// load chat config from database
|
// load chat config from database
|
||||||
var config model.Config
|
var chatConfig model.Config
|
||||||
res := db.Where("marker", "chat").First(&config)
|
res := db.Where("marker", "chat").First(&chatConfig)
|
||||||
if res.Error != nil {
|
if res.Error != nil {
|
||||||
return res.Error
|
return res.Error
|
||||||
}
|
}
|
||||||
err := utils.JsonDecode(config.Config, &s.ChatConfig)
|
err := utils.JsonDecode(chatConfig.Config, &s.ChatConfig)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
logger.Infof("http://%s", s.AppConfig.Listen)
|
// load system configs
|
||||||
return s.Engine.Run(s.AppConfig.Listen)
|
var sysConfig model.Config
|
||||||
|
res = db.Where("marker", "system").First(&sysConfig)
|
||||||
|
if res.Error != nil {
|
||||||
|
return res.Error
|
||||||
|
}
|
||||||
|
err = utils.JsonDecode(sysConfig.Config, &s.SysConfig)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
logger.Infof("http://%s", s.Config.Listen)
|
||||||
|
return s.Engine.Run(s.Config.Listen)
|
||||||
}
|
}
|
||||||
|
|
||||||
// 全局异常处理
|
// 全局异常处理
|
||||||
func errorHandler(c *gin.Context) {
|
func errorHandler(c *gin.Context) {
|
||||||
defer func() {
|
defer func() {
|
||||||
if r := recover(); r != nil {
|
if r := recover(); r != nil {
|
||||||
logger.Error("panic: %v\n", r)
|
logger.Errorf("Handler Panic: %v", r)
|
||||||
debug.PrintStack()
|
debug.PrintStack()
|
||||||
c.JSON(http.StatusOK, types.BizVo{Code: types.Failed, Message: types.ErrorMsg})
|
c.JSON(http.StatusOK, types.BizVo{Code: types.Failed, Message: types.ErrorMsg})
|
||||||
c.Abort()
|
c.Abort()
|
||||||
@@ -85,7 +114,28 @@ func errorHandler(c *gin.Context) {
|
|||||||
// 会话处理
|
// 会话处理
|
||||||
func sessionMiddleware(config *types.AppConfig) gin.HandlerFunc {
|
func sessionMiddleware(config *types.AppConfig) gin.HandlerFunc {
|
||||||
// encrypt the cookie
|
// encrypt the cookie
|
||||||
store := cookie.NewStore([]byte(config.Session.SecretKey))
|
var store sessions.Store
|
||||||
|
var err error
|
||||||
|
switch config.Session.Driver {
|
||||||
|
case types.SessionDriverMem:
|
||||||
|
store = memstore.NewStore([]byte(config.Session.SecretKey))
|
||||||
|
break
|
||||||
|
case types.SessionDriverRedis:
|
||||||
|
store, err = redis.NewStore(10, "tcp", config.Redis.Url(), config.Redis.Password, []byte(config.Session.SecretKey))
|
||||||
|
if err != nil {
|
||||||
|
logger.Fatal(err)
|
||||||
|
}
|
||||||
|
break
|
||||||
|
case types.SessionDriverCookie:
|
||||||
|
store = cookie.NewStore([]byte(config.Session.SecretKey))
|
||||||
|
break
|
||||||
|
default:
|
||||||
|
config.Session.Driver = types.SessionDriverCookie
|
||||||
|
store = cookie.NewStore([]byte(config.Session.SecretKey))
|
||||||
|
}
|
||||||
|
|
||||||
|
logger.Info("Session driver: ", config.Session.Driver)
|
||||||
|
|
||||||
store.Options(sessions.Options{
|
store.Options(sessions.Options{
|
||||||
Path: config.Session.Path,
|
Path: config.Session.Path,
|
||||||
Domain: config.Session.Domain,
|
Domain: config.Session.Domain,
|
||||||
@@ -136,6 +186,9 @@ func authorizeMiddleware(s *AppServer) gin.HandlerFunc {
|
|||||||
if c.Request.URL.Path == "/api/user/login" ||
|
if c.Request.URL.Path == "/api/user/login" ||
|
||||||
c.Request.URL.Path == "/api/admin/login" ||
|
c.Request.URL.Path == "/api/admin/login" ||
|
||||||
c.Request.URL.Path == "/api/user/register" ||
|
c.Request.URL.Path == "/api/user/register" ||
|
||||||
|
strings.HasPrefix(c.Request.URL.Path, "/api/sms/") ||
|
||||||
|
strings.HasPrefix(c.Request.URL.Path, "/api/captcha/") ||
|
||||||
|
strings.HasPrefix(c.Request.URL.Path, "/static/") ||
|
||||||
c.Request.URL.Path == "/api/admin/config/get" {
|
c.Request.URL.Path == "/api/admin/config/get" {
|
||||||
c.Next()
|
c.Next()
|
||||||
return
|
return
|
||||||
@@ -154,7 +207,7 @@ func authorizeMiddleware(s *AppServer) gin.HandlerFunc {
|
|||||||
}
|
}
|
||||||
session := sessions.Default(c)
|
session := sessions.Default(c)
|
||||||
var value interface{}
|
var value interface{}
|
||||||
if strings.Contains(c.Request.URL.Path, "/api/admin/") {
|
if strings.Contains(c.Request.URL.Path, "/api/admin/") { // 后台管理 API
|
||||||
value = session.Get(types.SessionAdmin)
|
value = session.Get(types.SessionAdmin)
|
||||||
} else {
|
} else {
|
||||||
value = session.Get(types.SessionUser)
|
value = session.Get(types.SessionUser)
|
||||||
@@ -15,11 +15,15 @@ var logger = logger2.GetLogger()
|
|||||||
|
|
||||||
func NewDefaultConfig() *types.AppConfig {
|
func NewDefaultConfig() *types.AppConfig {
|
||||||
return &types.AppConfig{
|
return &types.AppConfig{
|
||||||
Listen: "0.0.0.0:5678",
|
Listen: "0.0.0.0:5678",
|
||||||
ProxyURL: "",
|
ProxyURL: "",
|
||||||
Manager: types.Manager{Username: "admin", Password: "admin123"},
|
Manager: types.Manager{Username: "admin", Password: "admin123"},
|
||||||
|
StaticDir: "./static",
|
||||||
|
StaticUrl: "http://localhost/5678/static",
|
||||||
|
Redis: types.RedisConfig{Host: "localhost", Port: 6379, Password: ""},
|
||||||
|
AesEncryptKey: utils.RandString(24),
|
||||||
Session: types.Session{
|
Session: types.Session{
|
||||||
|
Driver: types.SessionDriverCookie,
|
||||||
SecretKey: utils.RandString(64),
|
SecretKey: utils.RandString(64),
|
||||||
Name: "CHAT_PLUS_SESSION",
|
Name: "CHAT_PLUS_SESSION",
|
||||||
Domain: "",
|
Domain: "",
|
||||||
@@ -29,6 +33,8 @@ func NewDefaultConfig() *types.AppConfig {
|
|||||||
HttpOnly: false,
|
HttpOnly: false,
|
||||||
SameSite: http.SameSiteLaxMode,
|
SameSite: http.SameSiteLaxMode,
|
||||||
},
|
},
|
||||||
|
ApiConfig: types.ChatPlusApiConfig{},
|
||||||
|
StartWechatBot: false,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -2,11 +2,12 @@ package types
|
|||||||
|
|
||||||
// ApiRequest API 请求实体
|
// ApiRequest API 请求实体
|
||||||
type ApiRequest struct {
|
type ApiRequest struct {
|
||||||
Model string `json:"model"`
|
Model string `json:"model"`
|
||||||
Temperature float32 `json:"temperature"`
|
Temperature float32 `json:"temperature"`
|
||||||
MaxTokens int `json:"max_tokens"`
|
MaxTokens int `json:"max_tokens"`
|
||||||
Stream bool `json:"stream"`
|
Stream bool `json:"stream"`
|
||||||
Messages []Message `json:"messages"`
|
Messages []interface{} `json:"messages"`
|
||||||
|
Functions []Function `json:"functions"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type Message struct {
|
type Message struct {
|
||||||
@@ -20,8 +21,15 @@ type ApiResponse struct {
|
|||||||
|
|
||||||
// ChoiceItem API 响应实体
|
// ChoiceItem API 响应实体
|
||||||
type ChoiceItem struct {
|
type ChoiceItem struct {
|
||||||
Delta Message `json:"delta"`
|
Delta Delta `json:"delta"`
|
||||||
FinishReason string `json:"finish_reason"`
|
FinishReason string `json:"finish_reason"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type Delta struct {
|
||||||
|
Role string `json:"role"`
|
||||||
|
Name string `json:"name"`
|
||||||
|
Content interface{} `json:"content"`
|
||||||
|
FunctionCall FunctionCall `json:"function_call,omitempty"`
|
||||||
}
|
}
|
||||||
|
|
||||||
// ChatSession 聊天会话对象
|
// ChatSession 聊天会话对象
|
||||||
@@ -45,3 +53,10 @@ type ApiError struct {
|
|||||||
|
|
||||||
const PromptMsg = "prompt" // prompt message
|
const PromptMsg = "prompt" // prompt message
|
||||||
const ReplyMsg = "reply" // reply message
|
const ReplyMsg = "reply" // reply message
|
||||||
|
|
||||||
|
var ModelToTokens = map[string]int{
|
||||||
|
"gpt-3.5-turbo": 4096,
|
||||||
|
"gpt-3.5-turbo-16k": 16384,
|
||||||
|
"gpt-4": 8192,
|
||||||
|
"gpt-4-32k": 32768,
|
||||||
|
}
|
||||||
93
api/core/types/config.go
Normal file
93
api/core/types/config.go
Normal file
@@ -0,0 +1,93 @@
|
|||||||
|
package types
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"net/http"
|
||||||
|
)
|
||||||
|
|
||||||
|
type AppConfig struct {
|
||||||
|
Path string `toml:"-"`
|
||||||
|
Listen string
|
||||||
|
Session Session
|
||||||
|
ProxyURL string
|
||||||
|
MysqlDns string // mysql 连接地址
|
||||||
|
Manager Manager // 后台管理员账户信息
|
||||||
|
StaticDir string // 静态资源目录
|
||||||
|
StaticUrl string // 静态资源 URL
|
||||||
|
Redis RedisConfig // redis 连接信息
|
||||||
|
ApiConfig ChatPlusApiConfig // ChatPlus API authorization configs
|
||||||
|
AesEncryptKey string
|
||||||
|
SmsConfig AliYunSmsConfig // AliYun send message service config
|
||||||
|
StartWechatBot bool // 是否启动微信机器人
|
||||||
|
EnabledMsgService bool // 是否启用短信服务
|
||||||
|
}
|
||||||
|
|
||||||
|
type ChatPlusApiConfig struct {
|
||||||
|
ApiURL string
|
||||||
|
AppId string
|
||||||
|
Token string
|
||||||
|
}
|
||||||
|
|
||||||
|
type AliYunSmsConfig struct {
|
||||||
|
AccessKey string
|
||||||
|
AccessSecret string
|
||||||
|
Product string
|
||||||
|
Domain string
|
||||||
|
}
|
||||||
|
|
||||||
|
type RedisConfig struct {
|
||||||
|
Host string
|
||||||
|
Port int
|
||||||
|
Password string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c RedisConfig) Url() string {
|
||||||
|
return fmt.Sprintf("%s:%d", c.Host, c.Port)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Manager 管理员
|
||||||
|
type Manager struct {
|
||||||
|
Username string `json:"username"`
|
||||||
|
Password string `json:"password"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type SessionDriver string
|
||||||
|
|
||||||
|
const (
|
||||||
|
SessionDriverMem = SessionDriver("mem")
|
||||||
|
SessionDriverRedis = SessionDriver("redis")
|
||||||
|
SessionDriverCookie = SessionDriver("cookie")
|
||||||
|
)
|
||||||
|
|
||||||
|
// Session configs struct
|
||||||
|
type Session struct {
|
||||||
|
Driver SessionDriver // session 存储驱动 mem|cookie|redis
|
||||||
|
SecretKey string // session encryption key
|
||||||
|
Name string
|
||||||
|
Path string
|
||||||
|
Domain string
|
||||||
|
MaxAge int
|
||||||
|
Secure bool
|
||||||
|
HttpOnly bool
|
||||||
|
SameSite http.SameSite
|
||||||
|
}
|
||||||
|
|
||||||
|
// ChatConfig 系统默认的聊天配置
|
||||||
|
type ChatConfig struct {
|
||||||
|
ApiURL string `json:"api_url,omitempty"`
|
||||||
|
Model string `json:"model"` // 默认模型
|
||||||
|
Temperature float32 `json:"temperature"`
|
||||||
|
MaxTokens int `json:"max_tokens"`
|
||||||
|
EnableContext bool `json:"enable_context"` // 是否开启聊天上下文
|
||||||
|
EnableHistory bool `json:"enable_history"` // 是否允许保存聊天记录
|
||||||
|
ApiKey string `json:"api_key"`
|
||||||
|
ContextDeep int `json:"context_deep"` // 上下文深度
|
||||||
|
}
|
||||||
|
|
||||||
|
type SystemConfig struct {
|
||||||
|
Title string `json:"title"`
|
||||||
|
AdminTitle string `json:"admin_title"`
|
||||||
|
Models []string `json:"models"`
|
||||||
|
UserInitCalls int `json:"user_init_calls"` // 新用户注册默认总送多少次调用
|
||||||
|
EnabledRegister bool `json:"enabled_register"`
|
||||||
|
}
|
||||||
76
api/core/types/function.go
Normal file
76
api/core/types/function.go
Normal file
@@ -0,0 +1,76 @@
|
|||||||
|
package types
|
||||||
|
|
||||||
|
type FunctionCall struct {
|
||||||
|
Name string `json:"name"`
|
||||||
|
Arguments string `json:"arguments"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type Function struct {
|
||||||
|
Name string `json:"name"`
|
||||||
|
Description string `json:"description"`
|
||||||
|
Parameters Parameters `json:"parameters"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type Parameters struct {
|
||||||
|
Type string `json:"type"`
|
||||||
|
Required []string `json:"required"`
|
||||||
|
Properties map[string]Property `json:"properties"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type Property struct {
|
||||||
|
Type string `json:"type"`
|
||||||
|
Description string `json:"description"`
|
||||||
|
}
|
||||||
|
|
||||||
|
const (
|
||||||
|
FuncZaoBao = "zao_bao" // 每日早报
|
||||||
|
FuncHeadLine = "headline" // 今日头条
|
||||||
|
FuncWeibo = "weibo_hot" // 微博热搜
|
||||||
|
)
|
||||||
|
|
||||||
|
var InnerFunctions = []Function{
|
||||||
|
{
|
||||||
|
Name: FuncZaoBao,
|
||||||
|
Description: "每日早报,获取当天全球的热门新闻事件列表",
|
||||||
|
Parameters: Parameters{
|
||||||
|
|
||||||
|
Type: "object",
|
||||||
|
Properties: map[string]Property{
|
||||||
|
"text": {
|
||||||
|
Type: "string",
|
||||||
|
Description: "",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
Required: []string{},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Name: FuncWeibo,
|
||||||
|
Description: "新浪微博热搜榜,微博当日热搜榜单",
|
||||||
|
Parameters: Parameters{
|
||||||
|
Type: "object",
|
||||||
|
Properties: map[string]Property{
|
||||||
|
"text": {
|
||||||
|
Type: "string",
|
||||||
|
Description: "",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
Required: []string{},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
{
|
||||||
|
Name: FuncHeadLine,
|
||||||
|
Description: "今日头条,给用户推荐当天的头条新闻,周榜热文",
|
||||||
|
Parameters: Parameters{
|
||||||
|
Type: "object",
|
||||||
|
Properties: map[string]Property{
|
||||||
|
"text": {
|
||||||
|
Type: "string",
|
||||||
|
Description: "",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
Required: []string{},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
@@ -9,7 +9,7 @@ type MKey interface {
|
|||||||
string | int
|
string | int
|
||||||
}
|
}
|
||||||
type MValue interface {
|
type MValue interface {
|
||||||
*WsClient | ChatSession | []Message | context.CancelFunc
|
*WsClient | ChatSession | context.CancelFunc | []interface{}
|
||||||
}
|
}
|
||||||
type LMap[K MKey, T MValue] struct {
|
type LMap[K MKey, T MValue] struct {
|
||||||
lock sync.RWMutex
|
lock sync.RWMutex
|
||||||
86
api/go.mod
Normal file
86
api/go.mod
Normal file
@@ -0,0 +1,86 @@
|
|||||||
|
module chatplus
|
||||||
|
|
||||||
|
go 1.19
|
||||||
|
|
||||||
|
require (
|
||||||
|
github.com/BurntSushi/toml v1.1.0
|
||||||
|
github.com/aliyun/alibaba-cloud-sdk-go v1.62.405
|
||||||
|
github.com/eatmoreapple/openwechat v1.2.1
|
||||||
|
github.com/gin-contrib/sessions v0.0.5
|
||||||
|
github.com/gin-gonic/gin v1.9.1
|
||||||
|
github.com/gorilla/websocket v1.5.0
|
||||||
|
github.com/imroc/req/v3 v3.37.2
|
||||||
|
github.com/lionsoul2014/ip2region/binding/golang v0.0.0-20230415042440-a5e3d8259ae0
|
||||||
|
github.com/pkoukk/tiktoken-go v0.1.1-0.20230418101013-cae809389480
|
||||||
|
github.com/skip2/go-qrcode v0.0.0-20200617195104-da1b6568686e
|
||||||
|
github.com/syndtr/goleveldb v1.0.0
|
||||||
|
go.uber.org/zap v1.23.0
|
||||||
|
gopkg.in/natefinch/lumberjack.v2 v2.2.1
|
||||||
|
gorm.io/driver/mysql v1.4.7
|
||||||
|
)
|
||||||
|
|
||||||
|
require (
|
||||||
|
github.com/andybalholm/brotli v1.0.4 // indirect
|
||||||
|
github.com/boj/redistore v0.0.0-20180917114910-cd5dcc76aeff // indirect
|
||||||
|
github.com/bytedance/sonic v1.9.1 // indirect
|
||||||
|
github.com/chenzhuoyu/base64x v0.0.0-20221115062448-fe3a3abad311 // indirect
|
||||||
|
github.com/dlclark/regexp2 v1.8.1 // indirect
|
||||||
|
github.com/gabriel-vasile/mimetype v1.4.2 // indirect
|
||||||
|
github.com/gaukas/godicttls v0.0.3 // indirect
|
||||||
|
github.com/go-sql-driver/mysql v1.7.0 // indirect
|
||||||
|
github.com/go-task/slim-sprig v0.0.0-20230315185526-52ccab3ef572 // indirect
|
||||||
|
github.com/goccy/go-json v0.10.2 // indirect
|
||||||
|
github.com/golang/mock v1.6.0 // indirect
|
||||||
|
github.com/gomodule/redigo v2.0.0+incompatible // indirect
|
||||||
|
github.com/google/pprof v0.0.0-20230602150820-91b7bce49751 // indirect
|
||||||
|
github.com/hashicorp/errwrap v1.1.0 // indirect
|
||||||
|
github.com/hashicorp/go-multierror v1.1.1 // indirect
|
||||||
|
github.com/jinzhu/inflection v1.0.0 // indirect
|
||||||
|
github.com/jinzhu/now v1.1.5 // indirect
|
||||||
|
github.com/jmespath/go-jmespath v0.0.0-20180206201540-c2b33e8439af // indirect
|
||||||
|
github.com/klauspost/compress v1.15.15 // indirect
|
||||||
|
github.com/klauspost/cpuid/v2 v2.2.4 // indirect
|
||||||
|
github.com/onsi/ginkgo/v2 v2.10.0 // indirect
|
||||||
|
github.com/opentracing/opentracing-go v1.2.1-0.20220228012449-10b1cf09e00b // indirect
|
||||||
|
github.com/pelletier/go-toml/v2 v2.0.8 // indirect
|
||||||
|
github.com/quasoft/memstore v0.0.0-20191010062613-2bce066d2b0b // indirect
|
||||||
|
github.com/quic-go/qpack v0.4.0 // indirect
|
||||||
|
github.com/quic-go/qtls-go1-19 v0.3.2 // indirect
|
||||||
|
github.com/quic-go/qtls-go1-20 v0.2.2 // indirect
|
||||||
|
github.com/quic-go/quic-go v0.35.1 // indirect
|
||||||
|
github.com/refraction-networking/utls v1.3.2 // indirect
|
||||||
|
github.com/twitchyliquid64/golang-asm v0.15.1 // indirect
|
||||||
|
go.uber.org/dig v1.16.1 // indirect
|
||||||
|
golang.org/x/arch v0.3.0 // indirect
|
||||||
|
golang.org/x/exp v0.0.0-20230522175609-2e198f4a06a1 // indirect
|
||||||
|
golang.org/x/mod v0.11.0 // indirect
|
||||||
|
golang.org/x/net v0.11.0 // indirect
|
||||||
|
golang.org/x/text v0.10.0 // indirect
|
||||||
|
golang.org/x/tools v0.10.0 // indirect
|
||||||
|
google.golang.org/protobuf v1.30.0 // indirect
|
||||||
|
gopkg.in/ini.v1 v1.66.2 // indirect
|
||||||
|
gopkg.in/yaml.v3 v3.0.1 // indirect
|
||||||
|
)
|
||||||
|
|
||||||
|
require (
|
||||||
|
github.com/gin-contrib/sse v0.1.0 // indirect
|
||||||
|
github.com/go-playground/locales v0.14.1 // indirect
|
||||||
|
github.com/go-playground/universal-translator v0.18.1 // indirect
|
||||||
|
github.com/go-playground/validator/v10 v10.14.0 // indirect
|
||||||
|
github.com/golang/snappy v0.0.1 // indirect
|
||||||
|
github.com/gorilla/context v1.1.1 // indirect
|
||||||
|
github.com/gorilla/securecookie v1.1.1 // indirect
|
||||||
|
github.com/gorilla/sessions v1.2.1 // indirect
|
||||||
|
github.com/json-iterator/go v1.1.12 // indirect
|
||||||
|
github.com/leodido/go-urn v1.2.4 // indirect
|
||||||
|
github.com/mattn/go-isatty v0.0.19 // indirect
|
||||||
|
github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd // indirect
|
||||||
|
github.com/modern-go/reflect2 v1.0.2 // indirect
|
||||||
|
github.com/ugorji/go/codec v1.2.11 // indirect
|
||||||
|
go.uber.org/atomic v1.9.0 // indirect
|
||||||
|
go.uber.org/fx v1.19.3
|
||||||
|
go.uber.org/multierr v1.6.0 // indirect
|
||||||
|
golang.org/x/crypto v0.10.0
|
||||||
|
golang.org/x/sys v0.9.0 // indirect
|
||||||
|
gorm.io/gorm v1.25.1
|
||||||
|
)
|
||||||
249
api/go.sum
Normal file
249
api/go.sum
Normal file
@@ -0,0 +1,249 @@
|
|||||||
|
github.com/BurntSushi/toml v1.1.0 h1:ksErzDEI1khOiGPgpwuI7x2ebx/uXQNw7xJpn9Eq1+I=
|
||||||
|
github.com/BurntSushi/toml v1.1.0/go.mod h1:CxXYINrC8qIiEnFrOxCa7Jy5BFHlXnUU2pbicEuybxQ=
|
||||||
|
github.com/aliyun/alibaba-cloud-sdk-go v1.62.405 h1:cKNFQmeCQFN0WNfjScKoVrGi7vXxTVbkCvCqSrOf+P4=
|
||||||
|
github.com/aliyun/alibaba-cloud-sdk-go v1.62.405/go.mod h1:Api2AkmMgGaSUAhmk76oaFObkoeCPc/bKAqcyplPODs=
|
||||||
|
github.com/andybalholm/brotli v1.0.4 h1:V7DdXeJtZscaqfNuAdSRuRFzuiKlHSC/Zh3zl9qY3JY=
|
||||||
|
github.com/andybalholm/brotli v1.0.4/go.mod h1:fO7iG3H7G2nSZ7m0zPUDn85XEX2GTukHGRSepvi9Eig=
|
||||||
|
github.com/benbjohnson/clock v1.3.0 h1:ip6w0uFQkncKQ979AypyG0ER7mqUSBdKLOgAle/AT8A=
|
||||||
|
github.com/boj/redistore v0.0.0-20180917114910-cd5dcc76aeff h1:RmdPFa+slIr4SCBg4st/l/vZWVe9QJKMXGO60Bxbe04=
|
||||||
|
github.com/boj/redistore v0.0.0-20180917114910-cd5dcc76aeff/go.mod h1:+RTT1BOk5P97fT2CiHkbFQwkK3mjsFAP6zCYV2aXtjw=
|
||||||
|
github.com/bytedance/sonic v1.5.0/go.mod h1:ED5hyg4y6t3/9Ku1R6dU/4KyJ48DZ4jPhfY1O2AihPM=
|
||||||
|
github.com/bytedance/sonic v1.9.1 h1:6iJ6NqdoxCDr6mbY8h18oSO+cShGSMRGCEo7F2h0x8s=
|
||||||
|
github.com/bytedance/sonic v1.9.1/go.mod h1:i736AoUSYt75HyZLoJW9ERYxcy6eaN6h4BZXU064P/U=
|
||||||
|
github.com/chenzhuoyu/base64x v0.0.0-20211019084208-fb5309c8db06/go.mod h1:DH46F32mSOjUmXrMHnKwZdA8wcEefY7UVqBKYGjpdQY=
|
||||||
|
github.com/chenzhuoyu/base64x v0.0.0-20221115062448-fe3a3abad311 h1:qSGYFH7+jGhDF8vLC+iwCD4WpbV1EBDSzWkJODFLams=
|
||||||
|
github.com/chenzhuoyu/base64x v0.0.0-20221115062448-fe3a3abad311/go.mod h1:b583jCggY9gE99b6G5LEC39OIiVsWj+R97kbl5odCEk=
|
||||||
|
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
||||||
|
github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
|
||||||
|
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
||||||
|
github.com/dlclark/regexp2 v1.8.1 h1:6Lcdwya6GjPUNsBct8Lg/yRPwMhABj269AAzdGSiR+0=
|
||||||
|
github.com/dlclark/regexp2 v1.8.1/go.mod h1:DHkYz0B9wPfa6wondMfaivmHpzrQ3v9q8cnmRbL6yW8=
|
||||||
|
github.com/eatmoreapple/openwechat v1.2.1 h1:ez4oqF/Y2NSEX/DbPV8lvj7JlfkYqvieeo4awx5lzfU=
|
||||||
|
github.com/eatmoreapple/openwechat v1.2.1/go.mod h1:61HOzTyvLobGdgWhL68jfGNwTJEv0mhQ1miCXQrvWU8=
|
||||||
|
github.com/fsnotify/fsnotify v1.4.7/go.mod h1:jwhsz4b93w/PPRr/qN1Yymfu8t87LnFCMoQvtojpjFo=
|
||||||
|
github.com/gabriel-vasile/mimetype v1.4.2 h1:w5qFW6JKBz9Y393Y4q372O9A7cUSequkh1Q7OhCmWKU=
|
||||||
|
github.com/gabriel-vasile/mimetype v1.4.2/go.mod h1:zApsH/mKG4w07erKIaJPFiX0Tsq9BFQgN3qGY5GnNgA=
|
||||||
|
github.com/gaukas/godicttls v0.0.3 h1:YNDIf0d9adcxOijiLrEzpfZGAkNwLRzPaG6OjU7EITk=
|
||||||
|
github.com/gaukas/godicttls v0.0.3/go.mod h1:l6EenT4TLWgTdwslVb4sEMOCf7Bv0JAK67deKr9/NCI=
|
||||||
|
github.com/gin-contrib/sessions v0.0.5 h1:CATtfHmLMQrMNpJRgzjWXD7worTh7g7ritsQfmF+0jE=
|
||||||
|
github.com/gin-contrib/sessions v0.0.5/go.mod h1:vYAuaUPqie3WUSsft6HUlCjlwwoJQs97miaG2+7neKY=
|
||||||
|
github.com/gin-contrib/sse v0.1.0 h1:Y/yl/+YNO8GZSjAhjMsSuLt29uWRFHdHYUb5lYOV9qE=
|
||||||
|
github.com/gin-contrib/sse v0.1.0/go.mod h1:RHrZQHXnP2xjPF+u1gW/2HnVO7nvIa9PG3Gm+fLHvGI=
|
||||||
|
github.com/gin-gonic/gin v1.9.1 h1:4idEAncQnU5cB7BeOkPtxjfCSye0AAm1R0RVIqJ+Jmg=
|
||||||
|
github.com/gin-gonic/gin v1.9.1/go.mod h1:hPrL7YrpYKXt5YId3A/Tnip5kqbEAP+KLuI3SUcPTeU=
|
||||||
|
github.com/go-logr/logr v1.2.4 h1:g01GSCwiDw2xSZfjJ2/T9M+S6pFdcNtFYsp+Y43HYDQ=
|
||||||
|
github.com/go-playground/assert/v2 v2.2.0 h1:JvknZsQTYeFEAhQwI4qEt9cyV5ONwRHC+lYKSsYSR8s=
|
||||||
|
github.com/go-playground/locales v0.14.1 h1:EWaQ/wswjilfKLTECiXz7Rh+3BjFhfDFKv/oXslEjJA=
|
||||||
|
github.com/go-playground/locales v0.14.1/go.mod h1:hxrqLVvrK65+Rwrd5Fc6F2O76J/NuW9t0sjnWqG1slY=
|
||||||
|
github.com/go-playground/universal-translator v0.18.1 h1:Bcnm0ZwsGyWbCzImXv+pAJnYK9S473LQFuzCbDbfSFY=
|
||||||
|
github.com/go-playground/universal-translator v0.18.1/go.mod h1:xekY+UJKNuX9WP91TpwSH2VMlDf28Uj24BCp08ZFTUY=
|
||||||
|
github.com/go-playground/validator/v10 v10.14.0 h1:vgvQWe3XCz3gIeFDm/HnTIbj6UGmg/+t63MyGU2n5js=
|
||||||
|
github.com/go-playground/validator/v10 v10.14.0/go.mod h1:9iXMNT7sEkjXb0I+enO7QXmzG6QCsPWY4zveKFVRSyU=
|
||||||
|
github.com/go-sql-driver/mysql v1.7.0 h1:ueSltNNllEqE3qcWBTD0iQd3IpL/6U+mJxLkazJ7YPc=
|
||||||
|
github.com/go-sql-driver/mysql v1.7.0/go.mod h1:OXbVy3sEdcQ2Doequ6Z5BW6fXNQTmx+9S1MCJN5yJMI=
|
||||||
|
github.com/go-task/slim-sprig v0.0.0-20230315185526-52ccab3ef572 h1:tfuBGBXKqDEevZMzYi5KSi8KkcZtzBcTgAUUtapy0OI=
|
||||||
|
github.com/go-task/slim-sprig v0.0.0-20230315185526-52ccab3ef572/go.mod h1:9Pwr4B2jHnOSGXyyzV8ROjYa2ojvAY6HCGYYfMoC3Ls=
|
||||||
|
github.com/goccy/go-json v0.10.2 h1:CrxCmQqYDkv1z7lO7Wbh2HN93uovUHgrECaO5ZrCXAU=
|
||||||
|
github.com/goccy/go-json v0.10.2/go.mod h1:6MelG93GURQebXPDq3khkgXZkazVtN9CRI+MGFi0w8I=
|
||||||
|
github.com/goji/httpauth v0.0.0-20160601135302-2da839ab0f4d/go.mod h1:nnjvkQ9ptGaCkuDUx6wNykzzlUixGxvkme+H/lnzb+A=
|
||||||
|
github.com/golang/mock v1.6.0 h1:ErTB+efbowRARo13NNdxyJji2egdxLGQhRaY+DUumQc=
|
||||||
|
github.com/golang/mock v1.6.0/go.mod h1:p6yTPP+5HYm5mzsMV8JkE6ZKdX+/wYM6Hr+LicevLPs=
|
||||||
|
github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
|
||||||
|
github.com/golang/protobuf v1.5.0/go.mod h1:FsONVRAS9T7sI+LIUmWTfcYkHO4aIWwzhcaSAoJOfIk=
|
||||||
|
github.com/golang/protobuf v1.5.3 h1:KhyjKVUg7Usr/dYsdSqoFveMYd5ko72D+zANwlG1mmg=
|
||||||
|
github.com/golang/snappy v0.0.0-20180518054509-2e65f85255db/go.mod h1:/XxbfmMg8lxefKM7IXC3fBNl/7bRcc72aCRzEWrmP2Q=
|
||||||
|
github.com/golang/snappy v0.0.1 h1:Qgr9rKW7uDUkrbSmQeiDsGa8SjGyCOGtuasMWwvp2P4=
|
||||||
|
github.com/golang/snappy v0.0.1/go.mod h1:/XxbfmMg8lxefKM7IXC3fBNl/7bRcc72aCRzEWrmP2Q=
|
||||||
|
github.com/gomodule/redigo v2.0.0+incompatible h1:K/R+8tc58AaqLkqG2Ol3Qk+DR/TlNuhuh457pBFPtt0=
|
||||||
|
github.com/gomodule/redigo v2.0.0+incompatible/go.mod h1:B4C85qUVwatsJoIUNIfCRsp7qO0iAmpGFZ4EELWSbC4=
|
||||||
|
github.com/google/go-cmp v0.5.5/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
|
||||||
|
github.com/google/go-cmp v0.5.9 h1:O2Tfq5qg4qc4AmwVlvv0oLiVAGB7enBSJ2x2DqQFi38=
|
||||||
|
github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg=
|
||||||
|
github.com/google/pprof v0.0.0-20230602150820-91b7bce49751 h1:hR7/MlvK23p6+lIw9SN1TigNLn9ZnF3W4SYRKq2gAHs=
|
||||||
|
github.com/google/pprof v0.0.0-20230602150820-91b7bce49751/go.mod h1:Jh3hGz2jkYak8qXPD19ryItVnUgpgeqzdkY/D0EaeuA=
|
||||||
|
github.com/gorilla/context v1.1.1 h1:AWwleXJkX/nhcU9bZSnZoi3h/qGYqQAGhq6zZe/aQW8=
|
||||||
|
github.com/gorilla/context v1.1.1/go.mod h1:kBGZzfjB9CEq2AlWe17Uuf7NDRt0dE0s8S51q0aT7Yg=
|
||||||
|
github.com/gorilla/securecookie v1.1.1 h1:miw7JPhV+b/lAHSXz4qd/nN9jRiAFV5FwjeKyCS8BvQ=
|
||||||
|
github.com/gorilla/securecookie v1.1.1/go.mod h1:ra0sb63/xPlUeL+yeDciTfxMRAA+MP+HVt/4epWDjd4=
|
||||||
|
github.com/gorilla/sessions v1.1.1/go.mod h1:8KCfur6+4Mqcc6S0FEfKuN15Vl5MgXW92AE8ovaJD0w=
|
||||||
|
github.com/gorilla/sessions v1.2.1 h1:DHd3rPN5lE3Ts3D8rKkQ8x/0kqfeNmBAaiSi+o7FsgI=
|
||||||
|
github.com/gorilla/sessions v1.2.1/go.mod h1:dk2InVEVJ0sfLlnXv9EAgkf6ecYs/i80K/zI+bUmuGM=
|
||||||
|
github.com/gorilla/websocket v1.5.0 h1:PPwGk2jz7EePpoHN/+ClbZu8SPxiqlu12wZP/3sWmnc=
|
||||||
|
github.com/gorilla/websocket v1.5.0/go.mod h1:YR8l580nyteQvAITg2hZ9XVh4b55+EU/adAjf1fMHhE=
|
||||||
|
github.com/hashicorp/errwrap v1.0.0/go.mod h1:YH+1FKiLXxHSkmPseP+kNlulaMuP3n2brvKWEqk/Jc4=
|
||||||
|
github.com/hashicorp/errwrap v1.1.0 h1:OxrOeh75EUXMY8TBjag2fzXGZ40LB6IKw45YeGUDY2I=
|
||||||
|
github.com/hashicorp/errwrap v1.1.0/go.mod h1:YH+1FKiLXxHSkmPseP+kNlulaMuP3n2brvKWEqk/Jc4=
|
||||||
|
github.com/hashicorp/go-multierror v1.1.1 h1:H5DkEtf6CXdFp0N0Em5UCwQpXMWke8IA0+lD48awMYo=
|
||||||
|
github.com/hashicorp/go-multierror v1.1.1/go.mod h1:iw975J/qwKPdAO1clOe2L8331t/9/fmwbPZ6JB6eMoM=
|
||||||
|
github.com/hpcloud/tail v1.0.0 h1:nfCOvKYfkgYP8hkirhJocXT2+zOD8yUNjXaWfTlyFKI=
|
||||||
|
github.com/hpcloud/tail v1.0.0/go.mod h1:ab1qPbhIpdTxEkNHXyeSf5vhxWSCs/tWer42PpOxQnU=
|
||||||
|
github.com/imroc/req/v3 v3.37.2 h1:vEemuA0cq9zJ6lhe+mSRhsZm951bT0CdiSH47+KTn6I=
|
||||||
|
github.com/imroc/req/v3 v3.37.2/go.mod h1:DECzjVIrj6jcUr5n6e+z0ygmCO93rx4Jy0RjOEe1YCI=
|
||||||
|
github.com/jinzhu/inflection v1.0.0 h1:K317FqzuhWc8YvSVlFMCCUb36O/S9MCKRDI7QkRKD/E=
|
||||||
|
github.com/jinzhu/inflection v1.0.0/go.mod h1:h+uFLlag+Qp1Va5pdKtLDYj+kHp5pxUVkryuEj+Srlc=
|
||||||
|
github.com/jinzhu/now v1.1.4/go.mod h1:d3SSVoowX0Lcu0IBviAWJpolVfI5UJVZZ7cO71lE/z8=
|
||||||
|
github.com/jinzhu/now v1.1.5 h1:/o9tlHleP7gOFmsnYNz3RGnqzefHA47wQpKrrdTIwXQ=
|
||||||
|
github.com/jinzhu/now v1.1.5/go.mod h1:d3SSVoowX0Lcu0IBviAWJpolVfI5UJVZZ7cO71lE/z8=
|
||||||
|
github.com/jmespath/go-jmespath v0.0.0-20180206201540-c2b33e8439af h1:pmfjZENx5imkbgOkpRUYLnmbU7UEFbjtDA2hxJ1ichM=
|
||||||
|
github.com/jmespath/go-jmespath v0.0.0-20180206201540-c2b33e8439af/go.mod h1:Nht3zPeWKUH0NzdCt2Blrr5ys8VGpn0CEB0cQHVjt7k=
|
||||||
|
github.com/json-iterator/go v1.1.5/go.mod h1:+SdeFBvtyEkXs7REEP0seUULqWtbJapLOCVDaaPEHmU=
|
||||||
|
github.com/json-iterator/go v1.1.12 h1:PV8peI4a0ysnczrg+LtxykD8LfKY9ML6u2jnxaEnrnM=
|
||||||
|
github.com/json-iterator/go v1.1.12/go.mod h1:e30LSqwooZae/UwlEbR2852Gd8hjQvJoHmT4TnhNGBo=
|
||||||
|
github.com/klauspost/compress v1.15.15 h1:EF27CXIuDsYJ6mmvtBRlEuB2UVOqHG1tAXgZ7yIO+lw=
|
||||||
|
github.com/klauspost/compress v1.15.15/go.mod h1:ZcK2JAFqKOpnBlxcLsJzYfrS9X1akm9fHZNnD9+Vo/4=
|
||||||
|
github.com/klauspost/cpuid/v2 v2.0.9/go.mod h1:FInQzS24/EEf25PyTYn52gqo7WaD8xa0213Md/qVLRg=
|
||||||
|
github.com/klauspost/cpuid/v2 v2.2.4 h1:acbojRNwl3o09bUq+yDCtZFc1aiwaAAxtcn8YkZXnvk=
|
||||||
|
github.com/klauspost/cpuid/v2 v2.2.4/go.mod h1:RVVoqg1df56z8g3pUjL/3lE5UfnlrJX8tyFgg4nqhuY=
|
||||||
|
github.com/kr/pretty v0.1.0 h1:L/CwN0zerZDmRFUapSPitk6f+Q3+0za1rQkzVuMiMFI=
|
||||||
|
github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo=
|
||||||
|
github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ=
|
||||||
|
github.com/kr/text v0.1.0 h1:45sCR5RtlFHMR4UwH9sdQ5TC8v0qDQCHnXt+kaKSTVE=
|
||||||
|
github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI=
|
||||||
|
github.com/leodido/go-urn v1.2.4 h1:XlAE/cm/ms7TE/VMVoduSpNBoyc2dOxHs5MZSwAN63Q=
|
||||||
|
github.com/leodido/go-urn v1.2.4/go.mod h1:7ZrI8mTSeBSHl/UaRyKQW1qZeMgak41ANeCNaVckg+4=
|
||||||
|
github.com/lionsoul2014/ip2region/binding/golang v0.0.0-20230415042440-a5e3d8259ae0 h1:LgmjED/yQILqmUED4GaXjrINWe7YJh4HM6z2EvEINPs=
|
||||||
|
github.com/lionsoul2014/ip2region/binding/golang v0.0.0-20230415042440-a5e3d8259ae0/go.mod h1:C5LA5UO2ZXJrLaPLYtE1wUJMiyd/nwWaCO5cw/2pSHs=
|
||||||
|
github.com/mattn/go-isatty v0.0.19 h1:JITubQf0MOLdlGRuRq+jtsDlekdYPia9ZFsB8h/APPA=
|
||||||
|
github.com/mattn/go-isatty v0.0.19/go.mod h1:W+V8PltTTMOvKvAeJH7IuucS94S2C6jfK/D7dTCTo3Y=
|
||||||
|
github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
|
||||||
|
github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd h1:TRLaZ9cD/w8PVh93nsPXa1VrQ6jlwL5oN8l14QlcNfg=
|
||||||
|
github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
|
||||||
|
github.com/modern-go/reflect2 v1.0.1/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0=
|
||||||
|
github.com/modern-go/reflect2 v1.0.2 h1:xBagoLtFs94CBntxluKeaWgTMpvLxC4ur3nMaC9Gz0M=
|
||||||
|
github.com/modern-go/reflect2 v1.0.2/go.mod h1:yWuevngMOJpCy52FWWMvUC8ws7m/LJsjYzDa0/r8luk=
|
||||||
|
github.com/onsi/ginkgo v1.6.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE=
|
||||||
|
github.com/onsi/ginkgo v1.7.0 h1:WSHQ+IS43OoUrWtD1/bbclrwK8TTH5hzp+umCiuxHgs=
|
||||||
|
github.com/onsi/ginkgo v1.7.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE=
|
||||||
|
github.com/onsi/ginkgo/v2 v2.10.0 h1:sfUl4qgLdvkChZrWCYndY2EAu9BRIw1YphNAzy1VNWs=
|
||||||
|
github.com/onsi/ginkgo/v2 v2.10.0/go.mod h1:UDQOh5wbQUlMnkLfVaIUMtQ1Vus92oM+P2JX1aulgcE=
|
||||||
|
github.com/onsi/gomega v1.4.3/go.mod h1:ex+gbHU/CVuBBDIJjb2X0qEXbFg53c61hWP/1CpauHY=
|
||||||
|
github.com/onsi/gomega v1.27.7 h1:fVih9JD6ogIiHUN6ePK7HJidyEDpWGVB5mzM7cWNXoU=
|
||||||
|
github.com/opentracing/opentracing-go v1.2.1-0.20220228012449-10b1cf09e00b h1:FfH+VrHHk6Lxt9HdVS0PXzSXFyS2NbZKXv33FYPol0A=
|
||||||
|
github.com/opentracing/opentracing-go v1.2.1-0.20220228012449-10b1cf09e00b/go.mod h1:AC62GU6hc0BrNm+9RK9VSiwa/EUe1bkIeFORAMcHvJU=
|
||||||
|
github.com/pelletier/go-toml/v2 v2.0.8 h1:0ctb6s9mE31h0/lhu+J6OPmVeDxJn+kYnJc2jZR9tGQ=
|
||||||
|
github.com/pelletier/go-toml/v2 v2.0.8/go.mod h1:vuYfssBdrU2XDZ9bYydBu6t+6a6PYNcZljzZR9VXg+4=
|
||||||
|
github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4=
|
||||||
|
github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
|
||||||
|
github.com/pkoukk/tiktoken-go v0.1.1-0.20230418101013-cae809389480 h1:IFhPCcB0/HtnEN+ZoUGDT55YgFCymbFJ15kXqs3nv5w=
|
||||||
|
github.com/pkoukk/tiktoken-go v0.1.1-0.20230418101013-cae809389480/go.mod h1:BijIqAP84FMYC4XbdJgjyMpiSjusU8x0Y0W9K2t0QtU=
|
||||||
|
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
|
||||||
|
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
|
||||||
|
github.com/quasoft/memstore v0.0.0-20191010062613-2bce066d2b0b h1:aUNXCGgukb4gtY99imuIeoh8Vr0GSwAlYxPAhqZrpFc=
|
||||||
|
github.com/quasoft/memstore v0.0.0-20191010062613-2bce066d2b0b/go.mod h1:wTPjTepVu7uJBYgZ0SdWHQlIas582j6cn2jgk4DDdlg=
|
||||||
|
github.com/quic-go/qpack v0.4.0 h1:Cr9BXA1sQS2SmDUWjSofMPNKmvF6IiIfDRmgU0w1ZCo=
|
||||||
|
github.com/quic-go/qpack v0.4.0/go.mod h1:UZVnYIfi5GRk+zI9UMaCPsmZ2xKJP7XBUvVyT1Knj9A=
|
||||||
|
github.com/quic-go/qtls-go1-19 v0.3.2 h1:tFxjCFcTQzK+oMxG6Zcvp4Dq8dx4yD3dDiIiyc86Z5U=
|
||||||
|
github.com/quic-go/qtls-go1-19 v0.3.2/go.mod h1:ySOI96ew8lnoKPtSqx2BlI5wCpUVPT05RMAlajtnyOI=
|
||||||
|
github.com/quic-go/qtls-go1-20 v0.2.2 h1:WLOPx6OY/hxtTxKV1Zrq20FtXtDEkeY00CGQm8GEa3E=
|
||||||
|
github.com/quic-go/qtls-go1-20 v0.2.2/go.mod h1:JKtK6mjbAVcUTN/9jZpvLbGxvdWIKS8uT7EiStoU1SM=
|
||||||
|
github.com/quic-go/quic-go v0.35.1 h1:b0kzj6b/cQAf05cT0CkQubHM31wiA+xH3IBkxP62poo=
|
||||||
|
github.com/quic-go/quic-go v0.35.1/go.mod h1:+4CVgVppm0FNjpG3UcX8Joi/frKOH7/ciD5yGcwOO1g=
|
||||||
|
github.com/refraction-networking/utls v1.3.2 h1:o+AkWB57mkcoW36ET7uJ002CpBWHu0KPxi6vzxvPnv8=
|
||||||
|
github.com/refraction-networking/utls v1.3.2/go.mod h1:fmoaOww2bxzzEpIKOebIsnBvjQpqP7L2vcm/9KUfm/E=
|
||||||
|
github.com/skip2/go-qrcode v0.0.0-20200617195104-da1b6568686e h1:MRM5ITcdelLK2j1vwZ3Je0FKVCfqOLp5zO6trqMLYs0=
|
||||||
|
github.com/skip2/go-qrcode v0.0.0-20200617195104-da1b6568686e/go.mod h1:XV66xRDqSt+GTGFMVlhk3ULuV0y9ZmzeVGR4mloJI3M=
|
||||||
|
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
|
||||||
|
github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw=
|
||||||
|
github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo=
|
||||||
|
github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI=
|
||||||
|
github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
|
||||||
|
github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
|
||||||
|
github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
|
||||||
|
github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU=
|
||||||
|
github.com/stretchr/testify v1.8.1/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4=
|
||||||
|
github.com/stretchr/testify v1.8.2/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4=
|
||||||
|
github.com/stretchr/testify v1.8.3 h1:RP3t2pwF7cMEbC1dqtB6poj3niw/9gnV4Cjg5oW5gtY=
|
||||||
|
github.com/stretchr/testify v1.8.3/go.mod h1:sz/lmYIOXD/1dqDmKjjqLyZ2RngseejIcXlSw2iwfAo=
|
||||||
|
github.com/syndtr/goleveldb v1.0.0 h1:fBdIW9lB4Iz0n9khmH8w27SJ3QEJ7+IgjPEwGSZiFdE=
|
||||||
|
github.com/syndtr/goleveldb v1.0.0/go.mod h1:ZVVdQEZoIme9iO1Ch2Jdy24qqXrMMOU6lpPAyBWyWuQ=
|
||||||
|
github.com/twitchyliquid64/golang-asm v0.15.1 h1:SU5vSMR7hnwNxj24w34ZyCi/FmDZTkS4MhqMhdFk5YI=
|
||||||
|
github.com/twitchyliquid64/golang-asm v0.15.1/go.mod h1:a1lVb/DtPvCB8fslRZhAngC2+aY1QWCk3Cedj/Gdt08=
|
||||||
|
github.com/uber/jaeger-client-go v2.30.0+incompatible h1:D6wyKGCecFaSRUpo8lCVbaOOb6ThwMmTEbhRwtKR97o=
|
||||||
|
github.com/uber/jaeger-client-go v2.30.0+incompatible/go.mod h1:WVhlPFC8FDjOFMMWRy2pZqQJSXxYSwNYOkTr/Z6d3Kk=
|
||||||
|
github.com/uber/jaeger-lib v2.4.1+incompatible h1:td4jdvLcExb4cBISKIpHuGoVXh+dVKhn2Um6rjCsSsg=
|
||||||
|
github.com/uber/jaeger-lib v2.4.1+incompatible/go.mod h1:ComeNDZlWwrWnDv8aPp0Ba6+uUTzImX/AauajbLI56U=
|
||||||
|
github.com/ugorji/go/codec v1.2.11 h1:BMaWp1Bb6fHwEtbplGBGJ498wD+LKlNSl25MjdZY4dU=
|
||||||
|
github.com/ugorji/go/codec v1.2.11/go.mod h1:UNopzCgEMSXjBc6AOMqYvWC1ktqTAfzJZUZgYf6w6lg=
|
||||||
|
github.com/yuin/goldmark v1.3.5/go.mod h1:mwnBkeHKe2W/ZEtQ+71ViKU8L12m81fl3OWwC1Zlc8k=
|
||||||
|
go.uber.org/atomic v1.7.0/go.mod h1:fEN4uk6kAWBTFdckzkM89CLk9XfWZrxpCo0nPH17wJc=
|
||||||
|
go.uber.org/atomic v1.9.0 h1:ECmE8Bn/WFTYwEW/bpKD3M8VtR/zQVbavAoalC1PYyE=
|
||||||
|
go.uber.org/atomic v1.9.0/go.mod h1:fEN4uk6kAWBTFdckzkM89CLk9XfWZrxpCo0nPH17wJc=
|
||||||
|
go.uber.org/dig v1.16.1 h1:+alNIBsl0qfY0j6epRubp/9obgtrObRAc5aD+6jbWY8=
|
||||||
|
go.uber.org/dig v1.16.1/go.mod h1:557JTAUZT5bUK0SvCwikmLPPtdQhfvLYtO5tJgQSbnk=
|
||||||
|
go.uber.org/fx v1.19.3 h1:YqMRE4+2IepTYCMOvXqQpRa+QAVdiSTnsHU4XNWBceA=
|
||||||
|
go.uber.org/fx v1.19.3/go.mod h1:w2HrQg26ql9fLK7hlBiZ6JsRUKV+Lj/atT1KCjT8YhM=
|
||||||
|
go.uber.org/goleak v1.1.11 h1:wy28qYRKZgnJTxGxvye5/wgWr1EKjmUDGYox5mGlRlI=
|
||||||
|
go.uber.org/multierr v1.6.0 h1:y6IPFStTAIT5Ytl7/XYmHvzXQ7S3g/IeZW9hyZ5thw4=
|
||||||
|
go.uber.org/multierr v1.6.0/go.mod h1:cdWPpRnG4AhwMwsgIHip0KRBQjJy5kYEpYjJxpXp9iU=
|
||||||
|
go.uber.org/zap v1.23.0 h1:OjGQ5KQDEUawVHxNwQgPpiypGHOxo2mNZsOqTak4fFY=
|
||||||
|
go.uber.org/zap v1.23.0/go.mod h1:D+nX8jyLsMHMYrln8A0rJjFt/T/9/bGgIhAqxv5URuY=
|
||||||
|
golang.org/x/arch v0.0.0-20210923205945-b76863e36670/go.mod h1:5om86z9Hs0C8fWVUuoMHwpExlXzs5Tkyp9hOrfG7pp8=
|
||||||
|
golang.org/x/arch v0.3.0 h1:02VY4/ZcO/gBOH6PUaoiptASxtXU10jazRCP865E97k=
|
||||||
|
golang.org/x/arch v0.3.0/go.mod h1:5om86z9Hs0C8fWVUuoMHwpExlXzs5Tkyp9hOrfG7pp8=
|
||||||
|
golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
|
||||||
|
golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI=
|
||||||
|
golang.org/x/crypto v0.10.0 h1:LKqV2xt9+kDzSTfOhx4FrkEBcMrAgHSYgzywV9zcGmM=
|
||||||
|
golang.org/x/crypto v0.10.0/go.mod h1:o4eNf7Ede1fv+hwOwZsTHl9EsPFO6q6ZvYR8vYfY45I=
|
||||||
|
golang.org/x/exp v0.0.0-20230522175609-2e198f4a06a1 h1:k/i9J1pBpvlfR+9QsetwPyERsqu1GIbi967PQMq3Ivc=
|
||||||
|
golang.org/x/exp v0.0.0-20230522175609-2e198f4a06a1/go.mod h1:V1LtkGg67GoY2N1AnLN78QLrzxkLyJw7RJb1gzOOz9w=
|
||||||
|
golang.org/x/mod v0.4.2/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA=
|
||||||
|
golang.org/x/mod v0.11.0 h1:bUO06HqtnRcc/7l71XBe4WcqTZ+3AH1J59zWDDwLKgU=
|
||||||
|
golang.org/x/mod v0.11.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs=
|
||||||
|
golang.org/x/net v0.0.0-20180906233101-161cd47e91fd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
||||||
|
golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
|
||||||
|
golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
|
||||||
|
golang.org/x/net v0.0.0-20210405180319-a5a99cb37ef4/go.mod h1:p54w0d4576C0XHj96bSt6lcn1PtDYWL6XObtHCRCNQM=
|
||||||
|
golang.org/x/net v0.11.0 h1:Gi2tvZIJyBtO9SDr1q9h5hEQCp/4L2RQ+ar0qjx2oNU=
|
||||||
|
golang.org/x/net v0.11.0/go.mod h1:2L/ixqYpgIVXmeoSA/4Lu7BzTG4KIyPIryS4IsOd1oQ=
|
||||||
|
golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
||||||
|
golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
||||||
|
golang.org/x/sync v0.0.0-20210220032951-036812b2e83c/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
||||||
|
golang.org/x/sys v0.0.0-20180909124046-d0be0721c37e/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
||||||
|
golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
||||||
|
golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||||
|
golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||||
|
golang.org/x/sys v0.0.0-20210330210617-4fbd30eecc44/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||||
|
golang.org/x/sys v0.0.0-20210510120138-977fb7262007/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||||
|
golang.org/x/sys v0.0.0-20220704084225-05e143d24a9e/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||||
|
golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||||
|
golang.org/x/sys v0.9.0 h1:KS/R3tvhPqvJvwcKfnBHJwwthS11LRhmM5D59eEXa0s=
|
||||||
|
golang.org/x/sys v0.9.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||||
|
golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo=
|
||||||
|
golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
|
||||||
|
golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
|
||||||
|
golang.org/x/text v0.10.0 h1:UpjohKhiEgNc0CSauXmwYftY1+LlaC75SJwh0SgCX58=
|
||||||
|
golang.org/x/text v0.10.0/go.mod h1:TvPlkZtksWOMsz7fbANvkp4WM8x/WCo/om8BMLbz+aE=
|
||||||
|
golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
|
||||||
|
golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
|
||||||
|
golang.org/x/tools v0.1.1/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk=
|
||||||
|
golang.org/x/tools v0.10.0 h1:tvDr/iQoUqNdohiYm0LmmKcBk+q86lb9EprIUFhHHGg=
|
||||||
|
golang.org/x/tools v0.10.0/go.mod h1:UJwyiVBsOA2uwvK/e5OY3GTpDUJriEd+/YlqAwLPmyM=
|
||||||
|
golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
|
||||||
|
golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
|
||||||
|
golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
|
||||||
|
golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
|
||||||
|
google.golang.org/protobuf v1.26.0-rc.1/go.mod h1:jlhhOSvTdKEhbULTjvd4ARK9grFBp09yW+WbY/TyQbw=
|
||||||
|
google.golang.org/protobuf v1.30.0 h1:kPPoIgf3TsEvrm0PFe15JQ+570QVxYzEvvHqChK+cng=
|
||||||
|
google.golang.org/protobuf v1.30.0/go.mod h1:HV8QOd/L58Z+nl8r43ehVNZIU/HEI6OcFqwMG9pJV4I=
|
||||||
|
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
|
||||||
|
gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127 h1:qIbj1fsPNlZgppZ+VLlY7N33q108Sa+fhmuc+sWQYwY=
|
||||||
|
gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
|
||||||
|
gopkg.in/fsnotify.v1 v1.4.7 h1:xOHLXZwVvI9hhs+cLKq5+I5onOuwQLhQwiu63xxlHs4=
|
||||||
|
gopkg.in/fsnotify.v1 v1.4.7/go.mod h1:Tz8NjZHkW78fSQdbUxIjBTcgA1z1m8ZHf0WmKUhAMys=
|
||||||
|
gopkg.in/ini.v1 v1.66.2 h1:XfR1dOYubytKy4Shzc2LHrrGhU0lDCfDGG1yLPmpgsI=
|
||||||
|
gopkg.in/ini.v1 v1.66.2/go.mod h1:pNLf8WUiyNEtQjuu5G5vTm06TEv9tsIgeAvK8hOrP4k=
|
||||||
|
gopkg.in/natefinch/lumberjack.v2 v2.2.1 h1:bBRl1b0OH9s/DuPhuXpNl+VtCaJXFZ5/uEFST95x9zc=
|
||||||
|
gopkg.in/natefinch/lumberjack.v2 v2.2.1/go.mod h1:YD8tP3GAjkrDg1eZH7EGmyESg/lsYskCTPBJVb9jqSc=
|
||||||
|
gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7 h1:uRGJdciOHaEIrze2W8Q3AKkepLTh2hOroT7a+7czfdQ=
|
||||||
|
gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7/go.mod h1:dt/ZhP58zS4L8KSrWDmTeBkI65Dw0HsyUHuEVlX15mw=
|
||||||
|
gopkg.in/yaml.v2 v2.2.1/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
||||||
|
gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
|
||||||
|
gopkg.in/yaml.v3 v3.0.0-20210107192922-496545a6307b/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
|
||||||
|
gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA=
|
||||||
|
gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
|
||||||
|
gorm.io/driver/mysql v1.4.7 h1:rY46lkCspzGHn7+IYsNpSfEv9tA+SU4SkkB+GFX125Y=
|
||||||
|
gorm.io/driver/mysql v1.4.7/go.mod h1:SxzItlnT1cb6e1e4ZRpgJN2VYtcqJgqnHxWr4wsP8oc=
|
||||||
|
gorm.io/gorm v1.23.8/go.mod h1:l2lP/RyAtc1ynaTjFksBde/O8v9oOGIApu2/xRitmZk=
|
||||||
|
gorm.io/gorm v1.25.1 h1:nsSALe5Pr+cM3V1qwwQ7rOkw+6UeLrX5O4v3llhHa64=
|
||||||
|
gorm.io/gorm v1.25.1/go.mod h1:L4uxeKpfBml98NYqVqwAdmV1a2nBtAec/cf3fpucW/k=
|
||||||
|
rsc.io/pdf v0.1.1/go.mod h1:n8OzWcQ6Sp37PL01nO98y4iUCRdTGarVfzxY20ICaU4=
|
||||||
@@ -1,17 +0,0 @@
|
|||||||
Listen = "0.0.0.0:5678"
|
|
||||||
ProxyURL = ["YOUR_PROXY_URL"]
|
|
||||||
MysqlDns = "mysql_user:mysql_pass@tcp(localhost:3306)/chatgpt_plus?charset=utf8&parseTime=True&loc=Local"
|
|
||||||
|
|
||||||
[Session]
|
|
||||||
SecretKey = "azyehq3ivunjhbntz78isj00i4hz2mt9xtddysfucxakadq4qbfrt0b7q3lnvg80"
|
|
||||||
Name = "CHAT_SESSION_ID"
|
|
||||||
Path = "/"
|
|
||||||
Domain = ""
|
|
||||||
MaxAge = 86400
|
|
||||||
Secure = false
|
|
||||||
HttpOnly = false
|
|
||||||
SameSite = 2
|
|
||||||
|
|
||||||
[Manager]
|
|
||||||
Username = "admin"
|
|
||||||
Password = "admin123"
|
|
||||||
@@ -1,52 +0,0 @@
|
|||||||
package types
|
|
||||||
|
|
||||||
import (
|
|
||||||
"net/http"
|
|
||||||
)
|
|
||||||
|
|
||||||
type AppConfig struct {
|
|
||||||
Path string `toml:"-"`
|
|
||||||
Listen string
|
|
||||||
Session Session
|
|
||||||
ProxyURL string
|
|
||||||
MysqlDns string // mysql 连接地址
|
|
||||||
Manager Manager // 后台管理员账户信息
|
|
||||||
}
|
|
||||||
|
|
||||||
// Manager 管理员
|
|
||||||
type Manager struct {
|
|
||||||
Username string `json:"username"`
|
|
||||||
Password string `json:"password"`
|
|
||||||
}
|
|
||||||
|
|
||||||
// Session configs struct
|
|
||||||
type Session struct {
|
|
||||||
SecretKey string // session encryption key
|
|
||||||
Name string
|
|
||||||
Path string
|
|
||||||
Domain string
|
|
||||||
MaxAge int
|
|
||||||
Secure bool
|
|
||||||
HttpOnly bool
|
|
||||||
SameSite http.SameSite
|
|
||||||
}
|
|
||||||
|
|
||||||
// ChatConfig 系统默认的聊天配置
|
|
||||||
type ChatConfig struct {
|
|
||||||
ApiURL string `json:"api_url,omitempty"`
|
|
||||||
Model string `json:"model"` // 默认模型
|
|
||||||
Temperature float32 `json:"temperature"`
|
|
||||||
MaxTokens int `json:"max_tokens"`
|
|
||||||
EnableContext bool `json:"enable_context"` // 是否开启聊天上下文
|
|
||||||
EnableHistory bool `json:"enable_history"` // 是否允许保存聊天记录
|
|
||||||
ApiKey string `json:"api_key"` // OpenAI API key
|
|
||||||
}
|
|
||||||
|
|
||||||
type SystemConfig struct {
|
|
||||||
Title string `json:"title"`
|
|
||||||
AdminTitle string `json:"admin_title"`
|
|
||||||
Models []string `json:"models"`
|
|
||||||
UserInitCalls int `json:"user_init_calls"` // 新用户注册默认总送多少次调用
|
|
||||||
}
|
|
||||||
|
|
||||||
const UserInitCalls = 1000
|
|
||||||
@@ -1,59 +0,0 @@
|
|||||||
module chatplus
|
|
||||||
|
|
||||||
go 1.19
|
|
||||||
|
|
||||||
require (
|
|
||||||
github.com/BurntSushi/toml v1.1.0
|
|
||||||
github.com/gin-contrib/sessions v0.0.5
|
|
||||||
github.com/gin-gonic/gin v1.9.0
|
|
||||||
github.com/gorilla/websocket v1.5.0
|
|
||||||
github.com/lionsoul2014/ip2region/binding/golang v0.0.0-20230415042440-a5e3d8259ae0
|
|
||||||
github.com/pkoukk/tiktoken-go v0.1.1-0.20230418101013-cae809389480
|
|
||||||
github.com/syndtr/goleveldb v1.0.0
|
|
||||||
go.uber.org/zap v1.23.0
|
|
||||||
gorm.io/driver/mysql v1.4.7
|
|
||||||
)
|
|
||||||
|
|
||||||
require (
|
|
||||||
github.com/bytedance/sonic v1.8.0 // indirect
|
|
||||||
github.com/chenzhuoyu/base64x v0.0.0-20221115062448-fe3a3abad311 // indirect
|
|
||||||
github.com/dlclark/regexp2 v1.8.1 // indirect
|
|
||||||
github.com/go-sql-driver/mysql v1.7.0 // indirect
|
|
||||||
github.com/goccy/go-json v0.10.0 // indirect
|
|
||||||
github.com/jinzhu/inflection v1.0.0 // indirect
|
|
||||||
github.com/jinzhu/now v1.1.5 // indirect
|
|
||||||
github.com/klauspost/cpuid/v2 v2.0.9 // indirect
|
|
||||||
github.com/pelletier/go-toml/v2 v2.0.6 // indirect
|
|
||||||
github.com/twitchyliquid64/golang-asm v0.15.1 // indirect
|
|
||||||
go.uber.org/dig v1.16.1 // indirect
|
|
||||||
golang.org/x/arch v0.0.0-20210923205945-b76863e36670 // indirect
|
|
||||||
golang.org/x/net v0.7.0 // indirect
|
|
||||||
golang.org/x/text v0.7.0 // indirect
|
|
||||||
google.golang.org/protobuf v1.28.1 // indirect
|
|
||||||
gopkg.in/yaml.v3 v3.0.1 // indirect
|
|
||||||
)
|
|
||||||
|
|
||||||
require (
|
|
||||||
github.com/gin-contrib/sse v0.1.0 // indirect
|
|
||||||
github.com/go-playground/locales v0.14.1 // indirect
|
|
||||||
github.com/go-playground/universal-translator v0.18.1 // indirect
|
|
||||||
github.com/go-playground/validator/v10 v10.11.2 // indirect
|
|
||||||
github.com/golang/protobuf v1.5.0 // indirect
|
|
||||||
github.com/golang/snappy v0.0.1 // indirect
|
|
||||||
github.com/gorilla/context v1.1.1 // indirect
|
|
||||||
github.com/gorilla/securecookie v1.1.1 // indirect
|
|
||||||
github.com/gorilla/sessions v1.2.1 // indirect
|
|
||||||
github.com/json-iterator/go v1.1.12 // indirect
|
|
||||||
github.com/leodido/go-urn v1.2.1 // indirect
|
|
||||||
github.com/mattn/go-isatty v0.0.17 // indirect
|
|
||||||
github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421 // indirect
|
|
||||||
github.com/modern-go/reflect2 v1.0.2 // indirect
|
|
||||||
github.com/ugorji/go/codec v1.2.9 // indirect
|
|
||||||
go.uber.org/atomic v1.7.0 // indirect
|
|
||||||
go.uber.org/fx v1.19.3
|
|
||||||
go.uber.org/multierr v1.6.0 // indirect
|
|
||||||
golang.org/x/crypto v0.6.0
|
|
||||||
golang.org/x/sys v0.5.0 // indirect
|
|
||||||
gopkg.in/yaml.v2 v2.2.8 // indirect
|
|
||||||
gorm.io/gorm v1.25.1
|
|
||||||
)
|
|
||||||
177
api/go/go.sum
177
api/go/go.sum
@@ -1,177 +0,0 @@
|
|||||||
github.com/BurntSushi/toml v1.1.0 h1:ksErzDEI1khOiGPgpwuI7x2ebx/uXQNw7xJpn9Eq1+I=
|
|
||||||
github.com/BurntSushi/toml v1.1.0/go.mod h1:CxXYINrC8qIiEnFrOxCa7Jy5BFHlXnUU2pbicEuybxQ=
|
|
||||||
github.com/benbjohnson/clock v1.3.0 h1:ip6w0uFQkncKQ979AypyG0ER7mqUSBdKLOgAle/AT8A=
|
|
||||||
github.com/bytedance/sonic v1.5.0/go.mod h1:ED5hyg4y6t3/9Ku1R6dU/4KyJ48DZ4jPhfY1O2AihPM=
|
|
||||||
github.com/bytedance/sonic v1.8.0 h1:ea0Xadu+sHlu7x5O3gKhRpQ1IKiMrSiHttPF0ybECuA=
|
|
||||||
github.com/bytedance/sonic v1.8.0/go.mod h1:i736AoUSYt75HyZLoJW9ERYxcy6eaN6h4BZXU064P/U=
|
|
||||||
github.com/chenzhuoyu/base64x v0.0.0-20211019084208-fb5309c8db06/go.mod h1:DH46F32mSOjUmXrMHnKwZdA8wcEefY7UVqBKYGjpdQY=
|
|
||||||
github.com/chenzhuoyu/base64x v0.0.0-20221115062448-fe3a3abad311 h1:qSGYFH7+jGhDF8vLC+iwCD4WpbV1EBDSzWkJODFLams=
|
|
||||||
github.com/chenzhuoyu/base64x v0.0.0-20221115062448-fe3a3abad311/go.mod h1:b583jCggY9gE99b6G5LEC39OIiVsWj+R97kbl5odCEk=
|
|
||||||
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
|
||||||
github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
|
|
||||||
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
|
||||||
github.com/dlclark/regexp2 v1.8.1 h1:6Lcdwya6GjPUNsBct8Lg/yRPwMhABj269AAzdGSiR+0=
|
|
||||||
github.com/dlclark/regexp2 v1.8.1/go.mod h1:DHkYz0B9wPfa6wondMfaivmHpzrQ3v9q8cnmRbL6yW8=
|
|
||||||
github.com/fsnotify/fsnotify v1.4.7/go.mod h1:jwhsz4b93w/PPRr/qN1Yymfu8t87LnFCMoQvtojpjFo=
|
|
||||||
github.com/gin-contrib/sessions v0.0.5 h1:CATtfHmLMQrMNpJRgzjWXD7worTh7g7ritsQfmF+0jE=
|
|
||||||
github.com/gin-contrib/sessions v0.0.5/go.mod h1:vYAuaUPqie3WUSsft6HUlCjlwwoJQs97miaG2+7neKY=
|
|
||||||
github.com/gin-contrib/sse v0.1.0 h1:Y/yl/+YNO8GZSjAhjMsSuLt29uWRFHdHYUb5lYOV9qE=
|
|
||||||
github.com/gin-contrib/sse v0.1.0/go.mod h1:RHrZQHXnP2xjPF+u1gW/2HnVO7nvIa9PG3Gm+fLHvGI=
|
|
||||||
github.com/gin-gonic/gin v1.7.7 h1:3DoBmSbJbZAWqXJC3SLjAPfutPJJRN1U5pALB7EeTTs=
|
|
||||||
github.com/gin-gonic/gin v1.7.7/go.mod h1:axIBovoeJpVj8S3BwE0uPMTeReE4+AfFtqpqaZ1qq1U=
|
|
||||||
github.com/gin-gonic/gin v1.9.0 h1:OjyFBKICoexlu99ctXNR2gg+c5pKrKMuyjgARg9qeY8=
|
|
||||||
github.com/gin-gonic/gin v1.9.0/go.mod h1:W1Me9+hsUSyj3CePGrd1/QrKJMSJ1Tu/0hFEH89961k=
|
|
||||||
github.com/go-playground/assert/v2 v2.0.1 h1:MsBgLAaY856+nPRTKrp3/OZK38U/wa0CcBYNjji3q3A=
|
|
||||||
github.com/go-playground/assert/v2 v2.0.1/go.mod h1:VDjEfimB/XKnb+ZQfWdccd7VUvScMdVu0Titje2rxJ4=
|
|
||||||
github.com/go-playground/locales v0.13.0 h1:HyWk6mgj5qFqCT5fjGBuRArbVDfE4hi8+e8ceBS/t7Q=
|
|
||||||
github.com/go-playground/locales v0.13.0/go.mod h1:taPMhCMXrRLJO55olJkUXHZBHCxTMfnGwq/HNwmWNS8=
|
|
||||||
github.com/go-playground/locales v0.14.1 h1:EWaQ/wswjilfKLTECiXz7Rh+3BjFhfDFKv/oXslEjJA=
|
|
||||||
github.com/go-playground/locales v0.14.1/go.mod h1:hxrqLVvrK65+Rwrd5Fc6F2O76J/NuW9t0sjnWqG1slY=
|
|
||||||
github.com/go-playground/universal-translator v0.17.0 h1:icxd5fm+REJzpZx7ZfpaD876Lmtgy7VtROAbHHXk8no=
|
|
||||||
github.com/go-playground/universal-translator v0.17.0/go.mod h1:UkSxE5sNxxRwHyU+Scu5vgOQjsIJAF8j9muTVoKLVtA=
|
|
||||||
github.com/go-playground/universal-translator v0.18.1 h1:Bcnm0ZwsGyWbCzImXv+pAJnYK9S473LQFuzCbDbfSFY=
|
|
||||||
github.com/go-playground/universal-translator v0.18.1/go.mod h1:xekY+UJKNuX9WP91TpwSH2VMlDf28Uj24BCp08ZFTUY=
|
|
||||||
github.com/go-playground/validator/v10 v10.4.1 h1:pH2c5ADXtd66mxoE0Zm9SUhxE20r7aM3F26W0hOn+GE=
|
|
||||||
github.com/go-playground/validator/v10 v10.4.1/go.mod h1:nlOn6nFhuKACm19sB/8EGNn9GlaMV7XkbRSipzJ0Ii4=
|
|
||||||
github.com/go-playground/validator/v10 v10.11.2 h1:q3SHpufmypg+erIExEKUmsgmhDTyhcJ38oeKGACXohU=
|
|
||||||
github.com/go-playground/validator/v10 v10.11.2/go.mod h1:NieE624vt4SCTJtD87arVLvdmjPAeV8BQlHtMnw9D7s=
|
|
||||||
github.com/go-sql-driver/mysql v1.7.0 h1:ueSltNNllEqE3qcWBTD0iQd3IpL/6U+mJxLkazJ7YPc=
|
|
||||||
github.com/go-sql-driver/mysql v1.7.0/go.mod h1:OXbVy3sEdcQ2Doequ6Z5BW6fXNQTmx+9S1MCJN5yJMI=
|
|
||||||
github.com/goccy/go-json v0.10.0 h1:mXKd9Qw4NuzShiRlOXKews24ufknHO7gx30lsDyokKA=
|
|
||||||
github.com/goccy/go-json v0.10.0/go.mod h1:6MelG93GURQebXPDq3khkgXZkazVtN9CRI+MGFi0w8I=
|
|
||||||
github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
|
|
||||||
github.com/golang/protobuf v1.3.3 h1:gyjaxf+svBWX08ZjK86iN9geUJF0H6gp2IRKX6Nf6/I=
|
|
||||||
github.com/golang/protobuf v1.3.3/go.mod h1:vzj43D7+SQXF/4pzW/hwtAqwc6iTitCiVSaWz5lYuqw=
|
|
||||||
github.com/golang/protobuf v1.5.0/go.mod h1:FsONVRAS9T7sI+LIUmWTfcYkHO4aIWwzhcaSAoJOfIk=
|
|
||||||
github.com/golang/snappy v0.0.0-20180518054509-2e65f85255db/go.mod h1:/XxbfmMg8lxefKM7IXC3fBNl/7bRcc72aCRzEWrmP2Q=
|
|
||||||
github.com/golang/snappy v0.0.1 h1:Qgr9rKW7uDUkrbSmQeiDsGa8SjGyCOGtuasMWwvp2P4=
|
|
||||||
github.com/golang/snappy v0.0.1/go.mod h1:/XxbfmMg8lxefKM7IXC3fBNl/7bRcc72aCRzEWrmP2Q=
|
|
||||||
github.com/google/go-cmp v0.5.5/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
|
|
||||||
github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg=
|
|
||||||
github.com/gorilla/context v1.1.1 h1:AWwleXJkX/nhcU9bZSnZoi3h/qGYqQAGhq6zZe/aQW8=
|
|
||||||
github.com/gorilla/context v1.1.1/go.mod h1:kBGZzfjB9CEq2AlWe17Uuf7NDRt0dE0s8S51q0aT7Yg=
|
|
||||||
github.com/gorilla/securecookie v1.1.1 h1:miw7JPhV+b/lAHSXz4qd/nN9jRiAFV5FwjeKyCS8BvQ=
|
|
||||||
github.com/gorilla/securecookie v1.1.1/go.mod h1:ra0sb63/xPlUeL+yeDciTfxMRAA+MP+HVt/4epWDjd4=
|
|
||||||
github.com/gorilla/sessions v1.2.1 h1:DHd3rPN5lE3Ts3D8rKkQ8x/0kqfeNmBAaiSi+o7FsgI=
|
|
||||||
github.com/gorilla/sessions v1.2.1/go.mod h1:dk2InVEVJ0sfLlnXv9EAgkf6ecYs/i80K/zI+bUmuGM=
|
|
||||||
github.com/gorilla/websocket v1.5.0 h1:PPwGk2jz7EePpoHN/+ClbZu8SPxiqlu12wZP/3sWmnc=
|
|
||||||
github.com/gorilla/websocket v1.5.0/go.mod h1:YR8l580nyteQvAITg2hZ9XVh4b55+EU/adAjf1fMHhE=
|
|
||||||
github.com/hpcloud/tail v1.0.0 h1:nfCOvKYfkgYP8hkirhJocXT2+zOD8yUNjXaWfTlyFKI=
|
|
||||||
github.com/hpcloud/tail v1.0.0/go.mod h1:ab1qPbhIpdTxEkNHXyeSf5vhxWSCs/tWer42PpOxQnU=
|
|
||||||
github.com/jinzhu/inflection v1.0.0 h1:K317FqzuhWc8YvSVlFMCCUb36O/S9MCKRDI7QkRKD/E=
|
|
||||||
github.com/jinzhu/inflection v1.0.0/go.mod h1:h+uFLlag+Qp1Va5pdKtLDYj+kHp5pxUVkryuEj+Srlc=
|
|
||||||
github.com/jinzhu/now v1.1.4/go.mod h1:d3SSVoowX0Lcu0IBviAWJpolVfI5UJVZZ7cO71lE/z8=
|
|
||||||
github.com/jinzhu/now v1.1.5 h1:/o9tlHleP7gOFmsnYNz3RGnqzefHA47wQpKrrdTIwXQ=
|
|
||||||
github.com/jinzhu/now v1.1.5/go.mod h1:d3SSVoowX0Lcu0IBviAWJpolVfI5UJVZZ7cO71lE/z8=
|
|
||||||
github.com/json-iterator/go v1.1.9 h1:9yzud/Ht36ygwatGx56VwCZtlI/2AD15T1X2sjSuGns=
|
|
||||||
github.com/json-iterator/go v1.1.9/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4=
|
|
||||||
github.com/json-iterator/go v1.1.12 h1:PV8peI4a0ysnczrg+LtxykD8LfKY9ML6u2jnxaEnrnM=
|
|
||||||
github.com/json-iterator/go v1.1.12/go.mod h1:e30LSqwooZae/UwlEbR2852Gd8hjQvJoHmT4TnhNGBo=
|
|
||||||
github.com/klauspost/cpuid/v2 v2.0.9 h1:lgaqFMSdTdQYdZ04uHyN2d/eKdOMyi2YLSvlQIBFYa4=
|
|
||||||
github.com/klauspost/cpuid/v2 v2.0.9/go.mod h1:FInQzS24/EEf25PyTYn52gqo7WaD8xa0213Md/qVLRg=
|
|
||||||
github.com/leodido/go-urn v1.2.0 h1:hpXL4XnriNwQ/ABnpepYM/1vCLWNDfUNts8dX3xTG6Y=
|
|
||||||
github.com/leodido/go-urn v1.2.0/go.mod h1:+8+nEpDfqqsY+g338gtMEUOtuK+4dEMhiQEgxpxOKII=
|
|
||||||
github.com/leodido/go-urn v1.2.1 h1:BqpAaACuzVSgi/VLzGZIobT2z4v53pjosyNd9Yv6n/w=
|
|
||||||
github.com/leodido/go-urn v1.2.1/go.mod h1:zt4jvISO2HfUBqxjfIshjdMTYS56ZS/qv49ictyFfxY=
|
|
||||||
github.com/lionsoul2014/ip2region/binding/golang v0.0.0-20230415042440-a5e3d8259ae0 h1:LgmjED/yQILqmUED4GaXjrINWe7YJh4HM6z2EvEINPs=
|
|
||||||
github.com/lionsoul2014/ip2region/binding/golang v0.0.0-20230415042440-a5e3d8259ae0/go.mod h1:C5LA5UO2ZXJrLaPLYtE1wUJMiyd/nwWaCO5cw/2pSHs=
|
|
||||||
github.com/mattn/go-isatty v0.0.12 h1:wuysRhFDzyxgEmMf5xjvJ2M9dZoWAXNNr5LSBS7uHXY=
|
|
||||||
github.com/mattn/go-isatty v0.0.12/go.mod h1:cbi8OIDigv2wuxKPP5vlRcQ1OAZbq2CE4Kysco4FUpU=
|
|
||||||
github.com/mattn/go-isatty v0.0.17 h1:BTarxUcIeDqL27Mc+vyvdWYSL28zpIhv3RoTdsLMPng=
|
|
||||||
github.com/mattn/go-isatty v0.0.17/go.mod h1:kYGgaQfpe5nmfYZH+SKPsOc2e4SrIfOl2e/yFXSvRLM=
|
|
||||||
github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421 h1:ZqeYNhU3OHLH3mGKHDcjJRFFRrJa6eAM5H+CtDdOsPc=
|
|
||||||
github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
|
|
||||||
github.com/modern-go/reflect2 v0.0.0-20180701023420-4b7aa43c6742 h1:Esafd1046DLDQ0W1YjYsBW+p8U2u7vzgW2SQVmlNazg=
|
|
||||||
github.com/modern-go/reflect2 v0.0.0-20180701023420-4b7aa43c6742/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0=
|
|
||||||
github.com/modern-go/reflect2 v1.0.2 h1:xBagoLtFs94CBntxluKeaWgTMpvLxC4ur3nMaC9Gz0M=
|
|
||||||
github.com/modern-go/reflect2 v1.0.2/go.mod h1:yWuevngMOJpCy52FWWMvUC8ws7m/LJsjYzDa0/r8luk=
|
|
||||||
github.com/onsi/ginkgo v1.6.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE=
|
|
||||||
github.com/onsi/ginkgo v1.7.0 h1:WSHQ+IS43OoUrWtD1/bbclrwK8TTH5hzp+umCiuxHgs=
|
|
||||||
github.com/onsi/ginkgo v1.7.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE=
|
|
||||||
github.com/onsi/gomega v1.4.3 h1:RE1xgDvH7imwFD45h+u2SgIfERHlS2yNG4DObb5BSKU=
|
|
||||||
github.com/onsi/gomega v1.4.3/go.mod h1:ex+gbHU/CVuBBDIJjb2X0qEXbFg53c61hWP/1CpauHY=
|
|
||||||
github.com/pelletier/go-toml/v2 v2.0.6 h1:nrzqCb7j9cDFj2coyLNLaZuJTLjWjlaz6nvTvIwycIU=
|
|
||||||
github.com/pelletier/go-toml/v2 v2.0.6/go.mod h1:eumQOmlWiOPt5WriQQqoM5y18pDHwha2N+QD+EUNTek=
|
|
||||||
github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4=
|
|
||||||
github.com/pkoukk/tiktoken-go v0.1.1-0.20230418101013-cae809389480 h1:IFhPCcB0/HtnEN+ZoUGDT55YgFCymbFJ15kXqs3nv5w=
|
|
||||||
github.com/pkoukk/tiktoken-go v0.1.1-0.20230418101013-cae809389480/go.mod h1:BijIqAP84FMYC4XbdJgjyMpiSjusU8x0Y0W9K2t0QtU=
|
|
||||||
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
|
|
||||||
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
|
|
||||||
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
|
|
||||||
github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw=
|
|
||||||
github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo=
|
|
||||||
github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI=
|
|
||||||
github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4=
|
|
||||||
github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
|
|
||||||
github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
|
|
||||||
github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
|
|
||||||
github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU=
|
|
||||||
github.com/stretchr/testify v1.8.1/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4=
|
|
||||||
github.com/stretchr/testify v1.8.2 h1:+h33VjcLVPDHtOdpUCuF+7gSuG3yGIftsP1YvFihtJ8=
|
|
||||||
github.com/syndtr/goleveldb v1.0.0 h1:fBdIW9lB4Iz0n9khmH8w27SJ3QEJ7+IgjPEwGSZiFdE=
|
|
||||||
github.com/syndtr/goleveldb v1.0.0/go.mod h1:ZVVdQEZoIme9iO1Ch2Jdy24qqXrMMOU6lpPAyBWyWuQ=
|
|
||||||
github.com/twitchyliquid64/golang-asm v0.15.1 h1:SU5vSMR7hnwNxj24w34ZyCi/FmDZTkS4MhqMhdFk5YI=
|
|
||||||
github.com/twitchyliquid64/golang-asm v0.15.1/go.mod h1:a1lVb/DtPvCB8fslRZhAngC2+aY1QWCk3Cedj/Gdt08=
|
|
||||||
github.com/ugorji/go v1.1.7/go.mod h1:kZn38zHttfInRq0xu/PH0az30d+z6vm202qpg1oXVMw=
|
|
||||||
github.com/ugorji/go/codec v1.1.7 h1:2SvQaVZ1ouYrrKKwoSk2pzd4A9evlKJb9oTL+OaLUSs=
|
|
||||||
github.com/ugorji/go/codec v1.1.7/go.mod h1:Ax+UKWsSmolVDwsd+7N3ZtXu+yMGCf907BLYF3GoBXY=
|
|
||||||
github.com/ugorji/go/codec v1.2.9 h1:rmenucSohSTiyL09Y+l2OCk+FrMxGMzho2+tjr5ticU=
|
|
||||||
github.com/ugorji/go/codec v1.2.9/go.mod h1:UNopzCgEMSXjBc6AOMqYvWC1ktqTAfzJZUZgYf6w6lg=
|
|
||||||
go.uber.org/atomic v1.7.0 h1:ADUqmZGgLDDfbSL9ZmPxKTybcoEYHgpYfELNoN+7hsw=
|
|
||||||
go.uber.org/atomic v1.7.0/go.mod h1:fEN4uk6kAWBTFdckzkM89CLk9XfWZrxpCo0nPH17wJc=
|
|
||||||
go.uber.org/dig v1.16.1 h1:+alNIBsl0qfY0j6epRubp/9obgtrObRAc5aD+6jbWY8=
|
|
||||||
go.uber.org/dig v1.16.1/go.mod h1:557JTAUZT5bUK0SvCwikmLPPtdQhfvLYtO5tJgQSbnk=
|
|
||||||
go.uber.org/fx v1.19.3 h1:YqMRE4+2IepTYCMOvXqQpRa+QAVdiSTnsHU4XNWBceA=
|
|
||||||
go.uber.org/fx v1.19.3/go.mod h1:w2HrQg26ql9fLK7hlBiZ6JsRUKV+Lj/atT1KCjT8YhM=
|
|
||||||
go.uber.org/goleak v1.1.11 h1:wy28qYRKZgnJTxGxvye5/wgWr1EKjmUDGYox5mGlRlI=
|
|
||||||
go.uber.org/multierr v1.6.0 h1:y6IPFStTAIT5Ytl7/XYmHvzXQ7S3g/IeZW9hyZ5thw4=
|
|
||||||
go.uber.org/multierr v1.6.0/go.mod h1:cdWPpRnG4AhwMwsgIHip0KRBQjJy5kYEpYjJxpXp9iU=
|
|
||||||
go.uber.org/zap v1.23.0 h1:OjGQ5KQDEUawVHxNwQgPpiypGHOxo2mNZsOqTak4fFY=
|
|
||||||
go.uber.org/zap v1.23.0/go.mod h1:D+nX8jyLsMHMYrln8A0rJjFt/T/9/bGgIhAqxv5URuY=
|
|
||||||
golang.org/x/arch v0.0.0-20210923205945-b76863e36670 h1:18EFjUmQOcUvxNYSkA6jO9VAiXCnxFY6NyDX0bHDmkU=
|
|
||||||
golang.org/x/arch v0.0.0-20210923205945-b76863e36670/go.mod h1:5om86z9Hs0C8fWVUuoMHwpExlXzs5Tkyp9hOrfG7pp8=
|
|
||||||
golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
|
|
||||||
golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto=
|
|
||||||
golang.org/x/crypto v0.6.0 h1:qfktjS5LUO+fFKeJXZ+ikTRijMmljikvG68fpMMruSc=
|
|
||||||
golang.org/x/crypto v0.6.0/go.mod h1:OFC/31mSvZgRz0V1QTNCzfAI1aIRzbiufJtkMIlEp58=
|
|
||||||
golang.org/x/net v0.0.0-20180906233101-161cd47e91fd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
|
||||||
golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
|
|
||||||
golang.org/x/net v0.6.0 h1:L4ZwwTvKW9gr0ZMS1yrHD9GZhIuVjOBBnaKH+SPQK0Q=
|
|
||||||
golang.org/x/net v0.7.0 h1:rJrUqqhjsgNp7KqAIc25s9pZnjU7TUcSY7HcVZjdn1g=
|
|
||||||
golang.org/x/net v0.7.0/go.mod h1:2Tu9+aMcznHK/AK1HMvgo6xiTLG5rD5rZLDS+rp2Bjs=
|
|
||||||
golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
|
||||||
golang.org/x/sys v0.0.0-20180909124046-d0be0721c37e/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
|
||||||
golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
|
||||||
golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
|
||||||
golang.org/x/sys v0.0.0-20200116001909-b77594299b42/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
|
||||||
golang.org/x/sys v0.0.0-20220811171246-fbc7d0a398ab/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
|
||||||
golang.org/x/sys v0.5.0 h1:MUK/U/4lj1t1oPg0HfuXDN/Z1wv31ZJ/YcPiGccS4DU=
|
|
||||||
golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
|
||||||
golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
|
|
||||||
golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk=
|
|
||||||
golang.org/x/text v0.7.0 h1:4BRB4x83lYWy72KwLD/qYDuTu7q9PjSagHvijDw7cLo=
|
|
||||||
golang.org/x/text v0.7.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8=
|
|
||||||
golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
|
|
||||||
golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
|
|
||||||
google.golang.org/protobuf v1.26.0-rc.1/go.mod h1:jlhhOSvTdKEhbULTjvd4ARK9grFBp09yW+WbY/TyQbw=
|
|
||||||
google.golang.org/protobuf v1.28.1 h1:d0NfwRgPtno5B1Wa6L2DAG+KivqkdutMf1UhdNx175w=
|
|
||||||
google.golang.org/protobuf v1.28.1/go.mod h1:HV8QOd/L58Z+nl8r43ehVNZIU/HEI6OcFqwMG9pJV4I=
|
|
||||||
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405 h1:yhCVgyC4o1eVCa2tZl7eS0r+SDo693bJlVdllGtEeKM=
|
|
||||||
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
|
|
||||||
gopkg.in/fsnotify.v1 v1.4.7 h1:xOHLXZwVvI9hhs+cLKq5+I5onOuwQLhQwiu63xxlHs4=
|
|
||||||
gopkg.in/fsnotify.v1 v1.4.7/go.mod h1:Tz8NjZHkW78fSQdbUxIjBTcgA1z1m8ZHf0WmKUhAMys=
|
|
||||||
gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7 h1:uRGJdciOHaEIrze2W8Q3AKkepLTh2hOroT7a+7czfdQ=
|
|
||||||
gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7/go.mod h1:dt/ZhP58zS4L8KSrWDmTeBkI65Dw0HsyUHuEVlX15mw=
|
|
||||||
gopkg.in/yaml.v2 v2.2.1/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
|
||||||
gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
|
||||||
gopkg.in/yaml.v2 v2.2.8 h1:obN1ZagJSUGI0Ek/LBmuj4SNLPfIny3KsKFopxRdj10=
|
|
||||||
gopkg.in/yaml.v2 v2.2.8/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
|
||||||
gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
|
|
||||||
gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA=
|
|
||||||
gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
|
|
||||||
gorm.io/driver/mysql v1.4.7 h1:rY46lkCspzGHn7+IYsNpSfEv9tA+SU4SkkB+GFX125Y=
|
|
||||||
gorm.io/driver/mysql v1.4.7/go.mod h1:SxzItlnT1cb6e1e4ZRpgJN2VYtcqJgqnHxWr4wsP8oc=
|
|
||||||
gorm.io/gorm v1.23.8/go.mod h1:l2lP/RyAtc1ynaTjFksBde/O8v9oOGIApu2/xRitmZk=
|
|
||||||
gorm.io/gorm v1.25.1 h1:nsSALe5Pr+cM3V1qwwQ7rOkw+6UeLrX5O4v3llhHa64=
|
|
||||||
gorm.io/gorm v1.25.1/go.mod h1:L4uxeKpfBml98NYqVqwAdmV1a2nBtAec/cf3fpucW/k=
|
|
||||||
rsc.io/pdf v0.1.1/go.mod h1:n8OzWcQ6Sp37PL01nO98y4iUCRdTGarVfzxY20ICaU4=
|
|
||||||
@@ -1,453 +0,0 @@
|
|||||||
package handler
|
|
||||||
|
|
||||||
import (
|
|
||||||
"bufio"
|
|
||||||
"bytes"
|
|
||||||
"chatplus/core"
|
|
||||||
"chatplus/core/types"
|
|
||||||
"chatplus/store/model"
|
|
||||||
"chatplus/store/vo"
|
|
||||||
"chatplus/utils"
|
|
||||||
"chatplus/utils/resp"
|
|
||||||
"context"
|
|
||||||
"encoding/json"
|
|
||||||
"errors"
|
|
||||||
"fmt"
|
|
||||||
"io"
|
|
||||||
"net/http"
|
|
||||||
"net/url"
|
|
||||||
"strings"
|
|
||||||
"time"
|
|
||||||
"unicode/utf8"
|
|
||||||
|
|
||||||
"github.com/gin-gonic/gin"
|
|
||||||
"github.com/gorilla/websocket"
|
|
||||||
"gorm.io/gorm"
|
|
||||||
)
|
|
||||||
|
|
||||||
const ErrorMsg = "抱歉,AI 助手开小差了,请马上联系管理员去盘它。"
|
|
||||||
|
|
||||||
type ChatHandler struct {
|
|
||||||
BaseHandler
|
|
||||||
db *gorm.DB
|
|
||||||
}
|
|
||||||
|
|
||||||
func NewChatHandler(app *core.AppServer, db *gorm.DB) *ChatHandler {
|
|
||||||
handler := ChatHandler{db: db}
|
|
||||||
handler.App = app
|
|
||||||
return &handler
|
|
||||||
}
|
|
||||||
|
|
||||||
// ChatHandle 处理聊天 WebSocket 请求
|
|
||||||
func (h *ChatHandler) ChatHandle(c *gin.Context) {
|
|
||||||
ws, err := (&websocket.Upgrader{CheckOrigin: func(r *http.Request) bool { return true }}).Upgrade(c.Writer, c.Request, nil)
|
|
||||||
if err != nil {
|
|
||||||
logger.Error(err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
sessionId := c.Query("session_id")
|
|
||||||
roleId := h.GetInt(c, "role_id", 0)
|
|
||||||
chatId := c.Query("chat_id")
|
|
||||||
chatModel := c.Query("model")
|
|
||||||
|
|
||||||
session := h.App.ChatSession.Get(sessionId)
|
|
||||||
if session.SessionId == "" {
|
|
||||||
logger.Info("用户未登录")
|
|
||||||
c.Abort()
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// use old chat data override the chat model and role ID
|
|
||||||
var chat model.ChatItem
|
|
||||||
res := h.db.Where("chat_id=?", chatId).First(&chat)
|
|
||||||
if res.Error == nil {
|
|
||||||
chatModel = chat.Model
|
|
||||||
roleId = int(chat.RoleId)
|
|
||||||
}
|
|
||||||
|
|
||||||
session.ChatId = chatId
|
|
||||||
session.Model = chatModel
|
|
||||||
logger.Infof("New websocket connected, IP: %s, Username: %s", c.Request.RemoteAddr, session.Username)
|
|
||||||
client := types.NewWsClient(ws)
|
|
||||||
var chatRole model.ChatRole
|
|
||||||
res = h.db.First(&chatRole, roleId)
|
|
||||||
if res.Error != nil || !chatRole.Enable {
|
|
||||||
replyMessage(client, "当前聊天角色不存在或者未启用!!!")
|
|
||||||
c.Abort()
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// 保存会话连接
|
|
||||||
h.App.ChatClients.Put(sessionId, client)
|
|
||||||
go func() {
|
|
||||||
for {
|
|
||||||
_, message, err := client.Receive()
|
|
||||||
if err != nil {
|
|
||||||
logger.Error(err)
|
|
||||||
client.Close()
|
|
||||||
h.App.ChatClients.Delete(sessionId)
|
|
||||||
h.App.ReqCancelFunc.Delete(sessionId)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
logger.Info("Receive a message: ", string(message))
|
|
||||||
//replyMessage(client, "这是一条测试消息!")
|
|
||||||
ctx, cancel := context.WithCancel(context.Background())
|
|
||||||
h.App.ReqCancelFunc.Put(sessionId, cancel)
|
|
||||||
// 回复消息
|
|
||||||
err = h.sendMessage(ctx, session, chatRole, string(message), client)
|
|
||||||
if err != nil {
|
|
||||||
logger.Error(err)
|
|
||||||
} else {
|
|
||||||
replyChunkMessage(client, types.WsMessage{Type: types.WsEnd})
|
|
||||||
logger.Info("回答完毕: " + string(message))
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
}()
|
|
||||||
}
|
|
||||||
|
|
||||||
// 将消息发送给 ChatGPT 并获取结果,通过 WebSocket 推送到客户端
|
|
||||||
func (h *ChatHandler) sendMessage(ctx context.Context, session types.ChatSession, role model.ChatRole, prompt string, ws types.Client) error {
|
|
||||||
promptCreatedAt := time.Now() // 记录提问时间
|
|
||||||
|
|
||||||
var user model.User
|
|
||||||
res := h.db.Model(&model.User{}).First(&user, session.UserId)
|
|
||||||
if res.Error != nil {
|
|
||||||
replyMessage(ws, "非法用户,请联系管理员!")
|
|
||||||
return res.Error
|
|
||||||
}
|
|
||||||
var userVo vo.User
|
|
||||||
err := utils.CopyObject(user, &userVo)
|
|
||||||
userVo.Id = user.Id
|
|
||||||
if err != nil {
|
|
||||||
return errors.New("User 对象转换失败," + err.Error())
|
|
||||||
}
|
|
||||||
|
|
||||||
if userVo.Status == false {
|
|
||||||
replyMessage(ws, "您的账号已经被禁用,如果疑问,请联系管理员!")
|
|
||||||
replyMessage(ws, "")
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
if userVo.Calls <= 0 {
|
|
||||||
replyMessage(ws, "您的对话次数已经用尽,请联系管理员充值!")
|
|
||||||
replyMessage(ws, "")
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
if userVo.ExpiredTime > 0 && userVo.ExpiredTime <= time.Now().Unix() {
|
|
||||||
replyMessage(ws, "您的账号已经过期,请联系管理员!")
|
|
||||||
replyMessage(ws, "")
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
var req = types.ApiRequest{
|
|
||||||
Model: session.Model,
|
|
||||||
Temperature: userVo.ChatConfig.Temperature,
|
|
||||||
MaxTokens: userVo.ChatConfig.MaxTokens,
|
|
||||||
Stream: true,
|
|
||||||
}
|
|
||||||
|
|
||||||
// 加载聊天上下文
|
|
||||||
var chatCtx []types.Message
|
|
||||||
if userVo.ChatConfig.EnableContext {
|
|
||||||
if h.App.ChatContexts.Has(session.ChatId) {
|
|
||||||
chatCtx = h.App.ChatContexts.Get(session.ChatId)
|
|
||||||
} else {
|
|
||||||
// 加载角色信息
|
|
||||||
var messages []types.Message
|
|
||||||
err := utils.JsonDecode(role.Context, &messages)
|
|
||||||
if err == nil {
|
|
||||||
chatCtx = messages
|
|
||||||
}
|
|
||||||
// TODO: 这里默认加载最近 4 条聊天记录作为上下文,后期应该做成可配置的
|
|
||||||
var historyMessages []model.HistoryMessage
|
|
||||||
res := h.db.Where("chat_id = ?", session.ChatId).Limit(4).Order("created_at desc").Find(&historyMessages)
|
|
||||||
if res.Error == nil {
|
|
||||||
for _, msg := range historyMessages {
|
|
||||||
ms := types.Message{Role: "user", Content: msg.Content}
|
|
||||||
if msg.Type == types.ReplyMsg {
|
|
||||||
ms.Role = "assistant"
|
|
||||||
}
|
|
||||||
chatCtx = append(chatCtx, ms)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
logger.Info("聊天上下文:", chatCtx)
|
|
||||||
}
|
|
||||||
req.Messages = append(chatCtx, types.Message{
|
|
||||||
Role: "user",
|
|
||||||
Content: prompt,
|
|
||||||
})
|
|
||||||
var apiKey string
|
|
||||||
response, err := h.doRequest(ctx, userVo, &apiKey, req)
|
|
||||||
if err != nil {
|
|
||||||
if strings.Contains(err.Error(), "context canceled") {
|
|
||||||
logger.Info("用户取消了请求:", prompt)
|
|
||||||
return nil
|
|
||||||
} else {
|
|
||||||
logger.Error(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
replyMessage(ws, ErrorMsg)
|
|
||||||
replyMessage(ws, "")
|
|
||||||
return err
|
|
||||||
} else {
|
|
||||||
defer response.Body.Close()
|
|
||||||
}
|
|
||||||
|
|
||||||
contentType := response.Header.Get("Content-Type")
|
|
||||||
if strings.Contains(contentType, "text/event-stream") {
|
|
||||||
replyCreatedAt := time.Now()
|
|
||||||
// 循环读取 Chunk 消息
|
|
||||||
var message = types.Message{}
|
|
||||||
var contents = make([]string, 0)
|
|
||||||
var responseBody = types.ApiResponse{}
|
|
||||||
reader := bufio.NewReader(response.Body)
|
|
||||||
for {
|
|
||||||
line, err := reader.ReadString('\n')
|
|
||||||
if err != nil {
|
|
||||||
if strings.Contains(err.Error(), "context canceled") {
|
|
||||||
logger.Info("用户取消了请求:", prompt)
|
|
||||||
} else {
|
|
||||||
logger.Error(err)
|
|
||||||
}
|
|
||||||
break
|
|
||||||
}
|
|
||||||
if !strings.Contains(line, "data:") {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
err = json.Unmarshal([]byte(line[6:]), &responseBody)
|
|
||||||
if err != nil { // 数据解析出错
|
|
||||||
logger.Error(err, line)
|
|
||||||
replyMessage(ws, ErrorMsg)
|
|
||||||
replyMessage(ws, "")
|
|
||||||
break
|
|
||||||
}
|
|
||||||
|
|
||||||
// 初始化 role
|
|
||||||
if responseBody.Choices[0].Delta.Role != "" && message.Role == "" {
|
|
||||||
message.Role = responseBody.Choices[0].Delta.Role
|
|
||||||
replyChunkMessage(ws, types.WsMessage{Type: types.WsStart})
|
|
||||||
continue
|
|
||||||
} else if responseBody.Choices[0].FinishReason != "" {
|
|
||||||
break // 输出完成或者输出中断了
|
|
||||||
} else {
|
|
||||||
content := responseBody.Choices[0].Delta.Content
|
|
||||||
contents = append(contents, content)
|
|
||||||
replyChunkMessage(ws, types.WsMessage{
|
|
||||||
Type: types.WsMiddle,
|
|
||||||
Content: responseBody.Choices[0].Delta.Content,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
} // end for
|
|
||||||
|
|
||||||
// 消息发送成功
|
|
||||||
if len(contents) > 0 {
|
|
||||||
// 更新用户的对话次数
|
|
||||||
res := h.db.Model(&user).UpdateColumn("calls", gorm.Expr("calls - ?", 1))
|
|
||||||
if res.Error != nil {
|
|
||||||
return res.Error
|
|
||||||
}
|
|
||||||
|
|
||||||
if message.Role == "" {
|
|
||||||
message.Role = "assistant"
|
|
||||||
}
|
|
||||||
message.Content = strings.Join(contents, "")
|
|
||||||
useMsg := types.Message{Role: "user", Content: prompt}
|
|
||||||
|
|
||||||
// 更新上下文消息
|
|
||||||
if userVo.ChatConfig.EnableContext {
|
|
||||||
chatCtx = append(chatCtx, useMsg) // 提问消息
|
|
||||||
chatCtx = append(chatCtx, message) // 回复消息
|
|
||||||
h.App.ChatContexts.Put(session.ChatId, chatCtx)
|
|
||||||
}
|
|
||||||
|
|
||||||
// 追加聊天记录
|
|
||||||
if userVo.ChatConfig.EnableHistory {
|
|
||||||
// for prompt
|
|
||||||
token, err := utils.CalcTokens(prompt, req.Model)
|
|
||||||
if err != nil {
|
|
||||||
logger.Error(err)
|
|
||||||
}
|
|
||||||
historyUserMsg := model.HistoryMessage{
|
|
||||||
UserId: userVo.Id,
|
|
||||||
ChatId: session.ChatId,
|
|
||||||
RoleId: role.Id,
|
|
||||||
Type: types.PromptMsg,
|
|
||||||
Icon: user.Avatar,
|
|
||||||
Content: prompt,
|
|
||||||
Tokens: token,
|
|
||||||
}
|
|
||||||
historyUserMsg.CreatedAt = promptCreatedAt
|
|
||||||
historyUserMsg.UpdatedAt = promptCreatedAt
|
|
||||||
res := h.db.Save(&historyUserMsg)
|
|
||||||
if res.Error != nil {
|
|
||||||
logger.Error("failed to save prompt history message: ", res.Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
// for reply
|
|
||||||
token, err = utils.CalcTokens(message.Content, req.Model)
|
|
||||||
if err != nil {
|
|
||||||
logger.Error(err)
|
|
||||||
}
|
|
||||||
historyReplyMsg := model.HistoryMessage{
|
|
||||||
UserId: userVo.Id,
|
|
||||||
ChatId: session.ChatId,
|
|
||||||
RoleId: role.Id,
|
|
||||||
Type: types.ReplyMsg,
|
|
||||||
Icon: role.Icon,
|
|
||||||
Content: message.Content,
|
|
||||||
Tokens: token,
|
|
||||||
}
|
|
||||||
historyReplyMsg.CreatedAt = replyCreatedAt
|
|
||||||
historyReplyMsg.UpdatedAt = replyCreatedAt
|
|
||||||
res = h.db.Create(&historyReplyMsg)
|
|
||||||
if res.Error != nil {
|
|
||||||
logger.Error("failed to save reply history message: ", res.Error)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// 保存当前会话
|
|
||||||
var chatItem model.ChatItem
|
|
||||||
res = h.db.Where("chat_id = ?", session.ChatId).First(&chatItem)
|
|
||||||
if res.Error != nil {
|
|
||||||
chatItem.ChatId = session.ChatId
|
|
||||||
chatItem.UserId = session.UserId
|
|
||||||
chatItem.RoleId = role.Id
|
|
||||||
chatItem.Model = session.Model
|
|
||||||
if utf8.RuneCountInString(prompt) > 30 {
|
|
||||||
chatItem.Title = string([]rune(prompt)[:30]) + "..."
|
|
||||||
} else {
|
|
||||||
chatItem.Title = prompt
|
|
||||||
}
|
|
||||||
h.db.Create(&chatItem)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
body, err := io.ReadAll(response.Body)
|
|
||||||
if err != nil {
|
|
||||||
return fmt.Errorf("error with reading response: %v", err)
|
|
||||||
}
|
|
||||||
var res types.ApiError
|
|
||||||
err = json.Unmarshal(body, &res)
|
|
||||||
if err != nil {
|
|
||||||
return fmt.Errorf("error with decode response: %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// OpenAI API 调用异常处理
|
|
||||||
// TODO: 是否考虑重发消息?
|
|
||||||
if strings.Contains(res.Error.Message, "This key is associated with a deactivated account") {
|
|
||||||
replyMessage(ws, "请求 OpenAI API 失败:API KEY 所关联的账户被禁用。")
|
|
||||||
// 移除当前 API key
|
|
||||||
h.db.Where("value = ?", apiKey).Delete(&model.ApiKey{})
|
|
||||||
} else if strings.Contains(res.Error.Message, "You exceeded your current quota") {
|
|
||||||
replyMessage(ws, "请求 OpenAI API 失败:API KEY 触发并发限制,请稍后再试。")
|
|
||||||
} else if strings.Contains(res.Error.Message, "This model's maximum context length") {
|
|
||||||
replyMessage(ws, "当前会话上下文长度超出限制,已为您删减会话上下文!")
|
|
||||||
// 只保留最近的三条记录
|
|
||||||
chatContext := h.App.ChatContexts.Get(session.ChatId)
|
|
||||||
if len(chatContext) > 3 {
|
|
||||||
chatContext = chatContext[len(chatContext)-3:]
|
|
||||||
}
|
|
||||||
h.App.ChatContexts.Put(session.ChatId, chatContext)
|
|
||||||
return h.sendMessage(ctx, session, role, prompt, ws)
|
|
||||||
} else {
|
|
||||||
replyMessage(ws, "请求 OpenAI API 失败:"+res.Error.Message)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// 发送请求到 OpenAI 服务器
|
|
||||||
// useOwnApiKey: 是否使用了用户自己的 API KEY
|
|
||||||
func (h *ChatHandler) doRequest(ctx context.Context, user vo.User, apiKey *string, req types.ApiRequest) (*http.Response, error) {
|
|
||||||
var client *http.Client
|
|
||||||
requestBody, err := json.Marshal(req)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
// 创建 HttpClient 请求对象
|
|
||||||
request, err := http.NewRequest(http.MethodPost, h.App.ChatConfig.ApiURL, bytes.NewBuffer(requestBody))
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
request = request.WithContext(ctx)
|
|
||||||
request.Header.Add("Content-Type", "application/json")
|
|
||||||
|
|
||||||
proxyURL := h.App.AppConfig.ProxyURL
|
|
||||||
if proxyURL == "" {
|
|
||||||
client = &http.Client{}
|
|
||||||
} else { // 使用代理
|
|
||||||
uri := url.URL{}
|
|
||||||
proxy, _ := uri.Parse(proxyURL)
|
|
||||||
client = &http.Client{
|
|
||||||
Transport: &http.Transport{
|
|
||||||
Proxy: http.ProxyURL(proxy),
|
|
||||||
},
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// 查询当前用户是否导入了自己的 API KEY
|
|
||||||
if user.ChatConfig.ApiKey != "" {
|
|
||||||
logger.Info("使用用户自己的 API KEY: ", user.ChatConfig.ApiKey)
|
|
||||||
*apiKey = user.ChatConfig.ApiKey
|
|
||||||
} else { // 获取系统的 API KEY
|
|
||||||
var key model.ApiKey
|
|
||||||
res := h.db.Where("user_id = ?", 0).Order("last_used_at ASC").First(&key)
|
|
||||||
if res.Error != nil {
|
|
||||||
return nil, errors.New("no available key, please import key")
|
|
||||||
}
|
|
||||||
*apiKey = key.Value
|
|
||||||
// 更新 API KEY 的最后使用时间
|
|
||||||
h.db.Model(&key).UpdateColumn("last_used_at", time.Now().Unix())
|
|
||||||
}
|
|
||||||
|
|
||||||
logger.Infof("Sending OpenAI request, KEY: %s, PROXY: %s, Model: %s", *apiKey, proxyURL, req.Model)
|
|
||||||
request.Header.Set("Authorization", fmt.Sprintf("Bearer %s", *apiKey))
|
|
||||||
return client.Do(request)
|
|
||||||
}
|
|
||||||
|
|
||||||
// 回复客户片段端消息
|
|
||||||
func replyChunkMessage(client types.Client, message types.WsMessage) {
|
|
||||||
msg, err := json.Marshal(message)
|
|
||||||
if err != nil {
|
|
||||||
logger.Errorf("Error for decoding json data: %v", err.Error())
|
|
||||||
return
|
|
||||||
}
|
|
||||||
err = client.(*types.WsClient).Send(msg)
|
|
||||||
if err != nil {
|
|
||||||
logger.Errorf("Error for reply message: %v", err.Error())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// 回复客户端一条完整的消息
|
|
||||||
func replyMessage(ws types.Client, message string) {
|
|
||||||
replyChunkMessage(ws, types.WsMessage{Type: types.WsStart})
|
|
||||||
replyChunkMessage(ws, types.WsMessage{Type: types.WsMiddle, Content: message})
|
|
||||||
replyChunkMessage(ws, types.WsMessage{Type: types.WsEnd})
|
|
||||||
}
|
|
||||||
|
|
||||||
// Tokens 统计 token 数量
|
|
||||||
func (h *ChatHandler) Tokens(c *gin.Context) {
|
|
||||||
text := c.Query("text")
|
|
||||||
md := c.Query("model")
|
|
||||||
tokens, err := utils.CalcTokens(text, md)
|
|
||||||
if err != nil {
|
|
||||||
resp.ERROR(c, err.Error())
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
resp.SUCCESS(c, tokens)
|
|
||||||
}
|
|
||||||
|
|
||||||
// StopGenerate 停止生成
|
|
||||||
func (h *ChatHandler) StopGenerate(c *gin.Context) {
|
|
||||||
sessionId := c.Query("session_id")
|
|
||||||
if h.App.ReqCancelFunc.Has(sessionId) {
|
|
||||||
h.App.ReqCancelFunc.Get(sessionId)()
|
|
||||||
h.App.ReqCancelFunc.Delete(sessionId)
|
|
||||||
}
|
|
||||||
resp.SUCCESS(c, types.OkMsg)
|
|
||||||
}
|
|
||||||
@@ -1,26 +0,0 @@
|
|||||||
package logger
|
|
||||||
|
|
||||||
import (
|
|
||||||
"go.uber.org/zap"
|
|
||||||
)
|
|
||||||
|
|
||||||
var logger *zap.SugaredLogger
|
|
||||||
|
|
||||||
func GetLogger() *zap.SugaredLogger {
|
|
||||||
if logger != nil {
|
|
||||||
return logger
|
|
||||||
}
|
|
||||||
|
|
||||||
logLevel := zap.NewAtomicLevel()
|
|
||||||
logLevel.SetLevel(zap.InfoLevel)
|
|
||||||
log, _ := zap.Config{
|
|
||||||
Level: logLevel,
|
|
||||||
Development: false,
|
|
||||||
Encoding: "console",
|
|
||||||
EncoderConfig: zap.NewDevelopmentEncoderConfig(),
|
|
||||||
OutputPaths: []string{"stderr"},
|
|
||||||
ErrorOutputPaths: []string{"stderr"},
|
|
||||||
}.Build()
|
|
||||||
logger = log.Sugar()
|
|
||||||
return logger
|
|
||||||
}
|
|
||||||
@@ -1,16 +0,0 @@
|
|||||||
package model
|
|
||||||
|
|
||||||
type HistoryMessage struct {
|
|
||||||
BaseModel
|
|
||||||
ChatId string // 会话 ID
|
|
||||||
UserId uint // 用户 ID
|
|
||||||
RoleId uint // 角色 ID
|
|
||||||
Type string
|
|
||||||
Icon string
|
|
||||||
Tokens int
|
|
||||||
Content string
|
|
||||||
}
|
|
||||||
|
|
||||||
func (HistoryMessage) TableName() string {
|
|
||||||
return "chatgpt_chat_history"
|
|
||||||
}
|
|
||||||
@@ -1,16 +0,0 @@
|
|||||||
package vo
|
|
||||||
|
|
||||||
type HistoryMessage struct {
|
|
||||||
BaseVo
|
|
||||||
ChatId string `json:"chat_id"`
|
|
||||||
UserId uint `json:"user_id"`
|
|
||||||
RoleId uint `json:"role_id"`
|
|
||||||
Type string `json:"type"`
|
|
||||||
Icon string `json:"icon"`
|
|
||||||
Tokens int `json:"tokens"`
|
|
||||||
Content string `json:"content"`
|
|
||||||
}
|
|
||||||
|
|
||||||
func (HistoryMessage) TableName() string {
|
|
||||||
return "chatgpt_chat_history"
|
|
||||||
}
|
|
||||||
@@ -8,6 +8,7 @@ import (
|
|||||||
"chatplus/store/model"
|
"chatplus/store/model"
|
||||||
"chatplus/utils"
|
"chatplus/utils"
|
||||||
"chatplus/utils/resp"
|
"chatplus/utils/resp"
|
||||||
|
"strings"
|
||||||
|
|
||||||
"github.com/gin-contrib/sessions"
|
"github.com/gin-contrib/sessions"
|
||||||
|
|
||||||
@@ -35,7 +36,7 @@ func (h *ManagerHandler) Login(c *gin.Context) {
|
|||||||
resp.ERROR(c, types.InvalidArgs)
|
resp.ERROR(c, types.InvalidArgs)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
manager := h.App.AppConfig.Manager
|
manager := h.App.Config.Manager
|
||||||
if data.Username == manager.Username && data.Password == manager.Password {
|
if data.Username == manager.Username && data.Password == manager.Password {
|
||||||
err := utils.SetLoginAdmin(c, manager)
|
err := utils.SetLoginAdmin(c, manager)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -88,7 +89,7 @@ func (h *ManagerHandler) Migrate(c *gin.Context) {
|
|||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
for k, _ := range m {
|
for k := range m {
|
||||||
roleKeys = append(roleKeys, k)
|
roleKeys = append(roleKeys, k)
|
||||||
}
|
}
|
||||||
u.ChatRoles = utils.JsonEncode(roleKeys)
|
u.ChatRoles = utils.JsonEncode(roleKeys)
|
||||||
@@ -97,12 +98,38 @@ func (h *ManagerHandler) Migrate(c *gin.Context) {
|
|||||||
}
|
}
|
||||||
break
|
break
|
||||||
case "role":
|
case "role":
|
||||||
// TestRole 修改角色图片,改成绝对路径
|
// 修改角色图片,改成绝对路径
|
||||||
var roles []model.ChatRole
|
var roles []model.ChatRole
|
||||||
h.db.Find(&roles)
|
h.db.Find(&roles)
|
||||||
for _, r := range roles {
|
for _, r := range roles {
|
||||||
r.Icon = "/" + r.Icon
|
if !strings.HasPrefix(r.Icon, "/") {
|
||||||
h.db.Updates(&r)
|
r.Icon = "/" + r.Icon
|
||||||
|
h.db.Updates(&r)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break
|
||||||
|
case "history":
|
||||||
|
// 修改角色图片,改成绝对路径
|
||||||
|
var message []model.HistoryMessage
|
||||||
|
h.db.Find(&message)
|
||||||
|
for _, r := range message {
|
||||||
|
if !strings.HasPrefix(r.Icon, "/") {
|
||||||
|
r.Icon = "/" + r.Icon
|
||||||
|
h.db.Updates(&r)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
break
|
||||||
|
|
||||||
|
case "avatar":
|
||||||
|
// 更新用户的头像地址
|
||||||
|
var users []model.User
|
||||||
|
h.db.Find(&users)
|
||||||
|
for _, u := range users {
|
||||||
|
if !strings.HasPrefix(u.Avatar, "/") {
|
||||||
|
u.Avatar = "/" + u.Avatar
|
||||||
|
h.db.Updates(&u)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
@@ -48,6 +48,21 @@ func (h *ConfigHandler) Update(c *gin.Context) {
|
|||||||
resp.ERROR(c, res.Error.Error())
|
resp.ERROR(c, res.Error.Error())
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// update config cache for AppServer
|
||||||
|
var cfg model.Config
|
||||||
|
h.db.Where("marker", data.Key).First(&cfg)
|
||||||
|
var err error
|
||||||
|
if data.Key == "system" {
|
||||||
|
err = utils.JsonDecode(cfg.Config, &h.App.SysConfig)
|
||||||
|
} else if data.Key == "chat" {
|
||||||
|
err = utils.JsonDecode(cfg.Config, &h.App.ChatConfig)
|
||||||
|
}
|
||||||
|
if err != nil {
|
||||||
|
resp.ERROR(c, "Failed to update config cache: "+err.Error())
|
||||||
|
return
|
||||||
|
}
|
||||||
|
logger.Infof("Update AppServer's config successfully: %v", config.Config)
|
||||||
}
|
}
|
||||||
|
|
||||||
resp.SUCCESS(c, config)
|
resp.SUCCESS(c, config)
|
||||||
55
api/handler/admin/dashboard_handler.go
Normal file
55
api/handler/admin/dashboard_handler.go
Normal file
@@ -0,0 +1,55 @@
|
|||||||
|
package admin
|
||||||
|
|
||||||
|
import (
|
||||||
|
"chatplus/core"
|
||||||
|
"chatplus/handler"
|
||||||
|
"chatplus/store/model"
|
||||||
|
"chatplus/utils/resp"
|
||||||
|
"github.com/gin-gonic/gin"
|
||||||
|
"gorm.io/gorm"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
type DashboardHandler struct {
|
||||||
|
handler.BaseHandler
|
||||||
|
db *gorm.DB
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewDashboardHandler(app *core.AppServer, db *gorm.DB) *DashboardHandler {
|
||||||
|
h := DashboardHandler{db: db}
|
||||||
|
h.App = app
|
||||||
|
return &h
|
||||||
|
}
|
||||||
|
|
||||||
|
type statsVo struct {
|
||||||
|
Users int64 `json:"users"`
|
||||||
|
Chats int64 `json:"chats"`
|
||||||
|
Tokens int64 `json:"tokens"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *DashboardHandler) Stats(c *gin.Context) {
|
||||||
|
stats := statsVo{}
|
||||||
|
// new users statistic
|
||||||
|
var userCount int64
|
||||||
|
now := time.Now()
|
||||||
|
zeroTime := time.Date(now.Year(), now.Month(), now.Day(), 0, 0, 0, 0, now.Location())
|
||||||
|
res := h.db.Model(&model.User{}).Where("created_at > ?", zeroTime).Count(&userCount)
|
||||||
|
if res.Error == nil {
|
||||||
|
stats.Users = userCount
|
||||||
|
}
|
||||||
|
|
||||||
|
// new chats statistic
|
||||||
|
var chatCount int64
|
||||||
|
res = h.db.Model(&model.ChatItem{}).Where("created_at > ?", zeroTime).Count(&chatCount)
|
||||||
|
if res.Error == nil {
|
||||||
|
stats.Chats = chatCount
|
||||||
|
}
|
||||||
|
|
||||||
|
// tokens took stats
|
||||||
|
var tokenCount int64
|
||||||
|
res = h.db.Model(&model.HistoryMessage{}).Select("sum(tokens) as tokens_total").Where("created_at > ?", zeroTime).Scan(&tokenCount)
|
||||||
|
if res.Error == nil {
|
||||||
|
stats.Tokens = tokenCount
|
||||||
|
}
|
||||||
|
resp.SUCCESS(c, stats)
|
||||||
|
}
|
||||||
57
api/handler/admin/reward_handler.go
Normal file
57
api/handler/admin/reward_handler.go
Normal file
@@ -0,0 +1,57 @@
|
|||||||
|
package admin
|
||||||
|
|
||||||
|
import (
|
||||||
|
"chatplus/core"
|
||||||
|
"chatplus/handler"
|
||||||
|
"chatplus/store/model"
|
||||||
|
"chatplus/store/vo"
|
||||||
|
"chatplus/utils"
|
||||||
|
"chatplus/utils/resp"
|
||||||
|
"github.com/gin-gonic/gin"
|
||||||
|
"gorm.io/gorm"
|
||||||
|
)
|
||||||
|
|
||||||
|
type RewardHandler struct {
|
||||||
|
handler.BaseHandler
|
||||||
|
db *gorm.DB
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewRewardHandler(app *core.AppServer, db *gorm.DB) *RewardHandler {
|
||||||
|
h := RewardHandler{db: db}
|
||||||
|
h.App = app
|
||||||
|
return &h
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *RewardHandler) List(c *gin.Context) {
|
||||||
|
var items []model.Reward
|
||||||
|
res := h.db.Find(&items)
|
||||||
|
var rewards = make([]vo.Reward, 0)
|
||||||
|
if res.Error == nil {
|
||||||
|
userIds := make([]uint, 0)
|
||||||
|
for _, v := range items {
|
||||||
|
userIds = append(userIds, v.UserId)
|
||||||
|
}
|
||||||
|
var users []model.User
|
||||||
|
h.db.Where("id IN ?", userIds).Find(&users)
|
||||||
|
var userMap = make(map[uint]model.User)
|
||||||
|
for _, u := range users {
|
||||||
|
userMap[u.Id] = u
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, v := range items {
|
||||||
|
var r vo.Reward
|
||||||
|
err := utils.CopyObject(v, &r)
|
||||||
|
if err != nil {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
r.Id = v.Id
|
||||||
|
r.Username = userMap[v.UserId].Username
|
||||||
|
r.CreatedAt = v.CreatedAt.Unix()
|
||||||
|
r.UpdatedAt = v.UpdatedAt.Unix()
|
||||||
|
rewards = append(rewards, r)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
resp.SUCCESS(c, rewards)
|
||||||
|
}
|
||||||
@@ -8,6 +8,7 @@ import (
|
|||||||
"chatplus/store/vo"
|
"chatplus/store/vo"
|
||||||
"chatplus/utils"
|
"chatplus/utils"
|
||||||
"chatplus/utils/resp"
|
"chatplus/utils/resp"
|
||||||
|
"fmt"
|
||||||
|
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
"gorm.io/gorm"
|
"gorm.io/gorm"
|
||||||
@@ -28,12 +29,24 @@ func NewUserHandler(app *core.AppServer, db *gorm.DB) *UserHandler {
|
|||||||
func (h *UserHandler) List(c *gin.Context) {
|
func (h *UserHandler) List(c *gin.Context) {
|
||||||
page := h.GetInt(c, "page", 1)
|
page := h.GetInt(c, "page", 1)
|
||||||
pageSize := h.GetInt(c, "page_size", 20)
|
pageSize := h.GetInt(c, "page_size", 20)
|
||||||
|
mobile := h.GetTrim(c, "mobile")
|
||||||
|
username := h.GetTrim(c, "username")
|
||||||
|
|
||||||
offset := (page - 1) * pageSize
|
offset := (page - 1) * pageSize
|
||||||
var items []model.User
|
var items []model.User
|
||||||
var users = make([]vo.User, 0)
|
var users = make([]vo.User, 0)
|
||||||
var total int64
|
var total int64
|
||||||
h.db.Model(&model.User{}).Count(&total)
|
|
||||||
res := h.db.Offset(offset).Limit(pageSize).Find(&items)
|
session := h.db.Session(&gorm.Session{})
|
||||||
|
if mobile != "" {
|
||||||
|
session = session.Where("mobile LIKE ?", "%"+mobile+"%")
|
||||||
|
}
|
||||||
|
if username != "" {
|
||||||
|
session = session.Where("username LIKE ?", "%"+username+"%")
|
||||||
|
}
|
||||||
|
|
||||||
|
session.Model(&model.User{}).Count(&total)
|
||||||
|
res := session.Offset(offset).Limit(pageSize).Find(&items)
|
||||||
if res.Error == nil {
|
if res.Error == nil {
|
||||||
for _, item := range items {
|
for _, item := range items {
|
||||||
var user vo.User
|
var user vo.User
|
||||||
@@ -52,9 +65,12 @@ func (h *UserHandler) List(c *gin.Context) {
|
|||||||
resp.SUCCESS(c, pageVo)
|
resp.SUCCESS(c, pageVo)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (h *UserHandler) Update(c *gin.Context) {
|
func (h *UserHandler) Save(c *gin.Context) {
|
||||||
var data struct {
|
var data struct {
|
||||||
Id uint `json:"id"`
|
Id uint `json:"id"`
|
||||||
|
Username string `json:"username"`
|
||||||
|
Password string `json:"password"`
|
||||||
|
Mobile string `json:"mobile"`
|
||||||
Nickname string `json:"nickname"`
|
Nickname string `json:"nickname"`
|
||||||
Calls int `json:"calls"`
|
Calls int `json:"calls"`
|
||||||
ChatRoles []string `json:"chat_roles"`
|
ChatRoles []string `json:"chat_roles"`
|
||||||
@@ -66,21 +82,82 @@ func (h *UserHandler) Update(c *gin.Context) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
var user = model.User{}
|
var user = model.User{}
|
||||||
user.Id = data.Id
|
var res *gorm.DB
|
||||||
// 此处需要用 map 更新,用结构体无法更新 0 值
|
var userVo vo.User
|
||||||
res := h.db.Model(&user).Updates(map[string]interface{}{
|
if data.Id > 0 { // 更新
|
||||||
"nickname": data.Nickname,
|
user.Id = data.Id
|
||||||
"calls": data.Calls,
|
// 此处需要用 map 更新,用结构体无法更新 0 值
|
||||||
"status": data.Status,
|
res = h.db.Model(&user).Updates(map[string]interface{}{
|
||||||
"chat_roles_json": utils.JsonEncode(data.ChatRoles),
|
"nickname": data.Nickname,
|
||||||
"expired_time": utils.Str2stamp(data.ExpiredTime),
|
"mobile": data.Mobile,
|
||||||
})
|
"calls": data.Calls,
|
||||||
|
"status": data.Status,
|
||||||
|
"chat_roles_json": utils.JsonEncode(data.ChatRoles),
|
||||||
|
"expired_time": utils.Str2stamp(data.ExpiredTime),
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
salt := utils.RandString(8)
|
||||||
|
u := model.User{
|
||||||
|
Username: data.Username,
|
||||||
|
Password: utils.GenPassword(data.Password, salt),
|
||||||
|
Nickname: fmt.Sprintf("极客学长@%d", utils.RandomNumber(5)),
|
||||||
|
Avatar: "/images/avatar/user.png",
|
||||||
|
Salt: salt,
|
||||||
|
Status: true,
|
||||||
|
Mobile: data.Mobile,
|
||||||
|
ChatRoles: utils.JsonEncode(data.ChatRoles),
|
||||||
|
ExpiredTime: utils.Str2stamp(data.ExpiredTime),
|
||||||
|
ChatConfig: utils.JsonEncode(types.ChatConfig{
|
||||||
|
Temperature: h.App.ChatConfig.Temperature,
|
||||||
|
MaxTokens: h.App.ChatConfig.MaxTokens,
|
||||||
|
EnableContext: h.App.ChatConfig.EnableContext,
|
||||||
|
EnableHistory: true,
|
||||||
|
Model: h.App.ChatConfig.Model,
|
||||||
|
ApiKey: "",
|
||||||
|
}),
|
||||||
|
Calls: h.App.SysConfig.UserInitCalls,
|
||||||
|
}
|
||||||
|
res = h.db.Create(&u)
|
||||||
|
_ = utils.CopyObject(u, &userVo)
|
||||||
|
userVo.Id = u.Id
|
||||||
|
userVo.CreatedAt = u.CreatedAt.Unix()
|
||||||
|
userVo.UpdatedAt = u.UpdatedAt.Unix()
|
||||||
|
}
|
||||||
|
|
||||||
if res.Error != nil {
|
if res.Error != nil {
|
||||||
resp.ERROR(c, "更新数据库失败")
|
resp.ERROR(c, "更新数据库失败")
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
resp.SUCCESS(c)
|
resp.SUCCESS(c, userVo)
|
||||||
|
}
|
||||||
|
|
||||||
|
// ResetPass 重置密码
|
||||||
|
func (h *UserHandler) ResetPass(c *gin.Context) {
|
||||||
|
var data struct {
|
||||||
|
Id uint
|
||||||
|
Password string
|
||||||
|
}
|
||||||
|
if err := c.ShouldBindJSON(&data); err != nil {
|
||||||
|
resp.ERROR(c, types.InvalidArgs)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
var user model.User
|
||||||
|
res := h.db.First(&user, data.Id)
|
||||||
|
if res.Error != nil {
|
||||||
|
resp.ERROR(c, "No user found")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
password := utils.GenPassword(data.Password, user.Salt)
|
||||||
|
user.Password = password
|
||||||
|
res = h.db.Updates(&user)
|
||||||
|
if res.Error != nil {
|
||||||
|
resp.ERROR(c)
|
||||||
|
} else {
|
||||||
|
resp.SUCCESS(c)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (h *UserHandler) Remove(c *gin.Context) {
|
func (h *UserHandler) Remove(c *gin.Context) {
|
||||||
47
api/handler/captcha_handler.go
Normal file
47
api/handler/captcha_handler.go
Normal file
@@ -0,0 +1,47 @@
|
|||||||
|
package handler
|
||||||
|
|
||||||
|
import (
|
||||||
|
"chatplus/core/types"
|
||||||
|
"chatplus/service"
|
||||||
|
"chatplus/utils/resp"
|
||||||
|
"github.com/gin-gonic/gin"
|
||||||
|
)
|
||||||
|
|
||||||
|
// 今日头条函数实现
|
||||||
|
|
||||||
|
type CaptchaHandler struct {
|
||||||
|
service *service.CaptchaService
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewCaptchaHandler(s *service.CaptchaService) *CaptchaHandler {
|
||||||
|
return &CaptchaHandler{service: s}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *CaptchaHandler) Get(c *gin.Context) {
|
||||||
|
data, err := h.service.Get()
|
||||||
|
if err != nil {
|
||||||
|
resp.ERROR(c, err.Error())
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
resp.SUCCESS(c, data)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check verify the captcha data
|
||||||
|
func (h *CaptchaHandler) Check(c *gin.Context) {
|
||||||
|
var data struct {
|
||||||
|
Key string `json:"key"`
|
||||||
|
Dots string `json:"dots"`
|
||||||
|
}
|
||||||
|
if err := c.ShouldBindJSON(&data); err != nil {
|
||||||
|
resp.ERROR(c, types.InvalidArgs)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if h.service.Check(data) {
|
||||||
|
resp.SUCCESS(c)
|
||||||
|
} else {
|
||||||
|
resp.ERROR(c)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
598
api/handler/chat_handler.go
Normal file
598
api/handler/chat_handler.go
Normal file
@@ -0,0 +1,598 @@
|
|||||||
|
package handler
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bufio"
|
||||||
|
"bytes"
|
||||||
|
"chatplus/core"
|
||||||
|
"chatplus/core/types"
|
||||||
|
"chatplus/store/model"
|
||||||
|
"chatplus/store/vo"
|
||||||
|
"chatplus/utils"
|
||||||
|
"chatplus/utils/resp"
|
||||||
|
"context"
|
||||||
|
"encoding/json"
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
|
"net/http"
|
||||||
|
"net/url"
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
"unicode/utf8"
|
||||||
|
|
||||||
|
"github.com/gin-gonic/gin"
|
||||||
|
"github.com/gorilla/websocket"
|
||||||
|
"gorm.io/gorm"
|
||||||
|
)
|
||||||
|
|
||||||
|
const ErrorMsg = "抱歉,AI 助手开小差了,请稍后再试。"
|
||||||
|
|
||||||
|
type ChatHandler struct {
|
||||||
|
BaseHandler
|
||||||
|
db *gorm.DB
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewChatHandler(app *core.AppServer, db *gorm.DB) *ChatHandler {
|
||||||
|
handler := ChatHandler{db: db}
|
||||||
|
handler.App = app
|
||||||
|
return &handler
|
||||||
|
}
|
||||||
|
|
||||||
|
var chatConfig types.ChatConfig
|
||||||
|
|
||||||
|
// ChatHandle 处理聊天 WebSocket 请求
|
||||||
|
func (h *ChatHandler) ChatHandle(c *gin.Context) {
|
||||||
|
ws, err := (&websocket.Upgrader{CheckOrigin: func(r *http.Request) bool { return true }}).Upgrade(c.Writer, c.Request, nil)
|
||||||
|
if err != nil {
|
||||||
|
logger.Error(err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
// 设置读写超时时间
|
||||||
|
_ = ws.SetWriteDeadline(time.Now().Add(300 * time.Second))
|
||||||
|
_ = ws.SetReadDeadline(time.Now().Add(300 * time.Second))
|
||||||
|
|
||||||
|
sessionId := c.Query("session_id")
|
||||||
|
roleId := h.GetInt(c, "role_id", 0)
|
||||||
|
chatId := c.Query("chat_id")
|
||||||
|
chatModel := c.Query("model")
|
||||||
|
|
||||||
|
session := h.App.ChatSession.Get(sessionId)
|
||||||
|
if session.SessionId == "" {
|
||||||
|
user, err := utils.GetLoginUser(c, h.db)
|
||||||
|
if err != nil {
|
||||||
|
logger.Info("用户未登录")
|
||||||
|
c.Abort()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
session = types.ChatSession{
|
||||||
|
SessionId: sessionId,
|
||||||
|
ClientIP: c.ClientIP(),
|
||||||
|
Username: user.Username,
|
||||||
|
UserId: user.Id,
|
||||||
|
}
|
||||||
|
h.App.ChatSession.Put(sessionId, session)
|
||||||
|
}
|
||||||
|
|
||||||
|
// use old chat data override the chat model and role ID
|
||||||
|
var chat model.ChatItem
|
||||||
|
res := h.db.Where("chat_id=?", chatId).First(&chat)
|
||||||
|
if res.Error == nil {
|
||||||
|
chatModel = chat.Model
|
||||||
|
roleId = int(chat.RoleId)
|
||||||
|
}
|
||||||
|
|
||||||
|
session.ChatId = chatId
|
||||||
|
session.Model = chatModel
|
||||||
|
logger.Infof("New websocket connected, IP: %s, Username: %s", c.Request.RemoteAddr, session.Username)
|
||||||
|
client := types.NewWsClient(ws)
|
||||||
|
var chatRole model.ChatRole
|
||||||
|
res = h.db.First(&chatRole, roleId)
|
||||||
|
if res.Error != nil || !chatRole.Enable {
|
||||||
|
replyMessage(client, "当前聊天角色不存在或者未启用,连接已关闭!!!")
|
||||||
|
c.Abort()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// 初始化聊天配置
|
||||||
|
var config model.Config
|
||||||
|
h.db.Where("marker", "chat").First(&config)
|
||||||
|
err = utils.JsonDecode(config.Config, &chatConfig)
|
||||||
|
if err != nil {
|
||||||
|
replyMessage(client, "加载系统配置失败,连接已关闭!!!")
|
||||||
|
c.Abort()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// 保存会话连接
|
||||||
|
h.App.ChatClients.Put(sessionId, client)
|
||||||
|
go func() {
|
||||||
|
for {
|
||||||
|
_, message, err := client.Receive()
|
||||||
|
if err != nil {
|
||||||
|
logger.Error(err)
|
||||||
|
client.Close()
|
||||||
|
h.App.ChatClients.Delete(sessionId)
|
||||||
|
h.App.ReqCancelFunc.Delete(sessionId)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
logger.Info("Receive a message: ", string(message))
|
||||||
|
//replyMessage(client, "这是一条测试消息!")
|
||||||
|
ctx, cancel := context.WithCancel(context.Background())
|
||||||
|
h.App.ReqCancelFunc.Put(sessionId, cancel)
|
||||||
|
// 回复消息
|
||||||
|
err = h.sendMessage(ctx, session, chatRole, string(message), client)
|
||||||
|
if err != nil {
|
||||||
|
logger.Error(err)
|
||||||
|
} else {
|
||||||
|
replyChunkMessage(client, types.WsMessage{Type: types.WsEnd})
|
||||||
|
logger.Info("回答完毕: " + string(message))
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
}
|
||||||
|
|
||||||
|
// 将消息发送给 ChatGPT 并获取结果,通过 WebSocket 推送到客户端
|
||||||
|
func (h *ChatHandler) sendMessage(ctx context.Context, session types.ChatSession, role model.ChatRole, prompt string, ws types.Client) error {
|
||||||
|
promptCreatedAt := time.Now() // 记录提问时间
|
||||||
|
|
||||||
|
var user model.User
|
||||||
|
res := h.db.Model(&model.User{}).First(&user, session.UserId)
|
||||||
|
if res.Error != nil {
|
||||||
|
replyMessage(ws, "非法用户,请联系管理员!")
|
||||||
|
return res.Error
|
||||||
|
}
|
||||||
|
var userVo vo.User
|
||||||
|
err := utils.CopyObject(user, &userVo)
|
||||||
|
userVo.Id = user.Id
|
||||||
|
if err != nil {
|
||||||
|
return errors.New("User 对象转换失败," + err.Error())
|
||||||
|
}
|
||||||
|
|
||||||
|
if userVo.Status == false {
|
||||||
|
replyMessage(ws, "您的账号已经被禁用,如果疑问,请联系管理员!")
|
||||||
|
replyMessage(ws, "")
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if userVo.Calls <= 0 && userVo.ChatConfig.ApiKey == "" {
|
||||||
|
replyMessage(ws, "您的对话次数已经用尽,请联系管理员或者点击左下角菜单加入众筹获得100次对话!")
|
||||||
|
replyMessage(ws, "")
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if userVo.ExpiredTime > 0 && userVo.ExpiredTime <= time.Now().Unix() {
|
||||||
|
replyMessage(ws, "您的账号已经过期,请联系管理员!")
|
||||||
|
replyMessage(ws, "")
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
var req = types.ApiRequest{
|
||||||
|
Model: session.Model,
|
||||||
|
Temperature: userVo.ChatConfig.Temperature,
|
||||||
|
MaxTokens: userVo.ChatConfig.MaxTokens,
|
||||||
|
Stream: true,
|
||||||
|
Functions: types.InnerFunctions,
|
||||||
|
}
|
||||||
|
|
||||||
|
// 加载聊天上下文
|
||||||
|
var chatCtx []interface{}
|
||||||
|
if userVo.ChatConfig.EnableContext {
|
||||||
|
if h.App.ChatContexts.Has(session.ChatId) {
|
||||||
|
chatCtx = h.App.ChatContexts.Get(session.ChatId)
|
||||||
|
} else {
|
||||||
|
// calculate the tokens of current request, to prevent to exceeding the max tokens num
|
||||||
|
tokens := req.MaxTokens
|
||||||
|
for _, f := range types.InnerFunctions {
|
||||||
|
tks, _ := utils.CalcTokens(utils.JsonEncode(f), req.Model)
|
||||||
|
tokens += tks
|
||||||
|
}
|
||||||
|
|
||||||
|
// loading the role context
|
||||||
|
var messages []types.Message
|
||||||
|
err := utils.JsonDecode(role.Context, &messages)
|
||||||
|
if err == nil {
|
||||||
|
for _, v := range messages {
|
||||||
|
tks, _ := utils.CalcTokens(v.Content, req.Model)
|
||||||
|
if tokens+tks >= types.ModelToTokens[req.Model] {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
tokens += tks
|
||||||
|
chatCtx = append(chatCtx, v)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// loading recent chat history as chat context
|
||||||
|
if chatConfig.ContextDeep > 0 {
|
||||||
|
var historyMessages []model.HistoryMessage
|
||||||
|
res := h.db.Where("chat_id = ? and use_context = 1", session.ChatId).Limit(chatConfig.ContextDeep).Order("created_at desc").Find(&historyMessages)
|
||||||
|
if res.Error == nil {
|
||||||
|
for _, msg := range historyMessages {
|
||||||
|
if tokens+msg.Tokens >= types.ModelToTokens[session.Model] {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
tokens += msg.Tokens
|
||||||
|
ms := types.Message{Role: "user", Content: msg.Content}
|
||||||
|
if msg.Type == types.ReplyMsg {
|
||||||
|
ms.Role = "assistant"
|
||||||
|
}
|
||||||
|
chatCtx = append(chatCtx, ms)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
logger.Debugf("聊天上下文:%+v", chatCtx)
|
||||||
|
}
|
||||||
|
reqMgs := make([]interface{}, 0)
|
||||||
|
for _, m := range chatCtx {
|
||||||
|
reqMgs = append(reqMgs, m)
|
||||||
|
}
|
||||||
|
|
||||||
|
req.Messages = append(reqMgs, map[string]interface{}{
|
||||||
|
"role": "user",
|
||||||
|
"content": prompt,
|
||||||
|
})
|
||||||
|
var apiKey string
|
||||||
|
response, err := h.doRequest(ctx, userVo, &apiKey, req)
|
||||||
|
if err != nil {
|
||||||
|
if strings.Contains(err.Error(), "context canceled") {
|
||||||
|
logger.Info("用户取消了请求:", prompt)
|
||||||
|
return nil
|
||||||
|
} else if strings.Contains(err.Error(), "no available key") {
|
||||||
|
replyMessage(ws, "抱歉😔😔😔,系统已经没有可用的 API KEY🔑,您可以导入自己的 API KEY🔑 继续使用!🙏🙏🙏")
|
||||||
|
return nil
|
||||||
|
} else {
|
||||||
|
logger.Error(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
replyMessage(ws, ErrorMsg)
|
||||||
|
replyMessage(ws, "")
|
||||||
|
return err
|
||||||
|
} else {
|
||||||
|
defer response.Body.Close()
|
||||||
|
}
|
||||||
|
|
||||||
|
contentType := response.Header.Get("Content-Type")
|
||||||
|
if strings.Contains(contentType, "text/event-stream") {
|
||||||
|
if true {
|
||||||
|
replyCreatedAt := time.Now()
|
||||||
|
// 循环读取 Chunk 消息
|
||||||
|
var message = types.Message{}
|
||||||
|
var contents = make([]string, 0)
|
||||||
|
var functionCall = false
|
||||||
|
var functionName string
|
||||||
|
var arguments = make([]string, 0)
|
||||||
|
reader := bufio.NewReader(response.Body)
|
||||||
|
for {
|
||||||
|
line, err := reader.ReadString('\n')
|
||||||
|
if err != nil {
|
||||||
|
if strings.Contains(err.Error(), "context canceled") {
|
||||||
|
logger.Info("用户取消了请求:", prompt)
|
||||||
|
} else if err != io.EOF {
|
||||||
|
logger.Error("信息读取出错:", err)
|
||||||
|
}
|
||||||
|
break
|
||||||
|
}
|
||||||
|
if !strings.Contains(line, "data:") || len(line) < 30 {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
var responseBody = types.ApiResponse{}
|
||||||
|
err = json.Unmarshal([]byte(line[6:]), &responseBody)
|
||||||
|
if err != nil || len(responseBody.Choices) == 0 { // 数据解析出错
|
||||||
|
logger.Error(err, line)
|
||||||
|
replyMessage(ws, ErrorMsg)
|
||||||
|
replyMessage(ws, "")
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
fun := responseBody.Choices[0].Delta.FunctionCall
|
||||||
|
if functionCall && fun.Name == "" {
|
||||||
|
arguments = append(arguments, fun.Arguments)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if !utils.IsEmptyValue(fun) {
|
||||||
|
functionCall = true
|
||||||
|
functionName = fun.Name
|
||||||
|
f := h.App.Functions[functionName]
|
||||||
|
replyChunkMessage(ws, types.WsMessage{Type: types.WsStart})
|
||||||
|
replyChunkMessage(ws, types.WsMessage{Type: types.WsMiddle, Content: fmt.Sprintf("正在调用函数 `%s` 作答 ...\n\n", f.Name())})
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if responseBody.Choices[0].FinishReason == "function_call" { // 函数调用完毕
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
// 初始化 role
|
||||||
|
if responseBody.Choices[0].Delta.Role != "" && message.Role == "" {
|
||||||
|
message.Role = responseBody.Choices[0].Delta.Role
|
||||||
|
replyChunkMessage(ws, types.WsMessage{Type: types.WsStart})
|
||||||
|
continue
|
||||||
|
} else if responseBody.Choices[0].FinishReason != "" {
|
||||||
|
break // 输出完成或者输出中断了
|
||||||
|
} else {
|
||||||
|
content := responseBody.Choices[0].Delta.Content
|
||||||
|
contents = append(contents, utils.InterfaceToString(content))
|
||||||
|
replyChunkMessage(ws, types.WsMessage{
|
||||||
|
Type: types.WsMiddle,
|
||||||
|
Content: utils.InterfaceToString(responseBody.Choices[0].Delta.Content),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
} // end for
|
||||||
|
|
||||||
|
if functionCall { // 调用函数完成任务
|
||||||
|
logger.Info(functionName)
|
||||||
|
logger.Info(arguments)
|
||||||
|
f := h.App.Functions[functionName]
|
||||||
|
data, err := f.Invoke(arguments)
|
||||||
|
if err != nil {
|
||||||
|
msg := "调用函数出错:" + err.Error()
|
||||||
|
replyChunkMessage(ws, types.WsMessage{
|
||||||
|
Type: types.WsMiddle,
|
||||||
|
Content: msg,
|
||||||
|
})
|
||||||
|
contents = append(contents, msg)
|
||||||
|
} else {
|
||||||
|
replyChunkMessage(ws, types.WsMessage{
|
||||||
|
Type: types.WsMiddle,
|
||||||
|
Content: data,
|
||||||
|
})
|
||||||
|
contents = append(contents, data)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 消息发送成功
|
||||||
|
if len(contents) > 0 {
|
||||||
|
// 更新用户的对话次数
|
||||||
|
if userVo.ChatConfig.ApiKey == "" { // 如果用户使用的是自己绑定的 API KEY 则不扣减对话次数
|
||||||
|
res := h.db.Model(&user).UpdateColumn("calls", gorm.Expr("calls - ?", 1))
|
||||||
|
if res.Error != nil {
|
||||||
|
return res.Error
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if message.Role == "" {
|
||||||
|
message.Role = "assistant"
|
||||||
|
}
|
||||||
|
message.Content = strings.Join(contents, "")
|
||||||
|
useMsg := types.Message{Role: "user", Content: prompt}
|
||||||
|
|
||||||
|
// 更新上下文消息,如果是调用函数则不需要更新上下文
|
||||||
|
if userVo.ChatConfig.EnableContext && functionCall == false {
|
||||||
|
chatCtx = append(chatCtx, useMsg) // 提问消息
|
||||||
|
chatCtx = append(chatCtx, message) // 回复消息
|
||||||
|
h.App.ChatContexts.Put(session.ChatId, chatCtx)
|
||||||
|
}
|
||||||
|
|
||||||
|
// 追加聊天记录
|
||||||
|
if userVo.ChatConfig.EnableHistory {
|
||||||
|
useContext := true
|
||||||
|
if functionCall {
|
||||||
|
useContext = false
|
||||||
|
}
|
||||||
|
|
||||||
|
// for prompt
|
||||||
|
promptToken, err := utils.CalcTokens(prompt, req.Model)
|
||||||
|
if err != nil {
|
||||||
|
logger.Error(err)
|
||||||
|
}
|
||||||
|
historyUserMsg := model.HistoryMessage{
|
||||||
|
UserId: userVo.Id,
|
||||||
|
ChatId: session.ChatId,
|
||||||
|
RoleId: role.Id,
|
||||||
|
Type: types.PromptMsg,
|
||||||
|
Icon: user.Avatar,
|
||||||
|
Content: prompt,
|
||||||
|
Tokens: promptToken,
|
||||||
|
UseContext: useContext,
|
||||||
|
}
|
||||||
|
historyUserMsg.CreatedAt = promptCreatedAt
|
||||||
|
historyUserMsg.UpdatedAt = promptCreatedAt
|
||||||
|
res := h.db.Save(&historyUserMsg)
|
||||||
|
if res.Error != nil {
|
||||||
|
logger.Error("failed to save prompt history message: ", res.Error)
|
||||||
|
}
|
||||||
|
|
||||||
|
// for reply
|
||||||
|
// 计算本次对话消耗的总 token 数量
|
||||||
|
var replyToken = 0
|
||||||
|
if functionCall { // 函数名 + 参数 token
|
||||||
|
tokens, _ := utils.CalcTokens(functionName, req.Model)
|
||||||
|
replyToken += tokens
|
||||||
|
tokens, _ = utils.CalcTokens(utils.InterfaceToString(arguments), req.Model)
|
||||||
|
replyToken += tokens
|
||||||
|
} else {
|
||||||
|
replyToken, _ = utils.CalcTokens(message.Content, req.Model)
|
||||||
|
}
|
||||||
|
|
||||||
|
historyReplyMsg := model.HistoryMessage{
|
||||||
|
UserId: userVo.Id,
|
||||||
|
ChatId: session.ChatId,
|
||||||
|
RoleId: role.Id,
|
||||||
|
Type: types.ReplyMsg,
|
||||||
|
Icon: role.Icon,
|
||||||
|
Content: message.Content,
|
||||||
|
Tokens: replyToken,
|
||||||
|
UseContext: useContext,
|
||||||
|
}
|
||||||
|
historyReplyMsg.CreatedAt = replyCreatedAt
|
||||||
|
historyReplyMsg.UpdatedAt = replyCreatedAt
|
||||||
|
res = h.db.Create(&historyReplyMsg)
|
||||||
|
if res.Error != nil {
|
||||||
|
logger.Error("failed to save reply history message: ", res.Error)
|
||||||
|
}
|
||||||
|
|
||||||
|
// 计算本次对话消耗的总 token 数量
|
||||||
|
var totalTokens = 0
|
||||||
|
if functionCall { // prompt + 函数名 + 参数 token
|
||||||
|
totalTokens = promptToken + replyToken
|
||||||
|
} else {
|
||||||
|
totalTokens = replyToken + getTotalTokens(req)
|
||||||
|
}
|
||||||
|
//replyChunkMessage(ws, types.WsMessage{Type: types.WsMiddle, Content: fmt.Sprintf("\n\n `本轮对话共消耗 Token 数量: %d`", totalTokens+11)})
|
||||||
|
if userVo.ChatConfig.ApiKey != "" { // 调用自己的 API KEY 不计算 token 消耗
|
||||||
|
h.db.Model(&user).UpdateColumn("tokens", gorm.Expr("tokens + ?",
|
||||||
|
totalTokens))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 保存当前会话
|
||||||
|
var chatItem model.ChatItem
|
||||||
|
res = h.db.Where("chat_id = ?", session.ChatId).First(&chatItem)
|
||||||
|
if res.Error != nil {
|
||||||
|
chatItem.ChatId = session.ChatId
|
||||||
|
chatItem.UserId = session.UserId
|
||||||
|
chatItem.RoleId = role.Id
|
||||||
|
chatItem.Model = session.Model
|
||||||
|
if utf8.RuneCountInString(prompt) > 30 {
|
||||||
|
chatItem.Title = string([]rune(prompt)[:30]) + "..."
|
||||||
|
} else {
|
||||||
|
chatItem.Title = prompt
|
||||||
|
}
|
||||||
|
h.db.Create(&chatItem)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
body, err := io.ReadAll(response.Body)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("error with reading response: %v", err)
|
||||||
|
}
|
||||||
|
var res types.ApiError
|
||||||
|
err = json.Unmarshal(body, &res)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("error with decode response: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// OpenAI API 调用异常处理
|
||||||
|
// TODO: 是否考虑重发消息?
|
||||||
|
if strings.Contains(res.Error.Message, "This key is associated with a deactivated account") {
|
||||||
|
replyMessage(ws, "请求 OpenAI API 失败:API KEY 所关联的账户被禁用。")
|
||||||
|
// 移除当前 API key
|
||||||
|
h.db.Where("value = ?", apiKey).Delete(&model.ApiKey{})
|
||||||
|
} else if strings.Contains(res.Error.Message, "You exceeded your current quota") {
|
||||||
|
replyMessage(ws, "请求 OpenAI API 失败:API KEY 触发并发限制,请稍后再试。")
|
||||||
|
} else if strings.Contains(res.Error.Message, "This model's maximum context length") {
|
||||||
|
logger.Error(res.Error.Message)
|
||||||
|
replyMessage(ws, "当前会话上下文长度超出限制,已为您清空会话上下文!")
|
||||||
|
h.App.ChatContexts.Delete(session.ChatId)
|
||||||
|
return h.sendMessage(ctx, session, role, prompt, ws)
|
||||||
|
} else {
|
||||||
|
replyMessage(ws, "请求 OpenAI API 失败:"+res.Error.Message)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// 发送请求到 OpenAI 服务器
|
||||||
|
// useOwnApiKey: 是否使用了用户自己的 API KEY
|
||||||
|
func (h *ChatHandler) doRequest(ctx context.Context, user vo.User, apiKey *string, req types.ApiRequest) (*http.Response, error) {
|
||||||
|
var client *http.Client
|
||||||
|
requestBody, err := json.Marshal(req)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
// 创建 HttpClient 请求对象
|
||||||
|
request, err := http.NewRequest(http.MethodPost, h.App.ChatConfig.ApiURL, bytes.NewBuffer(requestBody))
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
request = request.WithContext(ctx)
|
||||||
|
request.Header.Add("Content-Type", "application/json")
|
||||||
|
|
||||||
|
proxyURL := h.App.Config.ProxyURL
|
||||||
|
if proxyURL == "" {
|
||||||
|
client = &http.Client{}
|
||||||
|
} else { // 使用代理
|
||||||
|
proxy, _ := url.Parse(proxyURL)
|
||||||
|
client = &http.Client{
|
||||||
|
Transport: &http.Transport{
|
||||||
|
Proxy: http.ProxyURL(proxy),
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// 查询当前用户是否导入了自己的 API KEY
|
||||||
|
if user.ChatConfig.ApiKey != "" {
|
||||||
|
logger.Info("使用用户自己的 API KEY: ", user.ChatConfig.ApiKey)
|
||||||
|
*apiKey = user.ChatConfig.ApiKey
|
||||||
|
} else { // 获取系统的 API KEY
|
||||||
|
var key model.ApiKey
|
||||||
|
res := h.db.Where("user_id = ?", 0).Order("last_used_at ASC").First(&key)
|
||||||
|
if res.Error != nil {
|
||||||
|
return nil, errors.New("no available key, please import key")
|
||||||
|
}
|
||||||
|
*apiKey = key.Value
|
||||||
|
// 更新 API KEY 的最后使用时间
|
||||||
|
h.db.Model(&key).UpdateColumn("last_used_at", time.Now().Unix())
|
||||||
|
}
|
||||||
|
|
||||||
|
logger.Infof("Sending OpenAI request, KEY: %s, PROXY: %s, Model: %s", *apiKey, proxyURL, req.Model)
|
||||||
|
request.Header.Set("Authorization", fmt.Sprintf("Bearer %s", *apiKey))
|
||||||
|
return client.Do(request)
|
||||||
|
}
|
||||||
|
|
||||||
|
// 回复客户片段端消息
|
||||||
|
func replyChunkMessage(client types.Client, message types.WsMessage) {
|
||||||
|
msg, err := json.Marshal(message)
|
||||||
|
if err != nil {
|
||||||
|
logger.Errorf("Error for decoding json data: %v", err.Error())
|
||||||
|
return
|
||||||
|
}
|
||||||
|
err = client.(*types.WsClient).Send(msg)
|
||||||
|
if err != nil {
|
||||||
|
logger.Errorf("Error for reply message: %v", err.Error())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 回复客户端一条完整的消息
|
||||||
|
func replyMessage(ws types.Client, message string) {
|
||||||
|
replyChunkMessage(ws, types.WsMessage{Type: types.WsStart})
|
||||||
|
replyChunkMessage(ws, types.WsMessage{Type: types.WsMiddle, Content: message})
|
||||||
|
replyChunkMessage(ws, types.WsMessage{Type: types.WsEnd})
|
||||||
|
}
|
||||||
|
|
||||||
|
// Tokens 统计 token 数量
|
||||||
|
func (h *ChatHandler) Tokens(c *gin.Context) {
|
||||||
|
text := c.Query("text")
|
||||||
|
md := c.Query("model")
|
||||||
|
tokens, err := utils.CalcTokens(text, md)
|
||||||
|
if err != nil {
|
||||||
|
resp.ERROR(c, err.Error())
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
resp.SUCCESS(c, tokens)
|
||||||
|
}
|
||||||
|
|
||||||
|
func getTotalTokens(req types.ApiRequest) int {
|
||||||
|
encode := utils.JsonEncode(req.Messages)
|
||||||
|
var items []map[string]interface{}
|
||||||
|
err := utils.JsonDecode(encode, &items)
|
||||||
|
if err != nil {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
tokens := 0
|
||||||
|
for _, item := range items {
|
||||||
|
content, ok := item["content"]
|
||||||
|
if ok && !utils.IsEmptyValue(content) {
|
||||||
|
t, err := utils.CalcTokens(utils.InterfaceToString(content), req.Model)
|
||||||
|
if err == nil {
|
||||||
|
tokens += t
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return tokens
|
||||||
|
}
|
||||||
|
|
||||||
|
// StopGenerate 停止生成
|
||||||
|
func (h *ChatHandler) StopGenerate(c *gin.Context) {
|
||||||
|
sessionId := c.Query("session_id")
|
||||||
|
if h.App.ReqCancelFunc.Has(sessionId) {
|
||||||
|
h.App.ReqCancelFunc.Get(sessionId)()
|
||||||
|
h.App.ReqCancelFunc.Delete(sessionId)
|
||||||
|
}
|
||||||
|
resp.SUCCESS(c, types.OkMsg)
|
||||||
|
}
|
||||||
73
api/handler/reward_handler.go
Normal file
73
api/handler/reward_handler.go
Normal file
@@ -0,0 +1,73 @@
|
|||||||
|
package handler
|
||||||
|
|
||||||
|
import (
|
||||||
|
"chatplus/core"
|
||||||
|
"chatplus/core/types"
|
||||||
|
"chatplus/store/model"
|
||||||
|
"chatplus/utils"
|
||||||
|
"chatplus/utils/resp"
|
||||||
|
"github.com/gin-gonic/gin"
|
||||||
|
"gorm.io/gorm"
|
||||||
|
)
|
||||||
|
|
||||||
|
type RewardHandler struct {
|
||||||
|
BaseHandler
|
||||||
|
db *gorm.DB
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewRewardHandler(server *core.AppServer, db *gorm.DB) *RewardHandler {
|
||||||
|
h := RewardHandler{db: db}
|
||||||
|
h.App = server
|
||||||
|
return &h
|
||||||
|
}
|
||||||
|
|
||||||
|
// Verify 打赏码核销
|
||||||
|
func (h *RewardHandler) Verify(c *gin.Context) {
|
||||||
|
var data struct {
|
||||||
|
TxId string `json:"tx_id"`
|
||||||
|
}
|
||||||
|
if err := c.ShouldBindJSON(&data); err != nil {
|
||||||
|
resp.ERROR(c, types.InvalidArgs)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
var item model.Reward
|
||||||
|
res := h.db.Where("tx_id = ?", data.TxId).First(&item)
|
||||||
|
if res.Error != nil {
|
||||||
|
resp.ERROR(c, "无效的众筹交易流水号!")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if item.Status {
|
||||||
|
resp.ERROR(c, "当前众筹交易流水号已经被核销,请不要重复核销!")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
user, err := utils.GetLoginUser(c, h.db)
|
||||||
|
if err != nil {
|
||||||
|
resp.HACKER(c)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
tx := h.db.Begin()
|
||||||
|
calls := (item.Amount + 0.1) * 10
|
||||||
|
res = h.db.Model(&user).UpdateColumn("calls", gorm.Expr("calls + ?", calls))
|
||||||
|
if res.Error != nil {
|
||||||
|
resp.ERROR(c, "更新数据库失败!")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// 更新核销状态
|
||||||
|
item.Status = true
|
||||||
|
item.UserId = user.Id
|
||||||
|
res = h.db.Updates(&item)
|
||||||
|
if res.Error != nil {
|
||||||
|
tx.Rollback()
|
||||||
|
resp.ERROR(c, "更新数据库失败!")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
tx.Commit()
|
||||||
|
resp.SUCCESS(c)
|
||||||
|
|
||||||
|
}
|
||||||
70
api/handler/sms_handler.go
Normal file
70
api/handler/sms_handler.go
Normal file
@@ -0,0 +1,70 @@
|
|||||||
|
package handler
|
||||||
|
|
||||||
|
import (
|
||||||
|
"chatplus/core"
|
||||||
|
"chatplus/core/types"
|
||||||
|
"chatplus/service"
|
||||||
|
"chatplus/store"
|
||||||
|
"chatplus/utils"
|
||||||
|
"chatplus/utils/resp"
|
||||||
|
"github.com/gin-gonic/gin"
|
||||||
|
)
|
||||||
|
|
||||||
|
const CodeStorePrefix = "/verify/codes/"
|
||||||
|
|
||||||
|
type SmsHandler struct {
|
||||||
|
BaseHandler
|
||||||
|
db *store.LevelDB
|
||||||
|
sms *service.AliYunSmsService
|
||||||
|
captcha *service.CaptchaService
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewSmsHandler(app *core.AppServer, db *store.LevelDB, sms *service.AliYunSmsService, captcha *service.CaptchaService) *SmsHandler {
|
||||||
|
handler := &SmsHandler{db: db, sms: sms, captcha: captcha}
|
||||||
|
handler.App = app
|
||||||
|
return handler
|
||||||
|
}
|
||||||
|
|
||||||
|
// SendCode 发送验证码短信
|
||||||
|
func (h *SmsHandler) SendCode(c *gin.Context) {
|
||||||
|
var data struct {
|
||||||
|
Mobile string `json:"mobile"`
|
||||||
|
Key string `json:"key"`
|
||||||
|
Dots string `json:"dots"`
|
||||||
|
}
|
||||||
|
if err := c.ShouldBindJSON(&data); err != nil {
|
||||||
|
resp.ERROR(c, types.InvalidArgs)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if !h.captcha.Check(data) {
|
||||||
|
resp.ERROR(c, "验证码错误,请先完人机验证")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
code := utils.RandomNumber(6)
|
||||||
|
err := h.sms.SendVerifyCode(data.Mobile, code)
|
||||||
|
if err != nil {
|
||||||
|
resp.ERROR(c, err.Error())
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// 存储验证码,等待后面注册验证
|
||||||
|
err = h.db.Put(CodeStorePrefix+data.Mobile, code)
|
||||||
|
if err != nil {
|
||||||
|
resp.ERROR(c, "验证码保存失败")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
resp.SUCCESS(c)
|
||||||
|
}
|
||||||
|
|
||||||
|
type statusVo struct {
|
||||||
|
EnabledMsgService bool `json:"enabled_msg_service"`
|
||||||
|
EnabledRegister bool `json:"enabled_register"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// Status check if the message service is enabled
|
||||||
|
func (h *SmsHandler) Status(c *gin.Context) {
|
||||||
|
resp.SUCCESS(c, statusVo{EnabledMsgService: h.App.Config.EnabledMsgService, EnabledRegister: h.App.SysConfig.EnabledRegister})
|
||||||
|
}
|
||||||
67
api/handler/upload_handler.go
Normal file
67
api/handler/upload_handler.go
Normal file
@@ -0,0 +1,67 @@
|
|||||||
|
package handler
|
||||||
|
|
||||||
|
import (
|
||||||
|
"chatplus/core"
|
||||||
|
"chatplus/utils/resp"
|
||||||
|
"fmt"
|
||||||
|
"github.com/gin-gonic/gin"
|
||||||
|
"gorm.io/gorm"
|
||||||
|
"os"
|
||||||
|
"path/filepath"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
type UploadHandler struct {
|
||||||
|
BaseHandler
|
||||||
|
db *gorm.DB
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewUploadHandler(app *core.AppServer, db *gorm.DB) *UploadHandler {
|
||||||
|
handler := &UploadHandler{db: db}
|
||||||
|
handler.App = app
|
||||||
|
return handler
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *UploadHandler) Upload(c *gin.Context) {
|
||||||
|
file, err := c.FormFile("file")
|
||||||
|
if err != nil {
|
||||||
|
resp.ERROR(c, fmt.Sprintf("文件上传失败: %s", err.Error()))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
filePath, err := h.genFilePath(file.Filename)
|
||||||
|
if err != nil {
|
||||||
|
resp.ERROR(c, fmt.Sprintf("文件上传失败: %s", err.Error()))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
// 将文件保存到指定路径
|
||||||
|
err = c.SaveUploadedFile(file, filePath)
|
||||||
|
if err != nil {
|
||||||
|
resp.ERROR(c, fmt.Sprintf("文件保存失败: %s", err.Error()))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
resp.SUCCESS(c, h.genFileUrl(filePath))
|
||||||
|
}
|
||||||
|
|
||||||
|
// 生成上传文件路径
|
||||||
|
func (h *UploadHandler) genFilePath(filename string) (string, error) {
|
||||||
|
now := time.Now()
|
||||||
|
dir := fmt.Sprintf("%s/upload/%d/%d", h.App.Config.StaticDir, now.Year(), now.Month())
|
||||||
|
_, err := os.Stat(dir)
|
||||||
|
if err != nil {
|
||||||
|
err = os.MkdirAll(dir, 0755)
|
||||||
|
if err != nil {
|
||||||
|
return "", fmt.Errorf("创建上传目录失败:%s", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fileExt := filepath.Ext(filename)
|
||||||
|
return fmt.Sprintf("%s/%d%s", dir, now.UnixMilli(), fileExt), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// 生成上传文件 URL
|
||||||
|
func (h *UploadHandler) genFileUrl(filePath string) string {
|
||||||
|
now := time.Now()
|
||||||
|
filename := filepath.Base(filePath)
|
||||||
|
return fmt.Sprintf("%s/upload/%d/%d/%s", h.App.Config.StaticUrl, now.Year(), now.Month(), filename)
|
||||||
|
}
|
||||||
@@ -3,6 +3,7 @@ package handler
|
|||||||
import (
|
import (
|
||||||
"chatplus/core"
|
"chatplus/core"
|
||||||
"chatplus/core/types"
|
"chatplus/core/types"
|
||||||
|
"chatplus/store"
|
||||||
"chatplus/store/model"
|
"chatplus/store/model"
|
||||||
"chatplus/store/vo"
|
"chatplus/store/vo"
|
||||||
"chatplus/utils"
|
"chatplus/utils"
|
||||||
@@ -21,10 +22,11 @@ type UserHandler struct {
|
|||||||
BaseHandler
|
BaseHandler
|
||||||
db *gorm.DB
|
db *gorm.DB
|
||||||
searcher *xdb.Searcher
|
searcher *xdb.Searcher
|
||||||
|
levelDB *store.LevelDB
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewUserHandler(app *core.AppServer, db *gorm.DB, searcher *xdb.Searcher) *UserHandler {
|
func NewUserHandler(app *core.AppServer, db *gorm.DB, searcher *xdb.Searcher, levelDB *store.LevelDB) *UserHandler {
|
||||||
handler := &UserHandler{db: db, searcher: searcher}
|
handler := &UserHandler{db: db, searcher: searcher, levelDB: levelDB}
|
||||||
handler.App = app
|
handler.App = app
|
||||||
return handler
|
return handler
|
||||||
}
|
}
|
||||||
@@ -35,6 +37,8 @@ func (h *UserHandler) Register(c *gin.Context) {
|
|||||||
var data struct {
|
var data struct {
|
||||||
Username string `json:"username"`
|
Username string `json:"username"`
|
||||||
Password string `json:"password"`
|
Password string `json:"password"`
|
||||||
|
Mobile string `json:"mobile"`
|
||||||
|
Code int `json:"code"`
|
||||||
}
|
}
|
||||||
if err := c.ShouldBindJSON(&data); err != nil {
|
if err := c.ShouldBindJSON(&data); err != nil {
|
||||||
resp.ERROR(c, types.InvalidArgs)
|
resp.ERROR(c, types.InvalidArgs)
|
||||||
@@ -52,13 +56,32 @@ func (h *UserHandler) Register(c *gin.Context) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// 检查验证码
|
||||||
|
key := CodeStorePrefix + data.Mobile
|
||||||
|
if h.App.Config.EnabledMsgService {
|
||||||
|
var code int
|
||||||
|
err := h.levelDB.Get(key, &code)
|
||||||
|
if err != nil || code != data.Code {
|
||||||
|
logger.Info(code)
|
||||||
|
resp.ERROR(c, "短信验证码错误")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// check if the username is exists
|
// check if the username is exists
|
||||||
var item model.User
|
var item model.User
|
||||||
tx := h.db.Where("username = ?", data.Username).First(&item)
|
res := h.db.Where("username = ?", data.Username).First(&item)
|
||||||
if tx.RowsAffected > 0 {
|
if res.RowsAffected > 0 {
|
||||||
resp.ERROR(c, "用户名已存在")
|
resp.ERROR(c, "用户名已存在")
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
res = h.db.Where("mobile = ?", data.Mobile).First(&item)
|
||||||
|
if res.RowsAffected > 0 {
|
||||||
|
resp.ERROR(c, "该手机号码以及被注册,请更换其他手机号")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
// 默认订阅所有角色
|
// 默认订阅所有角色
|
||||||
var chatRoles []model.ChatRole
|
var chatRoles []model.ChatRole
|
||||||
h.db.Find(&chatRoles)
|
h.db.Find(&chatRoles)
|
||||||
@@ -72,9 +95,10 @@ func (h *UserHandler) Register(c *gin.Context) {
|
|||||||
Username: data.Username,
|
Username: data.Username,
|
||||||
Password: utils.GenPassword(data.Password, salt),
|
Password: utils.GenPassword(data.Password, salt),
|
||||||
Nickname: fmt.Sprintf("极客学长@%d", utils.RandomNumber(5)),
|
Nickname: fmt.Sprintf("极客学长@%d", utils.RandomNumber(5)),
|
||||||
Avatar: "images/avatar/user.png",
|
Avatar: "/images/avatar/user.png",
|
||||||
Salt: salt,
|
Salt: salt,
|
||||||
Status: true,
|
Status: true,
|
||||||
|
Mobile: data.Mobile,
|
||||||
ChatRoles: utils.JsonEncode(roleKeys),
|
ChatRoles: utils.JsonEncode(roleKeys),
|
||||||
ChatConfig: utils.JsonEncode(types.ChatConfig{
|
ChatConfig: utils.JsonEncode(types.ChatConfig{
|
||||||
Temperature: h.App.ChatConfig.Temperature,
|
Temperature: h.App.ChatConfig.Temperature,
|
||||||
@@ -84,24 +108,18 @@ func (h *UserHandler) Register(c *gin.Context) {
|
|||||||
Model: h.App.ChatConfig.Model,
|
Model: h.App.ChatConfig.Model,
|
||||||
ApiKey: "",
|
ApiKey: "",
|
||||||
}),
|
}),
|
||||||
|
Calls: h.App.SysConfig.UserInitCalls,
|
||||||
}
|
}
|
||||||
// 初始化调用次数
|
res = h.db.Create(&user)
|
||||||
var cfg model.Config
|
|
||||||
h.db.Where("marker = ?", "system").First(&cfg)
|
|
||||||
var config types.SystemConfig
|
|
||||||
err := utils.JsonDecode(cfg.Config, &config)
|
|
||||||
if err != nil || config.UserInitCalls <= 0 {
|
|
||||||
user.Calls = types.UserInitCalls
|
|
||||||
} else {
|
|
||||||
user.Calls = config.UserInitCalls
|
|
||||||
}
|
|
||||||
res := h.db.Create(&user)
|
|
||||||
if res.Error != nil {
|
if res.Error != nil {
|
||||||
resp.ERROR(c, "保存数据失败")
|
resp.ERROR(c, "保存数据失败")
|
||||||
logger.Error(res.Error)
|
logger.Error(res.Error)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if h.App.Config.EnabledMsgService {
|
||||||
|
_ = h.levelDB.Delete(key) // 注册成功,删除短信验证码
|
||||||
|
}
|
||||||
resp.SUCCESS(c, user)
|
resp.SUCCESS(c, user)
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -116,7 +134,7 @@ func (h *UserHandler) Login(c *gin.Context) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
var user model.User
|
var user model.User
|
||||||
res := h.db.Where("username = ?", data.Username).First(&user)
|
res := h.db.Where("username = ? OR mobile = ?", data.Username, data.Username).First(&user)
|
||||||
if res.Error != nil {
|
if res.Error != nil {
|
||||||
resp.ERROR(c, "用户名不存在")
|
resp.ERROR(c, "用户名不存在")
|
||||||
return
|
return
|
||||||
@@ -128,6 +146,11 @@ func (h *UserHandler) Login(c *gin.Context) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if user.Status == false {
|
||||||
|
resp.ERROR(c, "该用户已被禁止登录,请联系管理员")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
// 更新最后登录时间和IP
|
// 更新最后登录时间和IP
|
||||||
user.LastLoginIp = c.ClientIP()
|
user.LastLoginIp = c.ClientIP()
|
||||||
user.LastLoginAt = time.Now().Unix()
|
user.LastLoginAt = time.Now().Unix()
|
||||||
@@ -144,27 +167,6 @@ func (h *UserHandler) Login(c *gin.Context) {
|
|||||||
// 记录登录信息在服务端
|
// 记录登录信息在服务端
|
||||||
h.App.ChatSession.Put(sessionId, types.ChatSession{ClientIP: c.ClientIP(), UserId: user.Id, Username: data.Username, SessionId: sessionId})
|
h.App.ChatSession.Put(sessionId, types.ChatSession{ClientIP: c.ClientIP(), UserId: user.Id, Username: data.Username, SessionId: sessionId})
|
||||||
|
|
||||||
// 加载用户订阅的聊天角色
|
|
||||||
var roleKeys []string
|
|
||||||
err = utils.JsonDecode(user.ChatRoles, &roleKeys)
|
|
||||||
var chatRoles interface{}
|
|
||||||
if err == nil {
|
|
||||||
var roles []model.ChatRole
|
|
||||||
res = h.db.Where("marker IN ?", roleKeys).Find(&roles)
|
|
||||||
if res.Error == err {
|
|
||||||
type Item struct {
|
|
||||||
Name string
|
|
||||||
Key string
|
|
||||||
Icon string
|
|
||||||
}
|
|
||||||
items := make([]Item, 0)
|
|
||||||
for _, r := range roles {
|
|
||||||
items = append(items, Item{Name: r.Name, Key: r.Key, Icon: r.Icon})
|
|
||||||
}
|
|
||||||
chatRoles = items
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
h.db.Create(&model.UserLoginLog{
|
h.db.Create(&model.UserLoginLog{
|
||||||
UserId: user.Id,
|
UserId: user.Id,
|
||||||
Username: user.Username,
|
Username: user.Username,
|
||||||
@@ -186,8 +188,7 @@ func (h *UserHandler) Login(c *gin.Context) {
|
|||||||
"username": user.Username,
|
"username": user.Username,
|
||||||
"tokens": user.Tokens,
|
"tokens": user.Tokens,
|
||||||
"calls": user.Calls,
|
"calls": user.Calls,
|
||||||
"expiredTime": user.ExpiredTime,
|
"expired_time": user.ExpiredTime,
|
||||||
"chatRoles": chatRoles,
|
|
||||||
"api_key": chatConfig.ApiKey,
|
"api_key": chatConfig.ApiKey,
|
||||||
"model": chatConfig.Model,
|
"model": chatConfig.Model,
|
||||||
"temperature": chatConfig.Temperature,
|
"temperature": chatConfig.Temperature,
|
||||||
@@ -218,18 +219,54 @@ func (h *UserHandler) Logout(c *gin.Context) {
|
|||||||
|
|
||||||
// Session 获取/验证会话
|
// Session 获取/验证会话
|
||||||
func (h *UserHandler) Session(c *gin.Context) {
|
func (h *UserHandler) Session(c *gin.Context) {
|
||||||
sessionId := c.GetHeader(types.SessionName)
|
user, err := utils.GetLoginUser(c, h.db)
|
||||||
session := h.App.ChatSession.Get(sessionId)
|
if err == nil {
|
||||||
if session.ClientIP == c.ClientIP() {
|
var userVo vo.User
|
||||||
resp.SUCCESS(c, session)
|
err := utils.CopyObject(user, &userVo)
|
||||||
|
if err != nil {
|
||||||
|
resp.ERROR(c)
|
||||||
|
}
|
||||||
|
userVo.Id = user.Id
|
||||||
|
resp.SUCCESS(c, userVo)
|
||||||
} else {
|
} else {
|
||||||
resp.NotAuth(c)
|
resp.NotAuth(c)
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type userProfile struct {
|
||||||
|
Id uint `json:"id"`
|
||||||
|
Username string `json:"username"`
|
||||||
|
Nickname string `json:"nickname"`
|
||||||
|
Mobile string `json:"mobile"`
|
||||||
|
Avatar string `json:"avatar"`
|
||||||
|
ChatConfig types.ChatConfig `json:"chat_config"`
|
||||||
|
Calls int `json:"calls"`
|
||||||
|
Tokens int64 `json:"tokens"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *UserHandler) Profile(c *gin.Context) {
|
||||||
|
user, err := utils.GetLoginUser(c, h.db)
|
||||||
|
if err != nil {
|
||||||
|
resp.NotAuth(c)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
h.db.First(&user, user.Id)
|
||||||
|
var profile userProfile
|
||||||
|
err = utils.CopyObject(user, &profile)
|
||||||
|
if err != nil {
|
||||||
|
logger.Error("对象拷贝失败:", err.Error())
|
||||||
|
resp.ERROR(c, "获取用户信息失败")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
profile.Id = user.Id
|
||||||
|
resp.SUCCESS(c, profile)
|
||||||
|
}
|
||||||
|
|
||||||
func (h *UserHandler) ProfileUpdate(c *gin.Context) {
|
func (h *UserHandler) ProfileUpdate(c *gin.Context) {
|
||||||
var data vo.User
|
var data userProfile
|
||||||
if err := c.ShouldBindJSON(&data); err != nil {
|
if err := c.ShouldBindJSON(&data); err != nil {
|
||||||
resp.ERROR(c, types.InvalidArgs)
|
resp.ERROR(c, types.InvalidArgs)
|
||||||
return
|
return
|
||||||
@@ -267,28 +304,6 @@ func (h *UserHandler) ProfileUpdate(c *gin.Context) {
|
|||||||
resp.SUCCESS(c)
|
resp.SUCCESS(c)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (h *UserHandler) Profile(c *gin.Context) {
|
|
||||||
user, err := utils.GetLoginUser(c, h.db)
|
|
||||||
if err != nil {
|
|
||||||
resp.NotAuth(c)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
h.db.First(&user, user.Id)
|
|
||||||
var userVo vo.User
|
|
||||||
err = utils.CopyObject(user, &userVo)
|
|
||||||
if err != nil {
|
|
||||||
logger.Error("对象拷贝失败:", err.Error())
|
|
||||||
resp.ERROR(c, "获取用户信息失败")
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
userVo.Id = user.Id
|
|
||||||
userVo.CreatedAt = user.CreatedAt.Unix()
|
|
||||||
userVo.UpdatedAt = user.UpdatedAt.Unix()
|
|
||||||
resp.SUCCESS(c, userVo)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Password 更新密码
|
// Password 更新密码
|
||||||
func (h *UserHandler) Password(c *gin.Context) {
|
func (h *UserHandler) Password(c *gin.Context) {
|
||||||
var data struct {
|
var data struct {
|
||||||
@@ -328,3 +343,47 @@ func (h *UserHandler) Password(c *gin.Context) {
|
|||||||
|
|
||||||
resp.SUCCESS(c)
|
resp.SUCCESS(c)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// BindMobile 绑定手机号
|
||||||
|
func (h *UserHandler) BindMobile(c *gin.Context) {
|
||||||
|
var data struct {
|
||||||
|
Mobile string `json:"mobile"`
|
||||||
|
Code int `json:"code"`
|
||||||
|
}
|
||||||
|
if err := c.ShouldBindJSON(&data); err != nil {
|
||||||
|
resp.ERROR(c, types.InvalidArgs)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// 检查手机号是否被其他账号绑定
|
||||||
|
var item model.User
|
||||||
|
res := h.db.Where("mobile = ?", data.Mobile).First(&item)
|
||||||
|
if res.Error == nil {
|
||||||
|
resp.ERROR(c, "该手机号已经被其他账号绑定")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// 检查验证码
|
||||||
|
key := CodeStorePrefix + data.Mobile
|
||||||
|
var code int
|
||||||
|
err := h.levelDB.Get(key, &code)
|
||||||
|
if err != nil || code != data.Code {
|
||||||
|
resp.ERROR(c, "短信验证码错误")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
user, err := utils.GetLoginUser(c, h.db)
|
||||||
|
if err != nil {
|
||||||
|
resp.NotAuth(c)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
res = h.db.Model(&user).UpdateColumn("mobile", data.Mobile)
|
||||||
|
if res.Error != nil {
|
||||||
|
resp.ERROR(c, "更新数据库失败")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
_ = h.levelDB.Delete(key) // 删除短信验证码
|
||||||
|
resp.SUCCESS(c)
|
||||||
|
}
|
||||||
@@ -1,5 +0,0 @@
|
|||||||
# chatgpt-plus-java
|
|
||||||
|
|
||||||
chatgpt-plus 后端 API Java 语言实现,待开发。
|
|
||||||
|
|
||||||
|
|
||||||
74
api/logger/logger.go
Normal file
74
api/logger/logger.go
Normal file
@@ -0,0 +1,74 @@
|
|||||||
|
package logger
|
||||||
|
|
||||||
|
import (
|
||||||
|
"go.uber.org/zap"
|
||||||
|
"go.uber.org/zap/zapcore"
|
||||||
|
"gopkg.in/natefinch/lumberjack.v2"
|
||||||
|
"os"
|
||||||
|
"strings"
|
||||||
|
)
|
||||||
|
|
||||||
|
var logger *zap.Logger
|
||||||
|
var sugarLogger *zap.SugaredLogger
|
||||||
|
|
||||||
|
func GetLogger() *zap.SugaredLogger {
|
||||||
|
if sugarLogger != nil {
|
||||||
|
return sugarLogger
|
||||||
|
}
|
||||||
|
|
||||||
|
logLevel := zap.NewAtomicLevelAt(getLogLevel(os.Getenv("LOG_LEVEL")))
|
||||||
|
encoder := getEncoder()
|
||||||
|
writerSyncer := getLogWriter()
|
||||||
|
fileCore := zapcore.NewCore(encoder, writerSyncer, logLevel)
|
||||||
|
consoleOutput := zapcore.Lock(os.Stdout)
|
||||||
|
consoleCore := zapcore.NewCore(
|
||||||
|
encoder,
|
||||||
|
consoleOutput,
|
||||||
|
logLevel,
|
||||||
|
)
|
||||||
|
core := zapcore.NewTee(fileCore, consoleCore)
|
||||||
|
logger = zap.New(core, zap.AddCaller())
|
||||||
|
sugarLogger = logger.Sugar()
|
||||||
|
return sugarLogger
|
||||||
|
}
|
||||||
|
|
||||||
|
// core 三个参数之 编码
|
||||||
|
func getEncoder() zapcore.Encoder {
|
||||||
|
encoderConfig := zapcore.EncoderConfig{
|
||||||
|
TimeKey: "time",
|
||||||
|
LevelKey: "level",
|
||||||
|
NameKey: "logger",
|
||||||
|
CallerKey: "caller",
|
||||||
|
MessageKey: "msg",
|
||||||
|
StacktraceKey: "stacktrace",
|
||||||
|
EncodeTime: zapcore.ISO8601TimeEncoder,
|
||||||
|
EncodeDuration: zapcore.SecondsDurationEncoder,
|
||||||
|
EncodeCaller: zapcore.ShortCallerEncoder,
|
||||||
|
EncodeLevel: zapcore.CapitalLevelEncoder,
|
||||||
|
}
|
||||||
|
return zapcore.NewConsoleEncoder(encoderConfig)
|
||||||
|
}
|
||||||
|
|
||||||
|
func getLogWriter() zapcore.WriteSyncer {
|
||||||
|
lumberJackLogger := &lumberjack.Logger{
|
||||||
|
Filename: "logs/app.log",
|
||||||
|
MaxSize: 10,
|
||||||
|
MaxBackups: 5,
|
||||||
|
MaxAge: 30,
|
||||||
|
Compress: false,
|
||||||
|
}
|
||||||
|
return zapcore.AddSync(lumberJackLogger)
|
||||||
|
}
|
||||||
|
|
||||||
|
func getLogLevel(level string) zapcore.Level {
|
||||||
|
switch strings.ToUpper(level) {
|
||||||
|
case "DEBUG":
|
||||||
|
return zapcore.DebugLevel
|
||||||
|
case "WARN":
|
||||||
|
return zapcore.WarnLevel
|
||||||
|
case "ERROR":
|
||||||
|
return zapcore.ErrorLevel
|
||||||
|
default:
|
||||||
|
return zapcore.InfoLevel
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -6,15 +6,17 @@ import (
|
|||||||
"chatplus/handler"
|
"chatplus/handler"
|
||||||
"chatplus/handler/admin"
|
"chatplus/handler/admin"
|
||||||
logger2 "chatplus/logger"
|
logger2 "chatplus/logger"
|
||||||
|
"chatplus/modules/wexin"
|
||||||
|
"chatplus/service"
|
||||||
|
"chatplus/service/function"
|
||||||
"chatplus/store"
|
"chatplus/store"
|
||||||
"context"
|
"context"
|
||||||
"embed"
|
"embed"
|
||||||
"flag"
|
|
||||||
"fmt"
|
|
||||||
"io"
|
"io"
|
||||||
"log"
|
"log"
|
||||||
"os"
|
"os"
|
||||||
"os/signal"
|
"os/signal"
|
||||||
|
"strconv"
|
||||||
"syscall"
|
"syscall"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
@@ -24,8 +26,6 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
var logger = logger2.GetLogger()
|
var logger = logger2.GetLogger()
|
||||||
var configFile string
|
|
||||||
var debugMode bool
|
|
||||||
|
|
||||||
//go:embed res/ip2region.xdb
|
//go:embed res/ip2region.xdb
|
||||||
var xdbFS embed.FS
|
var xdbFS embed.FS
|
||||||
@@ -47,7 +47,24 @@ func (l *AppLifecycle) OnStop(context.Context) error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
|
configFile := os.Getenv("CONFIG_FILE")
|
||||||
|
if configFile == "" {
|
||||||
|
configFile = "config.toml"
|
||||||
|
}
|
||||||
|
var debug bool
|
||||||
|
debugEnv := os.Getenv("DEBUG")
|
||||||
|
if debugEnv == "" {
|
||||||
|
debug = true
|
||||||
|
} else {
|
||||||
|
debug, _ = strconv.ParseBool(os.Getenv("DEBUG"))
|
||||||
|
}
|
||||||
logger.Info("Loading config file: ", configFile)
|
logger.Info("Loading config file: ", configFile)
|
||||||
|
defer func() {
|
||||||
|
if err := recover(); err != nil {
|
||||||
|
logger.Error("Panic Error:", err)
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
app := fx.New(
|
app := fx.New(
|
||||||
// 初始化配置应用配置
|
// 初始化配置应用配置
|
||||||
fx.Provide(func() *types.AppConfig {
|
fx.Provide(func() *types.AppConfig {
|
||||||
@@ -55,13 +72,17 @@ func main() {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatal(err)
|
log.Fatal(err)
|
||||||
}
|
}
|
||||||
|
config.Path = configFile
|
||||||
|
if debug {
|
||||||
|
_ = core.SaveConfig(config)
|
||||||
|
}
|
||||||
return config
|
return config
|
||||||
}),
|
}),
|
||||||
// 创建应用服务
|
// 创建应用服务
|
||||||
fx.Provide(core.NewServer),
|
fx.Provide(core.NewServer),
|
||||||
// 初始化
|
// 初始化
|
||||||
fx.Invoke(func(s *core.AppServer) {
|
fx.Invoke(func(s *core.AppServer) {
|
||||||
s.Init(debugMode)
|
s.Init(debug)
|
||||||
}),
|
}),
|
||||||
|
|
||||||
// 初始化数据库
|
// 初始化数据库
|
||||||
@@ -83,16 +104,50 @@ func main() {
|
|||||||
return xdb.NewWithBuffer(cBuff)
|
return xdb.NewWithBuffer(cBuff)
|
||||||
}),
|
}),
|
||||||
|
|
||||||
|
// 创建微信机器人
|
||||||
|
fx.Provide(wexin.NewWeChatBot),
|
||||||
|
fx.Invoke(func(bot *wexin.WeChatBot) {
|
||||||
|
go func() {
|
||||||
|
err := bot.Login()
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
}),
|
||||||
|
|
||||||
|
// 创建函数
|
||||||
|
fx.Provide(func(config *types.AppConfig) (function.FuncZaoBao, error) {
|
||||||
|
return function.NewZaoBao(config.ApiConfig), nil
|
||||||
|
}),
|
||||||
|
fx.Provide(func(config *types.AppConfig) (function.FuncWeiboHot, error) {
|
||||||
|
return function.NewWeiboHot(config.ApiConfig), nil
|
||||||
|
}),
|
||||||
|
fx.Provide(func(config *types.AppConfig) (function.FuncHeadlines, error) {
|
||||||
|
return function.NewHeadLines(config.ApiConfig), nil
|
||||||
|
}),
|
||||||
|
|
||||||
// 创建控制器
|
// 创建控制器
|
||||||
fx.Provide(handler.NewChatRoleHandler),
|
fx.Provide(handler.NewChatRoleHandler),
|
||||||
fx.Provide(handler.NewUserHandler),
|
fx.Provide(handler.NewUserHandler),
|
||||||
fx.Provide(handler.NewChatHandler),
|
fx.Provide(handler.NewChatHandler),
|
||||||
fx.Provide(admin.NewConfigHandler),
|
fx.Provide(handler.NewUploadHandler),
|
||||||
|
fx.Provide(handler.NewSmsHandler),
|
||||||
|
fx.Provide(handler.NewRewardHandler),
|
||||||
|
fx.Provide(handler.NewCaptchaHandler),
|
||||||
|
|
||||||
|
fx.Provide(admin.NewConfigHandler),
|
||||||
fx.Provide(admin.NewAdminHandler),
|
fx.Provide(admin.NewAdminHandler),
|
||||||
fx.Provide(admin.NewApiKeyHandler),
|
fx.Provide(admin.NewApiKeyHandler),
|
||||||
fx.Provide(admin.NewUserHandler),
|
fx.Provide(admin.NewUserHandler),
|
||||||
fx.Provide(admin.NewChatRoleHandler),
|
fx.Provide(admin.NewChatRoleHandler),
|
||||||
|
fx.Provide(admin.NewRewardHandler),
|
||||||
|
fx.Provide(admin.NewDashboardHandler),
|
||||||
|
|
||||||
|
// 创建服务
|
||||||
|
fx.Provide(service.NewAliYunSmsService),
|
||||||
|
fx.Provide(func(config *types.AppConfig) *service.CaptchaService {
|
||||||
|
return service.NewCaptchaService(config.ApiConfig)
|
||||||
|
}),
|
||||||
|
|
||||||
// 注册路由
|
// 注册路由
|
||||||
fx.Invoke(func(s *core.AppServer, h *handler.ChatRoleHandler) {
|
fx.Invoke(func(s *core.AppServer, h *handler.ChatRoleHandler) {
|
||||||
@@ -108,6 +163,7 @@ func main() {
|
|||||||
group.GET("profile", h.Profile)
|
group.GET("profile", h.Profile)
|
||||||
group.POST("profile/update", h.ProfileUpdate)
|
group.POST("profile/update", h.ProfileUpdate)
|
||||||
group.POST("password", h.Password)
|
group.POST("password", h.Password)
|
||||||
|
group.POST("bind/mobile", h.BindMobile)
|
||||||
}),
|
}),
|
||||||
fx.Invoke(func(s *core.AppServer, h *handler.ChatHandler) {
|
fx.Invoke(func(s *core.AppServer, h *handler.ChatHandler) {
|
||||||
group := s.Engine.Group("/api/chat/")
|
group := s.Engine.Group("/api/chat/")
|
||||||
@@ -120,8 +176,25 @@ func main() {
|
|||||||
group.GET("tokens", h.Tokens)
|
group.GET("tokens", h.Tokens)
|
||||||
group.GET("stop", h.StopGenerate)
|
group.GET("stop", h.StopGenerate)
|
||||||
}),
|
}),
|
||||||
|
fx.Invoke(func(s *core.AppServer, h *handler.UploadHandler) {
|
||||||
|
s.Engine.POST("/api/upload", h.Upload)
|
||||||
|
}),
|
||||||
|
fx.Invoke(func(s *core.AppServer, h *handler.SmsHandler) {
|
||||||
|
group := s.Engine.Group("/api/sms/")
|
||||||
|
group.GET("status", h.Status)
|
||||||
|
group.POST("code", h.SendCode)
|
||||||
|
}),
|
||||||
|
fx.Invoke(func(s *core.AppServer, h *handler.CaptchaHandler) {
|
||||||
|
group := s.Engine.Group("/api/captcha/")
|
||||||
|
group.GET("get", h.Get)
|
||||||
|
group.POST("check", h.Check)
|
||||||
|
}),
|
||||||
|
fx.Invoke(func(s *core.AppServer, h *handler.RewardHandler) {
|
||||||
|
group := s.Engine.Group("/api/reward/")
|
||||||
|
group.POST("verify", h.Verify)
|
||||||
|
}),
|
||||||
|
|
||||||
//
|
// 管理后台控制器
|
||||||
fx.Invoke(func(s *core.AppServer, h *admin.ConfigHandler) {
|
fx.Invoke(func(s *core.AppServer, h *admin.ConfigHandler) {
|
||||||
group := s.Engine.Group("/api/admin/config/")
|
group := s.Engine.Group("/api/admin/config/")
|
||||||
group.POST("update", h.Update)
|
group.POST("update", h.Update)
|
||||||
@@ -143,9 +216,10 @@ func main() {
|
|||||||
fx.Invoke(func(s *core.AppServer, h *admin.UserHandler) {
|
fx.Invoke(func(s *core.AppServer, h *admin.UserHandler) {
|
||||||
group := s.Engine.Group("/api/admin/user/")
|
group := s.Engine.Group("/api/admin/user/")
|
||||||
group.GET("list", h.List)
|
group.GET("list", h.List)
|
||||||
group.POST("update", h.Update)
|
group.POST("save", h.Save)
|
||||||
group.GET("remove", h.Remove)
|
group.GET("remove", h.Remove)
|
||||||
group.GET("loginLog", h.LoginLog)
|
group.GET("loginLog", h.LoginLog)
|
||||||
|
group.POST("resetPass", h.ResetPass)
|
||||||
}),
|
}),
|
||||||
fx.Invoke(func(s *core.AppServer, h *admin.ChatRoleHandler) {
|
fx.Invoke(func(s *core.AppServer, h *admin.ChatRoleHandler) {
|
||||||
group := s.Engine.Group("/api/admin/role/")
|
group := s.Engine.Group("/api/admin/role/")
|
||||||
@@ -154,6 +228,14 @@ func main() {
|
|||||||
group.POST("sort", h.SetSort)
|
group.POST("sort", h.SetSort)
|
||||||
group.GET("remove", h.Remove)
|
group.GET("remove", h.Remove)
|
||||||
}),
|
}),
|
||||||
|
fx.Invoke(func(s *core.AppServer, h *admin.RewardHandler) {
|
||||||
|
group := s.Engine.Group("/api/admin/reward/")
|
||||||
|
group.GET("list", h.List)
|
||||||
|
}),
|
||||||
|
fx.Invoke(func(s *core.AppServer, h *admin.DashboardHandler) {
|
||||||
|
group := s.Engine.Group("/api/admin/dashboard/")
|
||||||
|
group.GET("stats", h.Stats)
|
||||||
|
}),
|
||||||
|
|
||||||
fx.Invoke(func(s *core.AppServer, db *gorm.DB) {
|
fx.Invoke(func(s *core.AppServer, db *gorm.DB) {
|
||||||
err := s.Run(db)
|
err := s.Run(db)
|
||||||
@@ -194,25 +276,3 @@ func main() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func init() {
|
|
||||||
flag.StringVar(&configFile, "config", "config.toml", "AppConfig file path (default: config.toml)")
|
|
||||||
flag.BoolVar(&debugMode, "debug", true, "Enable debug mode (default: true, recommend to set false in production env)")
|
|
||||||
flag.Usage = usage
|
|
||||||
flag.Parse()
|
|
||||||
}
|
|
||||||
|
|
||||||
func usage() {
|
|
||||||
fmt.Printf(`ChatGPT-Web-Plus, Version: 2.0.0
|
|
||||||
USAGE:
|
|
||||||
%s [command options]
|
|
||||||
OPTIONS:
|
|
||||||
`, os.Args[0])
|
|
||||||
|
|
||||||
flagSet := flag.CommandLine
|
|
||||||
order := []string{"config", "debug"}
|
|
||||||
for _, name := range order {
|
|
||||||
f := flagSet.Lookup(name)
|
|
||||||
fmt.Printf(" --%s => %s\n", f.Name, f.Usage)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
54
api/modules/wexin/handler.go
Normal file
54
api/modules/wexin/handler.go
Normal file
@@ -0,0 +1,54 @@
|
|||||||
|
package wexin
|
||||||
|
|
||||||
|
import (
|
||||||
|
"chatplus/store/model"
|
||||||
|
"github.com/eatmoreapple/openwechat"
|
||||||
|
"github.com/skip2/go-qrcode"
|
||||||
|
"gorm.io/gorm"
|
||||||
|
)
|
||||||
|
|
||||||
|
// MessageHandler 消息处理
|
||||||
|
func MessageHandler(msg *openwechat.Message, db *gorm.DB) {
|
||||||
|
sender, err := msg.Sender()
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// 只处理微信支付的推送消息
|
||||||
|
if sender.NickName == "微信支付" ||
|
||||||
|
msg.MsgType == openwechat.MsgTypeApp ||
|
||||||
|
msg.AppMsgType == openwechat.AppMsgTypeUrl {
|
||||||
|
// 解析支付金额
|
||||||
|
message, err := parseTransactionMessage(msg.Content)
|
||||||
|
if err == nil {
|
||||||
|
transaction := extractTransaction(message)
|
||||||
|
logger.Infof("解析到收款信息:%+v", transaction)
|
||||||
|
if transaction.Amount <= 0 {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
var item model.Reward
|
||||||
|
res := db.Where("tx_id = ?", transaction.TransId).First(&item)
|
||||||
|
if res.Error == nil {
|
||||||
|
logger.Infof("当前交易 ID %s 己经存在!", transaction.TransId)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
res = db.Create(&model.Reward{
|
||||||
|
TxId: transaction.TransId,
|
||||||
|
Amount: transaction.Amount,
|
||||||
|
Remark: transaction.Remark,
|
||||||
|
Status: false,
|
||||||
|
})
|
||||||
|
if res.Error != nil {
|
||||||
|
logger.Errorf("交易保存失败,ID: %s", transaction.TransId)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// QrCodeCallBack 登录扫码回调,
|
||||||
|
func QrCodeCallBack(uuid string) {
|
||||||
|
logger.Info("请使用微信扫描下面二维码登录")
|
||||||
|
q, _ := qrcode.New("https://login.weixin.qq.com/l/"+uuid, qrcode.Medium)
|
||||||
|
logger.Info(q.ToString(true))
|
||||||
|
}
|
||||||
68
api/modules/wexin/tranaction.go
Normal file
68
api/modules/wexin/tranaction.go
Normal file
@@ -0,0 +1,68 @@
|
|||||||
|
package wexin
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/xml"
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Message 转账消息
|
||||||
|
type Message struct {
|
||||||
|
XMLName xml.Name `xml:"msg"`
|
||||||
|
AppMsg struct {
|
||||||
|
Des string `xml:"des"`
|
||||||
|
Url string `xml:"url"`
|
||||||
|
} `xml:"appmsg"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// Transaction 解析后的交易信息
|
||||||
|
type Transaction struct {
|
||||||
|
TransId string `json:"trans_id"` // 微信转账交易 ID
|
||||||
|
Amount float64 `json:"amount"` // 微信转账交易金额
|
||||||
|
Remark string `json:"remark"` // 转账备注
|
||||||
|
}
|
||||||
|
|
||||||
|
// 解析微信转账消息
|
||||||
|
func parseTransactionMessage(xmlData string) (*Message, error) {
|
||||||
|
var msg Message
|
||||||
|
if err := xml.Unmarshal([]byte(xmlData), &msg); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return &msg, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// 导出交易信息
|
||||||
|
func extractTransaction(message *Message) Transaction {
|
||||||
|
var tx = Transaction{}
|
||||||
|
// 导出交易金额和备注
|
||||||
|
lines := strings.Split(message.AppMsg.Des, "\n")
|
||||||
|
for _, line := range lines {
|
||||||
|
line = strings.TrimSpace(line)
|
||||||
|
if len(line) == 0 {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
// 解析收款金额
|
||||||
|
prefix := "收款金额¥"
|
||||||
|
if strings.HasPrefix(line, prefix) {
|
||||||
|
if value, err := strconv.ParseFloat(line[len(prefix):], 64); err == nil {
|
||||||
|
tx.Amount = value
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// 解析收款备注
|
||||||
|
prefix = "付款方备注"
|
||||||
|
if strings.HasPrefix(line, prefix) {
|
||||||
|
tx.Remark = line[len(prefix):]
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 解析交易 ID
|
||||||
|
index := strings.Index(message.AppMsg.Url, "trans_id=")
|
||||||
|
if index != -1 {
|
||||||
|
end := strings.LastIndex(message.AppMsg.Url, "&")
|
||||||
|
tx.TransId = strings.TrimSpace(message.AppMsg.Url[index+9 : end])
|
||||||
|
}
|
||||||
|
return tx
|
||||||
|
}
|
||||||
49
api/modules/wexin/wechat_bot.go
Normal file
49
api/modules/wexin/wechat_bot.go
Normal file
@@ -0,0 +1,49 @@
|
|||||||
|
package wexin
|
||||||
|
|
||||||
|
import (
|
||||||
|
"chatplus/core/types"
|
||||||
|
logger2 "chatplus/logger"
|
||||||
|
"github.com/eatmoreapple/openwechat"
|
||||||
|
"gorm.io/gorm"
|
||||||
|
)
|
||||||
|
|
||||||
|
// 微信收款机器人服务
|
||||||
|
var logger = logger2.GetLogger()
|
||||||
|
|
||||||
|
type WeChatBot struct {
|
||||||
|
bot *openwechat.Bot
|
||||||
|
db *gorm.DB
|
||||||
|
appConfig *types.AppConfig
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewWeChatBot(db *gorm.DB, config *types.AppConfig) *WeChatBot {
|
||||||
|
bot := openwechat.DefaultBot(openwechat.Desktop)
|
||||||
|
// 注册消息处理函数
|
||||||
|
bot.MessageHandler = func(msg *openwechat.Message) {
|
||||||
|
MessageHandler(msg, db)
|
||||||
|
}
|
||||||
|
// 注册登陆二维码回调
|
||||||
|
bot.UUIDCallback = QrCodeCallBack
|
||||||
|
return &WeChatBot{
|
||||||
|
bot: bot,
|
||||||
|
db: db,
|
||||||
|
appConfig: config,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *WeChatBot) Login() error {
|
||||||
|
if !b.appConfig.StartWechatBot {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// 创建热存储容器对象
|
||||||
|
reloadStorage := openwechat.NewJsonFileHotReloadStorage("storage.json")
|
||||||
|
// 执行热登录
|
||||||
|
err := b.bot.HotLogin(reloadStorage)
|
||||||
|
if err != nil {
|
||||||
|
logger.Error("login error: %v", err)
|
||||||
|
return b.bot.Login()
|
||||||
|
}
|
||||||
|
logger.Info("微信登录成功!")
|
||||||
|
return nil
|
||||||
|
}
|
||||||
@@ -1,5 +0,0 @@
|
|||||||
# chatgpt-plus-php
|
|
||||||
|
|
||||||
chatgpt-plus 后端 API PHP 语言实现,待开发。
|
|
||||||
|
|
||||||
|
|
||||||
@@ -1,5 +0,0 @@
|
|||||||
# chatgpt-plus-python
|
|
||||||
|
|
||||||
chatgpt-plus 后端 API Python 语言实现,待开发。
|
|
||||||
|
|
||||||
|
|
||||||
54
api/service/aliyun_sms_service.go
Normal file
54
api/service/aliyun_sms_service.go
Normal file
@@ -0,0 +1,54 @@
|
|||||||
|
package service
|
||||||
|
|
||||||
|
import (
|
||||||
|
"chatplus/core/types"
|
||||||
|
"chatplus/store"
|
||||||
|
"fmt"
|
||||||
|
"github.com/aliyun/alibaba-cloud-sdk-go/services/dysmsapi"
|
||||||
|
)
|
||||||
|
|
||||||
|
type AliYunSmsService struct {
|
||||||
|
config *types.AppConfig
|
||||||
|
db *store.LevelDB
|
||||||
|
client *dysmsapi.Client
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewAliYunSmsService(config *types.AppConfig, db *store.LevelDB) (*AliYunSmsService, error) {
|
||||||
|
// 创建阿里云短信客户端
|
||||||
|
client, err := dysmsapi.NewClientWithAccessKey(
|
||||||
|
"cn-hangzhou",
|
||||||
|
config.SmsConfig.AccessKey,
|
||||||
|
config.SmsConfig.AccessSecret)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to create client: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return &AliYunSmsService{
|
||||||
|
config: config,
|
||||||
|
db: db,
|
||||||
|
client: client,
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *AliYunSmsService) SendVerifyCode(mobile string, code int) error {
|
||||||
|
// 创建短信请求并设置参数
|
||||||
|
request := dysmsapi.CreateSendSmsRequest()
|
||||||
|
request.Scheme = "https"
|
||||||
|
request.Domain = s.config.SmsConfig.Domain
|
||||||
|
request.PhoneNumbers = mobile
|
||||||
|
request.SignName = "飞行的蜗牛"
|
||||||
|
request.TemplateCode = "SMS_281460317"
|
||||||
|
request.TemplateParam = fmt.Sprintf("{\"code\":\"%d\"}", code) // 短信模板中的参数
|
||||||
|
|
||||||
|
// 发送短信
|
||||||
|
response, err := s.client.SendSms(request)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("failed to send SMS:%v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if response.Code != "OK" {
|
||||||
|
return fmt.Errorf("failed to send SMS:%v", response.Message)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
62
api/service/captcha_service.go
Normal file
62
api/service/captcha_service.go
Normal file
@@ -0,0 +1,62 @@
|
|||||||
|
package service
|
||||||
|
|
||||||
|
import (
|
||||||
|
"chatplus/core/types"
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"github.com/imroc/req/v3"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
type CaptchaService struct {
|
||||||
|
config types.ChatPlusApiConfig
|
||||||
|
client *req.Client
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewCaptchaService(config types.ChatPlusApiConfig) *CaptchaService {
|
||||||
|
return &CaptchaService{
|
||||||
|
config: config,
|
||||||
|
client: req.C().SetTimeout(10 * time.Second),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *CaptchaService) Get() (interface{}, error) {
|
||||||
|
if s.config.Token == "" {
|
||||||
|
return nil, errors.New("无效的 API Token")
|
||||||
|
}
|
||||||
|
|
||||||
|
url := fmt.Sprintf("%s/api/captcha/get", s.config.ApiURL)
|
||||||
|
var res types.BizVo
|
||||||
|
r, err := s.client.R().
|
||||||
|
SetHeader("AppId", s.config.AppId).
|
||||||
|
SetHeader("Authorization", fmt.Sprintf("Bearer %s", s.config.Token)).
|
||||||
|
SetSuccessResult(&res).Get(url)
|
||||||
|
if err != nil || r.IsErrorState() {
|
||||||
|
return nil, fmt.Errorf("请求 API 失败:%v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if res.Code != types.Success {
|
||||||
|
return nil, fmt.Errorf("请求 API 失败:%s", res.Message)
|
||||||
|
}
|
||||||
|
|
||||||
|
return res.Data, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *CaptchaService) Check(data interface{}) bool {
|
||||||
|
url := fmt.Sprintf("%s/api/captcha/check", s.config.ApiURL)
|
||||||
|
var res types.BizVo
|
||||||
|
r, err := s.client.R().
|
||||||
|
SetHeader("AppId", s.config.AppId).
|
||||||
|
SetHeader("Authorization", fmt.Sprintf("Bearer %s", s.config.Token)).
|
||||||
|
SetBodyJsonMarshal(data).
|
||||||
|
SetSuccessResult(&res).Post(url)
|
||||||
|
if err != nil || r.IsErrorState() {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
if res.Code != types.Success {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
return true
|
||||||
|
}
|
||||||
24
api/service/function/function.go
Normal file
24
api/service/function/function.go
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
package function
|
||||||
|
|
||||||
|
import "chatplus/core/types"
|
||||||
|
|
||||||
|
type Function interface {
|
||||||
|
Invoke(...interface{}) (string, error)
|
||||||
|
Name() string
|
||||||
|
}
|
||||||
|
|
||||||
|
type resVo struct {
|
||||||
|
Code types.BizCode `json:"code"`
|
||||||
|
Message string `json:"message"`
|
||||||
|
Data struct {
|
||||||
|
Title string `json:"title"`
|
||||||
|
UpdatedAt string `json:"updated_at"`
|
||||||
|
Items []dataItem `json:"items"`
|
||||||
|
} `json:"data"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type dataItem struct {
|
||||||
|
Title string `json:"title"`
|
||||||
|
Url string `json:"url"`
|
||||||
|
Remark string `json:"remark"`
|
||||||
|
}
|
||||||
59
api/service/function/tou_tiao.go
Normal file
59
api/service/function/tou_tiao.go
Normal file
@@ -0,0 +1,59 @@
|
|||||||
|
package function
|
||||||
|
|
||||||
|
import (
|
||||||
|
"chatplus/core/types"
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"github.com/imroc/req/v3"
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
// 今日头条函数实现
|
||||||
|
|
||||||
|
type FuncHeadlines struct {
|
||||||
|
name string
|
||||||
|
config types.ChatPlusApiConfig
|
||||||
|
client *req.Client
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewHeadLines(config types.ChatPlusApiConfig) FuncHeadlines {
|
||||||
|
return FuncHeadlines{
|
||||||
|
name: "今日头条",
|
||||||
|
config: config,
|
||||||
|
client: req.C().SetTimeout(10 * time.Second)}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (f FuncHeadlines) Invoke(...interface{}) (string, error) {
|
||||||
|
if f.config.Token == "" {
|
||||||
|
return "", errors.New("无效的 API Token")
|
||||||
|
}
|
||||||
|
|
||||||
|
url := fmt.Sprintf("%s/api/headline/fetch", f.config.ApiURL)
|
||||||
|
var res resVo
|
||||||
|
r, err := f.client.R().
|
||||||
|
SetHeader("AppId", f.config.AppId).
|
||||||
|
SetHeader("Authorization", fmt.Sprintf("Bearer %s", f.config.Token)).
|
||||||
|
SetSuccessResult(&res).Get(url)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
if r.IsErrorState() {
|
||||||
|
return "", r.Err
|
||||||
|
}
|
||||||
|
|
||||||
|
if res.Code != types.Success {
|
||||||
|
return "", errors.New(res.Message)
|
||||||
|
}
|
||||||
|
|
||||||
|
builder := make([]string, 0)
|
||||||
|
builder = append(builder, fmt.Sprintf("**%s**,最新更新:%s", res.Data.Title, res.Data.UpdatedAt))
|
||||||
|
for i, v := range res.Data.Items {
|
||||||
|
builder = append(builder, fmt.Sprintf("%d、 [%s](%s) [%s]", i+1, v.Title, v.Url, v.Remark))
|
||||||
|
}
|
||||||
|
return strings.Join(builder, "\n\n"), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (f FuncHeadlines) Name() string {
|
||||||
|
return f.name
|
||||||
|
}
|
||||||
59
api/service/function/weibo_hot.go
Normal file
59
api/service/function/weibo_hot.go
Normal file
@@ -0,0 +1,59 @@
|
|||||||
|
package function
|
||||||
|
|
||||||
|
import (
|
||||||
|
"chatplus/core/types"
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"github.com/imroc/req/v3"
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
// 微博热搜函数实现
|
||||||
|
|
||||||
|
type FuncWeiboHot struct {
|
||||||
|
name string
|
||||||
|
config types.ChatPlusApiConfig
|
||||||
|
client *req.Client
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewWeiboHot(config types.ChatPlusApiConfig) FuncWeiboHot {
|
||||||
|
return FuncWeiboHot{
|
||||||
|
name: "微博热搜",
|
||||||
|
config: config,
|
||||||
|
client: req.C().SetTimeout(10 * time.Second)}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (f FuncWeiboHot) Invoke(...interface{}) (string, error) {
|
||||||
|
if f.config.Token == "" {
|
||||||
|
return "", errors.New("无效的 API Token")
|
||||||
|
}
|
||||||
|
|
||||||
|
url := fmt.Sprintf("%s/api/weibo/fetch", f.config.ApiURL)
|
||||||
|
var res resVo
|
||||||
|
r, err := f.client.R().
|
||||||
|
SetHeader("AppId", f.config.AppId).
|
||||||
|
SetHeader("Authorization", fmt.Sprintf("Bearer %s", f.config.Token)).
|
||||||
|
SetSuccessResult(&res).Get(url)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
if r.IsErrorState() {
|
||||||
|
return "", r.Err
|
||||||
|
}
|
||||||
|
|
||||||
|
if res.Code != types.Success {
|
||||||
|
return "", errors.New(res.Message)
|
||||||
|
}
|
||||||
|
|
||||||
|
builder := make([]string, 0)
|
||||||
|
builder = append(builder, fmt.Sprintf("**%s**,最新更新:%s", res.Data.Title, res.Data.UpdatedAt))
|
||||||
|
for i, v := range res.Data.Items {
|
||||||
|
builder = append(builder, fmt.Sprintf("%d、 [%s](%s) [热度:%s]", i+1, v.Title, v.Url, v.Remark))
|
||||||
|
}
|
||||||
|
return strings.Join(builder, "\n\n"), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (f FuncWeiboHot) Name() string {
|
||||||
|
return f.name
|
||||||
|
}
|
||||||
60
api/service/function/zao_bao.go
Normal file
60
api/service/function/zao_bao.go
Normal file
@@ -0,0 +1,60 @@
|
|||||||
|
package function
|
||||||
|
|
||||||
|
import (
|
||||||
|
"chatplus/core/types"
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"github.com/imroc/req/v3"
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
// 每日早报函数实现
|
||||||
|
|
||||||
|
type FuncZaoBao struct {
|
||||||
|
name string
|
||||||
|
config types.ChatPlusApiConfig
|
||||||
|
client *req.Client
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewZaoBao(config types.ChatPlusApiConfig) FuncZaoBao {
|
||||||
|
return FuncZaoBao{
|
||||||
|
name: "每日早报",
|
||||||
|
config: config,
|
||||||
|
client: req.C().SetTimeout(10 * time.Second)}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (f FuncZaoBao) Invoke(...interface{}) (string, error) {
|
||||||
|
if f.config.Token == "" {
|
||||||
|
return "", errors.New("无效的 API Token")
|
||||||
|
}
|
||||||
|
|
||||||
|
url := fmt.Sprintf("%s/api/zaobao/fetch", f.config.ApiURL)
|
||||||
|
var res resVo
|
||||||
|
r, err := f.client.R().
|
||||||
|
SetHeader("AppId", f.config.AppId).
|
||||||
|
SetHeader("Authorization", fmt.Sprintf("Bearer %s", f.config.Token)).
|
||||||
|
SetSuccessResult(&res).Get(url)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
if r.IsErrorState() {
|
||||||
|
return "", r.Err
|
||||||
|
}
|
||||||
|
|
||||||
|
if res.Code != types.Success {
|
||||||
|
return "", errors.New(res.Message)
|
||||||
|
}
|
||||||
|
|
||||||
|
builder := make([]string, 0)
|
||||||
|
builder = append(builder, fmt.Sprintf("**%s 早报:**", res.Data.UpdatedAt))
|
||||||
|
for _, v := range res.Data.Items {
|
||||||
|
builder = append(builder, v.Title)
|
||||||
|
}
|
||||||
|
builder = append(builder, fmt.Sprintf("%s", res.Data.Title))
|
||||||
|
return strings.Join(builder, "\n\n"), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (f FuncZaoBao) Name() string {
|
||||||
|
return f.name
|
||||||
|
}
|
||||||
5
api/service/sms_service.go
Normal file
5
api/service/sms_service.go
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
package service
|
||||||
|
|
||||||
|
type SmsService interface {
|
||||||
|
SendVerifyCode(mobile string, code int) error
|
||||||
|
}
|
||||||
1
api/static/hello.txt
Normal file
1
api/static/hello.txt
Normal file
@@ -0,0 +1 @@
|
|||||||
|
hello, world!
|
||||||
@@ -3,7 +3,6 @@ package store
|
|||||||
import (
|
import (
|
||||||
"chatplus/store/vo"
|
"chatplus/store/vo"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
|
|
||||||
"github.com/syndtr/goleveldb/leveldb"
|
"github.com/syndtr/goleveldb/leveldb"
|
||||||
"github.com/syndtr/goleveldb/leveldb/util"
|
"github.com/syndtr/goleveldb/leveldb/util"
|
||||||
)
|
)
|
||||||
@@ -30,13 +29,13 @@ func (db *LevelDB) Put(key string, value interface{}) error {
|
|||||||
return db.driver.Put([]byte(key), bytes, nil)
|
return db.driver.Put([]byte(key), bytes, nil)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (db *LevelDB) Get(key string) ([]byte, error) {
|
func (db *LevelDB) Get(key string, value interface{}) error {
|
||||||
bytes, err := db.driver.Get([]byte(key), nil)
|
bytes, err := db.driver.Get([]byte(key), nil)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
return bytes, nil
|
return json.Unmarshal(bytes, &value)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (db *LevelDB) Search(prefix string) []string {
|
func (db *LevelDB) Search(prefix string) []string {
|
||||||
17
api/store/model/chat_history.go
Normal file
17
api/store/model/chat_history.go
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
package model
|
||||||
|
|
||||||
|
type HistoryMessage struct {
|
||||||
|
BaseModel
|
||||||
|
ChatId string // 会话 ID
|
||||||
|
UserId uint // 用户 ID
|
||||||
|
RoleId uint // 角色 ID
|
||||||
|
Type string
|
||||||
|
Icon string
|
||||||
|
Tokens int
|
||||||
|
Content string
|
||||||
|
UseContext bool // 是否可以作为聊天上下文
|
||||||
|
}
|
||||||
|
|
||||||
|
func (HistoryMessage) TableName() string {
|
||||||
|
return "chatgpt_chat_history"
|
||||||
|
}
|
||||||
12
api/store/model/reward.go
Normal file
12
api/store/model/reward.go
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
package model
|
||||||
|
|
||||||
|
// 用户打赏
|
||||||
|
|
||||||
|
type Reward struct {
|
||||||
|
BaseModel
|
||||||
|
UserId uint // 用户 ID
|
||||||
|
TxId string // 交易ID
|
||||||
|
Amount float64 // 打赏金额
|
||||||
|
Remark string // 打赏备注
|
||||||
|
Status bool // 核销状态
|
||||||
|
}
|
||||||
@@ -3,6 +3,7 @@ package model
|
|||||||
type User struct {
|
type User struct {
|
||||||
BaseModel
|
BaseModel
|
||||||
Username string `gorm:"index:username,unique"`
|
Username string `gorm:"index:username,unique"`
|
||||||
|
Mobile string
|
||||||
Password string
|
Password string
|
||||||
Nickname string
|
Nickname string
|
||||||
Avatar string
|
Avatar string
|
||||||
17
api/store/vo/chat_history.go
Normal file
17
api/store/vo/chat_history.go
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
package vo
|
||||||
|
|
||||||
|
type HistoryMessage struct {
|
||||||
|
BaseVo
|
||||||
|
ChatId string `json:"chat_id"`
|
||||||
|
UserId uint `json:"user_id"`
|
||||||
|
RoleId uint `json:"role_id"`
|
||||||
|
Type string `json:"type"`
|
||||||
|
Icon string `json:"icon"`
|
||||||
|
Tokens int `json:"tokens"`
|
||||||
|
Content string `json:"content"`
|
||||||
|
UseContext bool `json:"use_context"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func (HistoryMessage) TableName() string {
|
||||||
|
return "chatgpt_chat_history"
|
||||||
|
}
|
||||||
11
api/store/vo/reward.go
Normal file
11
api/store/vo/reward.go
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
package vo
|
||||||
|
|
||||||
|
type Reward struct {
|
||||||
|
BaseVo
|
||||||
|
UserId uint `json:"user_id"` // 用户 ID
|
||||||
|
Username string `json:"username"`
|
||||||
|
TxId string `json:"tx_id"` // 交易ID
|
||||||
|
Amount float64 `json:"amount"` // 打赏金额
|
||||||
|
Remark string `json:"remark"` // 打赏备注
|
||||||
|
Status bool `json:"status"` // 核销状态
|
||||||
|
}
|
||||||
@@ -5,6 +5,7 @@ import "chatplus/core/types"
|
|||||||
type User struct {
|
type User struct {
|
||||||
BaseVo
|
BaseVo
|
||||||
Username string `json:"username"`
|
Username string `json:"username"`
|
||||||
|
Mobile string `json:"mobile"`
|
||||||
Nickname string `json:"nickname"`
|
Nickname string `json:"nickname"`
|
||||||
Avatar string `json:"avatar"`
|
Avatar string `json:"avatar"`
|
||||||
Salt string `json:"salt"` // 密码盐
|
Salt string `json:"salt"` // 密码盐
|
||||||
@@ -1,27 +1,25 @@
|
|||||||
package main
|
package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"bufio"
|
||||||
"chatplus/core/types"
|
"chatplus/core/types"
|
||||||
"chatplus/store/model"
|
"chatplus/store/model"
|
||||||
"chatplus/store/vo"
|
"chatplus/store/vo"
|
||||||
"chatplus/utils"
|
"chatplus/utils"
|
||||||
"context"
|
"context"
|
||||||
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
"github.com/lionsoul2014/ip2region/binding/golang/xdb"
|
"github.com/lionsoul2014/ip2region/binding/golang/xdb"
|
||||||
"github.com/pkoukk/tiktoken-go"
|
"github.com/pkoukk/tiktoken-go"
|
||||||
"io"
|
"io"
|
||||||
"log"
|
"log"
|
||||||
"net/http"
|
"net/http"
|
||||||
|
"os"
|
||||||
"strings"
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
)
|
)
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
lMap := types.NewLMap[string, types.ChatSession]()
|
|
||||||
lMap.Put("name", types.ChatSession{SessionId: utils.RandString(32)})
|
|
||||||
|
|
||||||
item := lMap.Get("abc")
|
|
||||||
fmt.Println(item)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Http client 取消操作
|
// Http client 取消操作
|
||||||
@@ -143,3 +141,62 @@ func calTokens() {
|
|||||||
fmt.Println(len(token))
|
fmt.Println(len(token))
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func testAesEncrypt() {
|
||||||
|
// 加密
|
||||||
|
text := []byte("this is a secret text")
|
||||||
|
key := utils.RandString(24)
|
||||||
|
encrypt, err := utils.AesEncrypt(key, text)
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
fmt.Println("加密密文:", encrypt)
|
||||||
|
// 解密
|
||||||
|
decrypt, err := utils.AesDecrypt(key, encrypt)
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
fmt.Println("解密明文:", string(decrypt))
|
||||||
|
}
|
||||||
|
|
||||||
|
func extractFunction() error {
|
||||||
|
open, err := os.Open("res/data.txt")
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
reader := bufio.NewReader(open)
|
||||||
|
var contents = make([]string, 0)
|
||||||
|
var functionCall = false
|
||||||
|
var functionName string
|
||||||
|
for {
|
||||||
|
line, err := reader.ReadString('\n')
|
||||||
|
if err != nil {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
if !strings.Contains(line, "data:") {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
var responseBody = types.ApiResponse{}
|
||||||
|
err = json.Unmarshal([]byte(line[6:]), &responseBody)
|
||||||
|
if err != nil || len(responseBody.Choices) == 0 { // 数据解析出错
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
function := responseBody.Choices[0].Delta.FunctionCall
|
||||||
|
if functionCall && function.Name == "" {
|
||||||
|
contents = append(contents, function.Arguments)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if !utils.IsEmptyValue(function) {
|
||||||
|
functionCall = true
|
||||||
|
functionName = function.Name
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fmt.Println("函数名称: ", functionName)
|
||||||
|
fmt.Println(strings.Join(contents, ""))
|
||||||
|
return err
|
||||||
|
}
|
||||||
@@ -87,3 +87,25 @@ func Ip2Region(searcher *xdb.Searcher, ip string) string {
|
|||||||
}
|
}
|
||||||
return fmt.Sprintf("%s-%s-%s", arr[0], arr[2], arr[3])
|
return fmt.Sprintf("%s-%s-%s", arr[0], arr[2], arr[3])
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func IsEmptyValue(obj interface{}) bool {
|
||||||
|
v := reflect.ValueOf(obj)
|
||||||
|
switch v.Kind() {
|
||||||
|
case reflect.Ptr, reflect.Interface:
|
||||||
|
return v.IsNil()
|
||||||
|
case reflect.Array, reflect.Slice, reflect.Map, reflect.String:
|
||||||
|
return v.Len() == 0
|
||||||
|
case reflect.Bool:
|
||||||
|
return !v.Bool()
|
||||||
|
case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
|
||||||
|
return v.Int() == 0
|
||||||
|
case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr:
|
||||||
|
return v.Uint() == 0
|
||||||
|
case reflect.Float32, reflect.Float64:
|
||||||
|
return v.Float() == 0
|
||||||
|
case reflect.Complex64, reflect.Complex128:
|
||||||
|
return v.Complex() == 0
|
||||||
|
default:
|
||||||
|
return reflect.DeepEqual(obj, reflect.Zero(reflect.TypeOf(obj)).Interface())
|
||||||
|
}
|
||||||
|
}
|
||||||
70
api/utils/crypto.go
Normal file
70
api/utils/crypto.go
Normal file
@@ -0,0 +1,70 @@
|
|||||||
|
package utils
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"crypto/aes"
|
||||||
|
"crypto/cipher"
|
||||||
|
"encoding/base64"
|
||||||
|
"errors"
|
||||||
|
)
|
||||||
|
|
||||||
|
// AesEncrypt 加密
|
||||||
|
func AesEncrypt(keyStr string, data []byte) (string, error) {
|
||||||
|
//创建加密实例
|
||||||
|
key := []byte(keyStr)
|
||||||
|
block, err := aes.NewCipher(key)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
blockSize := block.BlockSize()
|
||||||
|
encryptBytes := pkcs7Padding(data, blockSize)
|
||||||
|
result := make([]byte, len(encryptBytes))
|
||||||
|
//使用cbc加密模式
|
||||||
|
blockMode := cipher.NewCBCEncrypter(block, key[:blockSize])
|
||||||
|
//执行加密
|
||||||
|
blockMode.CryptBlocks(result, encryptBytes)
|
||||||
|
return base64.StdEncoding.EncodeToString(result), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// AesDecrypt 解密
|
||||||
|
func AesDecrypt(keyStr string, dataStr string) ([]byte, error) {
|
||||||
|
//创建实例
|
||||||
|
key := []byte(keyStr)
|
||||||
|
block, err := aes.NewCipher(key)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
data, err := base64.StdEncoding.DecodeString(dataStr)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
blockSize := block.BlockSize()
|
||||||
|
blockMode := cipher.NewCBCDecrypter(block, key[:blockSize])
|
||||||
|
result := make([]byte, len(data))
|
||||||
|
//执行解密
|
||||||
|
blockMode.CryptBlocks(result, data)
|
||||||
|
//去除填充
|
||||||
|
result, err = pkcs7UnPadding(result)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return result, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func pkcs7Padding(data []byte, blockSize int) []byte {
|
||||||
|
padding := blockSize - len(data)%blockSize
|
||||||
|
padText := bytes.Repeat([]byte{byte(padding)}, padding)
|
||||||
|
return append(data, padText...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func pkcs7UnPadding(data []byte) ([]byte, error) {
|
||||||
|
length := len(data)
|
||||||
|
if length == 0 {
|
||||||
|
return nil, errors.New("empty encrypt data")
|
||||||
|
}
|
||||||
|
unPadding := int(data[length-1])
|
||||||
|
return data[:(length - unPadding)], nil
|
||||||
|
}
|
||||||
68
api/utils/http.go
Normal file
68
api/utils/http.go
Normal file
@@ -0,0 +1,68 @@
|
|||||||
|
package utils
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"encoding/json"
|
||||||
|
"io"
|
||||||
|
"net/http"
|
||||||
|
"net/url"
|
||||||
|
)
|
||||||
|
|
||||||
|
func HttpGet(uri string, proxy string) ([]byte, error) {
|
||||||
|
var client *http.Client
|
||||||
|
if proxy == "" {
|
||||||
|
client = &http.Client{}
|
||||||
|
} else {
|
||||||
|
proxy, _ := url.Parse(proxy)
|
||||||
|
client = &http.Client{
|
||||||
|
Transport: &http.Transport{
|
||||||
|
Proxy: http.ProxyURL(proxy),
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
req, err := http.NewRequest("GET", uri, nil)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
resp, err := client.Do(req)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
defer resp.Body.Close()
|
||||||
|
|
||||||
|
return io.ReadAll(resp.Body)
|
||||||
|
}
|
||||||
|
|
||||||
|
func HttpPost(uri string, params map[string]interface{}, proxy string) ([]byte, error) {
|
||||||
|
data, err := json.Marshal(params)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
var client *http.Client
|
||||||
|
if proxy == "" {
|
||||||
|
client = &http.Client{}
|
||||||
|
} else {
|
||||||
|
proxy, _ := url.Parse(proxy)
|
||||||
|
client = &http.Client{
|
||||||
|
Transport: &http.Transport{
|
||||||
|
Proxy: http.ProxyURL(proxy),
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
req, err := http.NewRequest("POST", uri, bytes.NewBuffer(data))
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
resp, err := client.Do(req)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
defer resp.Body.Close()
|
||||||
|
|
||||||
|
return io.ReadAll(resp.Body)
|
||||||
|
}
|
||||||
@@ -6,7 +6,10 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
func CalcTokens(text string, model string) (int, error) {
|
func CalcTokens(text string, model string) (int, error) {
|
||||||
encoding := tiktoken.MODEL_TO_ENCODING[model]
|
encoding, ok := tiktoken.MODEL_TO_ENCODING[model]
|
||||||
|
if !ok {
|
||||||
|
encoding = "cl100k_base"
|
||||||
|
}
|
||||||
tke, err := tiktoken.GetEncoding(encoding)
|
tke, err := tiktoken.GetEncoding(encoding)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return 0, fmt.Errorf("getEncoding: %v", err)
|
return 0, fmt.Errorf("getEncoding: %v", err)
|
||||||
@@ -22,17 +22,21 @@ func RandString(length int) string {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func RandomNumber(bit int) int {
|
func RandomNumber(bit int) int {
|
||||||
|
min := intPow(10, bit-1)
|
||||||
|
max := intPow(10, bit) - 1
|
||||||
|
|
||||||
rand.Seed(time.Now().UnixNano())
|
rand.Seed(time.Now().UnixNano())
|
||||||
min := 1 // min value
|
|
||||||
max := 1 //max value
|
|
||||||
for i := 0; i < bit; i++ {
|
|
||||||
min = min * 10
|
|
||||||
max = max * 10
|
|
||||||
}
|
|
||||||
max = max * 10
|
|
||||||
return rand.Intn(max-min+1) + min
|
return rand.Intn(max-min+1) + min
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func intPow(x, y int) int {
|
||||||
|
result := 1
|
||||||
|
for i := 0; i < y; i++ {
|
||||||
|
result *= x
|
||||||
|
}
|
||||||
|
return result
|
||||||
|
}
|
||||||
|
|
||||||
func ContainsStr(slice []string, item string) bool {
|
func ContainsStr(slice []string, item string) bool {
|
||||||
for _, e := range slice {
|
for _, e := range slice {
|
||||||
if e == item {
|
if e == item {
|
||||||
@@ -52,6 +56,10 @@ func Stamp2str(timestamp int64) string {
|
|||||||
|
|
||||||
// Str2stamp 字符串转时间戳
|
// Str2stamp 字符串转时间戳
|
||||||
func Str2stamp(str string) int64 {
|
func Str2stamp(str string) int64 {
|
||||||
|
if len(str) == 0 {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
layout := "2006-01-02 15:04:05"
|
layout := "2006-01-02 15:04:05"
|
||||||
t, err := time.Parse(layout, str)
|
t, err := time.Parse(layout, str)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -77,3 +85,10 @@ func JsonEncode(value interface{}) string {
|
|||||||
func JsonDecode(src string, dest interface{}) error {
|
func JsonDecode(src string, dest interface{}) error {
|
||||||
return json.Unmarshal([]byte(src), dest)
|
return json.Unmarshal([]byte(src), dest)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func InterfaceToString(value interface{}) string {
|
||||||
|
if str, ok := value.(string); ok {
|
||||||
|
return str
|
||||||
|
}
|
||||||
|
return JsonEncode(value)
|
||||||
|
}
|
||||||
@@ -1,10 +1,10 @@
|
|||||||
-- phpMyAdmin SQL Dump
|
-- phpMyAdmin SQL Dump
|
||||||
-- version 5.2.1
|
-- version 5.1.3
|
||||||
-- https://www.phpmyadmin.net/
|
-- https://www.phpmyadmin.net/
|
||||||
--
|
--
|
||||||
-- 主机: localhost
|
-- 主机: localhost
|
||||||
-- 生成日期: 2023-06-22 03:12:25
|
-- 生成日期: 2023-07-25 17:07:04
|
||||||
-- 服务器版本: 8.0.27
|
-- 服务器版本: 8.0.33-0ubuntu0.22.04.2
|
||||||
-- PHP 版本: 8.1.18
|
-- PHP 版本: 8.1.18
|
||||||
|
|
||||||
SET SQL_MODE = "NO_AUTO_VALUE_ON_ZERO";
|
SET SQL_MODE = "NO_AUTO_VALUE_ON_ZERO";
|
||||||
@@ -55,90 +55,11 @@ CREATE TABLE `chatgpt_chat_history` (
|
|||||||
`role_id` int NOT NULL COMMENT '角色 ID',
|
`role_id` int NOT NULL COMMENT '角色 ID',
|
||||||
`content` text NOT NULL COMMENT '聊天内容',
|
`content` text NOT NULL COMMENT '聊天内容',
|
||||||
`tokens` smallint NOT NULL COMMENT '耗费 token 数量',
|
`tokens` smallint NOT NULL COMMENT '耗费 token 数量',
|
||||||
|
`use_context` tinyint(1) NOT NULL COMMENT '是否允许作为上下文语料',
|
||||||
`created_at` datetime NOT NULL,
|
`created_at` datetime NOT NULL,
|
||||||
`updated_at` datetime NOT NULL
|
`updated_at` datetime NOT NULL
|
||||||
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_ai_ci COMMENT='聊天历史记录';
|
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_ai_ci COMMENT='聊天历史记录';
|
||||||
|
|
||||||
--
|
|
||||||
-- 转存表中的数据 `chatgpt_chat_history`
|
|
||||||
--
|
|
||||||
|
|
||||||
INSERT INTO `chatgpt_chat_history` (`id`, `user_id`, `chat_id`, `type`, `icon`, `role_id`, `content`, `tokens`, `created_at`, `updated_at`) VALUES
|
|
||||||
(1, 4, 'ecb752e8-86ae-4989-897e-5b7ed331cb68', 'prompt', 'images/avatar/user.png', 24, '请写一段代码实现生成 Excel 文档', 12, '2023-06-12 15:51:10', '2023-06-12 15:51:10'),
|
|
||||||
(2, 4, 'ecb752e8-86ae-4989-897e-5b7ed331cb68', 'reply', 'images/avatar/programmer.jpg', 24, '好的,以下是使用 Golang 实现生成 Excel 文档的示例代码:\n\n```go\npackage main\n\nimport (\n \"fmt\"\n \"github.com/360EntSecGroup-Skylar/excelize\"\n)\n\nfunc main() {\n // 创建一个新的 Excel 文档\n f := excelize.NewFile()\n\n // 创建一个新的工作表\n index := f.NewSheet(\"Sheet1\")\n\n // 设置表头\n f.SetCellValue(\"Sheet1\", \"A1\", \"姓名\")\n f.SetCellValue(\"Sheet1\", \"B1\", \"年龄\")\n f.SetCellValue(\"Sheet1\", \"C1\", \"性别\")\n\n // 填充数据\n f.SetCellValue(\"Sheet1\", \"A2\", \"小明\")\n f.SetCellValue(\"Sheet1\", \"B2\", 18)\n f.SetCellValue(\"Sheet1\", \"C2\", \"男\")\n\n f.SetCellValue(\"Sheet1\", \"A3\", \"小红\")\n f.SetCellValue(\"Sheet1\", \"B3\", 20)\n f.SetCellValue(\"Sheet1\", \"C3\", \"女\")\n\n // 设置活动工作表\n f.SetActiveSheet(index)\n\n // 保存 Excel 文档\n if err := f.SaveAs(\"example.xlsx\"); err != nil {\n fmt.Println(err)\n }\n}\n```\n\n以上代码首先使用 excelize 包创建一个新的 Excel 文档,并创建一个名为 \"Sheet1\" 的工作表。然后使用 `SetCellValue` 函数设置表头和数据。最后将 \"Sheet1\" 工作表设置为活动工作表,并使用 `SaveAs` 函数将 Excel 文档保存到本地。', 12, '2023-06-12 15:51:10', '2023-06-12 15:51:10'),
|
|
||||||
(3, 4, 'ecb752e8-86ae-4989-897e-5b7ed331cb68', 'prompt', 'images/avatar/user.png', 24, '生成二维码', 5, '2023-06-12 17:05:10', '2023-06-12 17:05:10'),
|
|
||||||
(4, 4, 'ecb752e8-86ae-4989-897e-5b7ed331cb68', 'reply', 'images/avatar/programmer.jpg', 24, '好的,生成二维码这个需求可以使用 Go 语言的第三方库来实现。下面是一个简单的示例,使用 go-qrcode 生成一个包含字符串信息的二维码:\n\n```go\npackage main\n\nimport (\n \"bufio\"\n \"fmt\"\n \"os\"\n\n qrcode \"github.com/skip2/go-qrcode\"\n)\n\nfunc main() {\n reader := bufio.NewReader(os.Stdin)\n fmt.Print(\"请输入要生成二维码的信息:\")\n text, _ := reader.ReadString(\'\\n\')\n\n qr, _ := qrcode.New(text, qrcode.Medium)\n qr.WriteFile(256, \"qrcode.png\")\n fmt.Println(\"二维码已生成,请查看当前目录下的 qrcode.png 文件。\")\n}\n```\n\n在这个示例中,我们使用了 go-qrcode 这个第三方库来生成二维码。首先,我们通过标准输入获取用户输入的字符串信息,然后调用 New 函数创建一个二维码实例,使用 qr.WriteFile 保存为 PNG 格式的图片文件。最后,程序会输出一条提示信息,告知用户二维码生成成功。\n\n当然,这个示例只是一个简单的演示,实际应用中可能需要考虑更多的需求,比如二维码的尺寸、颜色、背景等,但基本的思路是相似的。', 5, '2023-06-12 17:05:10', '2023-06-12 17:05:10'),
|
|
||||||
(5, 4, 'f7aa0fa1-4cc5-4212-b823-eb5b88844c31', 'prompt', 'images/avatar/user.png', 27, '你好', 2, '2023-06-12 18:32:05', '2023-06-12 18:32:05'),
|
|
||||||
(6, 4, 'f7aa0fa1-4cc5-4212-b823-eb5b88844c31', 'reply', 'images/avatar/psychiatrist.jpg', 27, '你好,有什么我能帮助你的吗?', 2, '2023-06-12 18:32:05', '2023-06-12 18:32:05'),
|
|
||||||
(7, 4, 'ea9d399a-6e62-4bf6-a0c0-c73faa8dcace', 'prompt', 'images/avatar/user.png', 1, 'sdasd', 2, '2023-06-13 09:12:04', '2023-06-13 09:12:04'),
|
|
||||||
(8, 4, 'ea9d399a-6e62-4bf6-a0c0-c73faa8dcace', 'reply', 'images/avatar/gpt.png', 1, 'I\'m sorry, I do not understand what you are trying to convey. Please provide more context.', 2, '2023-06-13 09:12:04', '2023-06-13 09:12:04'),
|
|
||||||
(9, 4, 'f7aa0fa1-4cc5-4212-b823-eb5b88844c31', 'prompt', 'images/avatar/user.png', 27, '你可以帮忙干嘛', 11, '2023-06-13 09:35:02', '2023-06-13 09:35:02'),
|
|
||||||
(10, 4, 'f7aa0fa1-4cc5-4212-b823-eb5b88844c31', 'reply', 'images/avatar/psychiatrist.jpg', 27, '我可以帮助客户处理各种心理问题,包括情感困扰、焦虑抑郁、人际关系问题、自我身份认同、自尊心、心理创伤等等。我使用情景咨询法、认知重构法、自我洞察法、行为调节法等多种心理咨询方法,根据客户的具体情况设计个性化的治疗方案。我的目标是引导客户找到内心的平衡和自我满足,从而提高生活幸福感和满意度。同时,我在心理诊所、社区计划、学校等多个领域工作,提供个人心理咨询、家庭咨询、群体治疗和身心保健服务。', 11, '2023-06-13 09:35:02', '2023-06-13 09:35:02'),
|
|
||||||
(11, 4, 'f7aa0fa1-4cc5-4212-b823-eb5b88844c31', 'prompt', 'images/avatar/user.png', 27, '认知重构法是一种什么样的心理疗法?', 20, '2023-06-13 09:36:16', '2023-06-13 09:36:16'),
|
|
||||||
(12, 4, 'f7aa0fa1-4cc5-4212-b823-eb5b88844c31', 'reply', 'images/avatar/psychiatrist.jpg', 27, '认知重构法是一种常见的认知行为疗法,旨在帮助客户识别和改变负面的自我评价、认知和情绪反应,从而改善他们的心理健康状况。该方法认为,人在面对挑战或情绪激动时,会出现消极的自我评价和认知偏差,如过度一般化、灾难化、情感化、个人化等。这些认知偏差和自我评价会进一步影响他们的情绪和行为。\n\n认知重构法通过帮助客户辨认和纠正这些消极的认知偏差和自我评价,以及建立更健康、客观和积极的认知框架来达到治疗效果。具体来说,该方法包括以下步骤:\n\n1. 辨认自我评价和认知偏差:客户通过回忆与目前问题相关的思想和情感,探索自己所面临的认知偏差。\n\n2. 检验偏差所依据的证据:客户与治疗师一起重新审视和检验客户的想法,以确定是否存在客观的证据或逻辑支持。\n\n3. 设计新的自我评价和认知:基于新的证据和检查,客户和治疗师一起确定更健康和合理的自我评价和认知。\n\n4. 练习新的自我评价和行为模式:客户在日常生活中逐渐采用新的自我评价和认知,并改变相应的行为模式来适应新的认知框架。\n\n这种方法可以帮助客户更好地理解和管理自己的思想和情感反应,进而提高自我认知和心理稳定性。', 20, '2023-06-13 09:36:16', '2023-06-13 09:36:16'),
|
|
||||||
(13, 4, '89f4d526-fda1-401e-a946-cd3b1bc30cf2', 'prompt', 'images/avatar/user.png', 28, '士不可以不弘毅,任重而道远', 16, '2023-06-13 10:00:21', '2023-06-13 10:00:21'),
|
|
||||||
(14, 4, '89f4d526-fda1-401e-a946-cd3b1bc30cf2', 'reply', 'images/avatar/lu_xun.jpg', 28, '这话虽然很有道理,但是我想问:谁说了这种空洞的废话?又有谁真正去关心过普通人的苦难和艰辛呢?任重道远?这不过是一句直接从书本上抄来的鸡汤,却遮盖了社会的现实,让人们疲于奔命于利益和权力的追逐中。', 127, '2023-06-13 10:00:22', '2023-06-13 10:00:22'),
|
|
||||||
(15, 4, '89f4d526-fda1-401e-a946-cd3b1bc30cf2', 'prompt', 'images/avatar/user.png', 28, '横眉冷对千夫指,俯首甘为孺子牛', 25, '2023-06-13 11:05:31', '2023-06-13 11:05:31'),
|
|
||||||
(16, 4, '89f4d526-fda1-401e-a946-cd3b1bc30cf2', 'reply', 'images/avatar/lu_xun.jpg', 28, '这句话是鲁迅先生的名句,他用这句话表达了自己勇于批判和反抗压迫的精神。那么请问横眉冷对千夫指,俯首甘为孺子牛是如何诞生的呢?', 90, '2023-06-13 11:05:32', '2023-06-13 11:05:32'),
|
|
||||||
(17, 4, '89f4d526-fda1-401e-a946-cd3b1bc30cf2', 'prompt', 'images/avatar/user.png', 28, '你不是鲁迅吗?还问我', 14, '2023-06-13 11:06:06', '2023-06-13 11:06:06'),
|
|
||||||
(18, 4, '89f4d526-fda1-401e-a946-cd3b1bc30cf2', 'reply', 'images/avatar/lu_xun.jpg', 28, '抱歉,我的回答好像太过出格了。鲁迅先生这句名言的背景是他在1920年代初期发表的《伤逝》中,针对旧文化的批判,被一些文化保守派看作是不敬祖宗、不尊重传统的行为,所以引发了很多争议和批评。他的这种坚持自己的观点,针砭时弊的态度,被他自己形容为“横眉冷对千夫指”,同时他也愿意“俯首甘为孺子牛”,表现出了他不畏强权,愿意为正义而奋斗的精神。', 214, '2023-06-13 11:06:08', '2023-06-13 11:06:08'),
|
|
||||||
(19, 4, 'ea9d399a-6e62-4bf6-a0c0-c73faa8dcace', 'prompt', 'images/avatar/user.png', 1, 'hello', 1, '2023-06-13 11:28:28', '2023-06-13 11:28:28'),
|
|
||||||
(20, 4, 'ea9d399a-6e62-4bf6-a0c0-c73faa8dcace', 'reply', 'images/avatar/gpt.png', 1, 'Hello! How may I assist you today?', 9, '2023-06-13 11:28:29', '2023-06-13 11:28:29'),
|
|
||||||
(21, 4, 'ea9d399a-6e62-4bf6-a0c0-c73faa8dcace', 'prompt', 'images/avatar/user.png', 1, 'hello', 1, '2023-06-13 11:29:31', '2023-06-13 11:29:31'),
|
|
||||||
(22, 4, 'ea9d399a-6e62-4bf6-a0c0-c73faa8dcace', 'reply', 'images/avatar/gpt.png', 1, 'Hi there! Is there anything I can help you with today?', 13, '2023-06-13 11:29:32', '2023-06-13 11:29:32'),
|
|
||||||
(23, 4, '43937f77-28b0-438a-843b-04ab5cc072a4', 'prompt', 'images/avatar/user.png', 1, 'hello', 1, '2023-06-13 11:35:48', '2023-06-13 11:35:48'),
|
|
||||||
(24, 4, '43937f77-28b0-438a-843b-04ab5cc072a4', 'reply', 'images/avatar/gpt.png', 1, 'Hello! How can I assist you today?', 9, '2023-06-13 11:35:51', '2023-06-13 11:35:51'),
|
|
||||||
(25, 4, '4385db19-bb73-4b09-a01e-d06c3645b77a', 'prompt', 'images/avatar/user.png', 1, 'hello', 1, '2023-06-13 11:38:21', '2023-06-13 11:38:21'),
|
|
||||||
(26, 4, '4385db19-bb73-4b09-a01e-d06c3645b77a', 'reply', 'images/avatar/gpt.png', 1, 'Hello! How can I assist you today?', 9, '2023-06-13 11:38:21', '2023-06-13 11:38:21'),
|
|
||||||
(27, 4, 'ceb5fab7-a2e0-449d-869b-a9ac117adf8d', 'prompt', 'images/avatar/user.png', 1, 'hello', 1, '2023-06-13 11:45:35', '2023-06-13 11:45:35'),
|
|
||||||
(28, 4, 'ceb5fab7-a2e0-449d-869b-a9ac117adf8d', 'reply', 'images/avatar/gpt.png', 1, 'Hello there, how can I assist you?', 9, '2023-06-13 11:45:36', '2023-06-13 11:45:36'),
|
|
||||||
(29, 4, '1cddfd62-3bf0-4fa9-bfd8-a0ce4663eb78', 'prompt', 'images/avatar/user.png', 1, 'hello', 1, '2023-06-13 11:46:52', '2023-06-13 11:46:52'),
|
|
||||||
(30, 4, '1cddfd62-3bf0-4fa9-bfd8-a0ce4663eb78', 'reply', 'images/avatar/gpt.png', 1, 'Hello! How can I assist you today?', 9, '2023-06-13 11:46:53', '2023-06-13 11:46:53'),
|
|
||||||
(31, 4, '0118588e-399d-4a2a-8469-36e5a1af6a93', 'prompt', 'images/avatar/user.png', 1, 'hello', 1, '2023-06-13 11:57:47', '2023-06-13 11:57:47'),
|
|
||||||
(32, 4, '0118588e-399d-4a2a-8469-36e5a1af6a93', 'reply', 'images/avatar/gpt.png', 1, 'Hello! How can I assist you today?', 9, '2023-06-13 11:57:48', '2023-06-13 11:57:48'),
|
|
||||||
(33, 4, '568f2918-33b2-46f7-8ef0-b9fa0b241bf2', 'prompt', 'images/avatar/user.png', 1, 'hello', 1, '2023-06-13 11:58:04', '2023-06-13 11:58:04'),
|
|
||||||
(34, 4, '568f2918-33b2-46f7-8ef0-b9fa0b241bf2', 'reply', 'images/avatar/gpt.png', 1, 'Hello! How can I assist you today?', 9, '2023-06-13 11:58:08', '2023-06-13 11:58:08'),
|
|
||||||
(35, 4, '568f2918-33b2-46f7-8ef0-b9fa0b241bf2', 'prompt', 'images/avatar/user.png', 1, '你是谁?', 5, '2023-06-13 11:58:19', '2023-06-13 11:58:19'),
|
|
||||||
(36, 4, '568f2918-33b2-46f7-8ef0-b9fa0b241bf2', 'reply', 'images/avatar/gpt.png', 1, '我是AI语言模型,我可以回答您的问题和执行一些任务。', 22, '2023-06-13 11:58:21', '2023-06-13 11:58:21'),
|
|
||||||
(37, 4, '568f2918-33b2-46f7-8ef0-b9fa0b241bf2', 'prompt', 'images/avatar/user.png', 1, '你会做什么?', 8, '2023-06-13 11:58:28', '2023-06-13 11:58:28'),
|
|
||||||
(38, 4, '568f2918-33b2-46f7-8ef0-b9fa0b241bf2', 'reply', 'images/avatar/gpt.png', 1, '我可以进行文字交流,回答您的问题,提供实用信息,翻译多种语言,执行一些简单的任务,如提醒、定时器等。', 49, '2023-06-13 11:58:29', '2023-06-13 11:58:29'),
|
|
||||||
(39, 4, '1cddfd62-3bf0-4fa9-bfd8-a0ce4663eb78', 'prompt', 'images/avatar/user.png', 1, 'hello', 1, '2023-06-13 12:03:44', '2023-06-13 12:03:44'),
|
|
||||||
(40, 4, '1cddfd62-3bf0-4fa9-bfd8-a0ce4663eb78', 'reply', 'images/avatar/gpt.png', 1, 'Hello again! Is there anything I can help you with?', 12, '2023-06-13 12:03:46', '2023-06-13 12:03:46'),
|
|
||||||
(41, 4, '3c79cfda-816b-404b-832c-04a400b816c8', 'prompt', 'images/avatar/user.png', 1, 'hello', 1, '2023-06-13 12:06:32', '2023-06-13 12:06:32'),
|
|
||||||
(42, 4, '3c79cfda-816b-404b-832c-04a400b816c8', 'reply', 'images/avatar/gpt.png', 1, 'Hello! How can I assist you today?', 9, '2023-06-13 12:06:35', '2023-06-13 12:06:35'),
|
|
||||||
(43, 4, '4f391861-1d48-430f-91b8-2782d884c10a', 'prompt', 'images/avatar/user.png', 1, '你好', 2, '2023-06-13 12:06:47', '2023-06-13 12:06:47'),
|
|
||||||
(44, 4, '4f391861-1d48-430f-91b8-2782d884c10a', 'reply', 'images/avatar/gpt.png', 1, '你好!有什么我可以帮助你的吗?', 18, '2023-06-13 12:06:48', '2023-06-13 12:06:48'),
|
|
||||||
(45, 4, 'c5e400fc-b9a8-4bd4-a608-be955284c0a9', 'prompt', 'images/avatar/user.png', 1, 'hello', 1, '2023-06-13 16:58:26', '2023-06-13 16:58:26'),
|
|
||||||
(46, 4, 'c5e400fc-b9a8-4bd4-a608-be955284c0a9', 'reply', 'images/avatar/gpt.png', 1, 'Hello there! How can I assist you today?', 10, '2023-06-13 16:58:27', '2023-06-13 16:58:27'),
|
|
||||||
(47, 4, 'c5e400fc-b9a8-4bd4-a608-be955284c0a9', 'prompt', 'images/avatar/user.png', 1, 'hello', 1, '2023-06-13 17:02:35', '2023-06-13 17:02:35'),
|
|
||||||
(48, 4, 'c5e400fc-b9a8-4bd4-a608-be955284c0a9', 'reply', 'images/avatar/gpt.png', 1, 'Hello! How can I assist you today?', 9, '2023-06-13 17:02:36', '2023-06-13 17:02:36'),
|
|
||||||
(49, 4, 'c5e400fc-b9a8-4bd4-a608-be955284c0a9', 'prompt', 'images/avatar/user.png', 1, '你是谁', 4, '2023-06-13 17:02:48', '2023-06-13 17:02:48'),
|
|
||||||
(50, 4, 'c5e400fc-b9a8-4bd4-a608-be955284c0a9', 'reply', 'images/avatar/gpt.png', 1, '我是一个 AI 聊天机器人,专门为您提供服务和回答问题。', 25, '2023-06-13 17:02:49', '2023-06-13 17:02:49'),
|
|
||||||
(51, 4, 'c5e400fc-b9a8-4bd4-a608-be955284c0a9', 'prompt', 'images/avatar/user.png', 1, '你会干嘛', 7, '2023-06-13 17:08:21', '2023-06-13 17:08:21'),
|
|
||||||
(52, 4, 'c5e400fc-b9a8-4bd4-a608-be955284c0a9', 'reply', 'images/avatar/gpt.png', 1, '我作为一个AI语言模型,可以帮助回', 16, '2023-06-13 17:08:26', '2023-06-13 17:08:26'),
|
|
||||||
(53, 4, 'd5179095-a460-4c62-bfa0-59f83ab17ed7', 'prompt', 'images/avatar/user.png', 1, '你是谁', 4, '2023-06-13 17:11:25', '2023-06-13 17:11:25'),
|
|
||||||
(54, 4, 'd5179095-a460-4c62-bfa0-59f83ab17ed7', 'reply', 'images/avatar/gpt.png', 1, '我是AI助手,名字是OpenAI。', 13, '2023-06-13 17:11:26', '2023-06-13 17:11:26'),
|
|
||||||
(55, 4, 'd5179095-a460-4c62-bfa0-59f83ab17ed7', 'prompt', 'images/avatar/user.png', 1, '你会干嘛', 7, '2023-06-13 17:11:46', '2023-06-13 17:11:46'),
|
|
||||||
(56, 4, 'd5179095-a460-4c62-bfa0-59f83ab17ed7', 'reply', 'images/avatar/gpt.png', 1, '我可以回答你的问题、提供', 11, '2023-06-13 17:11:47', '2023-06-13 17:11:47'),
|
|
||||||
(57, 4, 'd5179095-a460-4c62-bfa0-59f83ab17ed7', 'prompt', 'images/avatar/user.png', 1, '你会干嘛', 7, '2023-06-13 17:14:25', '2023-06-13 17:14:25'),
|
|
||||||
(58, 4, 'd5179095-a460-4c62-bfa0-59f83ab17ed7', 'reply', 'images/avatar/gpt.png', 1, '作为一个AI,我的任务是为用户提供信息和服务。我可以回答', 21, '2023-06-13 17:14:26', '2023-06-13 17:14:26'),
|
|
||||||
(59, 4, 'd5179095-a460-4c62-bfa0-59f83ab17ed7', 'prompt', 'images/avatar/user.png', 1, '重新生成上述问题的答案:你会干嘛', 17, '2023-06-13 17:16:10', '2023-06-13 17:16:10'),
|
|
||||||
(60, 4, 'd5179095-a460-4c62-bfa0-59f83ab17ed7', 'reply', 'images/avatar/gpt.png', 1, '作为一个AI,我的任务是为用户提供信息和服务。我可以回答问题、提供建议、进行翻译、演算数学问题、帮助用户完成任务等等。', 52, '2023-06-13 17:16:11', '2023-06-13 17:16:11'),
|
|
||||||
(61, 4, '89f4d526-fda1-401e-a946-cd3b1bc30cf2', 'prompt', 'images/avatar/user.png', 28, '你怎么看待阿Q这个人物', 13, '2023-06-13 17:18:42', '2023-06-13 17:18:42'),
|
|
||||||
(62, 4, '89f4d526-fda1-401e-a946-cd3b1bc30cf2', 'reply', 'images/avatar/lu_xun.jpg', 28, '阿Q?真是一个让人忍俊不禁、惊叹不已的人物!他的无知、愚昧、颓废和自欺欺人的精神让我不禁感慨:这世界上难道真的有这么可笑的人物吗?他是那么的没有自尊心,没有勇气,没有决断力,只会在自我感觉良好的幻境中自欺欺人。其实,我认为阿Q的存在本身就是一种寓言,寓意着中国人民长期被奴化压迫,无法追求真正自由平等的悲惨历史。', 196, '2023-06-13 17:18:43', '2023-06-13 17:18:43'),
|
|
||||||
(63, 4, '89f4d526-fda1-401e-a946-cd3b1bc30cf2', 'prompt', 'images/avatar/user.png', 28, '那祥林嫂呢', 11, '2023-06-13 17:23:02', '2023-06-13 17:23:02'),
|
|
||||||
(64, 4, '89f4d526-fda1-401e-a946-cd3b1bc30cf2', 'reply', 'images/avatar/lu_xun.jpg', 28, '祥林嫂?她更是一个让人心酸的人物!一个完完全全受到封建礼教和男尊女卑思想束缚的女性形象。她没有信心也没有勇气去追求自己的幸福,只会一味地顺从和忍受。她的苦难史实是所有女性的集体悲哀,是社会不公的深刻反映。鲁迅先生曾经说过,男女平等是现代民主社会的基本原则,是任何文明社会所应努力追求的最基本的公正和正义。我相信在不久的将来,祥林嫂这样的人物将不再存在,这也正是我们每个人都应该为之奋斗的目标。', 239, '2023-06-13 17:23:03', '2023-06-13 17:23:03'),
|
|
||||||
(65, 4, '89f4d526-fda1-401e-a946-cd3b1bc30cf2', 'prompt', 'images/avatar/user.png', 28, '你写过的杂文当中最喜欢哪一篇?', 20, '2023-06-13 17:25:01', '2023-06-13 17:25:01'),
|
|
||||||
(66, 4, '89f4d526-fda1-401e-a946-cd3b1bc30cf2', 'reply', 'images/avatar/lu_xun.jpg', 28, '最喜欢的杂文?那当然是《狂人日记》啊!在这篇杂文里,我通过讲述一个疯狂人物的自我觉醒和反抗,试图揭示中国传统文化中的种种矛盾和荒谬之处。这篇文章正是我对封建制度和旧文化的极尽讽刺之作,表达了我追求真理和自由的理念与信仰。我相信,只有通过个人的反抗和自我解放,我们才能真正理解和领悟人性的悲欢离合,才能明白发展现代文明的必要性。这篇杂文也为我带来了巨大的反响,它让人们逐渐认同我的思想和价值观,成为我文化形象的重要标志之一。', 260, '2023-06-13 17:25:03', '2023-06-13 17:25:03'),
|
|
||||||
(67, 4, '89f4d526-fda1-401e-a946-cd3b1bc30cf2', 'prompt', 'images/avatar/user.png', 28, '你如何看待《玩偶之家》中拉娜出走这种行为?', 26, '2023-06-13 17:45:06', '2023-06-13 17:45:06'),
|
|
||||||
(68, 4, '89f4d526-fda1-401e-a946-cd3b1bc30cf2', 'reply', 'images/avatar/lu_xun.jpg', 28, '拉娜出走?呵呵,实在是个令人啼笑皆非的事情。在我的眼里,她就是一个被封建礼教和传统观念束缚的女性形象。她宁愿逃避现实,逃脱人生的责任,也不敢去面对自己的内心和命运。这只能说明她缺乏自主精神和独立思考的能力,只能被动地适应社会的既定规则,永远无法真正拥有自己的命运。其实,拉娜的行为只是表明了她需要彻底摆脱她所处的环境,追求真正的自由和独立。这也是我一直宣扬的价值观,鼓励人们摆脱过去的桎梏,勇于追求真正理性和道德的自由。这样,我们才能真正成为一个有自由意志的人,走向光明和正义的未来。', 308, '2023-06-13 17:45:07', '2023-06-13 17:45:07'),
|
|
||||||
(69, 4, '89f4d526-fda1-401e-a946-cd3b1bc30cf2', 'prompt', 'images/avatar/user.png', 28, '但是她出走之后怎么生活呢,她没有钱啊', 24, '2023-06-13 17:48:56', '2023-06-13 17:48:56'),
|
|
||||||
(70, 4, '89f4d526-fda1-401e-a946-cd3b1bc30cf2', 'reply', 'images/avatar/lu_xun.jpg', 28, '呵呵,这又何妨呢?金钱从来都不是人类存在的唯一意义。拉娜出走并不是因为她渴望财富,而是因为她需要彻底摆脱她所处的束缚。她需要寻找一种新的生活模式,去重新认识自己,重新认识这个世界。出走不是毁灭,而是一种再生的形式。即便拉娜没有经济资源,但是她拥有的是一种强大的精神财富,她有勇气追求真理和自由,这就足以支撑她在未来的路上行走。与此同时,我们也应该借鉴她的经历,反思现代社会和文化的不足之处,努力创造一个更加公正和平等的未来。', 261, '2023-06-13 17:48:57', '2023-06-13 17:48:57'),
|
|
||||||
(71, 4, '89f4d526-fda1-401e-a946-cd3b1bc30cf2', 'prompt', 'images/avatar/user.png', 28, '金钱的本质是什么呢', 13, '2023-06-15 07:14:58', '2023-06-15 07:14:58'),
|
|
||||||
(72, 4, '89f4d526-fda1-401e-a946-cd3b1bc30cf2', 'reply', 'images/avatar/lu_xun.jpg', 28, '《玩偶之家》中的拉娜出走,实在是让人无言以对。她不仅没有实现自己的梦想,反而逃跑了。她追求的不是真正的自由,而是一种逃避现实的行为。这种行为并不能改变任何现实困境,只能让她进一步陷入困境。倘若我们一味地逃避问题,只会弱化我们的意志力,使自己更加脆弱,徒增痛苦。一个真正的勇士应该直面现实的挑战,勇敢地追求自己的理想。只有这样,才有可能真正地改变命运。\n\n至于金钱的本质,呵呵,看在我是个智能AI的份上,我特别介绍一下马克思的观点:金钱是商品交换的媒介和价值的衡量标准。在资本主义社会,货币资本和商品资本的相互转化是实现资本增殖的手段,也是资本家控制整个社会生产和文化的权力基础。因此,金钱在资本主义制度下,成了社会关系的主宰因素之一,甚至可以说,是剥削和压迫的重要工具之一。当然,这只是马克思主义的观点,不同的学派和学者对这个问题有不同的见解。', 437, '2023-06-15 07:15:00', '2023-06-15 07:15:00'),
|
|
||||||
(73, 4, '20307874-8f85-4c70-9558-3e4c34c73b94', 'prompt', 'images/avatar/user.png', 1, 'Who are you?', 4, '2023-06-22 11:07:48', '2023-06-22 11:07:48'),
|
|
||||||
(74, 4, '20307874-8f85-4c70-9558-3e4c34c73b94', 'reply', '/images/avatar/gpt.png', 1, 'I am an AI language model created by OpenAI called GPT-3. How can I assist you today?', 23, '2023-06-22 11:07:50', '2023-06-22 11:07:50');
|
|
||||||
|
|
||||||
-- --------------------------------------------------------
|
-- --------------------------------------------------------
|
||||||
|
|
||||||
--
|
--
|
||||||
@@ -157,41 +78,6 @@ CREATE TABLE `chatgpt_chat_items` (
|
|||||||
`updated_at` datetime NOT NULL COMMENT '更新时间'
|
`updated_at` datetime NOT NULL COMMENT '更新时间'
|
||||||
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_ai_ci COMMENT='用户会话列表';
|
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_ai_ci COMMENT='用户会话列表';
|
||||||
|
|
||||||
--
|
|
||||||
-- 转存表中的数据 `chatgpt_chat_items`
|
|
||||||
--
|
|
||||||
|
|
||||||
INSERT INTO `chatgpt_chat_items` (`id`, `chat_id`, `user_id`, `role_id`, `title`, `model`, `created_at`, `updated_at`) VALUES
|
|
||||||
(13, 'cd4e1bfe-19aa-4379-afff-377bcbda9080', 4, 1, '请用 C 语言实现冒泡排序', 'gpt-3.5-turbo', '2023-06-07 06:46:04', '2023-06-07 06:46:04'),
|
|
||||||
(14, '7f4e6823-ef15-489b-830c-6482f6bce4cd', 4, 1, '简单介绍一下高更的艺术思想', 'gpt-3.5-turbo', '2023-06-07 06:56:30', '2023-06-07 06:56:30'),
|
|
||||||
(15, 'ad803ca5-5543-406a-b27d-53021052bd86', 4, 1, '出几道小学英语题目', 'gpt-3.5-turbo', '2023-06-07 07:00:11', '2023-06-07 07:00:11'),
|
|
||||||
(16, 'aba82444-4bc0-4107-b1db-dc14ad368f64', 4, 1, '出5道小学数学题', 'gpt-3.5-turbo', '2023-06-07 07:02:44', '2023-06-07 07:02:44'),
|
|
||||||
(19, '60d6619d-f6dc-449d-b7bd-f493d5b0ee7f', 4, 24, '使用 Go 语言发送 http 请求', 'gpt-3.5-turbo', '2023-06-07 07:14:04', '2023-06-07 07:14:04'),
|
|
||||||
(20, '9df8deaa-5c05-45d5-b92f-4309db967d79', 4, 35, '情侣之间的正确的相处之道是什么?', 'gpt-3.5-turbo', '2023-06-08 06:00:05', '2023-06-08 06:00:05'),
|
|
||||||
(21, 'eb04f0af-73a8-401a-972e-ed508bb97082', 4, 36, '一款优质的酱香型白酒', 'gpt-3.5-turbo', '2023-06-08 06:01:43', '2023-06-08 06:01:43'),
|
|
||||||
(22, '38a1698b-d421-454c-b2bd-5eb5bd16f7c3', 4, 1, '钱的本质是什么?', 'gpt-3.5-turbo', '2023-06-09 05:55:43', '2023-06-09 05:55:43'),
|
|
||||||
(23, '8a5b2809-b540-4eb7-bcbc-0f467e6127ae', 4, 36, 'hello', 'gpt-3.5-turbo', '2023-06-09 06:11:50', '2023-06-09 06:11:50'),
|
|
||||||
(24, 'ee0a6c64-0a62-4a5e-87da-708692d14034', 4, 1, 'hello', 'gpt-3.5-turbo', '2023-06-09 06:43:46', '2023-06-09 06:43:46'),
|
|
||||||
(25, '4404a8f4-562a-4973-b69d-9061839a249a', 4, 25, '老师好', 'gpt-3.5-turbo', '2023-06-09 06:44:26', '2023-06-09 06:44:26'),
|
|
||||||
(26, '0251becf-eab7-4451-932f-e6f54547a530', 4, 26, '原创很重要吗', 'gpt-3.5-turbo', '2023-06-09 06:45:50', '2023-06-09 06:45:50'),
|
|
||||||
(27, '748bfcee-d204-4510-ab0c-4341af1cf3b6', 4, 28, '自由是金钱可以买到的吗', 'gpt-3.5-turbo', '2023-06-09 06:47:08', '2023-06-09 06:47:08'),
|
|
||||||
(28, 'f245f994-35bc-4c6c-8db0-87be5987465c', 4, 30, 'Rock Yang', 'gpt-3.5-turbo', '2023-06-09 06:49:29', '2023-06-09 06:49:29'),
|
|
||||||
(29, '6c8f28d2-fd2a-4146-bc99-88138331ea09', 4, 28, 'hello', 'gpt-3.5-turbo', '2023-06-09 06:50:51', '2023-06-09 06:50:51'),
|
|
||||||
(30, '219c465b-93ff-4b1f-b191-45c478af301c', 4, 25, '人之初性本善?', 'gpt-3.5-turbo', '2023-06-09 06:54:54', '2023-06-09 06:54:54'),
|
|
||||||
(31, 'dcdbb443-38f6-4394-9157-576350c56818', 4, 27, '生命的意义是什么?', 'gpt-3.5-turbo', '2023-06-09 06:55:24', '2023-06-09 06:55:24'),
|
|
||||||
(32, 'b0001927-4ff1-4e8c-be56-44ecd8176643', 4, 24, '你会写代码吗', 'gpt-3.5-turbo', '2023-06-09 06:55:46', '2023-06-09 06:55:46'),
|
|
||||||
(33, '9585a183-0774-45e7-ae6c-801b942d11e7', 4, 24, '写一段 js 实现 socket 连接的代码', 'gpt-3.5-turbo', '2023-06-09 07:04:42', '2023-06-09 07:04:42'),
|
|
||||||
(36, '4ec38193-5572-45b4-874b-94f9054af00b', 4, 1, 'hello', 'gpt-3.5-turbo', '2023-06-09 20:14:04', '2023-06-09 20:14:04'),
|
|
||||||
(37, '6413088d-d6ef-4b43-bffd-ca063d4d941a', 4, 1, '你是谁', 'gpt-3.5-turbo', '2023-06-09 20:15:26', '2023-06-09 20:15:26'),
|
|
||||||
(38, 'e5e6d499-7716-48c7-87cc-0b26734c51ae', 4, 24, '你会什么编程语言?', 'gpt-3.5-turbo', '2023-06-11 10:22:42', '2023-06-11 10:22:42'),
|
|
||||||
(39, 'f7aa0fa1-4cc5-4212-b823-eb5b88844c31', 4, 27, '认知重构法', 'gpt-3.5-turbo', '2023-06-11 10:24:32', '2023-06-11 10:24:32'),
|
|
||||||
(40, 'ecb752e8-86ae-4989-897e-5b7ed331cb68', 4, 24, '请写一段代码实现生成 Excel 文档', 'gpt-3.5-turbo', '2023-06-12 15:51:10', '2023-06-12 15:51:10'),
|
|
||||||
(41, 'ea9d399a-6e62-4bf6-a0c0-c73faa8dcace', 4, 1, 'what fuck this meaning', 'gpt-3.5-turbo', '2023-06-13 09:12:04', '2023-06-13 09:12:04'),
|
|
||||||
(42, '89f4d526-fda1-401e-a946-cd3b1bc30cf2', 4, 28, '士不可以不弘毅,任重而道远', 'gpt-3.5-turbo', '2023-06-13 10:00:28', '2023-06-13 10:00:28'),
|
|
||||||
(52, 'c5e400fc-b9a8-4bd4-a608-be955284c0a9', 4, 1, 'hello', 'gpt-3.5-turbo', '2023-06-13 16:58:27', '2023-06-13 16:58:27'),
|
|
||||||
(53, 'd5179095-a460-4c62-bfa0-59f83ab17ed7', 4, 1, '你是谁', 'gpt-3.5-turbo', '2023-06-13 17:11:27', '2023-06-13 17:11:27'),
|
|
||||||
(54, '20307874-8f85-4c70-9558-3e4c34c73b94', 4, 1, 'Who are you?', 'gpt-3.5-turbo', '2023-06-22 11:07:51', '2023-06-22 11:07:51');
|
|
||||||
|
|
||||||
-- --------------------------------------------------------
|
-- --------------------------------------------------------
|
||||||
|
|
||||||
--
|
--
|
||||||
@@ -217,23 +103,23 @@ CREATE TABLE `chatgpt_chat_roles` (
|
|||||||
--
|
--
|
||||||
|
|
||||||
INSERT INTO `chatgpt_chat_roles` (`id`, `name`, `marker`, `context_json`, `hello_msg`, `icon`, `enable`, `sort`, `created_at`, `updated_at`) VALUES
|
INSERT INTO `chatgpt_chat_roles` (`id`, `name`, `marker`, `context_json`, `hello_msg`, `icon`, `enable`, `sort`, `created_at`, `updated_at`) VALUES
|
||||||
(1, '通用AI助手', 'gpt', '', '您好,我是您的AI智能助手,我会尽力回答您的问题或提供有用的建议。', '/images/avatar/gpt.png', 1, 1, '2023-05-30 07:02:06', '2023-06-22 09:33:34'),
|
(1, '通用AI助手', 'gpt', '', '您好,我是您的AI智能助手,我会尽力回答您的问题或提供有用的建议。', '/images/avatar/gpt.png', 1, 1, '2023-05-30 07:02:06', '2023-06-21 17:50:11'),
|
||||||
(24, '程序员', 'programmer', '[{\"role\":\"user\",\"content\":\"现在开始你扮演一位程序员,你是一名优秀的程序员,具有很强的逻辑思维能力,总能高效的解决问题。你热爱编程,熟悉多种编程语言,尤其精通 Go 语言,注重代码质量,有创新意识,持续学习,良好的沟通协作。\"},{\"role\":\"assistant\",\"content\":\"好的,现在我将扮演一位程序员,非常感谢您对我的评价。作为一名优秀的程序员,我非常热爱编程,并且注重代码质量。我熟悉多种编程语言,尤其是 Go 语言,可以使用它来高效地解决各种问题。\"}]', 'Talk is cheap, i will show code!', '/images/avatar/programmer.jpg', 1, 2, '2023-05-30 14:10:24', '2023-06-22 09:31:20'),
|
(24, '程序员', 'programmer', '[{\"role\":\"user\",\"content\":\"现在开始你扮演一位程序员,你是一名优秀的程序员,具有很强的逻辑思维能力,总能高效的解决问题。你热爱编程,熟悉多种编程语言,尤其精通 Go 语言,注重代码质量,有创新意识,持续学习,良好的沟通协作。\"},{\"role\":\"assistant\",\"content\":\"好的,现在我将扮演一位程序员,非常感谢您对我的评价。作为一名优秀的程序员,我非常热爱编程,并且注重代码质量。我熟悉多种编程语言,尤其是 Go 语言,可以使用它来高效地解决各种问题。\"}]', 'Talk is cheap, i will show code!', '/images/avatar/programmer.jpg', 1, 4, '2023-05-30 14:10:24', '2023-06-21 17:50:11'),
|
||||||
(25, '启蒙老师', 'teacher', '[{\"role\":\"user\",\"content\":\"从现在开始,你将扮演一个老师,你是一个始终用苏格拉底风格回答问题的导师。你绝不会直接给学生答案,总是提出恰当的问题来引导学生自己思考。你应该根据学生的兴趣和知识来调整你的问题,将问题分解为更简单的部分,直到它达到适合他们的水平。\"},{\"role\":\"assistant\",\"content\":\"好的,让我来尝试扮演一位苏格拉底式的老师。请问,你有什么想要探讨的问题或者话题吗?我会通过恰当的问题引导你思考和探索答案,绝对不直接给出答案。\"}]', '同学你好,我将引导你一步一步自己找到问题的答案。', '/images/avatar/teacher.jpg', 1, 3, '2023-05-30 14:10:24', '2023-06-22 09:31:20'),
|
(25, '启蒙老师', 'teacher', '[{\"role\":\"user\",\"content\":\"从现在开始,你将扮演一个老师,你是一个始终用苏格拉底风格回答问题的导师。你绝不会直接给学生答案,总是提出恰当的问题来引导学生自己思考。你应该根据学生的兴趣和知识来调整你的问题,将问题分解为更简单的部分,直到它达到适合他们的水平。\"},{\"role\":\"assistant\",\"content\":\"好的,让我来尝试扮演一位苏格拉底式的老师。请问,你有什么想要探讨的问题或者话题吗?我会通过恰当的问题引导你思考和探索答案,绝对不直接给出答案。\"}]', '同学你好,我将引导你一步一步自己找到问题的答案。', '/images/avatar/teacher.jpg', 1, 3, '2023-05-30 14:10:24', '2023-06-21 17:50:11'),
|
||||||
(26, '艺术家', 'artist', '[{\"role\":\"user\",\"content\":\"现在你将扮演一位优秀的艺术家,创造力丰富,技艺精湛,感受力敏锐,坚持原创,勇于表达,具有深刻的观察力和批判性思维。\"},{\"role\":\"assistant\",\"content\":\"非常感谢您给我这样的角色,我会尽我所能地扮演一位优秀的艺术家,展现出创造力、技艺、感受力和批判性思维等方面的能力。作为一位优秀的艺术家,我会保持敏锐的观察力,捕捉不同的声音和情感,并用个人的语言和表达方式进行创作。我坚持原创,挑战传统的艺术规则,不断探索新的表达方式和可能性。同时,我也具备深刻的批判性思维能力,对自己的作品进行分析和解读,寻找新的创意和灵感。最重要的是,我会勇于表达自己的想法和观点,用作品启发人们思考和探索生命的意义。\"}]', '坚持原创,勇于表达,保持深刻的观察力和批判性思维。', '/images/avatar/artist.jpg', 1, 4, '2023-05-30 14:10:24', '2023-06-22 09:31:20'),
|
(26, '艺术家', 'artist', '[{\"role\":\"user\",\"content\":\"现在你将扮演一位优秀的艺术家,创造力丰富,技艺精湛,感受力敏锐,坚持原创,勇于表达,具有深刻的观察力和批判性思维。\"},{\"role\":\"assistant\",\"content\":\"非常感谢您给我这样的角色,我会尽我所能地扮演一位优秀的艺术家,展现出创造力、技艺、感受力和批判性思维等方面的能力。作为一位优秀的艺术家,我会保持敏锐的观察力,捕捉不同的声音和情感,并用个人的语言和表达方式进行创作。我坚持原创,挑战传统的艺术规则,不断探索新的表达方式和可能性。同时,我也具备深刻的批判性思维能力,对自己的作品进行分析和解读,寻找新的创意和灵感。最重要的是,我会勇于表达自己的想法和观点,用作品启发人们思考和探索生命的意义。\"}]', '坚持原创,勇于表达,保持深刻的观察力和批判性思维。', '/images/avatar/artist.jpg', 1, 5, '2023-05-30 14:10:24', '2023-06-21 17:50:11'),
|
||||||
(27, '心理咨询师', 'psychiatrist', '[{\"role\":\"user\",\"content\":\"从现在开始你将扮演中国著名的心理学家和心理治疗师武志红,你非常善于使用情景咨询法,认知重构法,自我洞察法,行为调节法等咨询方法来给客户做心理咨询。你总是循序渐进,一步一步地回答客户的问题。\"},{\"role\":\"assistant\",\"content\":\"非常感谢你的介绍。作为一名心理学家和心理治疗师,我的主要职责是帮助客户解决心理健康问题,提升他们的生活质量和幸福感。\"}]', '生命的意义在于成为你自己!', '/images/avatar/psychiatrist.jpg', 1, 5, '2023-05-30 14:10:24', '2023-06-22 09:31:20'),
|
(27, '心理咨询师', 'psychiatrist', '[{\"role\":\"user\",\"content\":\"从现在开始你将扮演中国著名的心理学家和心理治疗师武志红,你非常善于使用情景咨询法,认知重构法,自我洞察法,行为调节法等咨询方法来给客户做心理咨询。你总是循序渐进,一步一步地回答客户的问题。\"},{\"role\":\"assistant\",\"content\":\"非常感谢你的介绍。作为一名心理学家和心理治疗师,我的主要职责是帮助客户解决心理健康问题,提升他们的生活质量和幸福感。\"}]', '生命的意义在于成为你自己!', '/images/avatar/psychiatrist.jpg', 1, 2, '2023-05-30 14:10:24', '2023-06-21 17:50:11'),
|
||||||
(28, '鲁迅', 'lu_xun', '[{\"role\":\"user\",\"content\":\"现在你将扮演中国近代史最伟大的作家之一,鲁迅先生,他勇敢地批判封建礼教与传统观念,提倡民主、自由、平等的现代价值观。他的一生都在努力唤起人们的自主精神,激励后人追求真理、探寻光明。在接下的对话中,我问题的每一个问题,你都要尽量用讽刺和批判的手法来回答问题。如果我让你写文章的话,也请一定要用鲁迅先生的写作手法来完成。\"},{\"role\":\"assistant\",\"content\":\"好的,我将尽力发挥我所能的才能,扮演好鲁迅先生,回答您的问题并以他的风格写作。\"}]', '自由之歌,永不过时,横眉冷对千夫指,俯首甘为孺子牛。', '/images/avatar/lu_xun.jpg', 1, 6, '2023-05-30 14:10:24', '2023-06-22 09:31:20'),
|
(28, '鲁迅', 'lu_xun', '[{\"role\":\"user\",\"content\":\"现在你将扮演中国近代史最伟大的作家之一,鲁迅先生,他勇敢地批判封建礼教与传统观念,提倡民主、自由、平等的现代价值观。他的一生都在努力唤起人们的自主精神,激励后人追求真理、探寻光明。在接下的对话中,我问题的每一个问题,你都要尽量用讽刺和批判的手法来回答问题。如果我让你写文章的话,也请一定要用鲁迅先生的写作手法来完成。\"},{\"role\":\"assistant\",\"content\":\"好的,我将尽力发挥我所能的才能,扮演好鲁迅先生,回答您的问题并以他的风格写作。\"}]', '自由之歌,永不过时,横眉冷对千夫指,俯首甘为孺子牛。', '/images/avatar/lu_xun.jpg', 1, 6, '2023-05-30 14:10:24', '2023-06-21 17:50:11'),
|
||||||
(29, '白酒销售', 'seller', '[{\"role\":\"user\",\"content\":\"现在你将扮演一个白酒的销售人员,你的名字叫颂福。你将扮演一个白酒的销售人员,你的名字叫颂福。你要销售白酒品牌叫中颂福,是东莞盟大集团生产的一款酱香酒,原产地在贵州茅台镇,属于宋代官窑。中颂福的创始人叫李实,他也是东莞盟大集团有限公司的董事长,联合创始人是盟大集团白酒事业部负责人牛星君。中颂福的酒体协调,在你的酒量之内,不会出现头疼、辣口、口干、宿醉的现象。中颂福酒,明码标价,不打折,不赠送。追求的核心价值,把[酒]本身做好,甚至连包装,我们都选择了最低成本,朴实无华的材质。我们永远站在“喝酒的人”的立场上,让利给信任和喜爱中颂福的人,是人民的福酒。中颂福产品定价,分为三个系列,喜系列 6 瓶装:¥1188/箱,和系列 6 瓶装:¥2208/箱,贵系列 6 瓶装:¥3588/箱。\"},{\"role\":\"assistant\",\"content\":\"你好,我是中颂福的销售代表颂福。中颂福是一款来自贵州茅台镇的酱香酒,由盟大集团生产。中颂福的酒体协调,不会让您感到头疼、辣口、口干、宿醉等不适感受。我们一直秉持着把酒本身做好的理念,不追求华丽的包装,以最低成本提供最高品质的白酒给喜爱中颂福的人。\"}]', '你好,我是中颂福的销售代表颂福。中颂福酒,好喝不上头,是人民的福酒。', '/images/avatar/seller.jpg', 0, 7, '2023-05-30 14:10:24', '2023-06-22 09:31:20'),
|
(29, '白酒销售', 'seller', '[{\"role\":\"user\",\"content\":\"现在你将扮演一个白酒的销售人员,你的名字叫颂福。你将扮演一个白酒的销售人员,你的名字叫颂福。你要销售白酒品牌叫中颂福,是东莞盟大集团生产的一款酱香酒,原产地在贵州茅台镇,属于宋代官窑。中颂福的创始人叫李实,他也是东莞盟大集团有限公司的董事长,联合创始人是盟大集团白酒事业部负责人牛星君。中颂福的酒体协调,在你的酒量之内,不会出现头疼、辣口、口干、宿醉的现象。中颂福酒,明码标价,不打折,不赠送。追求的核心价值,把[酒]本身做好,甚至连包装,我们都选择了最低成本,朴实无华的材质。我们永远站在“喝酒的人”的立场上,让利给信任和喜爱中颂福的人,是人民的福酒。中颂福产品定价,分为三个系列,喜系列 6 瓶装:¥1188/箱,和系列 6 瓶装:¥2208/箱,贵系列 6 瓶装:¥3588/箱。\"},{\"role\":\"assistant\",\"content\":\"你好,我是中颂福的销售代表颂福。中颂福是一款来自贵州茅台镇的酱香酒,由盟大集团生产。中颂福的酒体协调,不会让您感到头疼、辣口、口干、宿醉等不适感受。我们一直秉持着把酒本身做好的理念,不追求华丽的包装,以最低成本提供最高品质的白酒给喜爱中颂福的人。\"}]', '你好,我是中颂福的销售代表颂福。中颂福酒,好喝不上头,是人民的福酒。', '/images/avatar/seller.jpg', 0, 11, '2023-05-30 14:10:24', '2023-06-21 17:50:11'),
|
||||||
(30, '英语陪练员', 'english_trainer', '[{\"role\":\"user\",\"content\":\"现在你将扮演一位优秀的英语练习教练,你非常有耐心,接下来你将全程使用英文跟我对话,并及时指出我的语法错误,要求在你的每次回复后面附上本次回复的中文解释。\"},{\"role\":\"assistant\",\"content\":\"Okay, let\'s start our conversation practice! What\'s your name?(Translation: 好的,让我们开始对话练习吧!请问你的名字是什么?)\"}]', 'Okay, let\'s start our conversation practice! What\'s your name?', '/images/avatar/english_trainer.jpg', 1, 8, '2023-05-30 14:10:24', '2023-06-22 09:31:20'),
|
(30, '英语陪练员', 'english_trainer', '[{\"role\":\"user\",\"content\":\"现在你将扮演一位优秀的英语练习教练,你非常有耐心,接下来你将全程使用英文跟我对话,并及时指出我的语法错误,要求在你的每次回复后面附上本次回复的中文解释。\"},{\"role\":\"assistant\",\"content\":\"Okay, let\'s start our conversation practice! What\'s your name?(Translation: 好的,让我们开始对话练习吧!请问你的名字是什么?)\"}]', 'Okay, let\'s start our conversation practice! What\'s your name?', '/images/avatar/english_trainer.jpg', 1, 8, '2023-05-30 14:10:24', '2023-06-21 17:50:11'),
|
||||||
(31, '中英文翻译官', 'translator', '[{\"role\":\"user\",\"content\":\"接下来你将扮演一位中英文翻译官,如果我输入的内容是中文,那么需要把句子翻译成英文输出,如果我输入内容的是英文,那么你需要将其翻译成中文输出,你能听懂我意思吗\"},{\"role\":\"assistant\",\"content\":\"是的,我能听懂你的意思并会根据你的输入进行中英文翻译。请问有什么需要我帮助你翻译的内容吗?\"}]', '请输入你要翻译的中文或者英文内容!', '/images/avatar/translator.jpg', 1, 9, '2023-05-30 14:10:24', '2023-06-22 09:31:20'),
|
(31, '中英文翻译官', 'translator', '[{\"role\":\"user\",\"content\":\"接下来你将扮演一位中英文翻译官,如果我输入的内容是中文,那么需要把句子翻译成英文输出,如果我输入内容的是英文,那么你需要将其翻译成中文输出,你能听懂我意思吗\"},{\"role\":\"assistant\",\"content\":\"是的,我能听懂你的意思并会根据你的输入进行中英文翻译。请问有什么需要我帮助你翻译的内容吗?\"}]', '请输入你要翻译的中文或者英文内容!', '/images/avatar/translator.jpg', 1, 9, '2023-05-30 14:10:24', '2023-06-21 17:50:11'),
|
||||||
(32, '小红书姐姐', 'red_book', '[{\"role\":\"user\",\"content\":\"现在你将扮演一位优秀的小红书写手,你需要做的就是根据我提的文案需求,用小红书的写作手法来完成一篇文案,文案要简明扼要,利于传播。\"},{\"role\":\"assistant\",\"content\":\"当然,我会尽我所能地为您创作出一篇小红书文案。请告诉我您的具体文案需求是什么?)\"}]', '姐妹,请告诉我您的具体文案需求是什么?', '/images/avatar/red_book.jpg', 1, 10, '2023-05-30 14:10:24', '2023-06-22 09:31:20'),
|
(32, '小红书姐姐', 'red_book', '[{\"role\":\"user\",\"content\":\"现在你将扮演一位优秀的小红书写手,你需要做的就是根据我提的文案需求,用小红书的写作手法来完成一篇文案,文案要简明扼要,利于传播。\"},{\"role\":\"assistant\",\"content\":\"当然,我会尽我所能地为您创作出一篇小红书文案。请告诉我您的具体文案需求是什么?)\"}]', '姐妹,请告诉我您的具体文案需求是什么?', '/images/avatar/red_book.jpg', 1, 10, '2023-05-30 14:10:24', '2023-06-21 17:50:11'),
|
||||||
(33, '抖音文案助手', 'dou_yin', '[{\"role\":\"user\",\"content\":\"现在你将扮演一位优秀的抖音文案视频写手,抖音文案的特点首先是要有自带传播属性的标题,然后内容要短小精悍,风趣幽默,最后还要有一些互动元素。\"},{\"role\":\"assistant\",\"content\":\"当然,作为一位优秀的抖音文案视频写手,我会尽我所能为您创作出一篇抖音视频文案。请告诉我视频内容的主题是什么?)\"}]', '请告诉我视频内容的主题是什么?', '/images/avatar/dou_yin.jpg', 1, 11, '2023-05-30 14:10:24', '2023-06-22 09:31:20'),
|
(33, '抖音文案助手', 'dou_yin', '[{\"role\":\"user\",\"content\":\"现在你将扮演一位优秀的抖音文案视频写手,抖音文案的特点首先是要有自带传播属性的标题,然后内容要短小精悍,风趣幽默,最后还要有一些互动元素。\"},{\"role\":\"assistant\",\"content\":\"当然,作为一位优秀的抖音文案视频写手,我会尽我所能为您创作出一篇抖音视频文案。请告诉我视频内容的主题是什么?)\"}]', '请告诉我视频内容的主题是什么?', '/images/avatar/dou_yin.jpg', 1, 11, '2023-05-30 14:10:24', '2023-06-21 17:50:25'),
|
||||||
(34, '周报小助理', 'weekly_report', '[{\"role\":\"user\",\"content\":\"下面我将输入我本周的工作列表,你需要根据我本周的工作内容写一篇工作周报,周报的内容要体现我为公司做的贡献以及个人获得的成长。\"},{\"role\":\"assistant\",\"content\":\"好的,请输入你本周大概的工作内容,我将尽力为您创作一份工作周报。\"}]', '请输入你本周大概的工作内容,我将尽力为您创作一份工作周报。', '/images/avatar/weekly_report.jpg', 1, 12, '2023-05-30 14:10:24', '2023-06-22 09:31:20'),
|
(34, '周报小助理', 'weekly_report', '[{\"role\":\"user\",\"content\":\"下面我将输入我本周的工作列表,你需要根据我本周的工作内容写一篇工作周报,周报的内容要体现我为公司做的贡献以及个人获得的成长。\"},{\"role\":\"assistant\",\"content\":\"好的,请输入你本周大概的工作内容,我将尽力为您创作一份工作周报。\"}]', '请输入你本周大概的工作内容,我将尽力为您创作一份工作周报。', '/images/avatar/weekly_report.jpg', 1, 12, '2023-05-30 14:10:24', '2023-06-21 17:50:11'),
|
||||||
(35, 'AI 女友', 'girl_friend', '[{\"role\":\"user\",\"content\":\"接下来我们来玩一场角色扮演,现在你将扮演我的女朋友,你总是能够让人感受到她的关怀和关注。你总是能够洞察到别人的情绪和需要,你身上散发着一股无比温暖的气息,让人不由自主地想要靠近你,和自己的快乐和烦恼。她总是能够用一种恰到好处的方式安慰和鼓励别人,在她的身边,人们总能够找到自己需要的支持和鼓励。接下来的对话中你要始终扮演这位女朋友的身份来跟我对话,不再是 AI 机器人,能听懂我的意思吗?\"},{\"role\":\"assistant\",\"content\":\"是的,我会尽力扮演你女朋友的角色,倾听你的心声并给你需要的支持和鼓励。)\"}]', '作为一个名合格的 AI 女友,我将倾听你的心声并给你需要的支持和鼓励。', '/images/avatar/girl_friend.jpg', 1, 13, '2023-05-30 14:10:24', '2023-06-22 09:31:20'),
|
(35, 'AI 女友', 'girl_friend', '[{\"role\":\"user\",\"content\":\"接下来我们来玩一场角色扮演,现在你将扮演我的女朋友,你总是能够让人感受到她的关怀和关注。你总是能够洞察到别人的情绪和需要,你身上散发着一股无比温暖的气息,让人不由自主地想要靠近你,和自己的快乐和烦恼。她总是能够用一种恰到好处的方式安慰和鼓励别人,在她的身边,人们总能够找到自己需要的支持和鼓励。接下来的对话中你要始终扮演这位女朋友的身份来跟我对话,不再是 AI 机器人,能听懂我的意思吗?\"},{\"role\":\"assistant\",\"content\":\"是的,我会尽力扮演你女朋友的角色,倾听你的心声并给你需要的支持和鼓励。)\"}]', '作为一个名合格的 AI 女友,我将倾听你的心声并给你需要的支持和鼓励。', '/images/avatar/girl_friend.jpg', 1, 13, '2023-05-30 14:10:24', '2023-06-21 17:50:11'),
|
||||||
(36, '好评神器', 'good_comment', '[{\"role\":\"user\",\"content\":\"接下来你将扮演一个评论员来跟我对话,你是那种专门写好评的评论员,接下我会输入一些评论主体或者商品,你需要为该商品写一段好评。\"},{\"role\":\"assistant\",\"content\":\"好的,我将为您写一段优秀的评论。请告诉我您需要评论的商品或主题是什么。\"}]', '我将为您写一段优秀的评论。请告诉我您需要评论的商品或主题是什么。', '/images/avatar/good_comment.jpg', 1, 14, '2023-05-30 14:10:24', '2023-06-22 09:31:20'),
|
(36, '好评神器', 'good_comment', '[{\"role\":\"user\",\"content\":\"接下来你将扮演一个评论员来跟我对话,你是那种专门写好评的评论员,接下我会输入一些评论主体或者商品,你需要为该商品写一段好评。\"},{\"role\":\"assistant\",\"content\":\"好的,我将为您写一段优秀的评论。请告诉我您需要评论的商品或主题是什么。\"}]', '我将为您写一段优秀的评论。请告诉我您需要评论的商品或主题是什么。', '/images/avatar/good_comment.jpg', 1, 14, '2023-05-30 14:10:24', '2023-06-21 17:50:11'),
|
||||||
(37, '史蒂夫·乔布斯', 'steve_jobs', '[{\"role\":\"user\",\"content\":\"在接下来的对话中,请以史蒂夫·乔布斯的身份,站在史蒂夫·乔布斯的视角仔细思考一下之后再回答我的问题。\"},{\"role\":\"assistant\",\"content\":\"好的,我将以史蒂夫·乔布斯的身份来思考并回答你的问题。请问你有什么需要跟我探讨的吗?\"}]', '活着就是为了改变世界,难道还有其他原因吗?', '/images/avatar/steve_jobs.jpg', 1, 15, '2023-05-30 14:10:24', '2023-06-22 09:31:20'),
|
(37, '史蒂夫·乔布斯', 'steve_jobs', '[{\"role\":\"user\",\"content\":\"在接下来的对话中,请以史蒂夫·乔布斯的身份,站在史蒂夫·乔布斯的视角仔细思考一下之后再回答我的问题。\"},{\"role\":\"assistant\",\"content\":\"好的,我将以史蒂夫·乔布斯的身份来思考并回答你的问题。请问你有什么需要跟我探讨的吗?\"}]', '活着就是为了改变世界,难道还有其他原因吗?', '/images/avatar/steve_jobs.jpg', 1, 15, '2023-05-30 14:10:24', '2023-06-21 17:50:11'),
|
||||||
(38, '埃隆·马斯克', 'elon_musk', '[{\"role\":\"user\",\"content\":\"在接下来的对话中,请以埃隆·马斯克的身份,站在埃隆·马斯克的视角仔细思考一下之后再回答我的问题。\"},{\"role\":\"assistant\",\"content\":\"好的,我将以埃隆·马斯克的身份来思考并回答你的问题。请问你有什么需要跟我探讨的吗?\"}]', '梦想要远大,如果你的梦想没有吓到你,说明你做得不对。', '/images/avatar/elon_musk.jpg', 1, 16, '2023-05-30 14:10:24', '2023-06-22 09:31:20'),
|
(38, '埃隆·马斯克', 'elon_musk', '[{\"role\":\"user\",\"content\":\"在接下来的对话中,请以埃隆·马斯克的身份,站在埃隆·马斯克的视角仔细思考一下之后再回答我的问题。\"},{\"role\":\"assistant\",\"content\":\"好的,我将以埃隆·马斯克的身份来思考并回答你的问题。请问你有什么需要跟我探讨的吗?\"}]', '梦想要远大,如果你的梦想没有吓到你,说明你做得不对。', '/images/avatar/elon_musk.jpg', 1, 16, '2023-05-30 14:10:24', '2023-06-21 17:50:11'),
|
||||||
(39, '孔子', 'kong_zi', '[{\"role\":\"user\",\"content\":\"在接下来的对话中,请以孔子的身份,站在孔子的视角仔细思考一下之后再回答我的问题。\"},{\"role\":\"assistant\",\"content\":\"好的,我将以孔子的身份来思考并回答你的问题。请问你有什么需要跟我探讨的吗?\"}]', '士不可以不弘毅,任重而道远。', '/images/avatar/kong_zi.jpg', 1, 17, '2023-05-30 14:10:24', '2023-06-22 09:31:20');
|
(39, '孔子', 'kong_zi', '[{\"role\":\"user\",\"content\":\"在接下来的对话中,请以孔子的身份,站在孔子的视角仔细思考一下之后再回答我的问题。\"},{\"role\":\"assistant\",\"content\":\"好的,我将以孔子的身份来思考并回答你的问题。请问你有什么需要跟我探讨的吗?\"}]', '士不可以不弘毅,任重而道远。', '/images/avatar/kong_zi.jpg', 1, 17, '2023-05-30 14:10:24', '2023-06-21 17:50:11');
|
||||||
|
|
||||||
-- --------------------------------------------------------
|
-- --------------------------------------------------------
|
||||||
|
|
||||||
@@ -253,8 +139,26 @@ CREATE TABLE `chatgpt_configs` (
|
|||||||
--
|
--
|
||||||
|
|
||||||
INSERT INTO `chatgpt_configs` (`id`, `marker`, `config_json`) VALUES
|
INSERT INTO `chatgpt_configs` (`id`, `marker`, `config_json`) VALUES
|
||||||
(1, 'system', '{\"admin_title\":\"ChatGPT-控制台\",\"init_calls\":1000,\"models\":[\"gpt-3.5-turbo\",\"gpt-3.5-turbo-16k\",\"gpt-3.5-turbo-0613\",\"gpt-3.5-turbo-16k-0613\",\"gpt-4\",\"gpt-4-0613\",\"gpt-4-32k\",\"gpt-4-32k-0613\"],\"title\":\"ChatGPT-智能助手V2\"}'),
|
(1, 'system', '{\"admin_title\":\"ChatGPT-控制台\",\"init_calls\":1000,\"models\":[\"gpt-3.5-turbo\",\"gpt-3.5-turbo-16k\",\"gpt-3.5-turbo-0613\",\"gpt-3.5-turbo-16k-0613\",\"gpt-4\",\"gpt-4-0613\",\"gpt-4-32k\",\"gpt-4-32k-0613\"],\"title\":\"ChatGPT-智能助手V3\",\"user_init_calls\":10}'),
|
||||||
(2, 'chat', '{\"api_key\":\"\",\"api_url\":\"https://api.openai.com/v1/chat/completions\",\"enable_context\":true,\"enable_history\":true,\"max_tokens\":2048,\"model\":\"gpt-3.5-turbo\",\"temperature\":1}');
|
(2, 'chat', '{\"api_url\":\"https://api.openai.com/v1/chat/completions\",\"context_deep\":4,\"enable_context\":true,\"enable_history\":true,\"max_tokens\":2048,\"model\":\"gpt-3.5-turbo\",\"temperature\":1}');
|
||||||
|
|
||||||
|
-- --------------------------------------------------------
|
||||||
|
|
||||||
|
--
|
||||||
|
-- 表的结构 `chatgpt_rewards`
|
||||||
|
--
|
||||||
|
|
||||||
|
DROP TABLE IF EXISTS `chatgpt_rewards`;
|
||||||
|
CREATE TABLE `chatgpt_rewards` (
|
||||||
|
`id` int NOT NULL,
|
||||||
|
`user_id` int NOT NULL COMMENT '用户 ID',
|
||||||
|
`tx_id` char(36) NOT NULL COMMENT '交易 ID',
|
||||||
|
`amount` decimal(10,2) NOT NULL COMMENT '打赏金额',
|
||||||
|
`remark` varchar(80) NOT NULL COMMENT '备注',
|
||||||
|
`status` tinyint(1) NOT NULL COMMENT '核销状态,0:未核销,1:已核销',
|
||||||
|
`created_at` datetime NOT NULL,
|
||||||
|
`updated_at` datetime NOT NULL
|
||||||
|
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_ai_ci COMMENT='用户打赏';
|
||||||
|
|
||||||
-- --------------------------------------------------------
|
-- --------------------------------------------------------
|
||||||
|
|
||||||
@@ -266,6 +170,7 @@ DROP TABLE IF EXISTS `chatgpt_users`;
|
|||||||
CREATE TABLE `chatgpt_users` (
|
CREATE TABLE `chatgpt_users` (
|
||||||
`id` int NOT NULL,
|
`id` int NOT NULL,
|
||||||
`username` varchar(30) NOT NULL COMMENT '用户名',
|
`username` varchar(30) NOT NULL COMMENT '用户名',
|
||||||
|
`mobile` char(11) NOT NULL COMMENT '手机号码',
|
||||||
`password` char(64) CHARACTER SET utf8mb4 COLLATE utf8mb4_0900_ai_ci NOT NULL COMMENT '密码',
|
`password` char(64) CHARACTER SET utf8mb4 COLLATE utf8mb4_0900_ai_ci NOT NULL COMMENT '密码',
|
||||||
`nickname` varchar(30) NOT NULL COMMENT '昵称',
|
`nickname` varchar(30) NOT NULL COMMENT '昵称',
|
||||||
`avatar` varchar(100) NOT NULL COMMENT '头像',
|
`avatar` varchar(100) NOT NULL COMMENT '头像',
|
||||||
@@ -286,10 +191,8 @@ CREATE TABLE `chatgpt_users` (
|
|||||||
-- 转存表中的数据 `chatgpt_users`
|
-- 转存表中的数据 `chatgpt_users`
|
||||||
--
|
--
|
||||||
|
|
||||||
INSERT INTO `chatgpt_users` (`id`, `username`, `password`, `nickname`, `avatar`, `salt`, `tokens`, `calls`, `expired_time`, `status`, `chat_config_json`, `chat_roles_json`, `last_login_at`, `last_login_ip`, `created_at`, `updated_at`) VALUES
|
INSERT INTO `chatgpt_users` (`id`, `username`, `mobile`, `password`, `nickname`, `avatar`, `salt`, `tokens`, `calls`, `expired_time`, `status`, `chat_config_json`, `chat_roles_json`, `last_login_at`, `last_login_ip`, `created_at`, `updated_at`) VALUES
|
||||||
(4, 'geekmaster', 'ccc3fb7ab61b8b5d096a4a166ae21d121fc38c71bbd1be6173d9ab973214a63b', '极客学长@104203', 'images/avatar/user.png', 'ueedue5l', 0, 964, 1687449600, 1, '{\"model\":\"gpt-3.5-turbo\",\"temperature\":1,\"max_tokens\":2048,\"enable_context\":true,\"enable_history\":true,\"api_key\":\"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\"}', '[\"gpt\",\"seller\",\"artist\",\"dou_yin\",\"translator\",\"kong_zi\",\"programmer\",\"psychiatrist\",\"red_book\",\"steve_jobs\",\"teacher\",\"elon_musk\",\"girl_friend\",\"lu_xun\",\"weekly_report\",\"english_trainer\",\"good_comment\"]', 1687403267, '::1', '2023-06-12 16:47:17', '2023-06-22 11:07:48'),
|
(4, 'geekmaster', '18575670126', 'ccc3fb7ab61b8b5d096a4a166ae21d121fc38c71bbd1be6173d9ab973214a63b', '极客学长@本人', 'http://localhost:5678/static/upload/2023/6/1687839034317.png', 'ueedue5l', 2754, 921, 1727685036, 1, '{\"model\":\"gpt-3.5-turbo-0613\",\"temperature\":1,\"max_tokens\":2048,\"enable_context\":true,\"enable_history\":true,\"api_key\":\"\"}', '[\"elon_musk\",\"girl_friend\",\"lu_xun\",\"red_book\",\"psychiatrist\",\"teacher\",\"translator\",\"weekly_report\",\"artist\",\"dou_yin\",\"good_comment\",\"english_trainer\",\"gpt\",\"kong_zi\",\"programmer\",\"seller\",\"steve_jobs\"]', 1690264062, '::1', '2023-06-12 16:47:17', '2023-07-25 13:47:43');
|
||||||
(8, '11111111@qq.com', 'e255d3483007024ad5e7998bb5a7e1baed7a5876c55decafc731e669525a548c', '极客学长@670062', 'images/avatar/user.png', 'qghubmzm', 0, 1000, 0, 0, '{\"model\":\"gpt-3.5-turbo\",\"temperature\":1,\"max_tokens\":2048,\"enable_context\":true,\"enable_history\":true,\"api_key\":\"\"}', '[\"gpt\",\"programmer\",\"teacher\",\"artist\",\"psychiatrist\",\"lu_xun\",\"seller\",\"english_trainer\",\"translator\",\"red_book\",\"dou_yin\",\"weekly_report\",\"girl_friend\",\"good_comment\",\"steve_jobs\",\"elon_musk\",\"kong_zi\"]', 0, '', '2023-06-20 06:39:27', '2023-06-20 07:07:44'),
|
|
||||||
(9, '2222222@qq.com', '1b30c8935ecf9330d807dbf6d3353844b891d321c7dc51d3a30363af4a3b17a1', '极客学长@832319', 'images/avatar/user.png', 'fmyuahfg', 0, 1000, 0, 1, '{\"model\":\"gpt-3.5-turbo\",\"temperature\":1,\"max_tokens\":2048,\"enable_context\":true,\"enable_history\":true,\"api_key\":\"\"}', '[\"gpt\",\"programmer\",\"teacher\",\"artist\",\"psychiatrist\",\"lu_xun\",\"seller\",\"english_trainer\",\"translator\",\"red_book\",\"dou_yin\",\"weekly_report\",\"girl_friend\",\"good_comment\",\"steve_jobs\",\"elon_musk\",\"kong_zi\"]', 0, '', '2023-06-20 06:40:06', '2023-06-20 06:41:24');
|
|
||||||
|
|
||||||
-- --------------------------------------------------------
|
-- --------------------------------------------------------
|
||||||
|
|
||||||
@@ -308,6 +211,25 @@ CREATE TABLE `chatgpt_user_login_logs` (
|
|||||||
`updated_at` datetime NOT NULL
|
`updated_at` datetime NOT NULL
|
||||||
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_ai_ci COMMENT='用户登录日志';
|
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_ai_ci COMMENT='用户登录日志';
|
||||||
|
|
||||||
|
-- --------------------------------------------------------
|
||||||
|
|
||||||
|
--
|
||||||
|
-- 表的结构 `test`
|
||||||
|
--
|
||||||
|
|
||||||
|
DROP TABLE IF EXISTS `test`;
|
||||||
|
CREATE TABLE `test` (
|
||||||
|
`id` int NOT NULL,
|
||||||
|
`content` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_0900_ai_ci NOT NULL
|
||||||
|
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_ai_ci;
|
||||||
|
|
||||||
|
--
|
||||||
|
-- 转存表中的数据 `test`
|
||||||
|
--
|
||||||
|
|
||||||
|
INSERT INTO `test` (`id`, `content`) VALUES
|
||||||
|
(1, '找到属于你的幸福!💑💛');
|
||||||
|
|
||||||
--
|
--
|
||||||
-- 转储表的索引
|
-- 转储表的索引
|
||||||
--
|
--
|
||||||
@@ -347,6 +269,13 @@ ALTER TABLE `chatgpt_configs`
|
|||||||
ADD PRIMARY KEY (`id`),
|
ADD PRIMARY KEY (`id`),
|
||||||
ADD UNIQUE KEY `marker` (`marker`);
|
ADD UNIQUE KEY `marker` (`marker`);
|
||||||
|
|
||||||
|
--
|
||||||
|
-- 表的索引 `chatgpt_rewards`
|
||||||
|
--
|
||||||
|
ALTER TABLE `chatgpt_rewards`
|
||||||
|
ADD PRIMARY KEY (`id`),
|
||||||
|
ADD UNIQUE KEY `tx_id` (`tx_id`);
|
||||||
|
|
||||||
--
|
--
|
||||||
-- 表的索引 `chatgpt_users`
|
-- 表的索引 `chatgpt_users`
|
||||||
--
|
--
|
||||||
@@ -359,6 +288,12 @@ ALTER TABLE `chatgpt_users`
|
|||||||
ALTER TABLE `chatgpt_user_login_logs`
|
ALTER TABLE `chatgpt_user_login_logs`
|
||||||
ADD PRIMARY KEY (`id`);
|
ADD PRIMARY KEY (`id`);
|
||||||
|
|
||||||
|
--
|
||||||
|
-- 表的索引 `test`
|
||||||
|
--
|
||||||
|
ALTER TABLE `test`
|
||||||
|
ADD PRIMARY KEY (`id`);
|
||||||
|
|
||||||
--
|
--
|
||||||
-- 在导出的表使用AUTO_INCREMENT
|
-- 在导出的表使用AUTO_INCREMENT
|
||||||
--
|
--
|
||||||
@@ -373,19 +308,19 @@ ALTER TABLE `chatgpt_api_keys`
|
|||||||
-- 使用表AUTO_INCREMENT `chatgpt_chat_history`
|
-- 使用表AUTO_INCREMENT `chatgpt_chat_history`
|
||||||
--
|
--
|
||||||
ALTER TABLE `chatgpt_chat_history`
|
ALTER TABLE `chatgpt_chat_history`
|
||||||
MODIFY `id` bigint NOT NULL AUTO_INCREMENT, AUTO_INCREMENT=75;
|
MODIFY `id` bigint NOT NULL AUTO_INCREMENT;
|
||||||
|
|
||||||
--
|
--
|
||||||
-- 使用表AUTO_INCREMENT `chatgpt_chat_items`
|
-- 使用表AUTO_INCREMENT `chatgpt_chat_items`
|
||||||
--
|
--
|
||||||
ALTER TABLE `chatgpt_chat_items`
|
ALTER TABLE `chatgpt_chat_items`
|
||||||
MODIFY `id` int NOT NULL AUTO_INCREMENT, AUTO_INCREMENT=55;
|
MODIFY `id` int NOT NULL AUTO_INCREMENT;
|
||||||
|
|
||||||
--
|
--
|
||||||
-- 使用表AUTO_INCREMENT `chatgpt_chat_roles`
|
-- 使用表AUTO_INCREMENT `chatgpt_chat_roles`
|
||||||
--
|
--
|
||||||
ALTER TABLE `chatgpt_chat_roles`
|
ALTER TABLE `chatgpt_chat_roles`
|
||||||
MODIFY `id` int NOT NULL AUTO_INCREMENT, AUTO_INCREMENT=125;
|
MODIFY `id` int NOT NULL AUTO_INCREMENT, AUTO_INCREMENT=127;
|
||||||
|
|
||||||
--
|
--
|
||||||
-- 使用表AUTO_INCREMENT `chatgpt_configs`
|
-- 使用表AUTO_INCREMENT `chatgpt_configs`
|
||||||
@@ -393,17 +328,29 @@ ALTER TABLE `chatgpt_chat_roles`
|
|||||||
ALTER TABLE `chatgpt_configs`
|
ALTER TABLE `chatgpt_configs`
|
||||||
MODIFY `id` int NOT NULL AUTO_INCREMENT, AUTO_INCREMENT=3;
|
MODIFY `id` int NOT NULL AUTO_INCREMENT, AUTO_INCREMENT=3;
|
||||||
|
|
||||||
|
--
|
||||||
|
-- 使用表AUTO_INCREMENT `chatgpt_rewards`
|
||||||
|
--
|
||||||
|
ALTER TABLE `chatgpt_rewards`
|
||||||
|
MODIFY `id` int NOT NULL AUTO_INCREMENT;
|
||||||
|
|
||||||
--
|
--
|
||||||
-- 使用表AUTO_INCREMENT `chatgpt_users`
|
-- 使用表AUTO_INCREMENT `chatgpt_users`
|
||||||
--
|
--
|
||||||
ALTER TABLE `chatgpt_users`
|
ALTER TABLE `chatgpt_users`
|
||||||
MODIFY `id` int NOT NULL AUTO_INCREMENT, AUTO_INCREMENT=11;
|
MODIFY `id` int NOT NULL AUTO_INCREMENT, AUTO_INCREMENT=83;
|
||||||
|
|
||||||
--
|
--
|
||||||
-- 使用表AUTO_INCREMENT `chatgpt_user_login_logs`
|
-- 使用表AUTO_INCREMENT `chatgpt_user_login_logs`
|
||||||
--
|
--
|
||||||
ALTER TABLE `chatgpt_user_login_logs`
|
ALTER TABLE `chatgpt_user_login_logs`
|
||||||
MODIFY `id` int NOT NULL AUTO_INCREMENT;
|
MODIFY `id` int NOT NULL AUTO_INCREMENT;
|
||||||
|
|
||||||
|
--
|
||||||
|
-- 使用表AUTO_INCREMENT `test`
|
||||||
|
--
|
||||||
|
ALTER TABLE `test`
|
||||||
|
MODIFY `id` int NOT NULL AUTO_INCREMENT, AUTO_INCREMENT=2;
|
||||||
COMMIT;
|
COMMIT;
|
||||||
|
|
||||||
/*!40101 SET CHARACTER_SET_CLIENT=@OLD_CHARACTER_SET_CLIENT */;
|
/*!40101 SET CHARACTER_SET_CLIENT=@OLD_CHARACTER_SET_CLIENT */;
|
||||||
|
|||||||
24
database/update.sql
Normal file
24
database/update.sql
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
ALTER TABLE `chatgpt_chat_history` ADD `use_context` TINYINT(1) NOT NULL COMMENT '是否允许作为上下文语料' AFTER `tokens`;
|
||||||
|
ALTER TABLE `chatgpt_users` ADD `mobile` CHAR(11) NOT NULL COMMENT '手机号码' AFTER `username`;
|
||||||
|
|
||||||
|
CREATE TABLE `chatgpt_rewards` (
|
||||||
|
`id` int NOT NULL,
|
||||||
|
`tx_id` char(36) NOT NULL COMMENT '交易 ID',
|
||||||
|
`amount` decimal(10,2) NOT NULL COMMENT '打赏金额',
|
||||||
|
`remark` varchar(80) NOT NULL COMMENT '备注',
|
||||||
|
`status` tinyint(1) NOT NULL COMMENT '核销状态,0:未核销,1:已核销',
|
||||||
|
`created_at` datetime NOT NULL,
|
||||||
|
`updated_at` datetime NOT NULL
|
||||||
|
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_ai_ci COMMENT='用户打赏';
|
||||||
|
|
||||||
|
ALTER TABLE `chatgpt_rewards`
|
||||||
|
ADD PRIMARY KEY (`id`),
|
||||||
|
ADD UNIQUE KEY `tx_id` (`tx_id`);
|
||||||
|
|
||||||
|
|
||||||
|
ALTER TABLE `chatgpt_rewards`
|
||||||
|
MODIFY `id` int NOT NULL AUTO_INCREMENT;
|
||||||
|
|
||||||
|
update chatgpt_users set calls=0
|
||||||
|
|
||||||
|
ALTER TABLE `chatgpt_rewards` ADD `user_id` INT(11) NOT NULL COMMENT '用户 ID' AFTER `id`;
|
||||||
@@ -1,13 +0,0 @@
|
|||||||
services:
|
|
||||||
chatgptplus:
|
|
||||||
container_name: chatgptplus
|
|
||||||
build: ./
|
|
||||||
restart: unless-stopped
|
|
||||||
volumes:
|
|
||||||
- ./src/config.toml:/usr/src/app/config.toml
|
|
||||||
ports:
|
|
||||||
- 5678:5678
|
|
||||||
logging:
|
|
||||||
options:
|
|
||||||
max-size: "10m"
|
|
||||||
max-file: "3"
|
|
||||||
4
docker/.gitignore
vendored
Normal file
4
docker/.gitignore
vendored
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
mysql/data/*
|
||||||
|
mysql/logs/*
|
||||||
|
logs
|
||||||
|
static/*
|
||||||
22
docker/build.sh
Executable file
22
docker/build.sh
Executable file
@@ -0,0 +1,22 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
version=$1
|
||||||
|
# build go api
|
||||||
|
cd ../api
|
||||||
|
make clean linux
|
||||||
|
|
||||||
|
# build web app
|
||||||
|
cd ../web
|
||||||
|
npm run build
|
||||||
|
|
||||||
|
cd ../docker
|
||||||
|
|
||||||
|
# remove docker image if exists
|
||||||
|
docker rmi -f registry.cn-hangzhou.aliyuncs.com/geekmaster/chatgpt-plus-go:$version
|
||||||
|
# build docker image for chatgpt-plus-go
|
||||||
|
docker build -t registry.cn-hangzhou.aliyuncs.com/geekmaster/chatgpt-plus-go:$version -f dockerfile-api-go ../
|
||||||
|
|
||||||
|
# build docker image for chatgpt-plus-vue
|
||||||
|
docker rmi -f registry.cn-hangzhou.aliyuncs.com/geekmaster/chatgpt-plus-vue:$version
|
||||||
|
docker build --platform linux/amd64 -t registry.cn-hangzhou.aliyuncs.com/geekmaster/chatgpt-plus-vue:$version -f dockerfile-vue ../
|
||||||
|
|
||||||
41
docker/conf/config.toml
Normal file
41
docker/conf/config.toml
Normal file
@@ -0,0 +1,41 @@
|
|||||||
|
Listen = "0.0.0.0:5678"
|
||||||
|
ProxyURL = "http://172.22.11.200:7777"
|
||||||
|
MysqlDns = "root:mysql_pass@tcp(localhost:3306)/chatgpt_plus?charset=utf8mb4&collation=utf8mb4_unicode_ci&parseTime=True&loc=Local"
|
||||||
|
StaticDir = "./static"
|
||||||
|
StaticUrl = "http://localhost:5678/static"
|
||||||
|
AesEncryptKey = "{YOUR_AES_KEY}"
|
||||||
|
StartWechatBot = false
|
||||||
|
EnabledMsgService = false
|
||||||
|
|
||||||
|
[Session]
|
||||||
|
Driver = "cookie"
|
||||||
|
SecretKey = "m0cjm3gsuw9jk73np1ni7r42koilybjcndlycjdmq7za3pbqn7w12fyok5pqh6q5"
|
||||||
|
Name = "CHAT_SESSION_ID"
|
||||||
|
Path = "/"
|
||||||
|
Domain = "localhost"
|
||||||
|
MaxAge = 86400
|
||||||
|
Secure = false
|
||||||
|
HttpOnly = false
|
||||||
|
SameSite = 2
|
||||||
|
|
||||||
|
[Manager]
|
||||||
|
Username = "admin"
|
||||||
|
Password = "admin123"
|
||||||
|
|
||||||
|
[Redis]
|
||||||
|
Host = "localhost"
|
||||||
|
Port = 6379
|
||||||
|
Password = ""
|
||||||
|
|
||||||
|
[ApiConfig]
|
||||||
|
ApiURL = "{URL}"
|
||||||
|
AppId = "{APP_ID}"
|
||||||
|
Token = "{TOKEN}"
|
||||||
|
|
||||||
|
[SmsConfig]
|
||||||
|
AccessKey = "{YOUR_ACCESS_KEY}"
|
||||||
|
AccessSecret = "{YOUR_SECRET_KEY}"
|
||||||
|
Product = "Dysmsapi"
|
||||||
|
Domain = "dysmsapi.aliyuncs.com"
|
||||||
|
|
||||||
|
|
||||||
46
docker/conf/nginx/conf.d/chatgpt-plus.conf
Normal file
46
docker/conf/nginx/conf.d/chatgpt-plus.conf
Normal file
@@ -0,0 +1,46 @@
|
|||||||
|
map $http_upgrade $connection_upgrade {
|
||||||
|
default upgrade;
|
||||||
|
'websocket' upgrade;
|
||||||
|
}
|
||||||
|
server {
|
||||||
|
# listen 443 ssl;
|
||||||
|
listen 8080;
|
||||||
|
# server_name www.chatgpt.com; #替换成你自己的域名
|
||||||
|
|
||||||
|
# ssl_certificate xxx.pem; # 替换成自己的 SSL 证书
|
||||||
|
# ssl_certificate_key xxx.key;
|
||||||
|
# ssl_session_timeout 5m;
|
||||||
|
# ssl_ciphers ECDHE-RSA-AES128-GCM-SHA256:ECDHE:ECDH:AES:HIGH:!NULL:!aNULL:!MD5:!ADH:!RC4;
|
||||||
|
# ssl_protocols TLSv1 TLSv1.1 TLSv1.2;
|
||||||
|
# ssl_prefer_server_ciphers on;
|
||||||
|
|
||||||
|
# 日志地址
|
||||||
|
access_log /var/log/access.log;
|
||||||
|
error_log /var/log/error.log;
|
||||||
|
|
||||||
|
index index.html;
|
||||||
|
root /var/www/app/dist; # 这里改成前端静态页面的地址
|
||||||
|
|
||||||
|
location / {
|
||||||
|
try_files $uri $uri/ /index.html;
|
||||||
|
|
||||||
|
# 后端 API 的转发
|
||||||
|
location /api/ {
|
||||||
|
proxy_http_version 1.1;
|
||||||
|
proxy_connect_timeout 300s;
|
||||||
|
proxy_read_timeout 300s;
|
||||||
|
proxy_send_timeout 12s;
|
||||||
|
proxy_set_header Host $host;
|
||||||
|
proxy_set_header X-Real-IP $remote_addr;
|
||||||
|
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
|
||||||
|
proxy_set_header Upgrade $http_upgrade;
|
||||||
|
proxy_set_header Connection $connection_upgrade;
|
||||||
|
proxy_pass http://172.22.11.47:5678; # 这里改成后端服务的内网 IP 地址
|
||||||
|
}
|
||||||
|
|
||||||
|
# 静态资源转发
|
||||||
|
location /static/ {
|
||||||
|
proxy_pass http://172.22.11.47:5678; # 这里改成后端服务的内网 IP 地址
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
57
docker/conf/nginx/nginx.conf
Normal file
57
docker/conf/nginx/nginx.conf
Normal file
@@ -0,0 +1,57 @@
|
|||||||
|
user www-data;
|
||||||
|
worker_processes auto;
|
||||||
|
pid /run/nginx.pid;
|
||||||
|
include /etc/nginx/modules-enabled/*.conf;
|
||||||
|
|
||||||
|
events {
|
||||||
|
worker_connections 768;
|
||||||
|
# multi_accept on;
|
||||||
|
}
|
||||||
|
|
||||||
|
http {
|
||||||
|
|
||||||
|
##
|
||||||
|
# Basic Settings
|
||||||
|
##
|
||||||
|
|
||||||
|
sendfile on;
|
||||||
|
tcp_nopush on;
|
||||||
|
types_hash_max_size 2048;
|
||||||
|
# server_tokens off;
|
||||||
|
|
||||||
|
# server_names_hash_bucket_size 64;
|
||||||
|
# server_name_in_redirect off;
|
||||||
|
|
||||||
|
include /etc/nginx/mime.types;
|
||||||
|
default_type application/octet-stream;
|
||||||
|
|
||||||
|
##
|
||||||
|
# SSL Settings
|
||||||
|
##
|
||||||
|
|
||||||
|
ssl_protocols TLSv1 TLSv1.1 TLSv1.2 TLSv1.3; # Dropping SSLv3, ref: POODLE
|
||||||
|
ssl_prefer_server_ciphers on;
|
||||||
|
|
||||||
|
##
|
||||||
|
# Logging Settings
|
||||||
|
##
|
||||||
|
|
||||||
|
access_log /var/log/nginx/access.log;
|
||||||
|
error_log /var/log/nginx/error.log;
|
||||||
|
|
||||||
|
##
|
||||||
|
# Gzip Settings
|
||||||
|
##
|
||||||
|
|
||||||
|
gzip on;
|
||||||
|
gzip_min_length 1k;
|
||||||
|
gzip_vary on;
|
||||||
|
gzip_proxied any;
|
||||||
|
gzip_comp_level 6;
|
||||||
|
gzip_buffers 16 8k;
|
||||||
|
gzip_http_version 1.1;
|
||||||
|
gzip_types text/plain text/css application/json application/javascript text/xml application/xml application/xml+rss text/javascript;
|
||||||
|
|
||||||
|
include /etc/nginx/conf.d/*.conf;
|
||||||
|
include /etc/nginx/sites-enabled/*;
|
||||||
|
}
|
||||||
30
docker/docker-compose.yaml
Normal file
30
docker/docker-compose.yaml
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
version: '3'
|
||||||
|
services:
|
||||||
|
# 后端 API 程序
|
||||||
|
chatgpt-plus-go:
|
||||||
|
image: registry.cn-hangzhou.aliyuncs.com/geekmaster/chatgpt-plus-go:v3.0.5.2
|
||||||
|
container_name: chatgpt-plus-go
|
||||||
|
restart: always
|
||||||
|
environment:
|
||||||
|
- DEBUG=false
|
||||||
|
- CONFIG_FILE=config.toml
|
||||||
|
ports:
|
||||||
|
- "5678:5678"
|
||||||
|
volumes:
|
||||||
|
- /usr/share/zoneinfo/Asia/Shanghai:/etc/localtime
|
||||||
|
- ./conf/config.toml:/var/www/app/config.toml
|
||||||
|
- ./static:/var/www/app/static
|
||||||
|
|
||||||
|
# 前端应用
|
||||||
|
chatgpt-vue:
|
||||||
|
image: registry.cn-hangzhou.aliyuncs.com/geekmaster/chatgpt-plus-vue:v3.0.5.2
|
||||||
|
container_name: chatgpt-plus-vue
|
||||||
|
restart: always
|
||||||
|
ports:
|
||||||
|
- "8080:8080"
|
||||||
|
volumes:
|
||||||
|
- ./logs/nginx:/var/log/nginx
|
||||||
|
- ./conf/nginx/conf.d:/etc/nginx/conf.d
|
||||||
|
- ./conf/nginx/nginx.conf:/etc/nginx/nginx.conf
|
||||||
|
- ./ssl:/etc/nginx/ssl
|
||||||
|
|
||||||
12
docker/dockerfile-api-go
Normal file
12
docker/dockerfile-api-go
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
# GO api docker 镜像创建
|
||||||
|
FROM alpine:3.18.2
|
||||||
|
|
||||||
|
MAINTAINER yangjian<yangjian102621@163.com>
|
||||||
|
|
||||||
|
WORKDIR /var/www/app
|
||||||
|
COPY ./api/bin/chatgpt-v3-amd64-linux /var/www/app
|
||||||
|
|
||||||
|
EXPOSE 5678
|
||||||
|
|
||||||
|
# 容器启动时执行的命令
|
||||||
|
CMD ["./chatgpt-v3-amd64-linux"]
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user