skip to content
snuow's brain

【python】VOICEVOXをpythonから使用する方法

VOICEVOXはpython製のFastAPIで動いているので、pythonからrequestが可能

概要

  • Voicevox という無料で使えるテキスト読み上げソフトウェアがあります。
  • バックエンドは FastAPI で動いているので POST することで、簡単に音声を合成できます。

VOICEVOX ENGINE

  • Githubのダウンロードからエンジンをダウンロードします。
  • 解凍したフォルダ内の、run.exeをダブルクリックし、バックエンドの FastAPI を立ち上げます。

ソースコード

  • バックエンドを立ち上げた状態で、下記のソースコードを実行すると、「こんにちは」と喋ります。
  • speak メソッドの speaker の id を変えることで、声を変更できます。
    • id はバックエンドに対してGET:/speakersすることで、一覧を出すことができます。
    • speaker のデフォルトが47のナースロボ_タイプ T になっているのは私の趣味です。
  • 声のサンプルは公式で聞くことができます。
main.py
import requests, json
import io
import wave
import pyaudio
import time

class Voicevox:
    def __init__(self,host="127.0.0.1",port=50021):
        self.host = host
        self.port = port

    def speak(self,text=None,speaker=47): # VOICEVOX:ナースロボ_タイプT

        params = (
            ("text", text),
            ("speaker", speaker)  # 音声の種類をInt型で指定
        )

        init_q = requests.post(
            f"http://{self.host}:{self.port}/audio_query",
            params=params
        )

        res = requests.post(
            f"http://{self.host}:{self.port}/synthesis",
            headers={"Content-Type": "application/json"},
            params=params,
            data=json.dumps(init_q.json())
        )

        # メモリ上で展開
        audio = io.BytesIO(res.content)

        with wave.open(audio,'rb') as f:
            # 以下再生用処理
            p = pyaudio.PyAudio()

            def _callback(in_data, frame_count, time_info, status):
                data = f.readframes(frame_count)
                return (data, pyaudio.paContinue)

            stream = p.open(format=p.get_format_from_width(width=f.getsampwidth()),
                            channels=f.getnchannels(),
                            rate=f.getframerate(),
                            output=True,
                            stream_callback=_callback)

            # Voice再生
            stream.start_stream()
            while stream.is_active():
                time.sleep(0.1)

            stream.stop_stream()
            stream.close()
            p.terminate()


def main():
    vv = Voicevox()
    vv.speak(text='こんにちは')


if __name__ == "__main__":
    main()