feat: 克隆音频完成能跑通

main
guofei 2024-12-03 14:58:40 +08:00
parent cfb38ccf8b
commit 1376f9375e
6 changed files with 501 additions and 150 deletions

View File

@ -27,3 +27,12 @@ pm2 start --name admin-banban-new-nest npm -- run start:prod
StatusCode: 1114, StatusCode: 1114,
StatusMessage: 'snr check failed, snr: 0.33, threshold: 5.00' StatusMessage: 'snr check failed, snr: 0.33, threshold: 5.00'
}, --> }, -->
const result = [
{
roleId: '6704bd0ef48326fe51ddb751',
roleName: '甘宁',
url: 'https://banban-systemcharter-speak.oss-cn-beijing.aliyuncs.com/test/WeChat_20241119150807_1.mp3',
},
];

View File

@ -32,6 +32,9 @@ model SystemCharter {
// 音色名称 // 音色名称
voiceName String? voiceName String?
// 剩余可克隆次数
remainingCloneCount Int @default(10)
// 原始音频 // 原始音频
originAudioUrl String? originAudioUrl String?
// 克隆音频 // 克隆音频
@ -45,6 +48,10 @@ model TaskQueue {
id String @id @default(auto()) @map("_id") @db.ObjectId id String @id @default(auto()) @map("_id") @db.ObjectId
// 任务类型 // 任务类型
type String type String
// 任务名称
roleName String?
// 角色id
roleId String?
// 任务数据 // 任务数据
data String data String
// 任务状态: pending/processing/completed/failed // 任务状态: pending/processing/completed/failed
@ -53,8 +60,21 @@ model TaskQueue {
error String? error String?
// 重试次数 // 重试次数
attempts Int @default(0) attempts Int @default(0)
// 最大重试次数 // 最大重试次数 (不需要这个逻辑了)
maxAttempts Int @default(3) maxAttempts Int @default(3)
createdAt DateTime @default(now()) createdAt DateTime @default(now())
updatedAt DateTime @updatedAt updatedAt DateTime @updatedAt
} }
model CloneHistory {
id String @id @default(auto()) @map("_id") @db.ObjectId
roleId String // 角色ID
roleName String // 角色名称
cloneUrl String // 克隆URL
speakerId String? // 克隆后的speakerId
status String // 任务状态pending, processing, success, failed
error String? // 错误信息
taskId String? // 关联的任务ID
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
}

View File

@ -12,10 +12,11 @@ export class RedisTaskService {
// 添加任务 // 添加任务
async addTask(data: any) { async addTask(data: any) {
// 先记录到数据库
const taskRecord = await this.dbService.taskQueue.create({ const taskRecord = await this.dbService.taskQueue.create({
data: { data: {
type: 'BATCH_CLONE_AUDIO', type: 'BATCH_CLONE_AUDIO',
roleId: data.roleId,
roleName: data.roleName,
data: JSON.stringify(data), data: JSON.stringify(data),
status: 'pending', status: 'pending',
}, },
@ -46,40 +47,69 @@ export class RedisTaskService {
}); });
} }
// 更新任务状态 任务状态pending, processing, success, failed
async updateTaskStatus(taskId: string, status: string, error?: string, speakerId?: string) {
try {
// 更新任务状态 // 更新任务状态
async updateTaskStatus(taskId: string, status: string, error?: string) { const task = await this.dbService.taskQueue.update({
return await this.dbService.taskQueue.update({
where: { id: taskId }, where: { id: taskId },
data: { data: {
status, status,
error, error,
attempts: { increment: 1 }, // 只有重新添加的任务才需要增加重试次数
attempts: { increment: status === 'processing' ? 1 : 0 },
},
});
const taskData = JSON.parse(task.data);
// 如果任务成功,减少剩余克隆次数
if (status === 'completed' && speakerId) {
// 减少剩余克隆次数
await this.dbService.systemCharter.update({
where: { id: taskData.roleId },
data: {
remainingCloneCount: {
decrement: 1,
},
}, },
}); });
} }
return task;
} catch (error) {
console.error('更新任务状态失败:', error);
throw error;
}
}
// 重试失败的任务 // 重试失败的任务
async retryTask(taskId: string) { async retryTask(taskId: string) {
try {
const task = await this.dbService.taskQueue.findUnique({ const task = await this.dbService.taskQueue.findUnique({
where: { id: taskId }, where: { id: taskId },
}); });
if (task && task.status === 'failed') { if (!task || task.status !== 'failed') {
// 重置任务状态 throw new Error('任务不存在或状态不是失败状态');
await this.dbService.taskQueue.update({ }
await this.dbService.$transaction(async (prisma) => {
// 更新任务状态为待处理
await prisma.taskQueue.update({
where: { id: taskId }, where: { id: taskId },
data: { data: {
status: 'pending', status: 'pending',
attempts: 0,
error: null, error: null,
}, },
}); });
// 重新加入队列 // 重新加入队列处理
const taskData = JSON.parse(task.data);
await this.taskQueue.add( await this.taskQueue.add(
'BATCH_CLONE_AUDIO', 'BATCH_CLONE_AUDIO',
{ {
...JSON.parse(task.data), ...taskData,
taskId: task.id, taskId: task.id,
}, },
{ {
@ -87,8 +117,12 @@ export class RedisTaskService {
attempts: 3, attempts: 3,
}, },
); );
});
return true; return true;
} } catch (error) {
return false; console.error('重试任务失败:', error);
throw error;
}
} }
} }

View File

@ -27,38 +27,43 @@ export class RedisTaskProcessor {
const volcenAudioSpeakService = new VolcenAudioSpeakService(); const volcenAudioSpeakService = new VolcenAudioSpeakService();
// voiceId = await volcenAudioSpeakService.getVoiceId(); // voiceId = await volcenAudioSpeakService.getVoiceId();
// const voiceId = 'S_FC60x0Gb1'; // const voiceId = 'S_FC60x0Gb1';
const voiceId = 'S_VK2Yw0Gb1'; // 剩余1次
// const voiceId = 'S_VK2Yw0Gb1';
voiceId = 'S_5QKWw0Gb1';
// 请求ossurl获取base64 // 请求ossurl获取base64
const base64 = await volcenAudioSpeakService.getAudioBase64(ossUrl); const base64 = await volcenAudioSpeakService.getAudioBase64(ossUrl);
// 克隆音频 // 克隆音频
volcenAudioSpeakService.speakClone(voiceId, base64); await volcenAudioSpeakService.speakClone(voiceId, base64);
// // 5秒后激活
setTimeout(async () => {
if (taskId) {
await this.redisTaskService.updateTaskStatus(taskId, 'completed'); await this.redisTaskService.updateTaskStatus(taskId, 'completed');
// 激活音频 (激活后就不能克隆了,所以要克隆后确定后再激活) const textToSpeechResult = await volcenAudioSpeakService.textToSpeech('今天有什么新鲜事吗?快给我讲讲', voiceId);
// await volcenAudioSpeakService.speakActivate(voiceId);
const textToSpeechResult = await volcenAudioSpeakService.textToSpeech(
'今天有什么新鲜事吗?快给我讲讲',
voiceId,
);
// 修改系统角色的 voiceId // 修改系统角色的 voiceId
await this.dbService.systemCharter.update({ await this.dbService.systemCharter.update({
where: { id: charterInfo.roleId }, where: { id: charterInfo.roleId },
data: { data: {
voiceId, voiceId,
voiceName: charterInfo.roleName, voiceName: charterInfo.roleName,
// 是否激活
activate: true,
originAudioUrl: ossUrl, originAudioUrl: ossUrl,
cloneAfterAudioUrl: textToSpeechResult.url, cloneAfterAudioUrl: textToSpeechResult.url,
remainingCloneCount: {
// 剩余可克隆次数减1
decrement: 1,
},
},
});
// 增加克隆人物音色记录
await this.dbService.cloneHistory.create({
data: {
roleId: charterInfo.roleId,
roleName: charterInfo.roleName,
cloneUrl: textToSpeechResult.url,
speakerId: voiceId,
status: 'success',
taskId,
}, },
}); });
}
console.log(`任务处理完成: ${job.data.roleName}`); console.log(`任务处理完成: ${job.data.roleName}`);
}, 1000 * 5);
} catch (error) { } catch (error) {
console.log(error.data); console.log('error', error);
console.log('voiceId=--------------', voiceId); console.log('voiceId=--------------', voiceId);
console.error(`任务处理失败 [第${job.attemptsMade + 1}次尝试]: ${error.message}`); console.error(`任务处理失败 [第${job.attemptsMade + 1}次尝试]: ${error.message}`);
if (job.data.taskId) { if (job.data.taskId) {

View File

@ -1,11 +1,22 @@
import { Body, Controller, Get, Post, Query } from '@nestjs/common'; import { Body, Controller, Get, Param, Post, Query, HttpException, HttpStatus, Delete } from '@nestjs/common';
import { SystemCharterlDto } from './dto/SystemCharter.dto'; import { SystemCharterlDto } from './dto/SystemCharter.dto';
import { Pagination } from 'src/common/pagination'; import { Pagination } from 'src/common/pagination';
import { DBService } from 'src/utils/db/DB.service'; import { DBService } from 'src/utils/db/DB.service';
import { ApiResponse } from 'src/utils/response/response'; import { ApiResponse } from 'src/utils/response/response';
import { RedisTaskService } from 'src/common/RedisTask/RedisTask.service'; import { RedisTaskService } from 'src/common/RedisTask/RedisTask.service';
import { CloneSpeakDto } from './dto/CloneSpeakDto.dto'; import { VolcenAudioSpeakService } from 'src/services/VolcenAudioSpeakService';
import axios from 'axios'; import axios from 'axios';
import { CloneSpeakDto } from './dto/CloneSpeakDto.dto';
// 添加任务查询DTO
export class TaskQueryDto {
current?: number;
pageSize?: number;
roleId?: string;
roleName?: string;
startTime?: string;
endTime?: string;
}
@Controller('/system/charter') @Controller('/system/charter')
export class SystemCharterController { export class SystemCharterController {
@ -28,6 +39,18 @@ export class SystemCharterController {
orderBy: { orderBy: {
createdAt: 'desc', createdAt: 'desc',
}, },
select: {
id: true,
roleName: true,
remainingCloneCount: true,
bg: true,
voiceId: true,
voiceName: true,
roleSetting: true,
originAudioUrl: true,
cloneAfterAudioUrl: true,
activate: true,
},
}); });
return ApiResponse.success({ return ApiResponse.success({
data: record, data: record,
@ -39,15 +62,25 @@ export class SystemCharterController {
@Post('/add') @Post('/add')
async addSystemCharter(@Body() object: SystemCharterlDto) { async addSystemCharter(@Body() object: SystemCharterlDto) {
try {
delete object.id; delete object.id;
// 添加默认的剩余可克隆次数
const newRecord = await this.dbService.systemCharter.create({ const newRecord = await this.dbService.systemCharter.create({
data: object, data: {
...object,
remainingCloneCount: 10, // 默认10次
},
}); });
return newRecord; return ApiResponse.successToMessage('添加成功', newRecord);
} catch (error) {
console.error('添加角色失败:', error);
return ApiResponse.failToMessage('添加失败:' + error.message);
}
} }
@Post('/update') @Post('/update')
async updateSystemCharter(@Body() object: SystemCharterlDto) { async updateSystemCharter(@Body() object: SystemCharterlDto) {
try {
const { id, ...other } = object; const { id, ...other } = object;
const newRecord = await this.dbService.systemCharter.update({ const newRecord = await this.dbService.systemCharter.update({
where: { where: {
@ -55,55 +88,305 @@ export class SystemCharterController {
}, },
data: other, data: other,
}); });
return newRecord; return ApiResponse.successToMessage('更新成功', newRecord);
} catch (error) {
console.error('更新角色失败:', error);
return ApiResponse.failToMessage('更新失败:' + error.message);
}
} }
@Get('cloneSpeaker') @Get('taskList')
async batchCloneAudio(@Body() object: CloneSpeakDto) { async getTaskList(@Query() query: TaskQueryDto) {
// try { try {
// if (!object.roleId || !object.roleName || !object.url) { const { current, pageSize, roleId, roleName } = query;
// return ApiResponse.failToMessage('参数错误');
// }
// // 验证音频URL是否可访问
// const response = await axios.head(object.url);
// if (response.status !== 200) {
// return ApiResponse.failToMessage('音频文件无法访问');
// }
// // 验证文件类型
// const contentType = response.headers['content-type'];
// if (!contentType.includes('audio')) {
// return ApiResponse.failToMessage('文件类型必须是音频');
// }
// this.redisTaskService.addTask(object);
// return ApiResponse.success(null, '任务添加成功');
// } catch (error) {
// console.error('添加克隆任务失败:', error);
// return ApiResponse.failToMessage('添加任务失败:' + error.message);
// }
}
@Get('test') // 构建查询条件
async batchCloneAudio1() { const where: any = {
const result = [ // 只查询非成功状态的任务
{ status: {
roleId: '6704bd0ef48326fe51ddb751', not: 'completed',
roleName: '甘宁', },
// url: 'https://banban-systemcharter-speak.oss-cn-beijing.aliyuncs.com/test/%E4%BB%A3%E5%8F%B7%E9%B8%A2%E5%AF%86%E6%8E%A2%E5%91%A8%E7%91%9C.mp3', };
// url: 'https://banban-systemcharter-speak.oss-cn-beijing.aliyuncs.com/test/%E4%BB%A3%E5%8F%B7%E9%B8%A2%E5%AF%86%E6%8E%A2%E7%94%98%E5%AE%81.mp3',
// url: 'https://banban-systemcharter-speak.oss-cn-beijing.aliyuncs.com/test/%E4%BB%A3%E5%8F%B7%E9%B8%A2%E5%AF%86%E6%8E%A2%E7%94%98%E5%AE%81%20-%20%E5%89%AF%E6%9C%AC.wav', if (roleId) {
// 郭德纲 where.roleId = roleId;
// url: 'https://banban-systemcharter-speak.oss-cn-beijing.aliyuncs.com/test/%E9%83%AD%E5%BE%B7%E7%BA%B2-%E5%A3%B0%E9%9F%B3%E5%85%8B%E9%9A%86.mp3', }
// 增以后的
url: 'https://banban-systemcharter-speak.oss-cn-beijing.aliyuncs.com/test/WeChat_20241119150807_1.mp3', if (roleName) {
where.roleName = {
// 周瑜 contains: roleName,
// url: 'https://banban-systemcharter-speak.os-cn-beijing.aliyuncs.com/test/%E4%BB%A3%E5%8F%B7%E9%B8%A2%E5%AF%86%E6%8E%A2%E5%91%A8%E7%91%9C.mp3', };
// 周瑜 提高分贝后 }
// url: 'https://banban-systemcharter-speak.oss-cn-beijing.aliyuncs.com/test/%E5%91%A8%E7%91%9C.mp3',
// 查询数据
const [tasks, total] = await Promise.all([
this.dbService.taskQueue.findMany({
where,
...Pagination.getPage(current, pageSize),
orderBy: {
createdAt: 'desc',
},
}),
this.dbService.taskQueue.count({ where }),
]);
return ApiResponse.success({
data: tasks,
meta: {
total,
current: current || 1,
pageSize: pageSize || 10,
}, },
];
result.forEach((item) => {
this.redisTaskService.addTask(item);
}); });
} catch (error) {
console.error('获取任务列表失败:', error);
return ApiResponse.failToMessage('获取任务列表失败:' + error.message);
}
}
@Get('activateVoice')
async activateVoice(@Query('speakerId') speakerId: string) {
try {
if (!speakerId) {
return ApiResponse.failToMessage('参数错误缺少speakerId');
}
const volcenAudioSpeakService = new VolcenAudioSpeakService();
const result = await volcenAudioSpeakService.speakActivate(speakerId);
return ApiResponse.successToMessage('音频激活成功', result);
} catch (error) {
console.error('激活音频失败:', error);
return ApiResponse.failToMessage('激活音频失败:' + error.message);
}
}
// 添加获取克隆历史记录的接口
@Get('cloneHistory/:id')
async getCloneHistory(@Param('id') id: string) {
try {
// 查询数据
const [histories, total] = await Promise.all([
this.dbService.cloneHistory.findMany({
where: {
roleId: id,
},
orderBy: {
createdAt: 'desc',
},
}),
this.dbService.cloneHistory.count({ where: { roleId: id } }),
]);
return ApiResponse.successToMessage('获取成功', {
data: histories,
meta: {
total,
},
});
} catch (error) {
console.error('获取克隆历史失败:', error);
return ApiResponse.failToMessage('获取克隆历史失败:' + error.message);
}
}
@Get('/detail/:id')
async getDetail(@Param('id') id: string) {
try {
if (!id) {
return ApiResponse.failToMessage('参数错误缺少id');
}
const record = await this.dbService.systemCharter.findUnique({
where: {
id: id,
},
});
if (!record) {
return ApiResponse.failToMessage('人设不存在');
}
return ApiResponse.successToMessage('获取成功', record);
} catch (error) {
console.error('获取人设详情失败:', error);
return ApiResponse.failToMessage('获取人设详情失败:' + error.message);
}
}
/**
*
* @param cloneSpeakDto
* @returns
*/
@Post('cloneSpeaker')
async cloneSpeaker(@Body() cloneSpeakDto: CloneSpeakDto) {
try {
const { roleId, roleName } = cloneSpeakDto;
// 检查是否存在未完成的相同角色任务
const existingTask = await this.dbService.taskQueue.findFirst({
where: {
roleId,
status: {
// pending / processing /completed/failed
in: ['pending', 'processing'],
},
},
});
if (existingTask) {
throw new HttpException(
{
message: '该角色已有正在进行的克隆任务,请等待任务完成后再试',
taskId: existingTask.id,
},
HttpStatus.BAD_REQUEST,
);
}
// 检查剩余克隆次数
const charter = await this.dbService.systemCharter.findUnique({
where: { id: roleId },
});
if (!charter) {
throw new HttpException('角色不存在', HttpStatus.BAD_REQUEST);
}
if (charter.remainingCloneCount <= 0) {
throw new HttpException('剩余克隆次数不足', HttpStatus.BAD_REQUEST);
}
// 添加任务到队列
await this.redisTaskService.addTask({
...cloneSpeakDto,
roleId,
roleName,
});
return {
message: '克隆任务已添加到队列',
};
} catch (error) {
throw error;
}
}
// 重试失败的任务
@Post('retryTask/:taskId')
async retryTask(@Param('taskId') taskId: string) {
try {
if (!taskId) {
return ApiResponse.failToMessage('参数错误缺少taskId');
}
await this.redisTaskService.retryTask(taskId);
return ApiResponse.successToMessage('重试任务添加成功');
} catch (error) {
console.error('重试任务失败:', error);
return ApiResponse.failToMessage('重试任务失败:' + error.message);
}
}
/**
*
* @param roleId
*/
@Get('checkCloneTask/:roleId')
async checkCloneTask(@Param('roleId') roleId: string) {
try {
const charter = await this.dbService.systemCharter.findUnique({
where: { id: roleId },
});
if (!charter) {
throw new HttpException('角色不存在', HttpStatus.BAD_REQUEST);
}
// 检查是否存在未完成的相同角色任务
const existingTask = await this.dbService.taskQueue.findFirst({
where: {
roleId,
status: {
in: ['pending', 'processing'],
},
},
});
if (existingTask) {
throw new HttpException('该角色已有正在进行的克隆任务,请等待任务完成后再试', HttpStatus.BAD_REQUEST);
}
// 检查是否存在失败的任务
const failedTask = await this.dbService.taskQueue.findFirst({
where: {
roleId,
status: 'failed',
},
});
// 处理失败任务的错误信息
if (failedTask) {
throw new HttpException('该角色存在失败的克隆任务,请先删除失败任务后再试', HttpStatus.BAD_REQUEST);
}
return ApiResponse.success();
} catch (error) {
throw error;
}
}
// 删除任务
@Delete('deleteTask/:taskId')
async deleteTask(@Param('taskId') taskId: string) {
try {
const task = await this.dbService.taskQueue.findUnique({
where: { id: taskId },
select: {
id: true,
status: true,
roleId: true,
roleName: true,
},
});
if (!task) {
throw new HttpException(
{
message: '任务不存在',
code: 'TASK_NOT_FOUND',
},
HttpStatus.NOT_FOUND,
);
}
if (task.status !== 'failed') {
throw new HttpException(
{
message: '只能删除失败状态的任务',
code: 'INVALID_TASK_STATUS',
currentStatus: task.status,
},
HttpStatus.BAD_REQUEST,
);
}
// 删除任务队列中的记录
await this.dbService.taskQueue.delete({
where: { id: taskId },
});
return ApiResponse.successToMessage('任务删除成功');
} catch (error) {
if (error instanceof HttpException) {
throw error;
}
throw new HttpException(
{
message: '删除任务失败',
error: error.message,
},
HttpStatus.INTERNAL_SERVER_ERROR,
);
}
} }
} }

View File

@ -89,7 +89,7 @@ export class VolcenAudioSpeakService {
speakerId: string, speakerId: string,
base64: string | Buffer, base64: string | Buffer,
): Promise<T> { ): Promise<T> {
return new Promise(async (resolve, reject) => { try {
const tiktokBody = { const tiktokBody = {
speaker_id: speakerId, speaker_id: speakerId,
appid: VolcenAudioSpeakService.TiktokAppId, appid: VolcenAudioSpeakService.TiktokAppId,
@ -97,41 +97,41 @@ export class VolcenAudioSpeakService {
{ {
audio_bytes: base64, audio_bytes: base64,
audio_format: 'wav', audio_format: 'wav',
// 添加音频参数要求 sample_rate: 16000,
// sample_rate: 16000, // 采样率必须是16kHz channels: 1,
// channels: 1, // 单声道 bits: 16,
// bits: 16, // 16位深度
}, },
], ],
source: 2, source: 2,
}; };
// 开始克隆声音,参数: { speaker_id: 'S_FC60x0Gb1', appid: '8167092294', audio_format: 'wav' }
console.log('开始克隆声音,参数:', { console.log('开始克隆声音,参数:', {
speaker_id: speakerId, speaker_id: speakerId,
appid: VolcenAudioSpeakService.TiktokAppId, appid: VolcenAudioSpeakService.TiktokAppId,
audio_format: 'wav', audio_format: 'wav',
}); });
axios
.post(VolcenAudioSpeakService.SPEAK_CLONE_API, tiktokBody, { const result = await axios.post(VolcenAudioSpeakService.SPEAK_CLONE_API, tiktokBody, {
headers: { headers: {
'Resource-Id': 'volc.megatts.voiceclone', 'Resource-Id': 'volc.megatts.voiceclone',
Authorization: `Bearer;${VolcenAudioSpeakService.TiktokAccessToken}`, Authorization: `Bearer;${VolcenAudioSpeakService.TiktokAccessToken}`,
'Content-Type': 'application/json', 'Content-Type': 'application/json',
}, },
}) });
.then((result) => {
console.log('result.status', result.status);
// 检查业务状态码 // 检查业务状态码
if (result.data?.BaseResp?.StatusCode !== 0) { if (result.data?.BaseResp?.StatusCode !== 0) {
console.error('克隆声音业务错误:', result.data); console.error('克隆声音业务错误:', result.data);
throw new Error(`训练声音业务错误: ${result.data?.BaseResp?.StatusMessage || '未知错误'}`); throw new Error(`训练声音业务错误: ${result.data?.BaseResp?.StatusMessage || '未知错误'}`);
} }
resolve(result.data as T);
}) return result.data as T;
.catch((error) => { } catch (error) {
reject(error); const responseResult = error.response.data;
}); throw new Error(JSON.stringify(responseResult));
}); }
} }
/** /**