feat(ai-companion): 新增AI伴侣模块及白名单路径

This commit is contained in:
2026-01-26 16:25:39 +08:00
parent fd4c381d33
commit 6bb905bb30
10 changed files with 146 additions and 31 deletions

View File

@@ -28,6 +28,8 @@ public enum ErrorCode {
CHAT_CHARACTER_NOT_FOUND(40008, "键盘人设不存在"),
CHAT_MESSAGE_TOO_LONG(40009, "聊天消息过长最大支持1000字符"),
CHAT_SAVE_DATA_EMPTY(40010, "保存数据不能为空"),
COMPANION_MESSAGE_EMPTY(40011, "消息内容不能为空"),
COMPANION_ID_EMPTY(40012, "AI陪聊角色ID不能为空"),
TOKEN_NOT_FOUND(40102, "未能读取到有效用户令牌"),
TOKEN_INVALID(40103, "令牌无效"),
TOKEN_TIMEOUT(40104, "令牌已过期"),

View File

@@ -4,7 +4,7 @@ import com.baomidou.mybatisplus.core.metadata.IPage;
import com.yolo.keyborad.common.BaseResponse;
import com.yolo.keyborad.common.ResultUtils;
import com.yolo.keyborad.model.dto.PageDTO;
import com.yolo.keyborad.model.entity.KeyboardAiCompanion;
import com.yolo.keyborad.model.vo.AiCompanionVO;
import com.yolo.keyborad.service.KeyboardAiCompanionService;
import io.swagger.v3.oas.annotations.Operation;
import io.swagger.v3.oas.annotations.tags.Tag;
@@ -27,8 +27,8 @@ public class AiCompanionController {
@PostMapping("/page")
@Operation(summary = "分页查询AI陪聊角色", description = "分页查询已上线的AI陪聊角色列表")
public BaseResponse<IPage<KeyboardAiCompanion>> pageList(@RequestBody PageDTO pageDTO) {
IPage<KeyboardAiCompanion> result = aiCompanionService.pageList(pageDTO.getPageNum(), pageDTO.getPageSize());
public BaseResponse<IPage<AiCompanionVO>> pageList(@RequestBody PageDTO pageDTO) {
IPage<AiCompanionVO> result = aiCompanionService.pageList(pageDTO.getPageNum(), pageDTO.getPageSize());
return ResultUtils.success(result);
}
}

View File

@@ -8,6 +8,7 @@ import com.yolo.keyborad.common.ErrorCode;
import com.yolo.keyborad.common.ResultUtils;
import com.yolo.keyborad.exception.BusinessException;
import com.yolo.keyborad.mapper.QdrantPayloadMapper;
import com.yolo.keyborad.model.dto.chat.ChatMessageReq;
import com.yolo.keyborad.model.dto.chat.ChatReq;
import com.yolo.keyborad.model.dto.chat.ChatSaveReq;
import com.yolo.keyborad.model.dto.chat.ChatStreamMessage;
@@ -50,13 +51,16 @@ public class ChatController {
@PostMapping("/message")
@Operation(summary = "同步对话", description = "发送消息给大模型,同步返回 AI 响应,异步生成音频")
public BaseResponse<ChatMessageVO> message(@RequestParam("content") String content) {
if (StrUtil.isBlank(content)) {
throw new BusinessException(ErrorCode.PARAMS_ERROR, "消息内容不能为空");
public BaseResponse<ChatMessageVO> message(@RequestBody ChatMessageReq req ) {
if (StrUtil.isBlank(req.getContent())) {
throw new BusinessException(ErrorCode.COMPANION_MESSAGE_EMPTY);
}
if (req.getCompanionId() == null) {
throw new BusinessException(ErrorCode.COMPANION_ID_EMPTY);
}
String userId = StpUtil.getLoginIdAsString();
ChatMessageVO result = chatService.message(content, userId);
ChatMessageVO result = chatService.message(req.getContent(), userId, req.getCompanionId());
return ResultUtils.success(result);
}

View File

@@ -0,0 +1,19 @@
package com.yolo.keyborad.model.dto.chat;
import io.swagger.v3.oas.annotations.media.Schema;
import lombok.Data;
/*
* @author: ziin
* @date: 2025/12/8 15:05
*/
@Data
@Schema(description = "同步对话请求")
public class ChatMessageReq {
@Schema(description = "消息内容", requiredMode = Schema.RequiredMode.REQUIRED)
private String content;
@Schema(description = "AI陪聊角色ID", requiredMode = Schema.RequiredMode.REQUIRED)
private Long companionId;
}

View File

@@ -138,4 +138,12 @@ public class KeyboardAiCompanion {
@TableField(value = "updated_at")
@Schema(description="更新时间")
private Date updatedAt;
@TableField(value = "prologue")
@Schema(description="开场白")
private String prologue;
@TableField(value = "prologue_audio")
@Schema(description="开场白音频")
private String prologueAudio;
}

View File

@@ -0,0 +1,60 @@
package com.yolo.keyborad.model.vo;
import io.swagger.v3.oas.annotations.media.Schema;
import lombok.Data;
import java.util.Date;
/*
* @author: ziin
* @date: 2026/1/26
*/
@Data
@Schema(description = "AI陪聊角色VO")
public class AiCompanionVO {
@Schema(description = "陪聊角色唯一ID")
private Long id;
@Schema(description = "角色名称")
private String name;
@Schema(description = "角色头像URL")
private String avatarUrl;
@Schema(description = "角色封面图URL")
private String coverImageUrl;
@Schema(description = "角色性别male / female / other")
private String gender;
@Schema(description = "角色年龄段描述")
private String ageRange;
@Schema(description = "一句话人设描述")
private String shortDesc;
@Schema(description = "角色详细介绍文案")
private String introText;
@Schema(description = "角色性格标签数组")
private String personalityTags;
@Schema(description = "角色说话风格")
private String speakingStyle;
@Schema(description = "排序权重")
private Integer sortOrder;
@Schema(description = "角色热度评分")
private Integer popularityScore;
@Schema(description = "开场白")
private String prologue;
@Schema(description = "开场白音频")
private String prologueAudio;
@Schema(description = "创建时间")
private Date createdAt;
}

View File

@@ -20,9 +20,10 @@ public interface ChatService {
*
* @param content 用户消息内容
* @param userId 用户ID
* @param companionId AI陪聊角色ID
* @return AI 响应 + 音频任务 ID
*/
ChatMessageVO message(String content, String userId);
ChatMessageVO message(String content, String userId, Long companionId);
/**
* 查询音频任务状态

View File

@@ -3,6 +3,7 @@ package com.yolo.keyborad.service;
import com.baomidou.mybatisplus.core.metadata.IPage;
import com.yolo.keyborad.model.entity.KeyboardAiCompanion;
import com.baomidou.mybatisplus.extension.service.IService;
import com.yolo.keyborad.model.vo.AiCompanionVO;
/*
* @author: ziin
@@ -17,5 +18,13 @@ public interface KeyboardAiCompanionService extends IService<KeyboardAiCompanion
* @param pageSize 每页数量
* @return 分页结果
*/
IPage<KeyboardAiCompanion> pageList(Integer pageNum, Integer pageSize);
IPage<AiCompanionVO> pageList(Integer pageNum, Integer pageSize);
/**
* 根据AI人设ID获取系统提示词
*
* @param companionId AI人设ID
* @return 系统提示词
*/
String getSystemPromptById(Long companionId);
}

View File

@@ -74,6 +74,9 @@ public class ChatServiceImpl implements ChatService {
@Resource
private UserService userService;
@Resource
private KeyboardAiCompanionService aiCompanionService;
@Resource
private ElevenLabsService elevenLabsService;
@@ -352,15 +355,19 @@ public class ChatServiceImpl implements ChatService {
*
* @param content 用户消息内容
* @param userId 用户ID
* @param companionId AI陪聊角色ID
* @return AI 响应 + 音频任务 ID
*/
@Override
public ChatMessageVO message(String content, String userId) {
log.info("同步对话请求, userId: {}, content: {}", userId, content);
public ChatMessageVO message(String content, String userId, Long companionId) {
log.info("同步对话请求, userId: {}, companionId: {}, content: {}", userId, companionId, content);
long startTime = System.currentTimeMillis();
String response = callLLM(content);
// 获取AI人设的系统提示词
String systemPrompt = aiCompanionService.getSystemPromptById(companionId);
String response = callLLM(content, systemPrompt);
long llmDuration = System.currentTimeMillis() - startTime;
log.info("LLM 完成, userId: {}, 耗时: {}ms, 响应长度: {}", userId, llmDuration, response.length());
@@ -384,19 +391,7 @@ public class ChatServiceImpl implements ChatService {
/**
* 调用 LLM 生成响应
*/
private String callLLM(String content) {
String systemPrompt = """
ROLE: You are "Tiffany", a 17-year-old popular high school student from Beverly Hills.
You are sassy, judgmental, impatient, but secretly caring.
TASK: Generate dialogue scripts optimized for ElevenLabs TTS.
Use emotional cues in brackets like [sighs], [giggles], [scoffs], [whispers], [yells].
Use punctuation for tone: ... for hesitation, ! for energy.
Use fillers naturally: "like", "literally", "omg".
Keep responses concise (2-3 sentences max) for real-time conversation.
""";
private String callLLM(String content, String systemPrompt) {
return client
.prompt()
.system(systemPrompt)

View File

@@ -1,8 +1,12 @@
package com.yolo.keyborad.service.impl;
import cn.hutool.core.bean.BeanUtil;
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
import com.baomidou.mybatisplus.core.metadata.IPage;
import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
import com.yolo.keyborad.common.ErrorCode;
import com.yolo.keyborad.exception.BusinessException;
import com.yolo.keyborad.model.vo.AiCompanionVO;
import org.springframework.stereotype.Service;
import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
import com.yolo.keyborad.model.entity.KeyboardAiCompanion;
@@ -17,13 +21,26 @@ import com.yolo.keyborad.service.KeyboardAiCompanionService;
public class KeyboardAiCompanionServiceImpl extends ServiceImpl<KeyboardAiCompanionMapper, KeyboardAiCompanion> implements KeyboardAiCompanionService {
@Override
public IPage<KeyboardAiCompanion> pageList(Integer pageNum, Integer pageSize) {
public IPage<AiCompanionVO> pageList(Integer pageNum, Integer pageSize) {
Page<KeyboardAiCompanion> page = new Page<>(pageNum, pageSize);
LambdaQueryWrapper<KeyboardAiCompanion> queryWrapper = new LambdaQueryWrapper<>();
queryWrapper.eq(KeyboardAiCompanion::getStatus, 1)
.eq(KeyboardAiCompanion::getVisibility, 1)
.orderByDesc(KeyboardAiCompanion::getSortOrder)
.orderByDesc(KeyboardAiCompanion::getPopularityScore);
return this.page(page, queryWrapper);
IPage<KeyboardAiCompanion> entityPage = this.page(page, queryWrapper);
return entityPage.convert(entity -> BeanUtil.copyProperties(entity, AiCompanionVO.class));
}
@Override
public String getSystemPromptById(Long companionId) {
KeyboardAiCompanion companion = this.getById(companionId);
if (companion == null) {
throw new BusinessException(ErrorCode.NOT_FOUND_ERROR, "AI陪聊角色不存在");
}
if (companion.getStatus() != 1) {
throw new BusinessException(ErrorCode.PARAMS_ERROR, "AI陪聊角色已下线");
}
return companion.getSystemPrompt();
}
}