post_api.py 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227
  1. import argparse
  2. import base64
  3. import wave
  4. import ormsgpack
  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. from tools.schema import ServeReferenceAudio, ServeTTSRequest
  11. def parse_args():
  12. parser = argparse.ArgumentParser(
  13. description="Send a WAV file and text to a server and receive synthesized audio.",
  14. formatter_class=argparse.RawTextHelpFormatter,
  15. )
  16. parser.add_argument(
  17. "--url",
  18. "-u",
  19. type=str,
  20. default="http://127.0.0.1:8080/v1/tts",
  21. help="URL of the server",
  22. )
  23. parser.add_argument(
  24. "--text", "-t", type=str, required=True, help="Text to be synthesized"
  25. )
  26. parser.add_argument(
  27. "--reference_id",
  28. "-id",
  29. type=str,
  30. default=None,
  31. help="ID of the reference model to be used for the speech\n(Local: name of folder containing audios and files)",
  32. )
  33. parser.add_argument(
  34. "--reference_audio",
  35. "-ra",
  36. type=str,
  37. nargs="+",
  38. default=None,
  39. help="Path to the audio file",
  40. )
  41. parser.add_argument(
  42. "--reference_text",
  43. "-rt",
  44. type=str,
  45. nargs="+",
  46. default=None,
  47. help="Reference text for voice synthesis",
  48. )
  49. parser.add_argument(
  50. "--output",
  51. "-o",
  52. type=str,
  53. default="generated_audio",
  54. help="Output audio file name",
  55. )
  56. parser.add_argument(
  57. "--play",
  58. type=bool,
  59. default=True,
  60. help="Whether to play audio after receiving data",
  61. )
  62. parser.add_argument("--normalize", type=bool, default=True)
  63. parser.add_argument(
  64. "--format", type=str, choices=["wav", "mp3", "flac"], default="wav"
  65. )
  66. parser.add_argument(
  67. "--mp3_bitrate", type=int, choices=[64, 128, 192], default=64, help="kHz"
  68. )
  69. parser.add_argument("--opus_bitrate", type=int, default=-1000)
  70. parser.add_argument(
  71. "--latency",
  72. type=str,
  73. default="normal",
  74. choices=["normal", "balanced"],
  75. help="Used in api.fish.audio/v1/tts",
  76. )
  77. parser.add_argument(
  78. "--max_new_tokens",
  79. type=int,
  80. default=0,
  81. help="Maximum new tokens to generate. \n0 means no limit.",
  82. )
  83. parser.add_argument(
  84. "--chunk_length", type=int, default=200, help="Chunk length for synthesis"
  85. )
  86. parser.add_argument(
  87. "--top_p", type=float, default=0.7, help="Top-p sampling for synthesis"
  88. )
  89. parser.add_argument(
  90. "--repetition_penalty",
  91. type=float,
  92. default=1.2,
  93. help="Repetition penalty for synthesis",
  94. )
  95. parser.add_argument(
  96. "--temperature", type=float, default=0.7, help="Temperature for sampling"
  97. )
  98. parser.add_argument(
  99. "--streaming", type=bool, default=False, help="Enable streaming response"
  100. )
  101. parser.add_argument(
  102. "--channels", type=int, default=1, help="Number of audio channels"
  103. )
  104. parser.add_argument("--rate", type=int, default=44100, help="Sample rate for audio")
  105. parser.add_argument(
  106. "--use_memory_cache",
  107. type=str,
  108. default="never",
  109. choices=["on-demand", "never"],
  110. help="Cache encoded references codes in memory.\n"
  111. "If `on-demand`, the server will use cached encodings\n "
  112. "instead of encoding reference audio again.",
  113. )
  114. parser.add_argument(
  115. "--seed",
  116. type=int,
  117. default=None,
  118. help="`None` means randomized inference, otherwise deterministic.\n"
  119. "It can't be used for fixing a timbre.",
  120. )
  121. return parser.parse_args()
  122. if __name__ == "__main__":
  123. args = parse_args()
  124. idstr: str | None = args.reference_id
  125. # priority: ref_id > [{text, audio},...]
  126. if idstr is None:
  127. ref_audios = args.reference_audio
  128. ref_texts = args.reference_text
  129. if ref_audios is None:
  130. byte_audios = []
  131. else:
  132. byte_audios = [audio_to_bytes(ref_audio) for ref_audio in ref_audios]
  133. if ref_texts is None:
  134. ref_texts = []
  135. else:
  136. ref_texts = [read_ref_text(ref_text) for ref_text in ref_texts]
  137. else:
  138. byte_audios = []
  139. ref_texts = []
  140. pass # in api.py
  141. data = {
  142. "text": args.text,
  143. "references": [
  144. ServeReferenceAudio(audio=ref_audio, text=ref_text)
  145. for ref_text, ref_audio in zip(ref_texts, byte_audios)
  146. ],
  147. "reference_id": idstr,
  148. "normalize": args.normalize,
  149. "format": args.format,
  150. "mp3_bitrate": args.mp3_bitrate,
  151. "opus_bitrate": args.opus_bitrate,
  152. "max_new_tokens": args.max_new_tokens,
  153. "chunk_length": args.chunk_length,
  154. "top_p": args.top_p,
  155. "repetition_penalty": args.repetition_penalty,
  156. "temperature": args.temperature,
  157. "streaming": args.streaming,
  158. "use_memory_cache": args.use_memory_cache,
  159. "seed": args.seed,
  160. }
  161. pydantic_data = ServeTTSRequest(**data)
  162. response = requests.post(
  163. args.url,
  164. data=ormsgpack.packb(pydantic_data, option=ormsgpack.OPT_SERIALIZE_PYDANTIC),
  165. stream=args.streaming,
  166. headers={
  167. "authorization": "Bearer YOUR_API_KEY",
  168. "content-type": "application/msgpack",
  169. },
  170. )
  171. if response.status_code == 200:
  172. if args.streaming:
  173. p = pyaudio.PyAudio()
  174. audio_format = pyaudio.paInt16 # Assuming 16-bit PCM format
  175. stream = p.open(
  176. format=audio_format, channels=args.channels, rate=args.rate, output=True
  177. )
  178. wf = wave.open(f"{args.output}.wav", "wb")
  179. wf.setnchannels(args.channels)
  180. wf.setsampwidth(p.get_sample_size(audio_format))
  181. wf.setframerate(args.rate)
  182. stream_stopped_flag = False
  183. try:
  184. for chunk in response.iter_content(chunk_size=1024):
  185. if chunk:
  186. stream.write(chunk)
  187. wf.writeframesraw(chunk)
  188. else:
  189. if not stream_stopped_flag:
  190. stream.stop_stream()
  191. stream_stopped_flag = True
  192. finally:
  193. stream.close()
  194. p.terminate()
  195. wf.close()
  196. else:
  197. audio_content = response.content
  198. audio_path = f"{args.output}.{args.format}"
  199. with open(audio_path, "wb") as audio_file:
  200. audio_file.write(audio_content)
  201. audio = AudioSegment.from_file(audio_path, format=args.format)
  202. if args.play:
  203. play(audio)
  204. print(f"Audio has been saved to '{audio_path}'.")
  205. else:
  206. print(f"Request failed with status code {response.status_code}")
  207. print(response.json())