File size: 5,602 Bytes
d5d0921 01e655b d2b7e94 01e655b d2b7e94 01e655b d5d0921 01e655b d5d0921 01e655b 02e90e4 01e655b 1df74c6 01e655b d5d0921 01e655b 627d3d7 d5d0921 01e655b 627d3d7 01e655b 627d3d7 01e655b ebc4336 01e655b 1df74c6 d5d0921 627d3d7 01e655b 627d3d7 c5458aa 627d3d7 01e655b d5d0921 1df74c6 01e655b ebc4336 c5458aa 627d3d7 c5458aa d5d0921 c5458aa d5d0921 c5458aa d5d0921 01e655b d5d0921 01e655b d5d0921 01e655b d5d0921 627d3d7 d5d0921 01e655b d5d0921 01e655b d5d0921 01e655b d5d0921 01e655b ebc4336 01e655b 02e90e4 |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 |
from typing import Union
from fastapi import HTTPException
from pydantic import BaseModel
from modules.api import utils as api_utils
from modules.api.Api import APIManager
from modules.api.impl.handler.SSMLHandler import SSMLHandler
from modules.api.impl.handler.TTSHandler import TTSHandler
from modules.api.impl.model.audio_model import AdjustConfig, AudioFormat
from modules.api.impl.model.chattts_model import ChatTTSConfig, InferConfig
from modules.api.impl.model.enhancer_model import EnhancerConfig
from modules.speaker import Speaker, speaker_mgr
class SynthesisInput(BaseModel):
text: Union[str, None] = None
ssml: Union[str, None] = None
class VoiceSelectionParams(BaseModel):
languageCode: str = "ZH-CN"
name: str = "female2"
style: str = ""
temperature: float = 0.3
topP: float = 0.7
topK: int = 20
seed: int = 42
# end_of_sentence
eos: str = "[uv_break]"
class AudioConfig(BaseModel):
audioEncoding: AudioFormat = AudioFormat.mp3
speakingRate: float = 1
pitch: float = 0
volumeGainDb: float = 0
sampleRateHertz: int = 24000
batchSize: int = 4
spliterThreshold: int = 100
class GoogleTextSynthesizeRequest(BaseModel):
input: SynthesisInput
voice: VoiceSelectionParams
audioConfig: AudioConfig
enhancerConfig: EnhancerConfig = None
class GoogleTextSynthesizeResponse(BaseModel):
audioContent: str
async def google_text_synthesize(request: GoogleTextSynthesizeRequest):
input = request.input
voice = request.voice
audioConfig = request.audioConfig
enhancerConfig = request.enhancerConfig
# 提取参数
# TODO 这个也许应该传给 normalizer
language_code = voice.languageCode
voice_name = voice.name
infer_seed = voice.seed or 42
eos = voice.eos or "[uv_break]"
audio_format = audioConfig.audioEncoding
if not isinstance(audio_format, AudioFormat) and isinstance(audio_format, str):
audio_format = AudioFormat(audio_format)
speaking_rate = audioConfig.speakingRate or 1
pitch = audioConfig.pitch or 0
volume_gain_db = audioConfig.volumeGainDb or 0
batch_size = audioConfig.batchSize or 1
spliter_threshold = audioConfig.spliterThreshold or 100
# TODO
sample_rate = audioConfig.sampleRateHertz or 24000
params = api_utils.calc_spk_style(spk=voice.name, style=voice.style)
# 虽然 calc_spk_style 可以解析 seed 形式,但是这个接口只准备支持 speakers list 中存在的 speaker
if speaker_mgr.get_speaker(voice_name) is None:
raise HTTPException(
status_code=422, detail="The specified voice name is not supported."
)
if not isinstance(params.get("spk"), Speaker):
raise HTTPException(
status_code=422, detail="The specified voice name is not supported."
)
speaker = params.get("spk")
tts_config = ChatTTSConfig(
style=params.get("style", ""),
temperature=voice.temperature,
top_k=voice.topK,
top_p=voice.topP,
)
infer_config = InferConfig(
batch_size=batch_size,
spliter_threshold=spliter_threshold,
eos=eos,
seed=infer_seed,
)
adjust_config = AdjustConfig(
speaking_rate=speaking_rate,
pitch=pitch,
volume_gain_db=volume_gain_db,
)
enhancer_config = enhancerConfig
mime_type = f"audio/{audio_format.value}"
if audio_format == AudioFormat.mp3:
mime_type = "audio/mpeg"
try:
if input.text:
text_content = input.text
handler = TTSHandler(
text_content=text_content,
spk=speaker,
tts_config=tts_config,
infer_config=infer_config,
adjust_config=adjust_config,
enhancer_config=enhancer_config,
)
base64_string = handler.enqueue_to_base64(format=audio_format)
return {"audioContent": f"data:{mime_type};base64,{base64_string}"}
elif input.ssml:
ssml_content = input.ssml
handler = SSMLHandler(
ssml_content=ssml_content,
infer_config=infer_config,
adjust_config=adjust_config,
enhancer_config=enhancer_config,
)
base64_string = handler.enqueue_to_base64(format=audio_format)
return {"audioContent": f"data:{mime_type};base64,{base64_string}"}
else:
raise HTTPException(
status_code=422, detail="Invalid input text or ssml specified."
)
except Exception as e:
import logging
logging.exception(e)
if isinstance(e, HTTPException):
raise e
else:
raise HTTPException(status_code=500, detail=str(e))
def setup(app: APIManager):
app.post(
"/v1/text:synthesize",
response_model=GoogleTextSynthesizeResponse,
description="""
google api document: <br/>
[https://cloud.google.com/text-to-speech/docs/reference/rest/v1/text/synthesize](https://cloud.google.com/text-to-speech/docs/reference/rest/v1/text/synthesize)
- 多个属性在本系统中无用仅仅是为了兼容google api
- voice 中的 topP, topK, temperature 为本系统中的参数
- voice.name 即 speaker name (或者speaker seed)
- voice.seed 为 infer seed (可在webui中测试具体作用)
- 编码格式影响的是 audioContent 的二进制格式,所以所有format都是返回带有base64数据的json
""",
)(google_text_synthesize)
|