| | |
| | | import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; |
| | | import com.vincent.rsf.framework.common.Cools; |
| | | import com.vincent.rsf.framework.exception.CoolException; |
| | | import com.vincent.rsf.server.ai.config.AiDefaults; |
| | | import com.vincent.rsf.server.ai.dto.AiChatMemoryDto; |
| | | import com.vincent.rsf.server.ai.dto.AiChatMessageDto; |
| | | import com.vincent.rsf.server.ai.dto.AiChatSessionPinRequest; |
| | | import com.vincent.rsf.server.ai.dto.AiChatSessionRenameRequest; |
| | | import com.vincent.rsf.server.ai.dto.AiChatSessionDto; |
| | | import com.vincent.rsf.server.ai.entity.AiChatMessage; |
| | | import com.vincent.rsf.server.ai.entity.AiChatSession; |
| | |
| | | private final AiChatSessionMapper aiChatSessionMapper; |
| | | private final AiChatMessageMapper aiChatMessageMapper; |
| | | |
| | | /** |
| | | * 读取会话记忆快照。 |
| | | * 返回结果同时包含完整落库历史、短期记忆窗口以及摘要/事实记忆, |
| | | * 便于调用方按不同用途选择数据粒度。 |
| | | */ |
| | | @Override |
| | | public AiChatMemoryDto getMemory(Long userId, Long tenantId, String promptCode, Long sessionId) { |
| | | ensureIdentity(userId, tenantId); |
| | |
| | | if (session == null) { |
| | | return AiChatMemoryDto.builder() |
| | | .sessionId(null) |
| | | .memorySummary(null) |
| | | .memoryFacts(null) |
| | | .recentMessageCount(0) |
| | | .persistedMessages(List.of()) |
| | | .shortMemoryMessages(List.of()) |
| | | .build(); |
| | | } |
| | | List<AiChatMessageDto> persistedMessages = listMessages(session.getId()); |
| | | List<AiChatMessageDto> shortMemoryMessages = tailMessagesByRounds(persistedMessages, AiDefaults.MEMORY_RECENT_ROUNDS); |
| | | return AiChatMemoryDto.builder() |
| | | .sessionId(session.getId()) |
| | | .persistedMessages(listMessages(session.getId())) |
| | | .memorySummary(session.getMemorySummary()) |
| | | .memoryFacts(session.getMemoryFacts()) |
| | | .recentMessageCount(shortMemoryMessages.size()) |
| | | .persistedMessages(persistedMessages) |
| | | .shortMemoryMessages(shortMemoryMessages) |
| | | .build(); |
| | | } |
| | | |
| | | /** |
| | | * 查询当前用户在某个 Prompt 下的会话列表。 |
| | | * 列表只返回用于侧边栏展示的摘要信息,不返回完整对话内容。 |
| | | */ |
| | | @Override |
| | | public List<AiChatSessionDto> listSessions(Long userId, Long tenantId, String promptCode) { |
| | | public List<AiChatSessionDto> listSessions(Long userId, Long tenantId, String promptCode, String keyword) { |
| | | ensureIdentity(userId, tenantId); |
| | | String resolvedPromptCode = requirePromptCode(promptCode); |
| | | List<AiChatSession> sessions = aiChatSessionMapper.selectList(new LambdaQueryWrapper<AiChatSession>() |
| | |
| | | .eq(AiChatSession::getPromptCode, resolvedPromptCode) |
| | | .eq(AiChatSession::getDeleted, 0) |
| | | .eq(AiChatSession::getStatus, StatusType.ENABLE.val) |
| | | .like(StringUtils.hasText(keyword), AiChatSession::getTitle, keyword == null ? null : keyword.trim()) |
| | | .orderByDesc(AiChatSession::getPinned) |
| | | .orderByDesc(AiChatSession::getLastMessageTime) |
| | | .orderByDesc(AiChatSession::getId)); |
| | | if (Cools.isEmpty(sessions)) { |
| | |
| | | } |
| | | List<AiChatSessionDto> result = new ArrayList<>(); |
| | | for (AiChatSession session : sessions) { |
| | | result.add(AiChatSessionDto.builder() |
| | | .sessionId(session.getId()) |
| | | .title(session.getTitle()) |
| | | .promptCode(session.getPromptCode()) |
| | | .lastMessageTime(session.getLastMessageTime()) |
| | | .build()); |
| | | result.add(buildSessionDto(session)); |
| | | } |
| | | return result; |
| | | } |
| | | |
| | | /** |
| | | * 解析本轮请求应该落到哪个会话。 |
| | | * 如果前端带了 sessionId 则做归属校验并复用;否则自动创建新会话。 |
| | | */ |
| | | @Override |
| | | public AiChatSession resolveSession(Long userId, Long tenantId, String promptCode, Long sessionId, String titleSeed) { |
| | | ensureIdentity(userId, tenantId); |
| | |
| | | .setUserId(userId) |
| | | .setTenantId(tenantId) |
| | | .setLastMessageTime(now) |
| | | .setPinned(0) |
| | | .setStatus(StatusType.ENABLE.val) |
| | | .setDeleted(0) |
| | | .setCreateBy(userId) |
| | |
| | | return session; |
| | | } |
| | | |
| | | /** |
| | | * 落库保存一整轮对话。 |
| | | * 这里会顺序写入本轮用户消息和模型回复,并在最后刷新会话标题、最后活跃时间和记忆画像。 |
| | | */ |
| | | @Override |
| | | public void saveRound(AiChatSession session, Long userId, Long tenantId, List<AiChatMessageDto> memoryMessages, String assistantContent) { |
| | | if (session == null || session.getId() == null) { |
| | |
| | | .setUpdateBy(userId) |
| | | .setUpdateTime(now); |
| | | aiChatSessionMapper.updateById(update); |
| | | refreshMemoryProfile(session.getId(), userId); |
| | | } |
| | | |
| | | /** 删除整个会话及其消息。 */ |
| | | @Override |
| | | public void removeSession(Long userId, Long tenantId, Long sessionId) { |
| | | ensureIdentity(userId, tenantId); |
| | |
| | | } |
| | | } |
| | | |
| | | /** 更新会话标题并返回最新会话摘要。 */ |
| | | @Override |
| | | public AiChatSessionDto renameSession(Long userId, Long tenantId, Long sessionId, AiChatSessionRenameRequest request) { |
| | | ensureIdentity(userId, tenantId); |
| | | if (request == null || !StringUtils.hasText(request.getTitle())) { |
| | | throw new CoolException("会话标题不能为空"); |
| | | } |
| | | AiChatSession session = requireOwnedSession(sessionId, userId, tenantId); |
| | | Date now = new Date(); |
| | | AiChatSession update = new AiChatSession() |
| | | .setId(sessionId) |
| | | .setTitle(buildSessionTitle(request.getTitle())) |
| | | .setUpdateBy(userId) |
| | | .setUpdateTime(now); |
| | | aiChatSessionMapper.updateById(update); |
| | | return buildSessionDto(requireOwnedSession(sessionId, userId, tenantId)); |
| | | } |
| | | |
| | | /** 更新会话置顶状态。 */ |
| | | @Override |
| | | public AiChatSessionDto pinSession(Long userId, Long tenantId, Long sessionId, AiChatSessionPinRequest request) { |
| | | ensureIdentity(userId, tenantId); |
| | | if (request == null || request.getPinned() == null) { |
| | | throw new CoolException("置顶状态不能为空"); |
| | | } |
| | | AiChatSession session = requireOwnedSession(sessionId, userId, tenantId); |
| | | Date now = new Date(); |
| | | AiChatSession update = new AiChatSession() |
| | | .setId(sessionId) |
| | | .setPinned(Boolean.TRUE.equals(request.getPinned()) ? 1 : 0) |
| | | .setUpdateBy(userId) |
| | | .setUpdateTime(now); |
| | | aiChatSessionMapper.updateById(update); |
| | | return buildSessionDto(requireOwnedSession(sessionId, userId, tenantId)); |
| | | } |
| | | |
| | | /** 清空某个会话的全部消息和派生记忆字段。 */ |
| | | @Override |
| | | public void clearSessionMemory(Long userId, Long tenantId, Long sessionId) { |
| | | ensureIdentity(userId, tenantId); |
| | | AiChatSession session = requireOwnedSession(sessionId, userId, tenantId); |
| | | List<AiChatMessage> messages = aiChatMessageMapper.selectList(new LambdaQueryWrapper<AiChatMessage>() |
| | | .eq(AiChatMessage::getSessionId, sessionId) |
| | | .eq(AiChatMessage::getDeleted, 0)); |
| | | for (AiChatMessage message : messages) { |
| | | aiChatMessageMapper.updateById(new AiChatMessage() |
| | | .setId(message.getId()) |
| | | .setDeleted(1)); |
| | | } |
| | | aiChatSessionMapper.updateById(new AiChatSession() |
| | | .setId(sessionId) |
| | | .setMemorySummary(null) |
| | | .setMemoryFacts(null) |
| | | .setUpdateBy(userId) |
| | | .setUpdateTime(new Date()) |
| | | .setLastMessageTime(session.getCreateTime())); |
| | | } |
| | | |
| | | /** 只保留最近一轮问答,用于手动裁剪长会话。 */ |
| | | @Override |
| | | public void retainLatestRound(Long userId, Long tenantId, Long sessionId) { |
| | | ensureIdentity(userId, tenantId); |
| | | requireOwnedSession(sessionId, userId, tenantId); |
| | | List<AiChatMessage> records = listMessageRecords(sessionId); |
| | | if (records.isEmpty()) { |
| | | return; |
| | | } |
| | | List<AiChatMessage> retained = tailMessageRecordsByRounds(records, 1); |
| | | for (AiChatMessage message : records) { |
| | | boolean shouldKeep = retained.stream().anyMatch(item -> item.getId().equals(message.getId())); |
| | | if (!shouldKeep) { |
| | | aiChatMessageMapper.updateById(new AiChatMessage() |
| | | .setId(message.getId()) |
| | | .setDeleted(1)); |
| | | } |
| | | } |
| | | refreshMemoryProfile(sessionId, userId); |
| | | } |
| | | |
| | | private AiChatSession findLatestSession(Long userId, Long tenantId, String promptCode) { |
| | | return aiChatSessionMapper.selectOne(new LambdaQueryWrapper<AiChatSession>() |
| | | .eq(AiChatSession::getUserId, userId) |
| | |
| | | return session; |
| | | } |
| | | |
| | | private AiChatSession requireOwnedSession(Long sessionId, Long userId, Long tenantId) { |
| | | if (sessionId == null) { |
| | | throw new CoolException("AI 会话 ID 不能为空"); |
| | | } |
| | | AiChatSession session = aiChatSessionMapper.selectOne(new LambdaQueryWrapper<AiChatSession>() |
| | | .eq(AiChatSession::getId, sessionId) |
| | | .eq(AiChatSession::getUserId, userId) |
| | | .eq(AiChatSession::getTenantId, tenantId) |
| | | .eq(AiChatSession::getDeleted, 0) |
| | | .eq(AiChatSession::getStatus, StatusType.ENABLE.val) |
| | | .last("limit 1")); |
| | | if (session == null) { |
| | | throw new CoolException("AI 会话不存在或无权访问"); |
| | | } |
| | | return session; |
| | | } |
| | | |
| | | private List<AiChatMessageDto> listMessages(Long sessionId) { |
| | | List<AiChatMessage> records = aiChatMessageMapper.selectList(new LambdaQueryWrapper<AiChatMessage>() |
| | | .eq(AiChatMessage::getSessionId, sessionId) |
| | | .eq(AiChatMessage::getDeleted, 0) |
| | | .orderByAsc(AiChatMessage::getSeqNo) |
| | | .orderByAsc(AiChatMessage::getId)); |
| | | List<AiChatMessage> records = listMessageRecords(sessionId); |
| | | if (Cools.isEmpty(records)) { |
| | | return List.of(); |
| | | } |
| | |
| | | } |
| | | |
| | | private List<AiChatMessageDto> normalizeMessages(List<AiChatMessageDto> memoryMessages) { |
| | | /** 清洗前端上传的内存消息,只允许 user/assistant 两类角色落库。 */ |
| | | List<AiChatMessageDto> normalized = new ArrayList<>(); |
| | | if (Cools.isEmpty(memoryMessages)) { |
| | | return normalized; |
| | |
| | | return normalized; |
| | | } |
| | | |
| | | private List<AiChatMessage> listMessageRecords(Long sessionId) { |
| | | return aiChatMessageMapper.selectList(new LambdaQueryWrapper<AiChatMessage>() |
| | | .eq(AiChatMessage::getSessionId, sessionId) |
| | | .eq(AiChatMessage::getDeleted, 0) |
| | | .orderByAsc(AiChatMessage::getSeqNo) |
| | | .orderByAsc(AiChatMessage::getId)); |
| | | } |
| | | |
| | | private int findNextSeqNo(Long sessionId) { |
| | | AiChatMessage lastMessage = aiChatMessageMapper.selectOne(new LambdaQueryWrapper<AiChatMessage>() |
| | | .eq(AiChatMessage::getSessionId, sessionId) |
| | |
| | | .setSeqNo(seqNo) |
| | | .setRole(role) |
| | | .setContent(content) |
| | | .setContentLength(content == null ? 0 : content.length()) |
| | | .setUserId(userId) |
| | | .setTenantId(tenantId) |
| | | .setDeleted(0) |
| | |
| | | } |
| | | |
| | | private String buildSessionTitle(String titleSeed) { |
| | | /** |
| | | * 把首轮用户问题压缩成适合作为会话标题的短摘要。 |
| | | * 这里会去掉换行、连续空白,并优先在自然语义断点处截断。 |
| | | */ |
| | | if (!StringUtils.hasText(titleSeed)) { |
| | | throw new CoolException("AI 会话标题不能为空"); |
| | | } |
| | | String title = titleSeed.trim().replace("\r", " ").replace("\n", " "); |
| | | return title.length() > 60 ? title.substring(0, 60) : title; |
| | | String title = titleSeed.trim() |
| | | .replace("\r", " ") |
| | | .replace("\n", " ") |
| | | .replaceAll("\\s+", " "); |
| | | int punctuationIndex = findSummaryBreakIndex(title); |
| | | if (punctuationIndex > 0) { |
| | | title = title.substring(0, punctuationIndex).trim(); |
| | | } |
| | | return title.length() > 48 ? title.substring(0, 48) : title; |
| | | } |
| | | |
| | | private int findSummaryBreakIndex(String title) { |
| | | String[] separators = {"。", "!", "?", ".", "!", "?"}; |
| | | int result = -1; |
| | | for (String separator : separators) { |
| | | int index = title.indexOf(separator); |
| | | if (index > 0 && (result < 0 || index < result)) { |
| | | result = index; |
| | | } |
| | | } |
| | | return result; |
| | | } |
| | | |
| | | private AiChatSessionDto buildSessionDto(AiChatSession session) { |
| | | AiChatMessage lastMessage = aiChatMessageMapper.selectOne(new LambdaQueryWrapper<AiChatMessage>() |
| | | .eq(AiChatMessage::getSessionId, session.getId()) |
| | | .eq(AiChatMessage::getDeleted, 0) |
| | | .orderByDesc(AiChatMessage::getSeqNo) |
| | | .orderByDesc(AiChatMessage::getId) |
| | | .last("limit 1")); |
| | | return AiChatSessionDto.builder() |
| | | .sessionId(session.getId()) |
| | | .title(session.getTitle()) |
| | | .promptCode(session.getPromptCode()) |
| | | .pinned(session.getPinned() != null && session.getPinned() == 1) |
| | | .lastMessagePreview(buildLastMessagePreview(lastMessage)) |
| | | .lastMessageTime(session.getLastMessageTime()) |
| | | .build(); |
| | | } |
| | | |
| | | private String buildLastMessagePreview(AiChatMessage message) { |
| | | if (message == null || !StringUtils.hasText(message.getContent())) { |
| | | return null; |
| | | } |
| | | String preview = message.getContent().trim() |
| | | .replace("\r", " ") |
| | | .replace("\n", " ") |
| | | .replaceAll("\\s+", " "); |
| | | String prefix = "assistant".equalsIgnoreCase(message.getRole()) ? "AI: " : "你: "; |
| | | String normalized = prefix + preview; |
| | | return normalized.length() > 80 ? normalized.substring(0, 80) : normalized; |
| | | } |
| | | |
| | | private void refreshMemoryProfile(Long sessionId, Long userId) { |
| | | /** |
| | | * 重新计算会话的摘要记忆和关键事实。 |
| | | * 这是“持久化消息”和“模型上下文治理”之间的桥梁方法。 |
| | | */ |
| | | List<AiChatMessageDto> messages = listMessages(sessionId); |
| | | List<AiChatMessageDto> shortMemoryMessages = tailMessagesByRounds(messages, AiDefaults.MEMORY_RECENT_ROUNDS); |
| | | List<AiChatMessageDto> historyMessages = messages.size() > shortMemoryMessages.size() |
| | | ? messages.subList(0, messages.size() - shortMemoryMessages.size()) |
| | | : List.of(); |
| | | String memorySummary = historyMessages.size() >= AiDefaults.MEMORY_SUMMARY_TRIGGER_MESSAGES |
| | | ? buildMemorySummary(historyMessages) |
| | | : null; |
| | | String memoryFacts = buildMemoryFacts(messages); |
| | | AiChatMessage lastMessage = aiChatMessageMapper.selectOne(new LambdaQueryWrapper<AiChatMessage>() |
| | | .eq(AiChatMessage::getSessionId, sessionId) |
| | | .eq(AiChatMessage::getDeleted, 0) |
| | | .orderByDesc(AiChatMessage::getSeqNo) |
| | | .orderByDesc(AiChatMessage::getId) |
| | | .last("limit 1")); |
| | | aiChatSessionMapper.updateById(new AiChatSession() |
| | | .setId(sessionId) |
| | | .setMemorySummary(memorySummary) |
| | | .setMemoryFacts(memoryFacts) |
| | | .setLastMessageTime(lastMessage == null ? null : lastMessage.getCreateTime()) |
| | | .setUpdateBy(userId) |
| | | .setUpdateTime(new Date())); |
| | | } |
| | | |
| | | private List<AiChatMessageDto> tailMessagesByRounds(List<AiChatMessageDto> source, int rounds) { |
| | | /** 按“用户发言轮次”裁剪最近消息,而不是简单按条数截断。 */ |
| | | if (Cools.isEmpty(source) || rounds <= 0) { |
| | | return List.of(); |
| | | } |
| | | int userCount = 0; |
| | | int startIndex = source.size(); |
| | | for (int i = source.size() - 1; i >= 0; i--) { |
| | | AiChatMessageDto item = source.get(i); |
| | | startIndex = i; |
| | | if (item != null && "user".equalsIgnoreCase(item.getRole())) { |
| | | userCount++; |
| | | if (userCount >= rounds) { |
| | | break; |
| | | } |
| | | } |
| | | } |
| | | return new ArrayList<>(source.subList(Math.max(0, startIndex), source.size())); |
| | | } |
| | | |
| | | private List<AiChatMessage> tailMessageRecordsByRounds(List<AiChatMessage> source, int rounds) { |
| | | if (Cools.isEmpty(source) || rounds <= 0) { |
| | | return List.of(); |
| | | } |
| | | int userCount = 0; |
| | | int startIndex = source.size(); |
| | | for (int i = source.size() - 1; i >= 0; i--) { |
| | | AiChatMessage item = source.get(i); |
| | | startIndex = i; |
| | | if (item != null && "user".equalsIgnoreCase(item.getRole())) { |
| | | userCount++; |
| | | if (userCount >= rounds) { |
| | | break; |
| | | } |
| | | } |
| | | } |
| | | return new ArrayList<>(source.subList(Math.max(0, startIndex), source.size())); |
| | | } |
| | | |
| | | private String buildMemorySummary(List<AiChatMessageDto> historyMessages) { |
| | | /** 为较早历史生成可直接插入系统消息的文本摘要。 */ |
| | | StringBuilder builder = new StringBuilder("较早对话摘要:\n"); |
| | | for (AiChatMessageDto item : historyMessages) { |
| | | if (item == null || !StringUtils.hasText(item.getContent())) { |
| | | continue; |
| | | } |
| | | String prefix = "assistant".equalsIgnoreCase(item.getRole()) ? "- AI: " : "- 用户: "; |
| | | String content = compactText(item.getContent(), 120); |
| | | if (!StringUtils.hasText(content)) { |
| | | continue; |
| | | } |
| | | builder.append(prefix).append(content).append("\n"); |
| | | if (builder.length() >= AiDefaults.MEMORY_SUMMARY_MAX_LENGTH) { |
| | | break; |
| | | } |
| | | } |
| | | return compactText(builder.toString(), AiDefaults.MEMORY_SUMMARY_MAX_LENGTH); |
| | | } |
| | | |
| | | private String buildMemoryFacts(List<AiChatMessageDto> messages) { |
| | | /** 从最近用户关注点中提炼关键事实,作为轻量持久记忆。 */ |
| | | if (Cools.isEmpty(messages)) { |
| | | return null; |
| | | } |
| | | StringBuilder builder = new StringBuilder("关键事实:\n"); |
| | | int userFacts = 0; |
| | | for (int i = messages.size() - 1; i >= 0 && userFacts < 4; i--) { |
| | | AiChatMessageDto item = messages.get(i); |
| | | if (item == null || !"user".equalsIgnoreCase(item.getRole()) || !StringUtils.hasText(item.getContent())) { |
| | | continue; |
| | | } |
| | | builder.append("- 用户关注: ").append(compactText(item.getContent(), 100)).append("\n"); |
| | | userFacts++; |
| | | } |
| | | return userFacts == 0 ? null : compactText(builder.toString(), AiDefaults.MEMORY_FACTS_MAX_LENGTH); |
| | | } |
| | | |
| | | private String compactText(String content, int maxLength) { |
| | | if (!StringUtils.hasText(content)) { |
| | | return null; |
| | | } |
| | | String normalized = content.trim() |
| | | .replace("\r", " ") |
| | | .replace("\n", " ") |
| | | .replaceAll("\\s+", " "); |
| | | return normalized.length() > maxLength ? normalized.substring(0, maxLength) : normalized; |
| | | } |
| | | |
| | | private void ensureIdentity(Long userId, Long tenantId) { |