直接先上gitee!!! 基于go-cqhttp的qqbot: java实现基于cqhttp,ws连接 由于当前chatgpt热度不断,网上众多的基于chatgpt提供的api做到chat网站层出不穷,但这些网址每次都得找,有些麻烦,我就在想能不能直接QQ接入chatgpt,想法一出,开始找资料开干!

1.申请一个openai官网的key

这一步首先你得有一个翻墙梯子,去openai官网来注册一个账号,这一步不会的猿猿先去找找这种注册的教程。 注册完毕后,我们去openai官网个人首页去申请一个API keys

到此,就拿到了openai的APIkey

2.go-cqhttp的使用

使用 mirai 以及 MiraiGo 开发的 cqhttp golang 原生实现, 并在 cqhttp 原版 的基础上做了部分修改和拓展。 使用这个的目的是为了接入qq。 直接上它的git: GitHub - Mrs4s/go-cqhttp: cqhttp的golang实现,轻量、原生跨平台. 之后下载它 Releases · Mrs4s/go-cqhttp · GitHub

系统类型 可执行文件 压缩文件
Intel 版 Macos Not available go-cqhttp_darwin_amd64.tar.gz
M1 版 Macos Not available go-cqhttp_darwin_arm64.tar.gz
32 位 Linux Not available go-cqhttp_linux_386.tar.gz
64 位 Linux Not available go-cqhttp_linux_amd64.tar.gz
arm64 Linux Not available go-cqhttp_linux_arm64.tar.gz
armv7 Linux Not available go-cqhttp_linux_armv7.tar.gz
32 位 Windows go-cqhttp_windows_386.exe go-cqhttp_windows_386.zip
64 位 Windows go-cqhttp_windows_amd64.exe go-cqhttp_windows_amd64.zip
arm64 Windows go-cqhttp_windows_arm64.exe go-cqhttp_windows_arm64.zip
armv7 Windows go-cqhttp_windows_armv7.exe go-cqhttp_windows_armv7.zip

我先用window来进行讲解了:

cmd运行它,我们先使用正向websocket连接 选择完毕,我们可以看到一个config.yml配置文件 我们来进行配置: 修改device.json文件: 将这个protocol设为2(手表)或0(mac)才可以扫码登录! 修改完后,重新启动该程序,完成qq的登录验证,正常完成验证的情况下,就会出现登录成功! 注意一定要用扫码登录!由于QQ风控越来越严了,所以用密码登录会报错45或235 到此,这个go-cqhttp服务就已经启动成功了! 由于我们采用的是websocket正向连接,所以,我们就需要写一个websocket客户端来进行接收:

@ClientEndpoint
@Slf4j
public class BotClient {private Session session;public static BotClient instance;public static boolean isOpen = false;/** 提供一个spring context上下文(解决方案)*/private static ApplicationContext applicationContext;public static Count count;//    单例模式,只生成一个客户端连接对象private BotClient(String url) {try {session = ContainerProvider.getWebSocketContainer().connectToServer(this, URI.create(url));} catch (Exception e) {e.printStackTrace();}}public static void setApplicationContext(ApplicationContext applicationContext) {BotClient.applicationContext = applicationContext;count = applicationContext.getBean(Count.class);}
​/*** 创建连接* @param url* @return*/public synchronized static boolean connect(String url) {instance = new  BotClient(url);return true;}
​/*** 连接前处理* @param session*/@OnOpenpublic void onOpen(Session session) {isOpen = true;log.info("连接成功!");}
​/*** 连接关闭处理* @param session*/@OnClosepublic void onClose(Session session) {isOpen = false;log.info("连接关闭!");}
​/*** 错误时处理* @param session* @param throwable*/@OnErrorpublic void onError(Session session, Throwable throwable) {log.info("连接错误!");}
​/*** 接收到消息时处理* @param message*/@OnMessagepublic void onMessage(String message) {
//        加好友if (message.contains("\"request_type\":\"friend\"")) {sendFriend(message);}
//        私信if (message.contains("\"post_type\":\"message\"") && message.contains("\"message_type\":\"private\"")) {sendMsg(message);}
//        群消息if (message.contains("\"post_type\":\"message\"")&& message.contains("\"message_type\":\"group\""))  {sendGroupMsg(message);}}
​/*** 好友请求*/private synchronized void sendFriend(String msg) {Friend parseObject = JSONObject.parseObject(msg, Friend.class);log.info("收到好友请求:" + parseObject.getUser_id() + ",验证消息:" + parseObject.getComment());Request<Object> paramsRequest = new Request<>();paramsRequest.setAction(BotActionEnum.ADD_FRIEND_ACTION.getAction());Map<String, Object> params = new HashMap<>();params.put("flag", parseObject.getFlag());if (parseObject.getComment().equals(count.getAuthor())) {params.put("approve", true);log.info("已同意好友请求:" + parseObject.getUser_id());} else {params.put("approve", false);log.info("已拒绝好友请求:" + parseObject.getUser_id());}paramsRequest.setParams(params);instance.session.getAsyncRemote().sendText(JSONObject.toJSONString(paramsRequest));}/*** 群消息*/public synchronized  void sendGroupMsg(String msg) {System.out.println(msg);Message parseObject = JSONObject.parseObject(msg, Message.class);log.info("收到群消息" + parseObject.getGroupId() + "的消息,发送者"+parseObject.getUserId()+"消息:"+ parseObject.getMessage()+"消息id:"+parseObject.getMessageId());System.out.println(parseObject.getMessage());String mes = parseObject.getMessage();if(mes.contains("[CQ:at,qq="+count.getBot_count()+"]")) {Request<Object> paramsRequest = new Request<>();paramsRequest.setAction(BotActionEnum.SEND_GROUP_MESSAGE.getAction());Map<String, Object> params = new HashMap<>();params.put("group_id", parseObject.getGroupId());String ai = AiOne(parseObject.getMessage(),parseObject.getMessageId());if (ai == null) {ai = "宝,回复失败!重新试试吧!";}params.put("message", ai);params.put("message_type", "group");params.put("auto_escape", false);paramsRequest.setParams(params);msg = JSONObject.toJSONString(paramsRequest);//        回复instance.session.getAsyncRemote().sendText(msg);}
​}/*** 私信好友消息*/public synchronized void sendMsg(String msg) {Message parseObject = JSONObject.parseObject(msg, Message.class);log.info("收到好友" + parseObject.getUserId() + "的消息:" + parseObject.getMessage());System.out.println("消息:"+parseObject.getMessage());Request<Object> paramsRequest = new Request<>();paramsRequest.setAction(BotActionEnum.SEND_PRIVATE_MESSAGE.getAction());Map<String, Object> params = new HashMap<>();params.put("user_id", parseObject.getUserId());String ai = AiOne(parseObject.getMessage(),null);if (ai == null) {ai = "宝,回复失败!重新试试吧!";}params.put("message", ai);params.put("message_type", "private");params.put("auto_escape", false);paramsRequest.setParams(params);msg = JSONObject.toJSONString(paramsRequest);
//        回复instance.session.getAsyncRemote().sendText(msg);}public  String AiOne(String sendMsg,String mes_id) {CloseableHttpClient httpClient = null;try {httpClient= HttpClientBuilder.create().build();GptUtils gptUtils = applicationContext.getBean(GptUtils.class);if(sendMsg.contains("画一幅图:")){log.info("进入到绘图......");String initBody = gptUtils.getImageURl(httpClient,sendMsg);if(Objects.isNull(mes_id)){String body =initBody;log.info("body:"+body);return body;}String body = "[CQ:reply,id="+mes_id+"]"+initBody;log.info("body:"+body);return body;}String initBody = gptUtils.getAnswer(httpClient,sendMsg);String body = null;if(Objects.isNull(mes_id)){body=initBody;log.info("body:"+body);return body;}body = "[CQ:reply,id="+mes_id+"]"+initBody;log.info("body:"+body);return body;} catch (Exception e) {log.error(e.toString());return null;}finally {try {httpClient.close();} catch (IOException e) {log.error("httpclient关闭失败");}}}
​
}

对于ws连接,我们也可以根据go-cqhttp的api文档来自行定制 api文档:API | go-cqhttp 帮助中心 目前我实现的是群聊,私信的聊天,及生成图片,以及好友请求同意。 项目的技术栈: websocket:连接go-cqhttp httpclient: 请求chatgptapi fastjson:进行序列化/反序列化 oss:图片存储

​<dependencies><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter</artifactId></dependency>​<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-devtools</artifactId><scope>runtime</scope><optional>true</optional></dependency><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-configuration-processor</artifactId><optional>true</optional></dependency><dependency><groupId>org.projectlombok</groupId><artifactId>lombok</artifactId><optional>true</optional></dependency><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-test</artifactId><scope>test</scope></dependency><!--           websocket--><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-websocket</artifactId></dependency><!--        fastjson--><dependency><groupId>com.alibaba</groupId><artifactId>fastjson</artifactId><version>1.2.76</version></dependency><!--       httpclient用来请求自动回复API --><dependency><groupId>org.apache.httpcomponents</groupId><artifactId>httpclient</artifactId><version>4.5.13</version></dependency><!--         阿里云 OSS--><dependency><groupId>com.aliyun.oss</groupId><artifactId>aliyun-sdk-oss</artifactId><version>3.10.2</version></dependency></dependencies>

3.openAIAPI的使用

这里重点来说说使用openai提供的API的使用: 先到openai官网上,我们先看看官方给的api文档:

请求:

curl https://api.openai.com/v1/chat/completions \-H "Content-Type: application/json" \-H "Authorization: Bearer $OPENAI_API_KEY" \-d '{"model": "gpt-3.5-turbo","messages": [{"role": "user", "content": "Hello!"}]}'

参数:

{"model": "gpt-3.5-turbo","messages": [{"role": "user", "content": "Hello!"}]
}

响应:

{"id": "chatcmpl-123","object": "chat.completion","created": 1677652288,"choices": [{"index": 0,"message": {"role": "assistant","content": "\n\nHello there, how may I assist you today?",},"finish_reason": "stop"}],"usage": {"prompt_tokens": 9,"completion_tokens": 12,"total_tokens": 21}
}

请求:

curl https://api.openai.com/v1/images/generations \-H "Content-Type: application/json" \-H "Authorization: Bearer $OPENAI_API_KEY" \-d '{"prompt": "A cute baby sea otter","n": 2,"size": "1024x1024"}'

参数:

{"prompt": "A cute baby sea otter","n": 2,"size": "1024x1024"
}

响应:

{"created": 1589478378,"data": [{"url": "https://..."},{"url": "https://..."}]
}

根据上面的请求响应,第一步肯定是编写VO,之后我们使用httpclient来进行请求的发送,和接收响应 VO类代码就不放这了,直接git上下载源码查看 由于 OpenAI 及 GFW 的双重限制,国内开发者无法访问 OpenAI 的 API,现提供代理服务地址供开发者免费使用. 教程:OpenAI API 代理 完毕,之后就能正常请求该接口了。 编写请求chat聊天的方法:

/*** 创建一个ChatGptRequestParameter,用于携带请求参数*/private static ChatGptRequestParameter chatGptRequestParameter = new ChatGptRequestParameter();public String getAnswer(CloseableHttpClient client, String question) {// 创建一个HttpPostHttpPost httpPost = new HttpPost(chatUrl);// 设置请求参数chatGptRequestParameter.addMessages(new ChatGptMessage(role, question));chatGptRequestParameter.setModel(model);HttpEntity httpEntity = null;try {// 对象转换为json字符串httpEntity = new StringEntity(JSON.toJSONString(chatGptRequestParameter), charset);} catch (Exception e) {log.info(question + "->json转换异常");return null;}httpPost.setEntity(httpEntity);// 设置请求头httpPost.setHeader(HttpHeaders.CONTENT_TYPE, "application/json");// 设置登录凭证httpPost.setHeader(HttpHeaders.AUTHORIZATION, "Bearer " + apiKey);// 用于设置超时时间RequestConfig config = RequestConfig.custom().setConnectTimeout(responseTimeout).build();httpPost.setConfig(config);
//        接收返回值CloseableHttpResponse response = null;HttpEntity entity = null;try {response = client.execute(httpPost);entity = response.getEntity();
//            反序列化String responseMes = EntityUtils.toString(entity);
//            转为对象ChatGptResponseParameter responseParameter = JSON.parseObject(responseMes, ChatGptResponseParameter.class);// 遍历所有的Choices(一般都只有一个)String ans = "";for (Choices choice : responseParameter.getChoices()) {log.info("size:" + responseParameter.getChoices().size());ChatGptMessage message = choice.getMessage();chatGptRequestParameter.addMessages(new ChatGptMessage(message.getRole(), message.getContent()));String s = message.getContent().replaceAll("\n+", "\n");ans += s;}return ans;} catch (Exception e) {e.printStackTrace();}finally {try {response.close();} catch (IOException e) {e.printStackTrace();}}
//        发生异常重置会话chatGptRequestParameter = new ChatGptRequestParameter();;return "您当前的网络无法访问,会话已重置";}

编写按描述生成图片接口:

public String getImageURl(CloseableHttpClient client, String describe) {HttpPost httpPost = new HttpPost(imageGPTUrl);ImageGptMessage imageGptMessage = new ImageGptMessage(describe, size, num);String jsonString = JSON.toJSONString(imageGptMessage);HttpEntity httpEntity = new StringEntity(jsonString,charset);//        设置请求参数httpPost.setEntity(httpEntity);// 设置请求头httpPost.setHeader(org.apache.http.HttpHeaders.CONTENT_TYPE, "application/json");// 设置登录凭证httpPost.setHeader(HttpHeaders.AUTHORIZATION, "Bearer " + apiKey);//        设置请求超时时间RequestConfig timeOut = RequestConfig.custom().setConnectTimeout(responseTimeout).build();httpPost.setConfig(timeOut);//        发送请求CloseableHttpResponse response = null;InputStream stream = null;try{response = client.execute(httpPost);HttpEntity entity = response.getEntity();
//            反序列化String responseMes = EntityUtils.toString(entity);
//            转为对象ImageGptResponseParameter responseParameter = JSON.parseObject(responseMes, ImageGptResponseParameter.class);// 遍历所有的image(一般都只有一个)for (Image image : responseParameter.getData()) {HttpGet httpGet = new HttpGet(image.getUrl());response = client.execute(httpGet);HttpEntity en = response.getEntity();stream= en.getContent();String fileName = UUID.randomUUID()+".png";String lastUrl = uploadStrategyContext.executeUploadStrategy(fileName, stream, FilePathEnum.GPTIMG.getPath());String url = "[CQ:image,file="+lastUrl+"]";
//                String url = "[CQ:image,file=https://leyasuzhou-blog.oss-cn-beijing.aliyuncs.com/img/test.png]";
//                String url = image.getUrl();System.out.println(url);return url;}}catch (Exception e){log.info("图片生成出错了");}finally {try {stream.close();response.close();} catch (IOException e) {e.printStackTrace();}}return null;}}

这里原本可以直接使用cq码来进行url封装,但直接用该接口生成图片的url会报错cq码解析超时,最后我想了个办法是先将图片放到oss上,之后再从oss上拿。但这样该方法就执行的很慢,如果有人知道这个cq码解析超时咋解决私信博主!

4.lunix部署

  1. 通过 SSH 连接到服务器

  2. cd到解压目录

  3. 输入 ./go-cqhttp, Enter运行 , 此时将提示

[WARNING]: 尝试加载配置文件 config.yml 失败: 文件不存在
[INFO]: 默认配置文件已生成,请编辑 config.yml 后重启程序.

之后继续像windows一样修改config.xml和device.json文件。 这里会有一个小问题,就是在扫码登录时,会出现风控,不让你登录,这时候怎么办呢? 首先,我们先在windows上扫码登录账号,登录成功后我们发现出现了一个session.token文件

这个文件保存了我们的登录信息,这时候,我们将这个文件页上传到lunix服务器上同样位置,就不用扫码登录了 到此,整个项目就跑起来了。 本项目是一个client单会话的,如果一个回复错误,我这边采取的是重置会话。 欢迎交流技术问题 来体验你的qqbot吧!!!

2023最新!QQ接入ChatGpt!!!保姆级教程相关推荐

  1. windows 10使用Pycharm从0到1搭建一个QQ聊天机器人保姆级教程【2023最新版】

    文章目录 选择Python环境 安装nb-cli 配置Go-cqhttp 大工告成! 选择Python环境 Nonebot2官方说,python版本要大于3.8,推荐大家使用虚拟环境,在Poetry. ...

  2. ChatGPT保姆级教程,一分钟学会使用ChatGPT!

    最近ChatGPT大火!微软退出首款ChatGPT搜索引擎,阿里等国内巨头也纷纷爆出自家产品,一夜之间,全球最大的科技公司仿佛都回到了自己年轻时的样子! 然而,ChatGPT这么火,这么好玩的东西,国 ...

  3. VMware虚拟机最新详细安装保姆级教程(2023年新版教程)

    VMware最新详细安装保姆级教程(2023年新版教程) 大家好,我是洲洲,欢迎关注,一个爱听周杰伦的程序员.关注公众号[程序员洲洲]即可获得10G学习资料.面试笔记.大厂独家学习体系路线等-还可以加 ...

  4. win11 安装Ubuntu加可视化桌面(最新保姆级教程)

    win11.win10 安装Ubuntu加可视化桌面(最新保姆级教程) 前言 保姆级安装流程 第一节·安装前准备 第二节·安装ubuntu本体 第三节·可视化安装 前言 win11.win10 安装U ...

  5. 【最新敲简单】浪漫微信早安推送 +页面总控,JAVA版本,一键部署运行——保姆级教程

    [最新敲简单]微信早安推送 +页面总控, JAVA版本,一键部署运行--保姆级教程 文章目录 [最新敲简单]微信早安推送 +页面总控, JAVA版本,一键部署运行--保姆级教程 一.项目简介 二.本地 ...

  6. 《软件安装与使用教程》— Windows操作系统快速安装MATLAB最新最全教程(保姆级教程以MATLAB2019A为例)

    <MATLAB教程>- Windows操作系统快速安装MATLAB(保姆级教程以MATLAB2019A为例) 目录 <MATLAB教程>- Windows操作系统快速安装MAT ...

  7. 数仓建设保姆级教程,离线和实时一网打尽(理论+实战)

    本文大纲: 因内容较多,带目录的PDF查看是比较方便的,点击下方链接获取完整PDF版: 数仓建设保姆级教程PDF文档 一.数仓基本概念 1. 数据仓库架构 我们在谈数仓之前,为了让大家有直观的认识,先 ...

  8. 50000字,数仓建设保姆级教程,离线和实时一网打尽(理论+实战) 下

    文档大纲: 本文上半部分之前已经发过了,传送门:50000字,数仓建设保姆级教程,离线和实时一网打尽(理论+实战) 上 此篇文章是整个文档的下半部分,将接着上半部分从第五章开始. 五.实时数仓建设核心 ...

  9. ac2100 反弹shell无法粘贴_手把手带你玩转NAS 篇二十一:小米Redmi AC2100路由器刷机padavan保姆级教程...

    手把手带你玩转NAS 篇二十一:小米Redmi AC2100路由器刷机padavan保姆级教程 2020-05-14 18:49:24 224点赞 1790收藏 241评论 你是AMD Yes党?还是 ...

最新文章

  1. Windows编程设备描述表的概念和在客户区绘制、在窗口标题栏绘制、在桌面绘制图解
  2. Android热修复升级探索——SO库修复方案 1
  3. python输入框_selenium+python 对输入框的输入处理方法
  4. 分享5个苹果系统超实用的黑科技APP,个个都是精品
  5. 自然资源部信息化建设总体方案摘要记录
  6. Access、Trunk和Hybird的个人理解
  7. Android 获取屏幕高度、宽度
  8. 智能网联时代汽车智能座舱操作系统的发展
  9. 工业B2B电商平台整合延伸工业供应链上下游,打造产业链闭环
  10. oracle 控制台使用手册,Oracle-ESS-入门手册
  11. Wireshark:抓取微信网址
  12. 基于注意力机制的超分辨率重建汇总
  13. 支持查看朋友圈的微信Mac版客户端
  14. flutter如何获取连接高德地图的SHA1和PackageName
  15. Java知识点总结《努力篇上》
  16. LeetCode 904. 水果成篮【fruit-into-baskets】
  17. Spring源码版本命名规则
  18. 分支定界 matlab,使用MATLAB实现分枝定界法求解整数规划的详细资料说明
  19. java下载我的世界1.11_我的世界Java版1.11
  20. java sca_SCA java编码入门

热门文章

  1. PM撸代码之Android【绝顶高手排行榜】
  2. HTML5和CSS3归纳
  3. numpy | 取矩阵非零元素
  4. 软件工程微信平台作业总结
  5. Unity之UGUI-特效遮挡问题2.0
  6. Linux和Windows系统基础操作命令
  7. 【Java】java.lang.Object
  8. JDBC实现多条件查询万能解决思路
  9. 卢克,学着去读源代码
  10. Mac使用数据线连接安卓手机传输文件