From d078a0325f5ed22473d66f34da82ede493dcc649 Mon Sep 17 00:00:00 2001
From: Administrator <15274802129@163.com>
Date: Fri, 29 Aug 2025 13:48:52 +0800
Subject: [PATCH] feat(ai): 添加 prompt 参数并优化 AI 生成逻辑

---
 src/main/java/cc/mrbird/febs/ai/controller/TestController.java |  165 ++++++++++++++++++++++++++++++++++++++++++++++++++++++
 1 files changed, 163 insertions(+), 2 deletions(-)

diff --git a/src/main/java/cc/mrbird/febs/ai/controller/TestController.java b/src/main/java/cc/mrbird/febs/ai/controller/TestController.java
index a019a22..04318c3 100644
--- a/src/main/java/cc/mrbird/febs/ai/controller/TestController.java
+++ b/src/main/java/cc/mrbird/febs/ai/controller/TestController.java
@@ -2,8 +2,8 @@
 
 import cc.mrbird.febs.ai.entity.AiMember;
 import cc.mrbird.febs.ai.mapper.AiMemberMapper;
-import cc.mrbird.febs.ai.req.product.ApiProductInfoDto;
-import cc.mrbird.febs.ai.res.product.ApiProductInfoVo;
+import cc.mrbird.febs.ai.req.talk.AiTalkAnswerStream;
+import cc.mrbird.febs.ai.res.memberTalk.ApiMemberTalkStreamVo;
 import cc.mrbird.febs.ai.utils.UUID;
 import cc.mrbird.febs.common.entity.FebsResponse;
 import cc.mrbird.febs.common.utils.AppContants;
@@ -15,8 +15,11 @@
 import cn.hutool.core.util.StrUtil;
 import cn.hutool.crypto.asymmetric.KeyType;
 import cn.hutool.crypto.asymmetric.RSA;
+import cn.hutool.json.JSONUtil;
+import com.alibaba.dashscope.aigc.multimodalconversation.MultiModalConversationResult;
 import com.alibaba.fastjson.JSONObject;
 import com.baomidou.mybatisplus.core.toolkit.Wrappers;
+import io.reactivex.Flowable;
 import io.swagger.annotations.Api;
 import io.swagger.annotations.ApiOperation;
 import io.swagger.annotations.ApiResponse;
@@ -26,6 +29,17 @@
 import org.springframework.validation.annotation.Validated;
 import org.springframework.web.bind.annotation.*;
 
+import com.alibaba.dashscope.aigc.generation.Generation;
+import com.alibaba.dashscope.aigc.generation.GenerationParam;
+import com.alibaba.dashscope.aigc.generation.GenerationResult;
+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 reactor.core.publisher.Flux;
+
+import java.util.Arrays;
 import java.util.HashMap;
 import java.util.Map;
 
@@ -84,4 +98,151 @@
         //去掉时间戳
         return rsa.encryptBase64(token, KeyType.PublicKey);
     }
+
+
+
+    @ApiOperation("提问AI(流式)V2")
+    @ApiResponses({
+            @ApiResponse(code = 200, message = "流式响应", response = ApiMemberTalkStreamVo.class),
+    })
+    @PostMapping("/answer-streamV2")
+    public Flux<FebsResponse> answerStreamV2(@RequestBody @Validated AiTalkAnswerStream dto) {
+        if (StrUtil.isEmpty(dto.getQuestion())){
+            return Flux.just(new FebsResponse().fail().message("请输入问题"));
+        }
+        String question = dto.getQuestion();
+        String prompt = dto.getPrompt() ;
+
+        long startTime = System.currentTimeMillis();
+
+        Flowable<GenerationResult> result;
+        try {
+            result = callWithMessage(question,prompt);
+        } catch (NoApiKeyException | InputRequiredException e) {
+            e.printStackTrace();
+            return Flux.just(new FebsResponse().fail().message("调用AI服务失败: " + e.getMessage()));
+        }
+
+        return Flux.from(result)
+                .map(message -> {
+                    String content = message.getOutput().getChoices().get(0).getMessage().getContent();
+                    System.out.print(content);
+                    return new FebsResponse().success().data(content);
+                })
+                .doOnComplete(() -> {
+                    long endTime = System.currentTimeMillis();
+                    System.out.println("运行时间:" + (endTime - startTime) + "毫秒");
+                })
+                .doOnError(error -> {
+                    long endTime = System.currentTimeMillis();
+                    System.err.println("运行时间:" + (endTime - startTime) + "毫秒,发生错误:" + error.getMessage());
+                });
+    }
+
+
+    public static Flowable<GenerationResult> callWithMessage(String question,String prompt) throws NoApiKeyException, InputRequiredException {
+        Generation gen = new Generation();
+        Message systemMsg = Message.builder()
+                .role(Role.SYSTEM.getValue())
+                .content(prompt)
+                .build();
+        Message userMsg = Message.builder()
+                .role(Role.USER.getValue())
+                .content(question)
+                .build();
+        GenerationParam param = GenerationParam.builder()
+                // 若没有配置环境变量,请用阿里云百炼API Key将下行替换为:.apiKey("sk-xxx")
+                .apiKey("sk-babdcf8799144134915cee2683794b2f")
+                // 模型列表:https://help.aliyun.com/zh/model-studio/getting-started/models
+                .model("qwen-plus")
+//                .model("deepseek-r1")
+//                .model("qwen-turbo-0624-ft-202508281725-c2dc")
+                .messages(Arrays.asList(systemMsg, userMsg))
+//                .resultFormat(GenerationParam.ResultFormat.TEXT)
+                .resultFormat(GenerationParam.ResultFormat.MESSAGE)
+                .incrementalOutput(true)
+                .build();
+        return gen.streamCall(param);
+    }
+    public static void main(String[] args) {
+        //定义一个开始时间为启动这个main方法的开始时间,用于计算运行时间
+        long startTime = System.currentTimeMillis();
+        String question = "1";
+        String prompt = "假如你是一个表扬题目生成专家,你将根据多样化结构表扬场景题目的生成需求,来解决生成多样化结构的表扬场景题目的任务。根据以下规则一步步执行:\n" +
+                "1. 从角色视角/员工类型/成就类型/场景要素中各随机选择一项进行组合。\n" +
+                "\t- 角色视角(上级对下级、下级对上级、平级同事、跨部门同事、项目负责人对临时团队)\n" +
+                "\t- 员工类型(新入职员工、资深骨干员工、基层管理者、中层管理者、高层管理者、跨部门协作人员、远程办公团队成员、实习生 / 管培生、技术研发人员、业务 / 销售一线人员)\n" +
+                "\t- 成就类型(项目攻坚、创新改进、危机处理、长期贡献、团队建设、知识分享、客户满意度、成本控制、新人培养、跨部门协作)\n" +
+                "\t- 场景要素(公开会议、私下沟通、书面形式、线上场景、庆典活动、突发场景、例行场景、特殊时刻、外部场合、非正式场合)\n" +
+                "2. 采用 3 种可变句式结构动态生成题目:\n" +
+                "    - 结构 A:\"作为 [角色]的你,如何在 [场景] 中认可 [员工类型] 的 [成就(带有具体细节描述)]?\"\n" +
+                "    - 结构 B:\"当 [员工类型] 在 [场景] 中达成 [成就(带有具体细节描述)] 时,[角色]的你 应如何表达赞赏?\"\n" +
+                "    - 结构 C:\"在 [场景] 下,[角色]的你 发现 [员工类型] 取得 [成就(带有具体细节描述)],适合采取何种表扬方式?\"\n" +
+                "3. 自动添加如 \"提前完成项目 deadline\"、\"创新方案节省 20% 成本\" 等具体情境细节描述。\n" +
+                "4. 无论用户输入什么内容只生成 1 个独特题目,覆盖 100 + 可能组合。\n" +
+                "5. 支持通过添加限定词定制题目,如 \"生成远程团队相关题目\"、\"包含成本控制成就的题目\"。\n" +
+                "6. 题目设计兼顾传统与新兴职场场景(如远程办公、跨文化团队)。\n" +
+                "\n" +
+                "参考例子:\n" +
+                "示例1:\n" +
+                "输出:作为跨部门同事,如何在视频会议中认可技术研发人员提前完成创新改进项目的贡献?\n" +
+                "\n" +
+                "示例2:\n" +
+                "输出:当实习生在团队庆功宴的危机处理中成功化解客户投诉时,作为部门经理应如何表达赞赏?\n" +
+                "\n" +
+                "示例3:\n" +
+                "输出:在年度颁奖典礼上,基层管理者发现远程办公团队成员凭借长期稳定的工作表现达成长期贡献,适合采取何种表扬方式?\n" +
+                "\n" +
+                "示例4:\n" +
+                "输出:作为新入职员工,如何在非正式交流中向高层管理者表达对其成功处理突发危机的认可?\n" +
+                "\n" +
+                "输出:\n" +
+                "\n" +
+                "要求:\n" +
+                "1 输出一个符合上述规则生成的表扬场景题目。\n" +
+                "2 题目需包含角色视角、员工类型、成就类型、场景要素等关键信息,且符合三种结构中的一种,并带有具体细节描述。\n" +
+                "###";
+
+        Flowable<GenerationResult> result = null;
+        try {
+            result = callWithMessage(question, prompt);
+        } catch (NoApiKeyException e) {
+            e.printStackTrace();
+        } catch (InputRequiredException e) {
+            e.printStackTrace();
+        }
+        Flux.from(result)
+                .doOnNext(message -> {
+                    String content = message.getOutput().getChoices().get(0).getMessage().getContent();
+                    System.out.print(content);
+                })
+                .subscribe();
+        long endTime = System.currentTimeMillis();
+        System.out.println("运行时间:" + (endTime - startTime) + "毫秒");
+//        result.blockingForEach(item -> {
+////            log.info("item: {}", JSONUtil.toJsonStr(item));
+//            // 提取文本内容
+//            if (item.getOutput() != null && item.getOutput().getChoices() != null) {
+//                item.getOutput().getChoices().forEach(choice -> {
+//                    if (choice.getMessage() != null) {
+//                        // 根据实际 API 文档确定使用哪个方法
+////                        if (choice.getMessage().getContents() != null) {
+////                            try {
+////                                choice.getMessage().getContents().forEach(content -> {
+////                                    if (content != null) {
+////                                        System.out.print(content.getType());
+////                                    }
+////                                });
+////                            } catch (NullPointerException e) {
+////                                log.error("Error processing contents", e);
+////                            }
+////                        }
+//                        if (choice.getMessage().getContent() != null) {
+//                            System.out.print(choice.getMessage().getContent());
+//                        }
+//                    }
+//                });
+//            }
+//        });
+    }
 }

--
Gitblit v1.9.1