feat(oa): 添加 AI 对话管理功能

- 新增 AI配置、对话历史、对话消息等相关的实体类、Mapper、Service 和 Controller
- 实现 AI 对话管理的基础功能,包括创建对话、发送消息、结束对话等- 集成 DeepSeek AI 服务,实现与 AI 模型的交互
- 添加 token消耗和费用计算相关逻辑
This commit is contained in:
2025-08-04 14:24:18 +08:00
parent 2fa67cb3c1
commit c93959e351
21 changed files with 1808 additions and 0 deletions

View File

@@ -0,0 +1,54 @@
package com.ruoyi.oa.service;
import com.ruoyi.oa.domain.bo.SysOaAiConfigBo;
import com.ruoyi.oa.domain.vo.SysOaAiConfigVo;
import com.ruoyi.common.core.page.TableDataInfo;
import com.ruoyi.common.core.domain.PageQuery;
import java.util.Collection;
import java.util.List;
/**
* AI配置Service接口
*
* @author ruoyi
* @date 2024-12-19
*/
public interface ISysOaAiConfigService {
/**
* 查询AI配置
*/
SysOaAiConfigVo queryById(Long configId);
/**
* 查询AI配置列表
*/
TableDataInfo<SysOaAiConfigVo> queryPageList(SysOaAiConfigBo bo, PageQuery pageQuery);
/**
* 查询AI配置列表
*/
List<SysOaAiConfigVo> queryList(SysOaAiConfigBo bo);
/**
* 新增AI配置
*/
Boolean insertByBo(SysOaAiConfigBo bo);
/**
* 修改AI配置
*/
Boolean updateByBo(SysOaAiConfigBo bo);
/**
* 校验并批量删除AI配置信息
*/
Boolean deleteWithValidByIds(Collection<Long> ids, Boolean isValid);
/**
* 获取启用的AI配置
*/
SysOaAiConfigVo getActiveConfig();
}

View File

@@ -0,0 +1,64 @@
package com.ruoyi.oa.service;
import com.ruoyi.oa.domain.bo.SysOaAiConversationBo;
import com.ruoyi.oa.domain.vo.SysOaAiConversationVo;
import com.ruoyi.common.core.page.TableDataInfo;
import com.ruoyi.common.core.domain.PageQuery;
import java.util.Collection;
import java.util.List;
/**
* AI对话历史Service接口
*
* @author ruoyi
* @date 2024-12-19
*/
public interface ISysOaAiConversationService {
/**
* 查询AI对话历史
*/
SysOaAiConversationVo queryById(Long conversationId);
/**
* 查询AI对话历史列表
*/
TableDataInfo<SysOaAiConversationVo> queryPageList(SysOaAiConversationBo bo, PageQuery pageQuery);
/**
* 查询AI对话历史列表
*/
List<SysOaAiConversationVo> queryList(SysOaAiConversationBo bo);
/**
* 新增AI对话历史
*/
Boolean insertByBo(SysOaAiConversationBo bo);
/**
* 修改AI对话历史
*/
Boolean updateByBo(SysOaAiConversationBo bo);
/**
* 校验并批量删除AI对话历史信息
*/
Boolean deleteWithValidByIds(Collection<Long> ids, Boolean isValid);
/**
* 根据用户ID查询对话列表
*/
List<SysOaAiConversationVo> queryByUserId(Long userId);
/**
* 创建新对话
*/
Long createConversation(Long userId, String title);
/**
* 结束对话
*/
Boolean endConversation(Long conversationId);
}

View File

@@ -0,0 +1,64 @@
package com.ruoyi.oa.service;
import com.ruoyi.oa.domain.bo.SysOaAiMessageBo;
import com.ruoyi.oa.domain.vo.SysOaAiMessageVo;
import com.ruoyi.common.core.page.TableDataInfo;
import com.ruoyi.common.core.domain.PageQuery;
import java.util.Collection;
import java.util.List;
/**
* AI对话详情Service接口
*
* @author ruoyi
* @date 2024-12-19
*/
public interface ISysOaAiMessageService {
/**
* 查询AI对话详情
*/
SysOaAiMessageVo queryById(Long messageId);
/**
* 查询AI对话详情列表
*/
TableDataInfo<SysOaAiMessageVo> queryPageList(SysOaAiMessageBo bo, PageQuery pageQuery);
/**
* 查询AI对话详情列表
*/
List<SysOaAiMessageVo> queryList(SysOaAiMessageBo bo);
/**
* 新增AI对话详情
*/
Boolean insertByBo(SysOaAiMessageBo bo);
/**
* 修改AI对话详情
*/
Boolean updateByBo(SysOaAiMessageBo bo);
/**
* 校验并批量删除AI对话详情信息
*/
Boolean deleteWithValidByIds(Collection<Long> ids, Boolean isValid);
/**
* 根据对话ID查询消息列表
*/
List<SysOaAiMessageVo> queryByConversationId(Long conversationId);
/**
* 添加用户消息
*/
Boolean addUserMessage(Long conversationId, String content);
/**
* 添加AI回复消息
*/
Boolean addAiMessage(Long conversationId, String content, Integer tokens, java.math.BigDecimal cost);
}

View File

@@ -0,0 +1,127 @@
package com.ruoyi.oa.service.impl;
import cn.hutool.core.bean.BeanUtil;
import com.ruoyi.common.utils.StringUtils;
import com.ruoyi.common.core.page.TableDataInfo;
import com.ruoyi.common.core.domain.PageQuery;
import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
import com.baomidou.mybatisplus.core.toolkit.Wrappers;
import lombok.RequiredArgsConstructor;
import org.springframework.stereotype.Service;
import com.ruoyi.oa.domain.bo.SysOaAiConfigBo;
import com.ruoyi.oa.domain.vo.SysOaAiConfigVo;
import com.ruoyi.oa.domain.SysOaAiConfig;
import com.ruoyi.oa.mapper.SysOaAiConfigMapper;
import com.ruoyi.oa.service.ISysOaAiConfigService;
import java.util.List;
import java.util.Map;
import java.util.Collection;
/**
* AI配置Service业务层处理
*
* @author ruoyi
* @date 2024-12-19
*/
@RequiredArgsConstructor
@Service
public class SysOaAiConfigServiceImpl implements ISysOaAiConfigService {
private final SysOaAiConfigMapper baseMapper;
/**
* 查询AI配置
*/
@Override
public SysOaAiConfigVo queryById(Long configId) {
return baseMapper.selectVoById(configId);
}
/**
* 查询AI配置列表
*/
@Override
public TableDataInfo<SysOaAiConfigVo> queryPageList(SysOaAiConfigBo bo, PageQuery pageQuery) {
LambdaQueryWrapper<SysOaAiConfig> lqw = buildQueryWrapper(bo);
Page<SysOaAiConfigVo> result = baseMapper.selectVoPage(pageQuery.build(), lqw);
return TableDataInfo.build(result);
}
/**
* 查询AI配置列表
*/
@Override
public List<SysOaAiConfigVo> queryList(SysOaAiConfigBo bo) {
LambdaQueryWrapper<SysOaAiConfig> lqw = buildQueryWrapper(bo);
return baseMapper.selectVoList(lqw);
}
private LambdaQueryWrapper<SysOaAiConfig> buildQueryWrapper(SysOaAiConfigBo bo) {
Map<String, Object> params = bo.getParams();
LambdaQueryWrapper<SysOaAiConfig> lqw = Wrappers.lambdaQuery();
lqw.like(StringUtils.isNotBlank(bo.getConfigName()), SysOaAiConfig::getConfigName, bo.getConfigName());
lqw.eq(StringUtils.isNotBlank(bo.getApiKey()), SysOaAiConfig::getApiKey, bo.getApiKey());
lqw.eq(StringUtils.isNotBlank(bo.getBaseUrl()), SysOaAiConfig::getBaseUrl, bo.getBaseUrl());
lqw.eq(StringUtils.isNotBlank(bo.getModelName()), SysOaAiConfig::getModelName, bo.getModelName());
lqw.eq(bo.getMaxRetries() != null, SysOaAiConfig::getMaxRetries, bo.getMaxRetries());
lqw.eq(bo.getTemperature() != null, SysOaAiConfig::getTemperature, bo.getTemperature());
lqw.eq(bo.getMaxTokens() != null, SysOaAiConfig::getMaxTokens, bo.getMaxTokens());
lqw.eq(bo.getStatus() != null, SysOaAiConfig::getStatus, bo.getStatus());
return lqw;
}
/**
* 新增AI配置
*/
@Override
public Boolean insertByBo(SysOaAiConfigBo bo) {
SysOaAiConfig add = BeanUtil.toBean(bo, SysOaAiConfig.class);
validEntityBeforeSave(add);
boolean flag = baseMapper.insert(add) > 0;
if (flag) {
bo.setConfigId(add.getConfigId());
}
return flag;
}
/**
* 修改AI配置
*/
@Override
public Boolean updateByBo(SysOaAiConfigBo bo) {
SysOaAiConfig update = BeanUtil.toBean(bo, SysOaAiConfig.class);
validEntityBeforeSave(update);
return baseMapper.updateById(update) > 0;
}
/**
* 保存前的数据校验
*/
private void validEntityBeforeSave(SysOaAiConfig entity) {
// TODO 做一些数据校验,如唯一约束
}
/**
* 批量删除AI配置
*/
@Override
public Boolean deleteWithValidByIds(Collection<Long> ids, Boolean isValid) {
if (isValid) {
// TODO 做一些业务上的校验,判断是否需要校验
}
return baseMapper.deleteBatchIds(ids) > 0;
}
/**
* 获取启用的AI配置
*/
@Override
public SysOaAiConfigVo getActiveConfig() {
LambdaQueryWrapper<SysOaAiConfig> lqw = Wrappers.lambdaQuery();
lqw.eq(SysOaAiConfig::getStatus, 1); // 启用状态
lqw.last("LIMIT 1");
return baseMapper.selectVoOne(lqw);
}
}

View File

@@ -0,0 +1,152 @@
package com.ruoyi.oa.service.impl;
import cn.hutool.core.bean.BeanUtil;
import com.ruoyi.common.utils.StringUtils;
import com.ruoyi.common.core.page.TableDataInfo;
import com.ruoyi.common.core.domain.PageQuery;
import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
import com.baomidou.mybatisplus.core.toolkit.Wrappers;
import lombok.RequiredArgsConstructor;
import org.springframework.stereotype.Service;
import com.ruoyi.oa.domain.bo.SysOaAiConversationBo;
import com.ruoyi.oa.domain.vo.SysOaAiConversationVo;
import com.ruoyi.oa.domain.SysOaAiConversation;
import com.ruoyi.oa.mapper.SysOaAiConversationMapper;
import com.ruoyi.oa.service.ISysOaAiConversationService;
import java.util.List;
import java.util.Map;
import java.util.Collection;
/**
* AI对话历史Service业务层处理
*
* @author ruoyi
* @date 2024-12-19
*/
@RequiredArgsConstructor
@Service
public class SysOaAiConversationServiceImpl implements ISysOaAiConversationService {
private final SysOaAiConversationMapper baseMapper;
/**
* 查询AI对话历史
*/
@Override
public SysOaAiConversationVo queryById(Long conversationId) {
return baseMapper.selectVoById(conversationId);
}
/**
* 查询AI对话历史列表
*/
@Override
public TableDataInfo<SysOaAiConversationVo> queryPageList(SysOaAiConversationBo bo, PageQuery pageQuery) {
LambdaQueryWrapper<SysOaAiConversation> lqw = buildQueryWrapper(bo);
Page<SysOaAiConversationVo> result = baseMapper.selectVoPage(pageQuery.build(), lqw);
return TableDataInfo.build(result);
}
/**
* 查询AI对话历史列表
*/
@Override
public List<SysOaAiConversationVo> queryList(SysOaAiConversationBo bo) {
LambdaQueryWrapper<SysOaAiConversation> lqw = buildQueryWrapper(bo);
return baseMapper.selectVoList(lqw);
}
private LambdaQueryWrapper<SysOaAiConversation> buildQueryWrapper(SysOaAiConversationBo bo) {
Map<String, Object> params = bo.getParams();
LambdaQueryWrapper<SysOaAiConversation> lqw = Wrappers.lambdaQuery();
lqw.like(StringUtils.isNotBlank(bo.getConversationTitle()), SysOaAiConversation::getConversationTitle, bo.getConversationTitle());
lqw.eq(bo.getUserId() != null, SysOaAiConversation::getUserId, bo.getUserId());
lqw.eq(StringUtils.isNotBlank(bo.getModelName()), SysOaAiConversation::getModelName, bo.getModelName());
lqw.eq(bo.getStatus() != null, SysOaAiConversation::getStatus, bo.getStatus());
lqw.eq(bo.getTotalTokens() != null, SysOaAiConversation::getTotalTokens, bo.getTotalTokens());
lqw.eq(bo.getTotalCost() != null, SysOaAiConversation::getTotalCost, bo.getTotalCost());
return lqw;
}
/**
* 新增AI对话历史
*/
@Override
public Boolean insertByBo(SysOaAiConversationBo bo) {
SysOaAiConversation add = BeanUtil.toBean(bo, SysOaAiConversation.class);
validEntityBeforeSave(add);
boolean flag = baseMapper.insert(add) > 0;
if (flag) {
bo.setConversationId(add.getConversationId());
}
return flag;
}
/**
* 修改AI对话历史
*/
@Override
public Boolean updateByBo(SysOaAiConversationBo bo) {
SysOaAiConversation update = BeanUtil.toBean(bo, SysOaAiConversation.class);
validEntityBeforeSave(update);
return baseMapper.updateById(update) > 0;
}
/**
* 保存前的数据校验
*/
private void validEntityBeforeSave(SysOaAiConversation entity) {
// TODO 做一些数据校验,如唯一约束
}
/**
* 批量删除AI对话历史
*/
@Override
public Boolean deleteWithValidByIds(Collection<Long> ids, Boolean isValid) {
if (isValid) {
// TODO 做一些业务上的校验,判断是否需要校验
}
return baseMapper.deleteBatchIds(ids) > 0;
}
/**
* 根据用户ID查询对话列表
*/
@Override
public List<SysOaAiConversationVo> queryByUserId(Long userId) {
LambdaQueryWrapper<SysOaAiConversation> lqw = Wrappers.lambdaQuery();
lqw.eq(SysOaAiConversation::getUserId, userId);
lqw.orderByDesc(SysOaAiConversation::getCreateTime);
return baseMapper.selectVoList(lqw);
}
/**
* 创建新对话
*/
@Override
public Long createConversation(Long userId, String title) {
SysOaAiConversation conversation = new SysOaAiConversation();
conversation.setUserId(userId);
conversation.setConversationTitle(title);
conversation.setModelName("deepseek-chat");
conversation.setStatus(1); // 进行中
conversation.setTotalTokens(0);
conversation.setTotalCost(java.math.BigDecimal.ZERO);
baseMapper.insert(conversation);
return conversation.getConversationId();
}
/**
* 结束对话
*/
@Override
public Boolean endConversation(Long conversationId) {
SysOaAiConversation conversation = new SysOaAiConversation();
conversation.setConversationId(conversationId);
conversation.setStatus(0); // 已结束
return baseMapper.updateById(conversation) > 0;
}
}

View File

@@ -0,0 +1,172 @@
package com.ruoyi.oa.service.impl;
import cn.hutool.core.bean.BeanUtil;
import com.ruoyi.common.utils.StringUtils;
import com.ruoyi.common.core.page.TableDataInfo;
import com.ruoyi.common.core.domain.PageQuery;
import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
import com.baomidou.mybatisplus.core.toolkit.Wrappers;
import lombok.RequiredArgsConstructor;
import org.springframework.stereotype.Service;
import com.ruoyi.oa.domain.bo.SysOaAiMessageBo;
import com.ruoyi.oa.domain.vo.SysOaAiMessageVo;
import com.ruoyi.oa.domain.SysOaAiMessage;
import com.ruoyi.oa.mapper.SysOaAiMessageMapper;
import com.ruoyi.oa.service.ISysOaAiMessageService;
import java.util.List;
import java.util.Map;
import java.util.Collection;
/**
* AI对话详情Service业务层处理
*
* @author ruoyi
* @date 2024-12-19
*/
@RequiredArgsConstructor
@Service
public class SysOaAiMessageServiceImpl implements ISysOaAiMessageService {
private final SysOaAiMessageMapper baseMapper;
/**
* 查询AI对话详情
*/
@Override
public SysOaAiMessageVo queryById(Long messageId) {
return baseMapper.selectVoById(messageId);
}
/**
* 查询AI对话详情列表
*/
@Override
public TableDataInfo<SysOaAiMessageVo> queryPageList(SysOaAiMessageBo bo, PageQuery pageQuery) {
LambdaQueryWrapper<SysOaAiMessage> lqw = buildQueryWrapper(bo);
Page<SysOaAiMessageVo> result = baseMapper.selectVoPage(pageQuery.build(), lqw);
return TableDataInfo.build(result);
}
/**
* 查询AI对话详情列表
*/
@Override
public List<SysOaAiMessageVo> queryList(SysOaAiMessageBo bo) {
LambdaQueryWrapper<SysOaAiMessage> lqw = buildQueryWrapper(bo);
return baseMapper.selectVoList(lqw);
}
private LambdaQueryWrapper<SysOaAiMessage> buildQueryWrapper(SysOaAiMessageBo bo) {
Map<String, Object> params = bo.getParams();
LambdaQueryWrapper<SysOaAiMessage> lqw = Wrappers.lambdaQuery();
lqw.eq(bo.getConversationId() != null, SysOaAiMessage::getConversationId, bo.getConversationId());
lqw.eq(StringUtils.isNotBlank(bo.getRole()), SysOaAiMessage::getRole, bo.getRole());
lqw.like(StringUtils.isNotBlank(bo.getContent()), SysOaAiMessage::getContent, bo.getContent());
lqw.eq(bo.getTokens() != null, SysOaAiMessage::getTokens, bo.getTokens());
lqw.eq(bo.getCost() != null, SysOaAiMessage::getCost, bo.getCost());
lqw.eq(bo.getMessageOrder() != null, SysOaAiMessage::getMessageOrder, bo.getMessageOrder());
return lqw;
}
/**
* 新增AI对话详情
*/
@Override
public Boolean insertByBo(SysOaAiMessageBo bo) {
SysOaAiMessage add = BeanUtil.toBean(bo, SysOaAiMessage.class);
validEntityBeforeSave(add);
boolean flag = baseMapper.insert(add) > 0;
if (flag) {
bo.setMessageId(add.getMessageId());
}
return flag;
}
/**
* 修改AI对话详情
*/
@Override
public Boolean updateByBo(SysOaAiMessageBo bo) {
SysOaAiMessage update = BeanUtil.toBean(bo, SysOaAiMessage.class);
validEntityBeforeSave(update);
return baseMapper.updateById(update) > 0;
}
/**
* 保存前的数据校验
*/
private void validEntityBeforeSave(SysOaAiMessage entity) {
// TODO 做一些数据校验,如唯一约束
}
/**
* 批量删除AI对话详情
*/
@Override
public Boolean deleteWithValidByIds(Collection<Long> ids, Boolean isValid) {
if (isValid) {
// TODO 做一些业务上的校验,判断是否需要校验
}
return baseMapper.deleteBatchIds(ids) > 0;
}
/**
* 根据对话ID查询消息列表
*/
@Override
public List<SysOaAiMessageVo> queryByConversationId(Long conversationId) {
return baseMapper.selectMessagesByConversationId(conversationId);
}
/**
* 添加用户消息
*/
@Override
public Boolean addUserMessage(Long conversationId, String content) {
SysOaAiMessage message = new SysOaAiMessage();
message.setConversationId(conversationId);
message.setRole("user");
message.setContent(content);
message.setTokens(0);
message.setCost(java.math.BigDecimal.ZERO);
// 获取当前消息顺序
LambdaQueryWrapper<SysOaAiMessage> lqw = Wrappers.lambdaQuery();
lqw.eq(SysOaAiMessage::getConversationId, conversationId);
lqw.orderByDesc(SysOaAiMessage::getMessageOrder);
lqw.last("LIMIT 1");
SysOaAiMessage lastMessage = baseMapper.selectOne(lqw);
int nextOrder = (lastMessage != null && lastMessage.getMessageOrder() != null) ?
lastMessage.getMessageOrder() + 1 : 1;
message.setMessageOrder(nextOrder);
return baseMapper.insert(message) > 0;
}
/**
* 添加AI回复消息
*/
@Override
public Boolean addAiMessage(Long conversationId, String content, Integer tokens, java.math.BigDecimal cost) {
SysOaAiMessage message = new SysOaAiMessage();
message.setConversationId(conversationId);
message.setRole("assistant");
message.setContent(content);
message.setTokens(tokens != null ? tokens : 0);
message.setCost(cost != null ? cost : java.math.BigDecimal.ZERO);
// 获取当前消息顺序
LambdaQueryWrapper<SysOaAiMessage> lqw = Wrappers.lambdaQuery();
lqw.eq(SysOaAiMessage::getConversationId, conversationId);
lqw.orderByDesc(SysOaAiMessage::getMessageOrder);
lqw.last("LIMIT 1");
SysOaAiMessage lastMessage = baseMapper.selectOne(lqw);
int nextOrder = (lastMessage != null && lastMessage.getMessageOrder() != null) ?
lastMessage.getMessageOrder() + 1 : 1;
message.setMessageOrder(nextOrder);
return baseMapper.insert(message) > 0;
}
}