package cc.mrbird.febs.ai.service.impl;
|
|
import cc.mrbird.febs.ai.entity.*;
|
import cc.mrbird.febs.ai.mapper.AiProductQuestionJobMapper;
|
import cc.mrbird.febs.ai.mapper.AiProductQuestionMapper;
|
import cc.mrbird.febs.ai.req.AiProductQuestionAiDto;
|
import cc.mrbird.febs.ai.service.AiProductCategoryService;
|
import cc.mrbird.febs.ai.service.AiProductQuestionItemService;
|
import cc.mrbird.febs.ai.service.AiProductQuestionService;
|
import cc.mrbird.febs.ai.service.AiService;
|
import cc.mrbird.febs.ai.util.UUID;
|
import cc.mrbird.febs.common.entity.FebsResponse;
|
import cc.mrbird.febs.common.entity.QueryRequest;
|
import cc.mrbird.febs.common.exception.FebsException;
|
import cc.mrbird.febs.common.utils.excl.ExcelSheetPO;
|
import cc.mrbird.febs.common.utils.excl.ExcelUtil;
|
import cc.mrbird.febs.common.utils.excl.ExcelVersion;
|
import cc.mrbird.febs.common.utils.excl.ResponseHeadUtil;
|
import cc.mrbird.febs.mall.dto.DeliverGoodsDto;
|
import cc.mrbird.febs.mall.entity.MallMember;
|
import cc.mrbird.febs.mall.entity.MallOrderInfo;
|
import cc.mrbird.febs.mall.entity.MallOrderItem;
|
import cc.mrbird.febs.rabbit.producter.AgentProducer;
|
import cn.hutool.core.collection.CollUtil;
|
import cn.hutool.core.date.DateUtil;
|
import cn.hutool.core.util.ObjectUtil;
|
import cn.hutool.core.util.StrUtil;
|
import cn.hutool.crypto.SecureUtil;
|
import cn.hutool.json.JSONArray;
|
import cn.hutool.json.JSONObject;
|
import cn.hutool.json.JSONUtil;
|
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
|
import com.baomidou.mybatisplus.core.conditions.update.LambdaUpdateWrapper;
|
import com.baomidou.mybatisplus.core.metadata.IPage;
|
import com.baomidou.mybatisplus.core.toolkit.Wrappers;
|
import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
|
import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
|
import com.fasterxml.jackson.databind.JsonNode;
|
import com.fasterxml.jackson.databind.ObjectMapper;
|
import com.fasterxml.jackson.databind.SerializationFeature;
|
import lombok.RequiredArgsConstructor;
|
import lombok.extern.slf4j.Slf4j;
|
import org.springframework.stereotype.Service;
|
import org.springframework.transaction.annotation.Transactional;
|
import org.springframework.web.multipart.MultipartFile;
|
|
import javax.servlet.http.HttpServletResponse;
|
import java.io.File;
|
import java.io.IOException;
|
import java.io.OutputStream;
|
import java.net.URLEncoder;
|
import java.util.*;
|
import java.util.stream.Collectors;
|
|
/**
|
* AI产品题目 Service实现类
|
*
|
* @author yourname
|
* @date 2025-07-29
|
*/
|
@Slf4j
|
@Service
|
@RequiredArgsConstructor
|
public class AiProductQuestionServiceImpl extends ServiceImpl<AiProductQuestionMapper, AiProductQuestion> implements AiProductQuestionService {
|
|
private final AiProductQuestionMapper aiProductQuestionMapper;
|
private final AiProductQuestionJobMapper aiProductQuestionJobMapper;
|
private final AiProductQuestionItemService aiProductQuestionItemService;
|
private final AiProductCategoryService aiProductCategoryService;
|
private final AiService aiService;
|
private final AgentProducer agentProducer;
|
|
@Override
|
public AiProductQuestion getById(String id) {
|
|
return aiProductQuestionMapper.selectById(id);
|
}
|
|
@Override
|
public IPage<AiProductQuestion> listInPage(AiProductQuestion dto, QueryRequest request) {
|
Page<AiProductQuestion> page = new Page<>(request.getPageNum(), request.getPageSize());
|
LambdaQueryWrapper<AiProductQuestion> query = Wrappers.lambdaQuery(AiProductQuestion.class);
|
if (StrUtil.isNotEmpty(dto.getCompanyId())){
|
query.eq(AiProductQuestion::getCompanyId, dto.getCompanyId());
|
}
|
if (StrUtil.isNotEmpty(dto.getProductCategoryId())){
|
query.eq(AiProductQuestion::getProductCategoryId, dto.getProductCategoryId());
|
}
|
query.ne(AiProductQuestion::getState, 2);
|
query.orderByDesc(AiProductQuestion::getCreatedTime);
|
Page<AiProductQuestion> pages = aiProductQuestionMapper.selectPage(page, query);
|
List<AiProductQuestion> records = pages.getRecords();
|
if (CollUtil.isNotEmpty( records)){
|
//stream流操作records,获取全部的productCategoryId的set集合
|
Set<String> productCategoryIds = records.stream().map(AiProductQuestion::getProductCategoryId).collect(Collectors.toSet());
|
if(CollUtil.isNotEmpty( productCategoryIds)){
|
Map<String,AiProductCategory> map = aiProductCategoryService.selectMapByIds(productCategoryIds);
|
for (AiProductQuestion record : records){
|
AiProductCategory orDefault = map.getOrDefault(record.getProductCategoryId(), null);
|
if(ObjectUtil.isNotNull(orDefault)){
|
record.setProductCategoryName(orDefault.getName());
|
}
|
}
|
}
|
}
|
return pages;
|
}
|
|
@Override
|
public IPage<AiProductQuestionJob> listJobInPage(AiProductQuestionJob dto, QueryRequest request) {
|
Page<AiProductQuestionJob> page = new Page<>(request.getPageNum(), request.getPageSize());
|
LambdaQueryWrapper<AiProductQuestionJob> query = Wrappers.lambdaQuery(AiProductQuestionJob.class);
|
if (StrUtil.isNotEmpty(dto.getCompanyId())){
|
query.eq(AiProductQuestionJob::getCompanyId, dto.getCompanyId());
|
}
|
if (StrUtil.isNotEmpty(dto.getProductCategoryId())){
|
query.eq(AiProductQuestionJob::getProductCategoryId, dto.getProductCategoryId());
|
}
|
query.orderByDesc(AiProductQuestionJob::getCreatedTime);
|
Page<AiProductQuestionJob> pages = aiProductQuestionJobMapper.selectPage(page, query);
|
return pages;
|
}
|
|
@Override
|
public FebsResponse changeState(String id, Integer state) {
|
AiProductQuestion entity = this.getById(id);
|
if(ObjectUtil.isNotEmpty(entity)){
|
this.update(null,
|
Wrappers.lambdaUpdate(AiProductQuestion.class)
|
.set(AiProductQuestion::getState, state)
|
.set(AiProductQuestion::getUpdatedTime, new Date())
|
.eq(AiProductQuestion::getId, id));
|
}
|
return new FebsResponse().success().message("操作成功");
|
}
|
|
@Override
|
public FebsResponse add(AiProductQuestion dto) {
|
|
List<AiProductQuestionItem> aiProductQuestionItems = dto.getAiProductQuestionItems();
|
if(CollUtil.isEmpty(aiProductQuestionItems)){
|
return new FebsResponse().fail().message("请添加答案选项");
|
}
|
//stream流操作aiProductQuestionItems,返回aiProductQuestionItems是否存在中correctAnswer为1的元素
|
boolean hasCorrectAnswer = aiProductQuestionItems.stream().anyMatch(item -> item != null && item.getCorrectAnswer() != null && item.getCorrectAnswer() == 1);
|
if (!hasCorrectAnswer) {
|
return new FebsResponse().fail().message("请添加一项为正确的答案");
|
}
|
|
AiProductQuestion entity = new AiProductQuestion();
|
entity.setId(UUID.getSimpleUUIDString());
|
entity.setCompanyId(dto.getCompanyId());
|
entity.setProductCategoryId(dto.getProductCategoryId());
|
entity.setTitle(dto.getTitle());
|
entity.setDifficulty(dto.getDifficulty());
|
entity.setCreatedTime(new Date());
|
this.save(entity);
|
|
|
for (AiProductQuestionItem item : dto.getAiProductQuestionItems()){
|
item.setId(UUID.getSimpleUUIDString());
|
item.setProductQuestionId(entity.getId());
|
item.setCompanyId(dto.getCompanyId());
|
item.setTitle(entity.getTitle());
|
item.setCreatedTime(new Date());
|
aiProductQuestionItemService.getBaseMapper().insert(item);
|
}
|
return new FebsResponse().success().message("操作成功");
|
}
|
|
|
|
@Override
|
public FebsResponse update(AiProductQuestion dto) {
|
String id = dto.getId();
|
AiProductQuestion entity = this.getById(id);
|
|
List<AiProductQuestionItem> aiProductQuestionItems = dto.getAiProductQuestionItems();
|
if(CollUtil.isEmpty(aiProductQuestionItems)){
|
return new FebsResponse().fail().message("请添加答案选项");
|
}
|
//stream流操作aiProductQuestionItems,返回aiProductQuestionItems是否存在中correctAnswer为1的元素
|
boolean hasCorrectAnswer = aiProductQuestionItems.stream().anyMatch(item -> item != null && item.getCorrectAnswer() != null && item.getCorrectAnswer() == 1);
|
if (!hasCorrectAnswer) {
|
return new FebsResponse().fail().message("请添加一项为正确的答案");
|
}
|
if (ObjectUtil.isNotNull( entity)){
|
this.update(null,
|
Wrappers.lambdaUpdate(AiProductQuestion.class)
|
.set(AiProductQuestion::getProductCategoryId, dto.getProductCategoryId())
|
.set(AiProductQuestion::getTitle, dto.getTitle())
|
.set(AiProductQuestion::getDifficulty, dto.getDifficulty())
|
.set(AiProductQuestion::getUpdatedTime, new Date())
|
.eq(AiProductQuestion::getId, id)
|
);
|
|
aiProductQuestionItemService.deleteByQuestionId(id);
|
|
for (AiProductQuestionItem item : dto.getAiProductQuestionItems()){
|
item.setId(UUID.getSimpleUUIDString());
|
item.setProductQuestionId(entity.getId());
|
item.setTitle(entity.getTitle());
|
item.setCreatedTime(new Date());
|
aiProductQuestionItemService.getBaseMapper().insert(item);
|
}
|
}
|
return new FebsResponse().success().message("操作成功");
|
}
|
|
@Override
|
public FebsResponse stateUpdate(String ids, Integer type) {
|
if (StrUtil.isEmpty(ids)) {
|
throw new FebsException("参数错误");
|
}
|
List<String> idList = StrUtil.split(ids, ',');
|
this.update(null,
|
Wrappers.lambdaUpdate(AiProductQuestion.class)
|
.set(AiProductQuestion::getState, type)
|
.set(AiProductQuestion::getUpdatedTime, new Date())
|
.in(AiProductQuestion::getId, idList));
|
return new FebsResponse().success().message("操作成功");
|
}
|
|
@Override
|
public FebsResponse delete(String id) {
|
AiProductQuestion entity = this.getById(id);
|
if(ObjectUtil.isNotNull( entity)){
|
this.update(null,
|
Wrappers.lambdaUpdate(AiProductQuestion.class)
|
.set(AiProductQuestion::getState, 2)
|
.set(AiProductQuestion::getUpdatedTime, new Date())
|
.eq(AiProductQuestion::getId, id));
|
|
}
|
return new FebsResponse().success().message("操作成功");
|
}
|
|
@Override
|
public FebsResponse productQuestionDelete(String ids) {
|
if (StrUtil.isEmpty(ids)) {
|
throw new FebsException("参数错误");
|
}
|
List<String> idList = StrUtil.split(ids, ',');
|
this.update(null,
|
Wrappers.lambdaUpdate(AiProductQuestion.class)
|
.set(AiProductQuestion::getState, 2)
|
.set(AiProductQuestion::getUpdatedTime, new Date())
|
.eq(AiProductQuestion::getState, 0)
|
.in(AiProductQuestion::getId, idList));
|
return new FebsResponse().success().message("操作成功");
|
}
|
|
@Override
|
public List<AiProductQuestion> questionTree(String companyId) {
|
LambdaQueryWrapper<AiProductQuestion> query = Wrappers.lambdaQuery(AiProductQuestion.class);
|
if (StrUtil.isNotEmpty(companyId)){
|
query.eq(AiProductQuestion::getCompanyId, companyId);
|
}
|
|
return aiProductQuestionMapper.selectList(query);
|
}
|
|
@Override
|
public List<AiProductQuestion> productQuestionTree(LambdaQueryWrapper<AiProductQuestion> aiProductQuestionLambdaQueryWrapper) {
|
return aiProductQuestionMapper.selectList(aiProductQuestionLambdaQueryWrapper);
|
}
|
|
@Override
|
public FebsResponse aiAdd(AiProductQuestionAiDto dto) {
|
|
String jsonFormat = "{\"question_list\":[{\"title\": \"消费者对透明质酸的主要担忧是什么?\",\"answer_list\":[{\"answer\": \"消费者担心透明质酸维持时间短,效果不明显。\",\"type\": 1,\"analysis\": \"\"},{\"answer\": \"消费者担心透明质酸注射后可能出现移位、凹陷、馒化、僵硬以及炎症反应。\",\"type\": 1,\"analysis\": \"\"},{\"answer\": \"消费者的主要担忧包括效果不明显、维持时间短、注射后出现移位、凹陷、馒化、僵硬以及炎症反应。他们希望既达到理想效果,又避免这些副作用。\",\"type\": 2,\"analysis\": \"标准答案涵盖了消费者对透明质酸的所有主要担忧,既包括效果问题,也涵盖副作用风险,全面反映消费者心理需求。\"}]}]} ";
|
dto.setJsonFormat(jsonFormat);
|
String questionAndAnswerStr = aiService.llmInvokeNonStreaming(dto);
|
System.out.println(questionAndAnswerStr);
|
// 解析AI返回的结果
|
List<JSONObject> questionList = parseAiQuestionResponse(questionAndAnswerStr);
|
|
String productCategoryId = dto.getProductCategoryId();
|
Integer difficulty = dto.getDifficulty();
|
Date createdTime = new Date();
|
// 处理解析后的问题列表
|
for (JSONObject questionObj : questionList) {
|
String title = questionObj.getStr("title");
|
|
AiProductQuestion aiProductQuestion = new AiProductQuestion();
|
aiProductQuestion.setId(UUID.getSimpleUUIDString());
|
aiProductQuestion.setCompanyId(dto.getCompanyId());
|
aiProductQuestion.setProductCategoryId(productCategoryId);
|
aiProductQuestion.setTitle(title);
|
aiProductQuestion.setDifficulty(difficulty);
|
aiProductQuestion.setCreatedTime(createdTime);
|
this.save(aiProductQuestion);
|
|
JSONArray answerList = questionObj.getJSONArray("answer_list");
|
System.out.println("问题: " + title);
|
for (int i = 0; i < answerList.size(); i++) {
|
JSONObject answer = answerList.getJSONObject(i);
|
System.out.println("答案" + (i+1) + ": " + answer.getStr("answer"));
|
System.out.println("类型: " + answer.getStr("type"));
|
System.out.println("分析: " + answer.getStr("analysis"));
|
AiProductQuestionItem aiProductQuestionItem = new AiProductQuestionItem();
|
aiProductQuestionItem.setId(UUID.getSimpleUUIDString());
|
aiProductQuestionItem.setCompanyId(dto.getCompanyId());
|
aiProductQuestionItem.setProductQuestionId(aiProductQuestion.getId());
|
aiProductQuestionItem.setTitle(aiProductQuestion.getTitle());
|
aiProductQuestionItem.setAnswer(answer.getStr("answer"));
|
aiProductQuestionItem.setCorrectAnswer(answer.getInt("type") == 2 ? 1 : 0);
|
aiProductQuestionItem.setAnswerAnalysis(answer.getStr("analysis"));
|
aiProductQuestionItem.setCreatedTime(createdTime);
|
aiProductQuestionItemService.getBaseMapper().insert(aiProductQuestionItem);
|
}
|
}
|
return new FebsResponse().success().message("操作成功");
|
}
|
|
@Override
|
public FebsResponse aiAddV2(AiProductQuestionAiDto dto) {
|
String companyId = dto.getCompanyId();
|
String productCategoryId = dto.getProductCategoryId();
|
if (StrUtil.isEmpty(productCategoryId)){
|
return new FebsResponse().fail().message("请选择产品分类");
|
}
|
Integer questionCnt = dto.getQuestionCnt();
|
String query = dto.getQuery();
|
Integer difficulty = dto.getDifficulty();
|
String promptAiSystem = dto.getPromptAiSystem();
|
Date createdTime = new Date();
|
AiProductQuestionJob aiProductQuestionJob = new AiProductQuestionJob();
|
aiProductQuestionJob.setId(UUID.getSimpleUUIDString());
|
aiProductQuestionJob.setCompanyId(companyId);
|
aiProductQuestionJob.setProductCategoryId(productCategoryId);
|
aiProductQuestionJob.setTitle(query);
|
aiProductQuestionJob.setPromptAiSystem(promptAiSystem);
|
aiProductQuestionJob.setQuestionCnt(questionCnt);
|
aiProductQuestionJob.setDifficulty(difficulty);
|
aiProductQuestionJob.setCreatedTime(createdTime);
|
aiProductQuestionJobMapper.insert(aiProductQuestionJob);
|
|
agentProducer.sendAddQuestionJob(aiProductQuestionJob.getId());
|
return new FebsResponse().success().message("操作成功");
|
}
|
|
@Override
|
public void getAddQuestion(String id) {
|
try {
|
AiProductQuestionJob aiProductQuestionJob = aiProductQuestionJobMapper.selectById(id);
|
if (ObjectUtil.isEmpty(aiProductQuestionJob)){
|
return;
|
}
|
Integer state = aiProductQuestionJob.getState();
|
if (2 == state){
|
return;
|
}
|
Integer questionCnt = aiProductQuestionJob.getQuestionCnt();
|
Integer questionDoneCnt = aiProductQuestionJob.getQuestionDoneCnt();
|
if (questionCnt <= questionDoneCnt){
|
aiProductQuestionJob.setState(2);
|
aiProductQuestionJobMapper.update(null,
|
Wrappers.lambdaUpdate(AiProductQuestionJob.class)
|
.set(AiProductQuestionJob::getState, 2)
|
.set(AiProductQuestionJob::getUpdatedTime, new Date())
|
.eq(AiProductQuestionJob::getId, id));
|
return;
|
}
|
int i = questionCnt - questionDoneCnt;
|
if (i >= 5){
|
i = 5;
|
}
|
AiProductQuestionAiDto dto = new AiProductQuestionAiDto();
|
String jsonFormat = "{\"question_list\":[{\"title\": \"消费者对透明质酸的主要担忧是什么?\",\"answer_list\":[{\"answer\": \"消费者担心透明质酸维持时间短,效果不明显。\",\"type\": 1,\"analysis\": \"\"},{\"answer\": \"消费者担心透明质酸注射后可能出现移位、凹陷、馒化、僵硬以及炎症反应。\",\"type\": 1,\"analysis\": \"\"},{\"answer\": \"消费者的主要担忧包括效果不明显、维持时间短、注射后出现移位、凹陷、馒化、僵硬以及炎症反应。他们希望既达到理想效果,又避免这些副作用。\",\"type\": 2,\"analysis\": \"标准答案涵盖了消费者对透明质酸的所有主要担忧,既包括效果问题,也涵盖副作用风险,全面反映消费者心理需求。\"}]}]} ";
|
dto.setJsonFormat(jsonFormat);
|
dto.setCompanyId(aiProductQuestionJob.getCompanyId());
|
dto.setProductCategoryId(aiProductQuestionJob.getProductCategoryId());
|
dto.setQuery(aiProductQuestionJob.getTitle());
|
dto.setPromptAiSystem(aiProductQuestionJob.getPromptAiSystem());
|
dto.setQuestionCnt(i);
|
dto.setDifficulty(aiProductQuestionJob.getDifficulty());
|
|
String questionAndAnswerStr = aiService.llmInvokeNonStreaming(dto);
|
log.info("问题答案: " + questionAndAnswerStr);
|
// 解析AI返回的结果
|
List<JSONObject> questionList = parseAiQuestionResponse(questionAndAnswerStr);
|
if (CollUtil.isNotEmpty(questionList)){
|
String productCategoryId = dto.getProductCategoryId();
|
Integer difficulty = dto.getDifficulty();
|
Date createdTime = new Date();
|
// 处理解析后的问题列表
|
for (JSONObject questionObj : questionList) {
|
String title = questionObj.getStr("title");
|
|
AiProductQuestion aiProductQuestion = new AiProductQuestion();
|
aiProductQuestion.setId(UUID.getSimpleUUIDString());
|
aiProductQuestion.setCompanyId(dto.getCompanyId());
|
aiProductQuestion.setProductCategoryId(productCategoryId);
|
aiProductQuestion.setTitle(title);
|
aiProductQuestion.setDifficulty(difficulty);
|
aiProductQuestion.setCreatedTime(createdTime);
|
this.save(aiProductQuestion);
|
JSONArray answerList = questionObj.getJSONArray("answer_list");
|
for (int j = 0; j < answerList.size(); j++) {
|
JSONObject answer = answerList.getJSONObject(j);
|
System.out.println("答案" + (j+1) + ": " + answer.getStr("answer"));
|
System.out.println("类型: " + answer.getStr("type"));
|
System.out.println("分析: " + answer.getStr("analysis"));
|
AiProductQuestionItem aiProductQuestionItem = new AiProductQuestionItem();
|
aiProductQuestionItem.setId(UUID.getSimpleUUIDString());
|
aiProductQuestionItem.setCompanyId(dto.getCompanyId());
|
aiProductQuestionItem.setProductQuestionId(aiProductQuestion.getId());
|
aiProductQuestionItem.setTitle(aiProductQuestion.getTitle());
|
aiProductQuestionItem.setAnswer(answer.getStr("answer"));
|
aiProductQuestionItem.setCorrectAnswer(answer.getInt("type") == 2 ? 1 : 0);
|
aiProductQuestionItem.setAnswerAnalysis(answer.getStr("analysis"));
|
aiProductQuestionItem.setCreatedTime(createdTime);
|
aiProductQuestionItemService.getBaseMapper().insert(aiProductQuestionItem);
|
}
|
}
|
aiProductQuestionJobMapper.update(null,
|
Wrappers.lambdaUpdate(AiProductQuestionJob.class)
|
.set(AiProductQuestionJob::getQuestionDoneCnt, questionDoneCnt + questionList.size())
|
.set(AiProductQuestionJob::getUpdatedTime, new Date())
|
.set(AiProductQuestionJob::getState, 1)
|
.eq(AiProductQuestionJob::getId, id));
|
}
|
AiProductQuestionJob aiProductQuestionJobDone = aiProductQuestionJobMapper.selectById(id);
|
if (2 != aiProductQuestionJobDone.getState()){
|
agentProducer.sendAddQuestionJob(id);
|
}
|
} catch (Exception e) {
|
agentProducer.sendAddQuestionJob(id);
|
log.error("知识库异常", e);
|
}
|
}
|
|
@Override
|
public void exportProductQuestion(AiProductQuestion dto, HttpServletResponse response) {
|
try {
|
List<ExcelSheetPO> res = new ArrayList<>();
|
ExcelSheetPO orderSheet = new ExcelSheetPO();
|
String title = "单选题目列表";
|
orderSheet.setSheetName(title);
|
orderSheet.setTitle(title);
|
String[] header = {"答案ID", "题目ID", "题目", "答案", "正确答案", "解析"};
|
orderSheet.setHeaders(header);
|
|
QueryRequest request = new QueryRequest();
|
request.setPageNum(1);
|
request.setPageSize(9999);
|
String questionIds = dto.getIds();
|
List<String> ids = StrUtil.splitTrim(questionIds, ",");
|
List<AiProductQuestionItem> aiProductQuestionItems = aiProductQuestionItemService.getListByQuery(
|
Wrappers.lambdaQuery(AiProductQuestionItem.class)
|
.in(AiProductQuestionItem::getProductQuestionId, ids)
|
.orderByAsc(AiProductQuestionItem::getProductQuestionId)
|
);
|
|
List<List<Object>> list = new ArrayList<>();
|
if (CollUtil.isNotEmpty(aiProductQuestionItems)) {
|
for (AiProductQuestionItem item : aiProductQuestionItems) {
|
List<Object> temp = new ArrayList<>();
|
temp.add(item.getId());
|
temp.add(item.getProductQuestionId());
|
temp.add(item.getTitle());
|
temp.add(item.getAnswer());
|
temp.add(item.getCorrectAnswer() == 1 ? "是" : "否");
|
temp.add(item.getAnswerAnalysis());
|
list.add(temp);
|
}
|
}
|
orderSheet.setDataList(list);
|
res.add(orderSheet);
|
|
// 设置响应头
|
response = ResponseHeadUtil.setExcelHead(response);
|
String fileName = title + DateUtil.format(new Date(), "yyyyMMddHHmmss") + ".xlsx";
|
response.setHeader("Content-Disposition",
|
"attachment;filename=" + URLEncoder.encode(fileName, "UTF-8"));
|
|
// 写入 Excel 文件
|
try (OutputStream os = response.getOutputStream()) {
|
ExcelUtil.createWorkbookAtOutStream(ExcelVersion.V2007, res, os, true);
|
}
|
} catch (Exception e) {
|
log.error("导出产品题目失败", e);
|
throw new RuntimeException("导出失败,请稍后重试");
|
}
|
}
|
|
@Override
|
@Transactional
|
public FebsResponse importDeliver(MultipartFile file) {
|
try {
|
if (file.isEmpty()) {
|
return new FebsResponse().fail();
|
}
|
|
String fileName = file.getOriginalFilename();
|
String dirPath = "/home/javaweb/webresource/ai/import/";
|
|
File saveFile = new File(new File(dirPath).getAbsolutePath() + File.separator + fileName);
|
if (!saveFile.exists()) {
|
if (!saveFile.getParentFile().exists()) {
|
saveFile.getParentFile().mkdirs();
|
}
|
}
|
file.transferTo(saveFile);
|
|
List<ExcelSheetPO> data = ExcelUtil.readExcel(saveFile, null, null);
|
if (CollUtil.isEmpty(data)) {
|
return new FebsResponse().fail();
|
}
|
|
List<List<Object>> dataList = data.get(0).getDataList();
|
// String[] header = {"答案ID", "题目ID", "题目", "答案", "正确答案", "解析"};
|
int questionItemIdIndex = -1;
|
int questionIdIndex = -1;
|
int titleIndex = -1;
|
int answerIndex = -1;
|
int correctAnswerIndex = -1;
|
int answerAnalysisIndex = -1;
|
for (int i = 1; i < dataList.size(); i++) {
|
List<Object> objects = dataList.get(i);
|
|
String questionItemId = "";
|
String questionId = "";
|
String title = "";
|
String answer = "";
|
Integer correctAnswer = 0;
|
String answerAnalysis = "";
|
for (int j = 0; j < objects.size(); j++) {
|
Object obj = objects.get(j);
|
if ("答案ID".equals(obj)) {
|
questionItemIdIndex = j;
|
}
|
if ("题目ID".equals(obj)) {
|
questionIdIndex = j;
|
}
|
if ("题目".equals(obj)) {
|
titleIndex = j;
|
}
|
if ("答案".equals(obj)) {
|
answerIndex = j;
|
}
|
if ("正确答案".equals(obj)) {
|
correctAnswerIndex = j;
|
}
|
if ("解析".equals(obj)) {
|
answerAnalysisIndex = j;
|
}
|
if (j == questionItemIdIndex) {
|
questionItemId = (String) objects.get(j);
|
}
|
if (j == questionIdIndex) {
|
questionId = (String) objects.get(j);
|
}
|
if (j == titleIndex) {
|
title = (String) objects.get(j);
|
}
|
if (j == answerIndex) {
|
answer = (String) objects.get(j);
|
}
|
if (j == correctAnswerIndex) {
|
correctAnswer = "是".equals((String) objects.get(j)) ? 1 : 0;
|
}
|
if (j == answerAnalysisIndex) {
|
answerAnalysis = (String) objects.get(j);
|
}
|
}
|
|
if (
|
StrUtil.isNotBlank(questionItemId)
|
&& StrUtil.isNotBlank(questionId)
|
&& StrUtil.isNotBlank(title)
|
&& StrUtil.isNotBlank(answer)
|
){
|
aiProductQuestionItemService.getBaseMapper().update(null,
|
Wrappers.lambdaUpdate(AiProductQuestionItem.class)
|
.set(AiProductQuestionItem::getTitle, title)
|
.set(AiProductQuestionItem::getAnswer, answer)
|
.set(AiProductQuestionItem::getCorrectAnswer, correctAnswer)
|
.set(AiProductQuestionItem::getAnswerAnalysis, answerAnalysis)
|
.set(AiProductQuestionItem::getUpdatedTime, new Date())
|
.eq(AiProductQuestionItem::getId, questionItemId)
|
.eq(AiProductQuestionItem::getProductQuestionId, questionId));
|
this.update(null,
|
Wrappers.lambdaUpdate(AiProductQuestion.class)
|
.set(AiProductQuestion::getTitle, title)
|
.eq(AiProductQuestion::getId, questionId));
|
}
|
}
|
return new FebsResponse().success();
|
} catch (IOException e) {
|
e.printStackTrace();
|
throw new RuntimeException("导入失败,请稍后重试");
|
}
|
}
|
|
|
/**
|
* 解析AI返回的包含问题和答案的JSON字符串
|
* @param aiResponse AI返回的原始响应字符串
|
* @return 解析后的问题列表
|
*/
|
public static List<JSONObject> parseAiQuestionResponse(String aiResponse) {
|
|
try {
|
// 解析外层JSON
|
JSONObject outerJson = JSONUtil.parseObj(aiResponse);
|
|
// 提取output字段
|
String output = outerJson.getStr("output");
|
log.info("原始输出内容: {}", output);
|
|
// 去除<start>和</start>标签,并清理多余字符
|
String jsonContent = output.replace("<start>", "")
|
.replace("</start>", "")
|
.replace("\\n", "")
|
.replace("\\\"", "\"")
|
.trim();
|
|
log.info("清理后JSON内容: {}", jsonContent);
|
// 解析内部JSON
|
JSONObject innerJson = JSONUtil.parseObj(jsonContent);
|
|
// 提取question_list
|
JSONArray questionList = innerJson.getJSONArray("question_list");
|
|
// 转换为List<JSONObject>返回
|
return questionList.toList(JSONObject.class);
|
|
} catch (Exception e) {
|
log.error("解析AI问题响应失败: ", e);
|
throw new RuntimeException("解析AI响应失败", e);
|
}
|
}
|
|
public static void main(String[] args) {
|
String s = "{\"output\":\"<start>{\\\"question_list\\\":[{\\\"title\\\":\\\"根据知识库内容,调整T区的三大作用分别是什么?请逐条准确列举。\\\",\\\"answer_list\\\":[{\\\"answer\\\":\\\"1.提升折叠度;2.调整面部比例;3.提升骨性支撑力\\\",\\\"type\\\":2,\\\"analysis\\\":\\\"原文明确列出:'1.提升折叠度(增加正面、侧面转折落差,减少扁平感);2.调整面部比例(通过T区的纵向与横线的调整,优化三庭五眼、四高三底);3.提升骨性支撑力(为软组织提供坚实的\\\\\\\"地基\\\\\\\"抗衰并提升高级的骨感)'\\\"},{\\\"answer\\\":\\\"1.提升折叠度;2.调整三庭五眼;3.增强软组织弹性\\\",\\\"type\\\":1,\\\"analysis\\\":\\\"\\\"},{\\\"answer\\\":\\\"1.提升立体感;2.优化面部比例;3.强化韧带支撑\\\",\\\"type\\\":1,\\\"analysis\\\":\\\"\\\"}]}]}</start>\"}";
|
List<JSONObject> questionList = parseAiQuestionResponse(s);
|
System.out.println(questionList);
|
|
}
|
|
}
|