|
@@ -0,0 +1,334 @@
|
|
|
+package com.usky.ai.controller.web;
|
|
|
+
|
|
|
+import com.alibaba.dashscope.aigc.generation.Generation;
|
|
|
+import com.alibaba.dashscope.aigc.generation.GenerationParam;
|
|
|
+import com.alibaba.dashscope.common.Message;
|
|
|
+import com.alibaba.dashscope.common.Role;
|
|
|
+import com.alibaba.dashscope.exception.ApiException;
|
|
|
+import com.alibaba.dashscope.exception.InputRequiredException;
|
|
|
+import com.alibaba.dashscope.exception.NoApiKeyException;
|
|
|
+import com.alibaba.fastjson.JSONObject;
|
|
|
+import com.fasterxml.jackson.databind.ObjectMapper;
|
|
|
+import com.usky.ai.mapper.AiQuestionMapper;
|
|
|
+import com.usky.ai.mapper.AiSessionMapper;
|
|
|
+import com.usky.ai.service.AiQuestion;
|
|
|
+import com.usky.ai.service.AiSession;
|
|
|
+import com.usky.ai.service.vo.AiStreamOutputVO;
|
|
|
+import com.usky.common.core.bean.ApiResult;
|
|
|
+import com.usky.common.security.utils.SecurityUtils;
|
|
|
+import lombok.extern.slf4j.Slf4j;
|
|
|
+import org.springframework.beans.factory.annotation.Autowired;
|
|
|
+import org.springframework.beans.factory.annotation.Value;
|
|
|
+import org.springframework.http.MediaType;
|
|
|
+import org.springframework.http.ResponseEntity;
|
|
|
+import org.springframework.web.bind.annotation.*;
|
|
|
+import org.springframework.web.servlet.mvc.method.annotation.StreamingResponseBody;
|
|
|
+
|
|
|
+import javax.annotation.Resource;
|
|
|
+import java.io.IOException;
|
|
|
+import java.nio.charset.StandardCharsets;
|
|
|
+import java.time.LocalDateTime;
|
|
|
+import java.util.List;
|
|
|
+import java.util.stream.Collectors;
|
|
|
+
|
|
|
+@Slf4j
|
|
|
+@RestController
|
|
|
+@RequestMapping("/ai")
|
|
|
+public class AiChatController {
|
|
|
+
|
|
|
+ @Value("${ai.api.key}")
|
|
|
+ private String apiKey;
|
|
|
+
|
|
|
+ @Value("${airole}")
|
|
|
+ private String aiRole;
|
|
|
+
|
|
|
+ @Value("${aliTyqw.model}")
|
|
|
+ private String tyqwModel;
|
|
|
+
|
|
|
+ @Value("${aliDpsk.model}")
|
|
|
+ private String dpskModel;
|
|
|
+
|
|
|
+ @Value("${ai.historyLimit}")
|
|
|
+ private int Limit;
|
|
|
+
|
|
|
+ @Resource
|
|
|
+ private Generation generation;
|
|
|
+
|
|
|
+ @Autowired
|
|
|
+ private AiQuestionMapper aiQuestionMapper;
|
|
|
+
|
|
|
+ @Autowired
|
|
|
+ private AiSessionMapper aiSessionMapper;
|
|
|
+
|
|
|
+ private final ObjectMapper objectMapper = new ObjectMapper();
|
|
|
+
|
|
|
+ // 阿里百炼通义千问大模型
|
|
|
+ @PostMapping(value = "/aliTyqw", produces = MediaType.TEXT_EVENT_STREAM_VALUE)
|
|
|
+ public ResponseEntity<StreamingResponseBody> send1(@RequestBody JSONObject object )throws NoApiKeyException, InputRequiredException {
|
|
|
+ // 获取当前登录用户的信息
|
|
|
+ Long userId = SecurityUtils.getUserId();
|
|
|
+ String userName = SecurityUtils.getLoginUser().getSysUser().getNickName();
|
|
|
+ String sessionId = null;
|
|
|
+ String content = object.get("content").toString();
|
|
|
+
|
|
|
+ // 如果没有传入 sessionId,则创建一个新的会话ID
|
|
|
+ if (object.containsKey("sessionId")) {
|
|
|
+ sessionId = object.get("sessionId").toString();
|
|
|
+ } else {
|
|
|
+ sessionId = java.util.UUID.randomUUID().toString();
|
|
|
+ }
|
|
|
+
|
|
|
+ // 解析 JSON 并提取 "content" 字段的值
|
|
|
+ String questionText;
|
|
|
+ questionText = content; // 提取 "content" 字段的值
|
|
|
+
|
|
|
+ // 检查是否已经存在相同的 sessionId
|
|
|
+ boolean exists = aiSessionMapper.existsBySessionId(sessionId);
|
|
|
+
|
|
|
+ if (!exists) {
|
|
|
+ // 创建新的 AiSession 实体并存入数据库
|
|
|
+ AiSession aiSession = new AiSession();
|
|
|
+ aiSession.setSessionId(sessionId);
|
|
|
+ aiSession.setUserId(userId);
|
|
|
+ aiSession.setUserName(userName);
|
|
|
+ aiSession.setQuestion(questionText);
|
|
|
+ aiSession.setAskTime(LocalDateTime.now());
|
|
|
+ aiSessionMapper.save(aiSession);
|
|
|
+ }
|
|
|
+
|
|
|
+ // 获取当前用户的对话历史
|
|
|
+ List<AiQuestion> conversationHistory = aiQuestionMapper.findByUserIdAndSessionId(sessionId, userId);
|
|
|
+
|
|
|
+// // 只保留最近的几轮对话
|
|
|
+// int historyLimit = Limit;
|
|
|
+// if (conversationHistory.size() > historyLimit) {
|
|
|
+// conversationHistory = conversationHistory.subList(conversationHistory.size() - historyLimit, conversationHistory.size());
|
|
|
+// }
|
|
|
+
|
|
|
+ // 构建对话历史消息
|
|
|
+ List<Message> messages = conversationHistory.stream()
|
|
|
+ .map(q -> Message.builder()
|
|
|
+ .role(q.getUserId().equals(userId) ? Role.USER.getValue() : Role.ASSISTANT.getValue())
|
|
|
+ .content(q.getUserId().equals(userId) ? q.getQuestion() : q.getAnswer())
|
|
|
+ .build())
|
|
|
+ .collect(Collectors.toList());
|
|
|
+
|
|
|
+ // 插入角色定义(在对话历史的开头)
|
|
|
+ Message roleDefinition = Message.builder()
|
|
|
+ .role(Role.SYSTEM.getValue()) // 使用系统角色
|
|
|
+ .content(aiRole) // 定义角色的行为和风格
|
|
|
+ .build();
|
|
|
+ messages.add(0, roleDefinition); // 将角色定义插入到对话历史的开头
|
|
|
+
|
|
|
+ // 添加用户的新消息
|
|
|
+ Message userMessage = Message.builder()
|
|
|
+ .role(Role.USER.getValue())
|
|
|
+ .content(questionText) // 使用提取的文本
|
|
|
+ .build();
|
|
|
+ messages.add(userMessage);
|
|
|
+
|
|
|
+ // 构建模型调用参数
|
|
|
+ GenerationParam param = GenerationParam.builder()
|
|
|
+ .model(tyqwModel)
|
|
|
+ .messages(messages)
|
|
|
+ .resultFormat(GenerationParam.ResultFormat.MESSAGE)
|
|
|
+ .topP(0.8)
|
|
|
+ .apiKey(apiKey)
|
|
|
+ .incrementalOutput(true) // 开启增量输出
|
|
|
+ .build();
|
|
|
+
|
|
|
+ String finalSessionId = sessionId;
|
|
|
+ StringBuilder completeAnswer = new StringBuilder(); // 用于收集完整的回答内容
|
|
|
+ return ResponseEntity.ok()
|
|
|
+ .contentType(MediaType.TEXT_EVENT_STREAM)
|
|
|
+ .body(outputStream -> {
|
|
|
+ try {
|
|
|
+ // 调用流式接口
|
|
|
+ generation.streamCall(param).blockingForEach(chunk -> {
|
|
|
+ try {
|
|
|
+ // 获取每次生成的内容
|
|
|
+ String partialAnswer = chunk.getOutput().getChoices().get(0).getMessage().getContent();
|
|
|
+
|
|
|
+ // 检查内容是否为空,如果为空则跳过
|
|
|
+ if (partialAnswer == null || partialAnswer.trim().isEmpty()) {
|
|
|
+ return; // 如果内容为空,直接返回,不进行后续操作
|
|
|
+ }
|
|
|
+
|
|
|
+ // 构建输出对象
|
|
|
+ AiStreamOutputVO aiStreamOutputVO = new AiStreamOutputVO();
|
|
|
+ aiStreamOutputVO.setSessionId(finalSessionId);
|
|
|
+ aiStreamOutputVO.setReasoningContent(partialAnswer);
|
|
|
+
|
|
|
+ // 转换为 JSON 字符串
|
|
|
+ String newString = objectMapper.writeValueAsString(aiStreamOutputVO);
|
|
|
+
|
|
|
+ // 写入输出流
|
|
|
+ outputStream.write(("data: " + newString + "\n\n").getBytes(StandardCharsets.UTF_8));
|
|
|
+// outputStream.write(ApiResult.success(newString).toString().getBytes(StandardCharsets.UTF_8));
|
|
|
+ outputStream.flush(); // 确保立即发送到前端
|
|
|
+
|
|
|
+ // 累加到完整回答内容中
|
|
|
+ completeAnswer.append(partialAnswer);
|
|
|
+
|
|
|
+ } catch (Exception e) {
|
|
|
+ log.error("Error processing chunk", e);
|
|
|
+ outputStream.write(("data: Error processing chunk\n\n").getBytes(StandardCharsets.UTF_8));
|
|
|
+ outputStream.flush();
|
|
|
+ }
|
|
|
+ });
|
|
|
+
|
|
|
+ // 流式接口调用完成后,将完整回答存入数据库
|
|
|
+ AiQuestion aiQuestion = new AiQuestion();
|
|
|
+ aiQuestion.setModel(tyqwModel);
|
|
|
+ aiQuestion.setSessionId(finalSessionId);
|
|
|
+ aiQuestion.setUserId(userId);
|
|
|
+ aiQuestion.setUserName(userName);
|
|
|
+ aiQuestion.setQuestion(questionText);
|
|
|
+ aiQuestion.setAnswer(completeAnswer.toString());
|
|
|
+ aiQuestion.setAskTime(LocalDateTime.now());
|
|
|
+ aiQuestionMapper.save(aiQuestion);
|
|
|
+
|
|
|
+ } catch (ApiException e) {
|
|
|
+ log.error("Error processing request", e);
|
|
|
+ outputStream.write(("data: Error processing request\n\n").getBytes(StandardCharsets.UTF_8));
|
|
|
+ outputStream.flush();
|
|
|
+ } catch (NoApiKeyException | InputRequiredException e) {
|
|
|
+ throw new RuntimeException(e);
|
|
|
+ }
|
|
|
+ });
|
|
|
+ }
|
|
|
+
|
|
|
+ // 阿里百炼DeepSeek大模型
|
|
|
+ @PostMapping(value = "/aliDeepSeek", produces = MediaType.TEXT_EVENT_STREAM_VALUE)
|
|
|
+ public ResponseEntity<StreamingResponseBody> send2(@RequestBody JSONObject object )throws NoApiKeyException, InputRequiredException {
|
|
|
+ // 获取当前登录用户的信息
|
|
|
+ Long userId = SecurityUtils.getUserId();
|
|
|
+ String userName = SecurityUtils.getLoginUser().getSysUser().getNickName();
|
|
|
+ String sessionId = null;
|
|
|
+ String content = object.get("content").toString();
|
|
|
+
|
|
|
+ // 如果没有传入 sessionId,则创建一个新的会话ID
|
|
|
+ if (object.containsKey("sessionId")) {
|
|
|
+ sessionId = object.get("sessionId").toString();
|
|
|
+ } else {
|
|
|
+ sessionId = java.util.UUID.randomUUID().toString();
|
|
|
+ }
|
|
|
+
|
|
|
+ // 解析 JSON 并提取 "content" 字段的值
|
|
|
+ String questionText;
|
|
|
+ questionText = content; // 提取 "content" 字段的值
|
|
|
+
|
|
|
+ // 检查是否已经存在相同的 sessionId
|
|
|
+ boolean exists = aiSessionMapper.existsBySessionId(sessionId);
|
|
|
+
|
|
|
+ if (!exists) {
|
|
|
+ // 创建新的 AiSession 实体并存入数据库
|
|
|
+ AiSession aiSession = new AiSession();
|
|
|
+ aiSession.setSessionId(sessionId);
|
|
|
+ aiSession.setUserId(userId);
|
|
|
+ aiSession.setUserName(userName);
|
|
|
+ aiSession.setQuestion(questionText);
|
|
|
+ aiSession.setAskTime(LocalDateTime.now());
|
|
|
+ aiSessionMapper.save(aiSession);
|
|
|
+ }
|
|
|
+
|
|
|
+ // 获取当前用户的对话历史
|
|
|
+ List<AiQuestion> conversationHistory = aiQuestionMapper.findByUserIdAndSessionId(sessionId, userId);
|
|
|
+
|
|
|
+ // 只保留最近的几轮对话
|
|
|
+// int historyLimit = Limit;
|
|
|
+// if (conversationHistory.size() > historyLimit) {
|
|
|
+// conversationHistory = conversationHistory.subList(conversationHistory.size() - historyLimit, conversationHistory.size());
|
|
|
+// }
|
|
|
+
|
|
|
+ // 构建对话历史消息
|
|
|
+ List<Message> messages = conversationHistory.stream()
|
|
|
+ .map(q -> Message.builder()
|
|
|
+ .role(q.getUserId().equals(userId) ? Role.USER.getValue() : Role.ASSISTANT.getValue())
|
|
|
+ .content(q.getUserId().equals(userId) ? q.getQuestion() : q.getAnswer())
|
|
|
+ .build())
|
|
|
+ .collect(Collectors.toList());
|
|
|
+
|
|
|
+ // 插入角色定义(在对话历史的开头)
|
|
|
+ Message roleDefinition = Message.builder()
|
|
|
+ .role(Role.SYSTEM.getValue()) // 使用系统角色
|
|
|
+ .content(aiRole) // 定义角色的行为和风格
|
|
|
+ .build();
|
|
|
+ messages.add(0, roleDefinition); // 将角色定义插入到对话历史的开头
|
|
|
+
|
|
|
+ // 添加用户的新消息
|
|
|
+ Message userMessage = Message.builder()
|
|
|
+ .role(Role.USER.getValue())
|
|
|
+ .content(questionText) // 使用提取的文本
|
|
|
+ .build();
|
|
|
+ messages.add(userMessage);
|
|
|
+
|
|
|
+ // 构建模型调用参数
|
|
|
+ GenerationParam param = GenerationParam.builder()
|
|
|
+ .model(dpskModel)
|
|
|
+ .messages(messages)
|
|
|
+ .resultFormat(GenerationParam.ResultFormat.MESSAGE)
|
|
|
+ .apiKey(apiKey)
|
|
|
+ .incrementalOutput(true) // 开启增量输出[^1^]
|
|
|
+ .build();
|
|
|
+
|
|
|
+ String finalSessionId = sessionId;
|
|
|
+ StringBuilder completeAnswer = new StringBuilder(); // 用于收集完整的回答内容
|
|
|
+ return ResponseEntity.ok()
|
|
|
+ .contentType(MediaType.TEXT_EVENT_STREAM)
|
|
|
+ .body(outputStream -> {
|
|
|
+ try {
|
|
|
+ // 调用流式接口
|
|
|
+ generation.streamCall(param).blockingForEach(chunk -> {
|
|
|
+ try {
|
|
|
+ // 获取每次生成的内容
|
|
|
+ String partialAnswer = chunk.getOutput().getChoices().get(0).getMessage().getContent();
|
|
|
+
|
|
|
+ // 检查内容是否为空,如果为空则跳过
|
|
|
+ if (partialAnswer == null || partialAnswer.trim().isEmpty()) {
|
|
|
+ return; // 如果内容为空,直接返回,不进行后续操作
|
|
|
+ }
|
|
|
+
|
|
|
+ // 构建输出对象
|
|
|
+ AiStreamOutputVO aiStreamOutputVO = new AiStreamOutputVO();
|
|
|
+ aiStreamOutputVO.setSessionId(finalSessionId);
|
|
|
+ aiStreamOutputVO.setReasoningContent(partialAnswer);
|
|
|
+
|
|
|
+ // 转换为 JSON 字符串
|
|
|
+ String newString = objectMapper.writeValueAsString(aiStreamOutputVO);
|
|
|
+
|
|
|
+ // 写入输出流
|
|
|
+ outputStream.write(("data: " + newString + "\n\n").getBytes(StandardCharsets.UTF_8));
|
|
|
+ outputStream.flush(); // 确保立即发送到前端
|
|
|
+
|
|
|
+ // 累加到完整回答内容中
|
|
|
+ completeAnswer.append(partialAnswer);
|
|
|
+
|
|
|
+ } catch (Exception e) {
|
|
|
+ log.error("Error processing chunk", e);
|
|
|
+ outputStream.write(("data: Error processing chunk\n\n").getBytes(StandardCharsets.UTF_8));
|
|
|
+ outputStream.flush();
|
|
|
+ }
|
|
|
+ });
|
|
|
+
|
|
|
+ // 流式接口调用完成后,将完整回答存入数据库
|
|
|
+ AiQuestion aiQuestion = new AiQuestion();
|
|
|
+ aiQuestion.setModel(dpskModel);
|
|
|
+ aiQuestion.setSessionId(finalSessionId);
|
|
|
+ aiQuestion.setUserId(userId);
|
|
|
+ aiQuestion.setUserName(userName);
|
|
|
+ aiQuestion.setQuestion(questionText);
|
|
|
+ aiQuestion.setAnswer(completeAnswer.toString());
|
|
|
+ aiQuestion.setAskTime(LocalDateTime.now());
|
|
|
+ aiQuestionMapper.save(aiQuestion);
|
|
|
+
|
|
|
+ } catch (ApiException e) {
|
|
|
+ log.error("Error processing request", e);
|
|
|
+ outputStream.write(("data: Error processing request\n\n").getBytes(StandardCharsets.UTF_8));
|
|
|
+ outputStream.flush();
|
|
|
+ } catch (NoApiKeyException | InputRequiredException e) {
|
|
|
+ throw new RuntimeException(e);
|
|
|
+ }
|
|
|
+ });
|
|
|
+ }
|
|
|
+}
|