From 2f80b7085c4eabce06d3491306b75eecc275275f Mon Sep 17 00:00:00 2001
From: 云 <2163098428@qq.com>
Date: 星期四, 30 四月 2026 17:31:57 +0800
Subject: [PATCH] Merge remote-tracking branch 'origin/dev_New_pro' into dev_New_pro

---
 src/main/java/com/ruoyi/ai/controller/XiaozhiController.java |  136 +++++++++++++++++++++++++++++++++++++++++++-
 1 files changed, 131 insertions(+), 5 deletions(-)

diff --git a/src/main/java/com/ruoyi/ai/controller/XiaozhiController.java b/src/main/java/com/ruoyi/ai/controller/XiaozhiController.java
index e1223cc..eac43b2 100644
--- a/src/main/java/com/ruoyi/ai/controller/XiaozhiController.java
+++ b/src/main/java/com/ruoyi/ai/controller/XiaozhiController.java
@@ -2,35 +2,161 @@
 
 import com.ruoyi.ai.assistant.ApproveTodoAgent;
 import com.ruoyi.ai.assistant.ApproveTodoIntentExecutor;
+import com.ruoyi.ai.assistant.FileAnalyzeAgent;
 import com.ruoyi.ai.bean.ChatForm;
+import com.ruoyi.ai.context.AiSessionUserContext;
+import com.ruoyi.ai.service.AiChatSessionService;
+import com.ruoyi.ai.service.AiFileTextExtractor;
+import com.ruoyi.ai.store.MongoChatMemoryStore;
+import com.ruoyi.common.utils.SecurityUtils;
+import com.ruoyi.common.utils.StringUtils;
+import com.ruoyi.framework.security.LoginUser;
+import com.ruoyi.framework.web.controller.BaseController;
+import com.ruoyi.framework.web.domain.AjaxResult;
+import dev.langchain4j.data.message.AiMessage;
+import dev.langchain4j.data.message.UserMessage;
 import io.swagger.v3.oas.annotations.Operation;
 import io.swagger.v3.oas.annotations.tags.Tag;
+import org.springframework.web.bind.annotation.DeleteMapping;
+import org.springframework.web.bind.annotation.GetMapping;
+import org.springframework.web.bind.annotation.PathVariable;
 import org.springframework.web.bind.annotation.PostMapping;
 import org.springframework.web.bind.annotation.RequestBody;
 import org.springframework.web.bind.annotation.RequestMapping;
+import org.springframework.web.bind.annotation.RequestParam;
 import org.springframework.web.bind.annotation.RestController;
+import org.springframework.web.multipart.MultipartFile;
 import reactor.core.publisher.Flux;
+
+import java.io.IOException;
+import java.util.List;
+import java.util.NoSuchElementException;
+import java.util.UUID;
 
 @Tag(name = "鍗忓悓鍔炲叕鍔╂墜")
 @RestController
 @RequestMapping("/xiaozhi")
-public class XiaozhiController {
+public class XiaozhiController extends BaseController {
+
+    private static final String FILE_ANALYZE_MEMORY_PREFIX = "file-analyze::";
 
     private final ApproveTodoAgent approveTodoAgent;
     private final ApproveTodoIntentExecutor approveTodoIntentExecutor;
+    private final FileAnalyzeAgent fileAnalyzeAgent;
+    private final AiSessionUserContext aiSessionUserContext;
+    private final MongoChatMemoryStore mongoChatMemoryStore;
+    private final AiFileTextExtractor aiFileTextExtractor;
+    private final AiChatSessionService aiChatSessionService;
 
-    public XiaozhiController(ApproveTodoAgent approveTodoAgent, ApproveTodoIntentExecutor approveTodoIntentExecutor) {
+    public XiaozhiController(ApproveTodoAgent approveTodoAgent,
+                             ApproveTodoIntentExecutor approveTodoIntentExecutor,
+                             FileAnalyzeAgent fileAnalyzeAgent,
+                             AiSessionUserContext aiSessionUserContext,
+                             MongoChatMemoryStore mongoChatMemoryStore,
+                             AiFileTextExtractor aiFileTextExtractor,
+                             AiChatSessionService aiChatSessionService) {
         this.approveTodoAgent = approveTodoAgent;
         this.approveTodoIntentExecutor = approveTodoIntentExecutor;
+        this.fileAnalyzeAgent = fileAnalyzeAgent;
+        this.aiSessionUserContext = aiSessionUserContext;
+        this.mongoChatMemoryStore = mongoChatMemoryStore;
+        this.aiFileTextExtractor = aiFileTextExtractor;
+        this.aiChatSessionService = aiChatSessionService;
     }
 
     @Operation(summary = "瀵硅瘽")
     @PostMapping(value = "/chat", produces = "text/stream;charset=utf-8")
     public Flux<String> chat(@RequestBody ChatForm chatForm) {
-        String directResponse = approveTodoIntentExecutor.tryExecute(chatForm.getMessage());
-        if (directResponse != null) {
+        if (!StringUtils.hasText(chatForm.getMemoryId())) {
+            return Flux.just("memoryId涓嶈兘涓虹┖");
+        }
+        if (!StringUtils.hasText(chatForm.getMessage())) {
+            return Flux.just("message涓嶈兘涓虹┖");
+        }
+        LoginUser loginUser = SecurityUtils.getLoginUser();
+        String memoryId = chatForm.getMemoryId();
+        String userMessage = chatForm.getMessage();
+
+        aiSessionUserContext.bind(memoryId, loginUser);
+        aiChatSessionService.touchSession(memoryId, loginUser, userMessage);
+
+        String directResponse = approveTodoIntentExecutor.tryExecute(memoryId, userMessage);
+        if (StringUtils.isNotEmpty(directResponse)) {
+            mongoChatMemoryStore.appendMessages(
+                    memoryId,
+                    List.of(UserMessage.from(userMessage), AiMessage.from(directResponse))
+            );
+            aiChatSessionService.refreshSessionStats(memoryId, loginUser);
             return Flux.just(directResponse);
         }
-        return approveTodoAgent.chat(chatForm.getMemoryId(), chatForm.getMessage());
+
+        return approveTodoAgent.chat(memoryId, userMessage)
+                .doOnComplete(() -> aiChatSessionService.refreshSessionStats(memoryId, loginUser))
+                .doOnError(ex -> aiChatSessionService.refreshSessionStats(memoryId, loginUser));
+    }
+
+    @Operation(summary = "涓婁紶鏂囦欢鍒嗘瀽")
+    @PostMapping(value = "/analyze-file", consumes = "multipart/form-data", produces = "text/stream;charset=utf-8")
+    public Flux<String> analyzeFile(@RequestParam("file") MultipartFile file,
+                                    @RequestParam(value = "message", required = false) String message,
+                                    @RequestParam(value = "memoryId", required = false) String memoryId) {
+        String rawMemoryId = StringUtils.hasText(memoryId) ? memoryId : UUID.randomUUID().toString();
+        String finalMemoryId = rawMemoryId.startsWith(FILE_ANALYZE_MEMORY_PREFIX)
+                ? rawMemoryId
+                : FILE_ANALYZE_MEMORY_PREFIX + rawMemoryId;
+
+        LoginUser loginUser = SecurityUtils.getLoginUser();
+        aiSessionUserContext.bind(finalMemoryId, loginUser);
+
+        String finalMessage = StringUtils.hasText(message) ? message : "璇峰垎鏋愯繖涓枃浠剁殑鏍稿績鍐呭骞剁粰鍑烘�荤粨";
+        String fileText;
+        try {
+            fileText = aiFileTextExtractor.extractText(file);
+        } catch (IllegalArgumentException ex) {
+            return Flux.just(ex.getMessage());
+        } catch (IOException ex) {
+            return Flux.just("鏂囦欢璇诲彇澶辫触");
+        }
+
+        if (!StringUtils.hasText(fileText)) {
+            return Flux.just("鏈彁鍙栧埌鏈夋晥鏂囦欢鍐呭");
+        }
+
+        String limitedContent = fileText.length() > 12000 ? fileText.substring(0, 12000) : fileText;
+        String userPrompt = "浣犲皢鍒嗘瀽鐢ㄦ埛涓婁紶鐨勬枃浠跺唴瀹广�俓n"
+                + "鏂囦欢鍚�: " + file.getOriginalFilename() + "\n"
+                + "鐢ㄦ埛闂: " + finalMessage + "\n"
+                + "鏂囦欢鍐呭濡備笅:\n"
+                + limitedContent;
+
+        aiChatSessionService.touchSession(finalMemoryId, loginUser, "鏂囦欢鍒嗘瀽: " + finalMessage);
+
+        return Flux.defer(() -> fileAnalyzeAgent.chat(finalMemoryId, userPrompt))
+                .onErrorResume(NoSuchElementException.class, ex -> {
+                    // DashScope 鍦ㄥ巻鍙叉秷鎭吋瀹瑰満鏅笅鍙兘鎶� NoSuchElementException锛屾竻绌哄悗閲嶈瘯涓�娆�
+                    mongoChatMemoryStore.deleteMessages(finalMemoryId);
+                    return fileAnalyzeAgent.chat(finalMemoryId, userPrompt);
+                })
+                .doOnComplete(() -> aiChatSessionService.refreshSessionStats(finalMemoryId, loginUser))
+                .doOnError(ex -> aiChatSessionService.refreshSessionStats(finalMemoryId, loginUser));
+    }
+
+    @Operation(summary = "浼氳瘽鍒楄〃")
+    @GetMapping("/history/sessions")
+    public AjaxResult listSessions() {
+        return success(aiChatSessionService.listCurrentUserSessions(SecurityUtils.getLoginUser()));
+    }
+
+    @Operation(summary = "浼氳瘽娑堟伅")
+    @GetMapping("/history/messages/{memoryId}")
+    public AjaxResult listMessages(@PathVariable String memoryId) {
+        return success(aiChatSessionService.listCurrentUserMessages(memoryId, SecurityUtils.getLoginUser()));
+    }
+
+    @Operation(summary = "鍒犻櫎浼氳瘽")
+    @DeleteMapping("/history/{memoryId}")
+    public AjaxResult deleteSession(@PathVariable String memoryId) {
+        aiSessionUserContext.remove(memoryId);
+        return toAjax(aiChatSessionService.deleteCurrentUserSession(memoryId, SecurityUtils.getLoginUser()));
     }
 }

--
Gitblit v1.9.3