post_api.py 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188
  1. import argparse
  2. import base64
  3. import wave
  4. from pathlib import Path
  5. import pyaudio
  6. import requests
  7. from pydub import AudioSegment
  8. from pydub.playback import play
  9. from tools.file import audio_to_bytes, read_ref_text
  10. def parse_args():
  11. parser = argparse.ArgumentParser(
  12. description="Send a WAV file and text to a server and receive synthesized audio."
  13. )
  14. parser.add_argument(
  15. "--url",
  16. "-u",
  17. type=str,
  18. default="http://127.0.0.1:8080/v1/tts",
  19. help="URL of the server",
  20. )
  21. parser.add_argument(
  22. "--text", "-t", type=str, required=True, help="Text to be synthesized"
  23. )
  24. parser.add_argument(
  25. "--reference_id",
  26. "-id",
  27. type=str,
  28. default=None,
  29. help="ID of the reference model o be used for the speech",
  30. )
  31. parser.add_argument(
  32. "--reference_audio",
  33. "-ra",
  34. type=str,
  35. nargs="+",
  36. default=None,
  37. help="Path to the WAV file",
  38. )
  39. parser.add_argument(
  40. "--reference_text",
  41. "-rt",
  42. type=str,
  43. nargs="+",
  44. default=None,
  45. help="Reference text for voice synthesis",
  46. )
  47. parser.add_argument(
  48. "--output",
  49. "-o",
  50. type=str,
  51. default="generated_audio",
  52. help="Output audio file name",
  53. )
  54. parser.add_argument(
  55. "--play",
  56. type=bool,
  57. default=True,
  58. help="Whether to play audio after receiving data",
  59. )
  60. parser.add_argument("--normalize", type=bool, default=True)
  61. parser.add_argument(
  62. "--format", type=str, choices=["wav", "mp3", "flac"], default="wav"
  63. )
  64. parser.add_argument("--mp3_bitrate", type=int, default=64)
  65. parser.add_argument("--opus_bitrate", type=int, default=-1000)
  66. parser.add_argument("--latency", type=str, default="normal", help="延迟选项")
  67. parser.add_argument(
  68. "--max_new_tokens",
  69. type=int,
  70. default=1024,
  71. help="Maximum new tokens to generate",
  72. )
  73. parser.add_argument(
  74. "--chunk_length", type=int, default=100, help="Chunk length for synthesis"
  75. )
  76. parser.add_argument(
  77. "--top_p", type=float, default=0.7, help="Top-p sampling for synthesis"
  78. )
  79. parser.add_argument(
  80. "--repetition_penalty",
  81. type=float,
  82. default=1.2,
  83. help="Repetition penalty for synthesis",
  84. )
  85. parser.add_argument(
  86. "--temperature", type=float, default=0.7, help="Temperature for sampling"
  87. )
  88. parser.add_argument(
  89. "--speaker", type=str, default=None, help="Speaker ID for voice synthesis"
  90. )
  91. parser.add_argument("--emotion", type=str, default=None, help="Speaker's Emotion")
  92. parser.add_argument(
  93. "--streaming", type=bool, default=False, help="Enable streaming response"
  94. )
  95. parser.add_argument(
  96. "--channels", type=int, default=1, help="Number of audio channels"
  97. )
  98. parser.add_argument("--rate", type=int, default=44100, help="Sample rate for audio")
  99. return parser.parse_args()
  100. if __name__ == "__main__":
  101. args = parse_args()
  102. idstr: str | None = args.reference_id
  103. # priority: ref_id > [{text, audio},...]
  104. if idstr is None:
  105. base64_audios = [
  106. audio_to_bytes(ref_audio) for ref_audio in args.reference_audio
  107. ]
  108. ref_texts = [read_ref_text(ref_text) for ref_text in args.reference_text]
  109. else:
  110. base64_audios = []
  111. ref_texts = []
  112. pass # in api.py
  113. data = {
  114. "text": args.text,
  115. "references": [
  116. dict(text=ref_text, audio=ref_audio)
  117. for ref_text, ref_audio in zip(ref_texts, base64_audios)
  118. ],
  119. "reference_id": idstr,
  120. "normalize": args.normalize,
  121. "format": args.format,
  122. "mp3_bitrate": args.mp3_bitrate,
  123. "opus_bitrate": args.opus_bitrate,
  124. "max_new_tokens": args.max_new_tokens,
  125. "chunk_length": args.chunk_length,
  126. "top_p": args.top_p,
  127. "repetition_penalty": args.repetition_penalty,
  128. "temperature": args.temperature,
  129. "speaker": args.speaker,
  130. "emotion": args.emotion,
  131. "streaming": args.streaming,
  132. }
  133. response = requests.post(args.url, json=data, stream=args.streaming)
  134. if response.status_code == 200:
  135. if args.streaming:
  136. p = pyaudio.PyAudio()
  137. audio_format = pyaudio.paInt16 # Assuming 16-bit PCM format
  138. stream = p.open(
  139. format=audio_format, channels=args.channels, rate=args.rate, output=True
  140. )
  141. wf = wave.open(f"{args.output}.wav", "wb")
  142. wf.setnchannels(args.channels)
  143. wf.setsampwidth(p.get_sample_size(audio_format))
  144. wf.setframerate(args.rate)
  145. stream_stopped_flag = False
  146. try:
  147. for chunk in response.iter_content(chunk_size=1024):
  148. if chunk:
  149. stream.write(chunk)
  150. wf.writeframesraw(chunk)
  151. else:
  152. if not stream_stopped_flag:
  153. stream.stop_stream()
  154. stream_stopped_flag = True
  155. finally:
  156. stream.close()
  157. p.terminate()
  158. wf.close()
  159. else:
  160. audio_content = response.content
  161. audio_path = f"{args.output}.{args.format}"
  162. with open(audio_path, "wb") as audio_file:
  163. audio_file.write(audio_content)
  164. audio = AudioSegment.from_file(audio_path, format=args.format)
  165. if args.play:
  166. play(audio)
  167. print(f"Audio has been saved to '{audio_path}'.")
  168. else:
  169. print(f"Request failed with status code {response.status_code}")
  170. print(response.json())