From 3732bde983f34f5cb729ac1724b9795fb97d8941 Mon Sep 17 00:00:00 2001
From: admin <cgf12138@163.com>
Date: 星期六, 07 六月 2025 18:12:05 +0800
Subject: [PATCH] 0606

---
 aiflowy-modules/aiflowy-module-ai/src/main/java/tech/aiflowy/ai/controller/AiBotController.java |  377 ++++++++++++++++++++++++++++++++++++++++-------------
 1 files changed, 286 insertions(+), 91 deletions(-)

diff --git a/aiflowy-modules/aiflowy-module-ai/src/main/java/tech/aiflowy/ai/controller/AiBotController.java b/aiflowy-modules/aiflowy-module-ai/src/main/java/tech/aiflowy/ai/controller/AiBotController.java
index d56bd2d..d5543f2 100644
--- a/aiflowy-modules/aiflowy-module-ai/src/main/java/tech/aiflowy/ai/controller/AiBotController.java
+++ b/aiflowy-modules/aiflowy-module-ai/src/main/java/tech/aiflowy/ai/controller/AiBotController.java
@@ -2,7 +2,6 @@
 
 import cn.dev33.satoken.annotation.SaIgnore;
 import cn.hutool.core.util.ObjectUtil;
-import cn.hutool.core.util.StrUtil;
 import com.agentsflex.core.llm.ChatContext;
 import com.agentsflex.core.llm.Llm;
 import com.agentsflex.core.llm.StreamResponseListener;
@@ -13,25 +12,39 @@
 import com.agentsflex.core.message.HumanMessage;
 import com.agentsflex.core.message.SystemMessage;
 import com.agentsflex.core.prompt.HistoriesPrompt;
+import com.agentsflex.core.prompt.ToolPrompt;
 import com.agentsflex.core.util.CollectionUtil;
 import com.alibaba.fastjson.JSON;
+import com.alibaba.fastjson.JSONObject;
 import com.alibaba.fastjson.serializer.SerializeConfig;
-import com.alibaba.fastjson2.JSONObject;
+import com.fasterxml.jackson.databind.ObjectMapper;
+import com.google.gson.Gson;
+import com.google.gson.GsonBuilder;
+import com.google.gson.JsonObject;
 import com.mybatisflex.core.query.QueryWrapper;
-import org.springframework.http.HttpStatus;
-import org.springframework.http.ResponseEntity;
+import com.mybatisflex.core.table.TableInfo;
+import com.mybatisflex.core.table.TableInfoFactory;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.http.*;
 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.RestController;
+import org.springframework.web.client.RestTemplate;
 import org.springframework.web.context.request.RequestContextHolder;
 import org.springframework.web.context.request.ServletRequestAttributes;
 import org.springframework.web.servlet.mvc.method.annotation.SseEmitter;
+import tech.aiflowy.ai.config.DifyStreamClient;
+import tech.aiflowy.ai.config.FileReference;
 import tech.aiflowy.ai.entity.*;
 import tech.aiflowy.ai.mapper.AiBotConversationMessageMapper;
 import tech.aiflowy.ai.service.*;
 import tech.aiflowy.common.ai.ChatManager;
 import tech.aiflowy.common.ai.MySseEmitter;
 import tech.aiflowy.common.domain.Result;
+import tech.aiflowy.common.entity.LoginAccount;
 import tech.aiflowy.common.satoken.util.SaTokenUtil;
 import tech.aiflowy.common.util.StringUtil;
 import tech.aiflowy.common.web.controller.BaseCurdController;
@@ -43,10 +56,7 @@
 import javax.servlet.http.HttpServletRequest;
 import javax.servlet.http.HttpServletResponse;
 import java.math.BigInteger;
-import java.util.Date;
-import java.util.HashMap;
-import java.util.List;
-import java.util.Map;
+import java.util.*;
 
 /**
  * 鎺у埗灞傘��
@@ -69,6 +79,10 @@
     @Resource
     private AiBotConversationMessageMapper aiBotConversationMessageMapper;
 
+    private static final Logger logger = LoggerFactory.getLogger(AiBotController.class);
+    @Autowired
+    private RestTemplate restTemplate;
+
     public AiBotController(AiBotService service, AiLlmService aiLlmService, AiBotWorkflowService aiBotWorkflowService, AiBotKnowledgeService aiBotKnowledgeService, AiBotMessageService aiBotMessageService) {
         super(service);
         this.aiLlmService = aiLlmService;
@@ -77,8 +91,6 @@
         this.aiBotMessageService = aiBotMessageService;
     }
 
-    @Resource
-    private AiPluginsService aiPluginsService;
     @Resource
     private AiBotPluginsService aiBotPluginsService;
     @Resource
@@ -125,79 +137,207 @@
      * @param response
      * @return
      */
+
+    @Autowired
+    private ObjectMapper objectMapper;
+
+
     @PostMapping("chat")
     public SseEmitter chat(@JsonBody(value = "prompt", required = true) String prompt,
                            @JsonBody(value = "botId", required = true) BigInteger botId,
                            @JsonBody(value = "sessionId", required = true) String sessionId,
                            @JsonBody(value = "isExternalMsg") int isExternalMsg,
+                           @JsonBody(value = "file") String file,
                            HttpServletResponse response) {
         response.setContentType("text/event-stream");
         AiBot aiBot = service.getById(botId);
         if (aiBot == null) {
             return ChatManager.getInstance().sseEmitterForContent("鏈哄櫒浜轰笉瀛樺湪");
         }
-
         Map<String, Object> llmOptions = aiBot.getLlmOptions();
         String systemPrompt = llmOptions != null ? (String) llmOptions.get("systemPrompt") : null;
-        AiLlm aiLlm = aiLlmService.getById(aiBot.getLlmId());
 
-        if (aiLlm == null) {
-            return ChatManager.getInstance().sseEmitterForContent("LLM涓嶅瓨鍦�");
-        }
+        if (StringUtil.hasText(aiBot.getModelAPI())){
+            if (aiBot.getBotTypeId() == 2) {
 
-        Llm llm = aiLlm.toLlm();
+                String apiUrl = aiBot.getModelAPI()+"/workflows/run"; // 鏇挎崲涓哄疄闄匒PI URL
+                String apiKey = aiBot.getModelKEY(); // 鏇挎崲涓哄疄闄匒PI Key
 
-        if (llm == null) {
-            return ChatManager.getInstance().sseEmitterForContent("LLM鑾峰彇涓虹┖");
-        }
+                DifyStreamClient client = new DifyStreamClient(apiUrl, apiKey, aiBotMessageService);
+                DifyStreamClient uploadClient = new DifyStreamClient(aiBot.getModelAPI()+"/files/upload", apiKey, aiBotMessageService);
+                String test = uploadClient.fileUpload(SaTokenUtil.getLoginAccount().getId() + "", file);
+                System.out.println(test);
+                Gson gson = new GsonBuilder().setPrettyPrinting().create();
+                JsonObject fileJson = gson.fromJson(test, JsonObject.class);
+                String fileId = fileJson.get("id").getAsString();
 
-        AiBotMessageMemory memory = new AiBotMessageMemory(botId, SaTokenUtil.getLoginAccount().getId(),
-                sessionId, isExternalMsg, aiBotMessageService, aiBotConversationMessageMapper,
-                aiBotConversationMessageService);
+                // 2. 鏋勫缓鏂囦欢鍙傛暟瀵硅薄
+                Map<String, Object> fileParam = new HashMap<>();
+                fileParam.put("transfer_method", "local_file");
+                fileParam.put("upload_file_id", fileId);
+                String[] split = file.split("\\.");
+//                fileParam.put("type", fileJson.get("extension").getAsString()); // 渚嬪 "excel"銆�"pdf" 绛�
+                fileParam.put("type", "document"); // 渚嬪 "excel"銆�"pdf" 绛�
 
-        final HistoriesPrompt historiesPrompt = new HistoriesPrompt();
-        if (systemPrompt != null) {
-            historiesPrompt.setSystemMessage(SystemMessage.of(systemPrompt));
-        }
+                // 3. 缁勮 inputs 鍙傛暟
+                Map<String, Object> inputs = new HashMap<>();
+                inputs.put("w", fileParam); // 娣诲姞鏂囦欢鍙傛暟锛寁ariableName 濡� "document"
 
-        historiesPrompt.setMemory(memory);
 
-        HumanMessage humanMessage = new HumanMessage(prompt);
 
-        // 娣诲姞鎻掍欢鐩稿叧鐨刦unction calling
-        appendPluginToolFunction(botId, humanMessage);
+                AiBotMessageMemory memory = new AiBotMessageMemory(botId, SaTokenUtil.getLoginAccount().getId(),
+                        sessionId, isExternalMsg, aiBotMessageService, aiBotConversationMessageMapper,
+                        aiBotConversationMessageService);
 
-        //娣诲姞宸ヤ綔娴佺浉鍏崇殑 Function Calling
-        appendWorkflowFunctions(botId, humanMessage);
+                final HistoriesPrompt historiesPrompt = new HistoriesPrompt();
+                if (systemPrompt != null) {
+                    historiesPrompt.setSystemMessage(SystemMessage.of(systemPrompt));
+                }
 
-        //娣诲姞鐭ヨ瘑搴撶浉鍏崇殑 Function Calling
-        appendKnowledgeFunctions(botId, humanMessage);
+                historiesPrompt.setMemory(memory);
 
-        historiesPrompt.addMessage(humanMessage);
+                HumanMessage humanMessage = new HumanMessage(prompt);
 
-        MySseEmitter emitter = new MySseEmitter((long) (1000 * 60 * 2));
+                // 娣诲姞鎻掍欢鐩稿叧鐨刦unction calling
+                appendPluginToolFunction(botId, humanMessage);
 
-        final Boolean[] needClose = {true};
-        if (humanMessage.getFunctions() != null && !humanMessage.getFunctions().isEmpty()) {
+                //娣诲姞宸ヤ綔娴佺浉鍏崇殑 Function Calling
+                appendWorkflowFunctions(botId, humanMessage);
+
+                //娣诲姞鐭ヨ瘑搴撶浉鍏崇殑 Function Calling
+                appendKnowledgeFunctions(botId, humanMessage);
+
+                historiesPrompt.addMessage(humanMessage);
+
+
+                final Boolean[] needClose = {true};
+
+                ServletRequestAttributes sra = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes();
+
+
+                MySseEmitter emitter = new MySseEmitter(1000L * 60 * 2); // 2鍒嗛挓瓒呮椂
+
+                try {
+                    String userId = SaTokenUtil.getLoginAccount().getId() + "";
+                    client.runWorkflow(inputs, prompt, userId, emitter, sessionId, botId);
+                } catch (Exception e) {
+                    emitter.completeWithError(e);
+                }
+//                System.out.println(emitter.toString());
+
+                return emitter;
+            }
+            aiBot.setModelAPI(aiBot.getModelAPI()+"/chat-messages");
+            String apiUrl = aiBot.getModelAPI(); // 鏇挎崲涓哄疄闄匒PI URL
+            String apiKey = aiBot.getModelKEY(); // 鏇挎崲涓哄疄闄匒PI Key
+
+            DifyStreamClient client = new DifyStreamClient(apiUrl, apiKey, aiBotMessageService);
+            AiBotMessageMemory memory = new AiBotMessageMemory(botId, SaTokenUtil.getLoginAccount().getId(),
+                    sessionId, isExternalMsg, aiBotMessageService, aiBotConversationMessageMapper,
+                    aiBotConversationMessageService);
+
+            final HistoriesPrompt historiesPrompt = new HistoriesPrompt();
+            if (systemPrompt != null) {
+                historiesPrompt.setSystemMessage(SystemMessage.of(systemPrompt));
+            }
+
+            historiesPrompt.setMemory(memory);
+
+            HumanMessage humanMessage = new HumanMessage(prompt);
+
+            // 娣诲姞鎻掍欢鐩稿叧鐨刦unction calling
+            appendPluginToolFunction(botId, humanMessage);
+
+            //娣诲姞宸ヤ綔娴佺浉鍏崇殑 Function Calling
+            appendWorkflowFunctions(botId, humanMessage);
+
+            //娣诲姞鐭ヨ瘑搴撶浉鍏崇殑 Function Calling
+            appendKnowledgeFunctions(botId, humanMessage);
+
+            historiesPrompt.addMessage(humanMessage);
+
+
+            final Boolean[] needClose = {true};
+
+            ServletRequestAttributes sra = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes();
+
+
+            MySseEmitter emitter = new MySseEmitter(1000L * 60 * 2); // 2鍒嗛挓瓒呮椂
+
             try {
-                AiMessageResponse aiMessageResponse = llm.chat(historiesPrompt);
-                function_call(aiMessageResponse, emitter, needClose, historiesPrompt, llm, prompt, false);
+                String userId = SaTokenUtil.getLoginAccount().getId() + "";
+                client.chatStream(prompt, userId, emitter, sessionId, botId);
             } catch (Exception e) {
                 emitter.completeWithError(e);
             }
+            System.out.println(emitter.toString());
+            return emitter;
+        }
+        else{
+            AiLlm aiLlm = aiLlmService.getById(aiBot.getLlmId());
 
-            if (needClose[0]) {
-                System.out.println("function chat complete");
-                emitter.complete();
+            if (aiLlm == null) {
+                return ChatManager.getInstance().sseEmitterForContent("LLM涓嶅瓨鍦�");
             }
-        } else {
 
+            Llm llm = aiLlm.toLlm();
+            if (llm == null) {
+                return ChatManager.getInstance().sseEmitterForContent("LLM鑾峰彇涓虹┖");
+            }
+
+            AiBotMessageMemory memory = new AiBotMessageMemory(botId, SaTokenUtil.getLoginAccount().getId(),
+                    sessionId, isExternalMsg, aiBotMessageService, aiBotConversationMessageMapper,
+                    aiBotConversationMessageService);
+
+            final HistoriesPrompt historiesPrompt = new HistoriesPrompt();
+            if (systemPrompt != null) {
+                historiesPrompt.setSystemMessage(SystemMessage.of(systemPrompt));
+            }
+
+            historiesPrompt.setMemory(memory);
+
+            HumanMessage humanMessage = new HumanMessage(prompt);
+
+            // 娣诲姞鎻掍欢鐩稿叧鐨刦unction calling
+            appendPluginToolFunction(botId, humanMessage);
+
+            //娣诲姞宸ヤ綔娴佺浉鍏崇殑 Function Calling
+            appendWorkflowFunctions(botId, humanMessage);
+
+            //娣诲姞鐭ヨ瘑搴撶浉鍏崇殑 Function Calling
+            appendKnowledgeFunctions(botId, humanMessage);
+
+            historiesPrompt.addMessage(humanMessage);
+
+            MySseEmitter emitter = new MySseEmitter((long) (1000 * 60 * 2));
+
+            final Boolean[] needClose = {true};
+
+            ServletRequestAttributes sra = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes();
+            // 缁熶竴浣跨敤娴佸紡澶勭悊锛屾棤璁烘槸鍚︽湁 Function Calling
             llm.chatStream(historiesPrompt, new StreamResponseListener() {
                 @Override
                 public void onMessage(ChatContext context, AiMessageResponse response) {
                     try {
+                        RequestContextHolder.setRequestAttributes(sra, true);
+                        if (response != null) {
+                            // 妫�鏌ユ槸鍚﹂渶瑕佽Е鍙� Function Calling
+                            if (response.getFunctionCallers() != null && CollectionUtil.hasItems(response.getFunctionCallers())) {
+                                needClose[0] = false;
+                                function_call(response, emitter, needClose, historiesPrompt, llm, prompt, false);
+                            } else {
+                                // 寮哄埗娴佸紡杩斿洖锛屽嵆浣挎湁 Function Calling 涔熷厛杩斿洖閮ㄥ垎缁撴灉
+                                if (response.getMessage() != null) {
+                                    String content = response.getMessage().getContent();
+                                    if (StringUtil.hasText(content)) {
+                                        System.out.println(response);
+                                        emitter.send(JSON.toJSONString(response.getMessage()));
+                                    }
+                                }
+                            }
+                        }
 
-                        function_call(response, emitter, needClose, historiesPrompt, llm, prompt, false);
+
                     } catch (Exception e) {
                         emitter.completeWithError(e);
                     }
@@ -206,7 +346,6 @@
                 @Override
                 public void onStop(ChatContext context) {
                     if (needClose[0]) {
-                        System.out.println("normal chat complete");
                         emitter.complete();
                     }
                 }
@@ -216,10 +355,74 @@
                     emitter.completeWithError(throwable);
                 }
             });
+
+            System.out.println(emitter.toString());
+            return emitter;
         }
 
-        return emitter;
     }
+
+    @PostMapping("files/upload")
+    public Result filesUpload(@JsonBody(value = "botId", required = true) BigInteger botId,
+                           String file,
+                           HttpServletResponse response){
+        try{
+            String userId = SaTokenUtil.getLoginAccount().getId() + "";
+            response.setContentType("text/event-stream");
+            AiBot aiBot = service.getById(botId);
+            aiBot.setModelAPI(aiBot.getModelAPI()+"/files/upload");
+            String apiUrl = aiBot.getModelAPI(); // 鏇挎崲涓哄疄闄匒PI URL
+            String apiKey = aiBot.getModelKEY(); // 鏇挎崲涓哄疄闄匒PI Key
+            DifyStreamClient client = new DifyStreamClient(apiUrl, apiKey, aiBotMessageService);
+            client.fileUpload(userId,file);
+            return Result.success("鎴愬姛锛�");
+        }catch (Exception e){
+            return Result.fail(400, String.valueOf(e));
+        }
+    }
+
+
+
+    public Result save(@RequestBody String jsonStr) {
+        // 瑙f瀽JSON
+        JSONObject json = JSONObject.parseObject(jsonStr);
+
+        // 鍚堝苟鎵�鏈塻econdMenuId*瀛楁
+        List<Integer> menuIds = new ArrayList<>();
+        for (String key : json.keySet()) {
+            if (key.startsWith("secondMenuId")) {
+                Object value = json.get(key);
+                if (value instanceof Integer) {
+                    menuIds.add((Integer) value);
+                }
+            }
+        }
+
+        // 淇濈暀绗竴涓狪D锛堟牴鎹渶瑕佽皟鏁达級
+        if (!menuIds.isEmpty()) {
+            json.put("secondMenuId", menuIds.get(0));
+        }
+
+        // 杞崲涓哄疄浣撶被
+        AiBot entity = json.toJavaObject(AiBot.class);
+
+        // 鍚庣画澶勭悊淇濇寔涓嶅彉
+        Result result = onSaveOrUpdateBefore(entity, true);
+        if (result != null) return result;
+
+        if (entity == null) {
+            throw new NullPointerException("entity is null");
+        }
+        LoginAccount loginAccount = SaTokenUtil.getLoginAccount();
+        commonFiled(entity,loginAccount.getId(),loginAccount.getTenantId(),loginAccount.getDeptId());
+        boolean success = service.save(entity);
+        onSaveOrUpdateAfter(entity, true);
+        TableInfo tableInfo = TableInfoFactory.ofEntityClass(entity.getClass());
+        Object[] pkArgs = tableInfo.buildPkSqlArgs(entity);
+        return Result.create(success).set("id", pkArgs);
+    }
+
+
 
     /**
      * 澶栭儴鐢ㄦ埛璋冪敤鏅鸿兘浣撹繘琛屽璇�
@@ -362,9 +565,15 @@
         AiBotExternalMsgJsonResult messageResult = new AiBotExternalMsgJsonResult();
         messageResult.setCreated(new Date().getTime());
         AiBotExternalMsgJsonResult.Usage usage = new AiBotExternalMsgJsonResult.Usage();
-        usage.setTotalTokens(aiMessage.getTotalTokens());
-        usage.setCompletionTokens(aiMessage.getCompletionTokens());
-        usage.setPromptTokens(aiMessage.getPromptTokens());
+        if (aiMessage.getTotalTokens() != null){
+            usage.setTotalTokens(aiMessage.getTotalTokens());
+        }
+        if (aiMessage.getCompletionTokens() != null){
+            usage.setCompletionTokens(aiMessage.getCompletionTokens());
+        }
+        if (aiMessage.getPromptTokens() != null){
+            usage.setPromptTokens(aiMessage.getPromptTokens());
+        }
         messageResult.setUsage(usage);
         AiBotExternalMsgJsonResult.Choice choice = new AiBotExternalMsgJsonResult.Choice();
         AiBotExternalMsgJsonResult.Message message = new AiBotExternalMsgJsonResult.Message();
@@ -411,9 +620,6 @@
             // 濡傛灉鏄閮ㄧ郴缁熻皟鐢╟hat
             if (isExternalChatApi) {
                 AiBotExternalMsgJsonResult result = handleMessageStreamJsonResult(aiMessageResponse.getMessage());
-                System.out.println(">>>>>>>>>>>>>>>>>>>>>>>>>>>>>");
-                System.out.println(JSON.toJSONString(result, new SerializeConfig()));
-                System.out.println("鏈畬娴嬭瘯");
 
                 emitter.send(JSON.toJSONString(result, new SerializeConfig()));
             } else {
@@ -421,45 +627,34 @@
             }
 
         }
-        List<FunctionCaller> functionCallers = aiMessageResponse.getFunctionCallers();
-        if (CollectionUtil.hasItems(functionCallers)) {
-            needClose[0] = false;
-            for (FunctionCaller functionCaller : functionCallers) {
-                Object result = functionCaller.call();
-                if (ObjectUtil.isNotEmpty(result)) {
-
-                    String newPrompt = "璇锋牴鎹互涓嬪唴瀹瑰洖绛旂敤鎴凤紝鍐呭鏄�:\n" + result + "\n 鐢ㄦ埛鐨勯棶棰樻槸锛�" + prompt;
-                    historiesPrompt.addMessageTemporary(new HumanMessage(newPrompt));
-
-                    llm.chatStream(historiesPrompt, new StreamResponseListener() {
-                        @Override
-                        public void onMessage(ChatContext context, AiMessageResponse response) {
-                            needClose[0] = true;
-                            String content = response.getMessage().getContent();
-                            Object messageContent = response.getMessage();
-                            if (StringUtil.hasText(content)) {
-                                String jsonResult = JSON.toJSONString(messageContent);
-                                emitter.send(jsonResult);
-                            }
-                        }
-
-                        @Override
-                        public void onStop(ChatContext context) {
-                            if (needClose[0]) {
-                                System.out.println("function chat complete");
-                                emitter.complete();
-                            }
-                            historiesPrompt.clearTemporaryMessages();
-                        }
-
-                        @Override
-                        public void onFailure(ChatContext context, Throwable throwable) {
-                            emitter.completeWithError(throwable);
-                        }
-                    });
+        System.out.println("function call 鎺ユ敹鍒扮殑鍙傛暟message锛�" + aiMessageResponse);
+        llm.chatStream(ToolPrompt.of(aiMessageResponse), new StreamResponseListener() {
+            @Override
+            public void onMessage(ChatContext context, AiMessageResponse response) {
+                System.out.println("function call <UNK>message<UNK>" + aiMessageResponse);
+                String content = response.getMessage().getContent();
+                if (StringUtil.hasText(content)) {
+                    System.out.println("if content"  + content);
+                    emitter.send(JSON.toJSONString(response.getMessage()));
                 }
             }
-        }
+
+            @Override
+            public void onStop(ChatContext context) {
+                System.out.println("function call complete");
+                emitter.complete();
+            }
+
+            @Override
+            public void onFailure(ChatContext context, Throwable throwable) {
+                logger.error("function_call鎶ラ敊:",throwable);
+                AiMessage aiMessage = new AiMessage();
+                aiMessage.setContent("鏈煡璇㈠埌鐩稿叧淇℃伅...");
+                emitter.send(JSON.toJSONString(aiMessage));
+                System.out.println("function call complete with error");
+            }
+        });
+
         return JSON.toJSONString(messageContent);
     }
 
@@ -550,4 +745,4 @@
 
 
     }
-}
+}
\ No newline at end of file

--
Gitblit v1.8.0