diff --git a/notebooks/piper_inference_(ONNX).ipynb b/notebooks/piper_inference_(ONNX).ipynb new file mode 100644 index 0000000..be48807 --- /dev/null +++ b/notebooks/piper_inference_(ONNX).ipynb @@ -0,0 +1,475 @@ +{ + "nbformat": 4, + "nbformat_minor": 0, + "metadata": { + "colab": { + "provenance": [], + "authorship_tag": "ABX9TyPUlEPUiRIAfFt3VCxArrAD", + "include_colab_link": true + }, + "kernelspec": { + "name": "python3", + "display_name": "Python 3" + }, + "language_info": { + "name": "python" + } + }, + "cells": [ + { + "cell_type": "markdown", + "metadata": { + "id": "view-in-github", + "colab_type": "text" + }, + "source": [ + "\"Open" + ] + }, + { + "cell_type": "markdown", + "source": [ + "# **[Piper](https://github.com/rhasspy/piper) inferencing notebook.**\n", + "## ![Piper logo](https://contribute.rhasspy.org/img/logo.png)\n", + "\n", + "---\n", + "\n", + "- Notebook made by [rmcpantoja](http://github.com/rmcpantoja)\n", + "- Collaborator: [Xx_Nessu_xX](https://fakeyou.com/profile/Xx_Nessu_xX)" + ], + "metadata": { + "id": "eK3nmYDB6C1a" + } + }, + { + "cell_type": "markdown", + "source": [ + "# First steps" + ], + "metadata": { + "id": "9wIvcSmOby84" + } + }, + { + "cell_type": "code", + "source": [ + "#@title Install software\n", + "#@markdown The speech synthesizer and other important dependencies will be installed in this cell.\n", + "\n", + "#@markdown #### Enhable Enhanced Accessibility?\n", + "#@markdown This Enhanced Accessibility functionality is designed for the visually impaired, in which most of the interface can be used by voice guides.\n", + "enhanced_accessibility = True #@param {type:\"boolean\"}\n", + "#@markdown ---\n", + "\n", + "#@markdown #### Please select your language:\n", + "lang_select = \"Spanish\" #@param [\"English\", \"Spanish\"]\n", + "if lang_select == \"English\":\n", + " lang = \"en\"\n", + "elif lang_select == \"Spanish\":\n", + " lang = \"es\"\n", + "else:\n", + " raise Exception(\"Language not supported.\")\n", + "\n", + "if enhanced_accessibility:\n", + " from google.colab import output\n", + " guideurl = f\"https://github.com/rmcpantoja/piper/blob/master/notebooks/wav/{lang}\"\n", + " def playaudio(filename, extension = \"wav\"):\n", + " return output.eval_js(f'new Audio(\"{guideurl}/{filename}.{extension}?raw=true\").play()')\n", + "\n", + "%cd /content\n", + "print(\"Installing...\")\n", + "if enhanced_accessibility:\n", + " playaudio(\"installing\")\n", + "!git clone -q https://github.com/rmcpantoja/piper\n", + "%cd /content/piper/src/python\n", + "!pip install -q -r requirements.txt\n", + "!bash build_monotonic_align.sh\n", + "!apt-get install -q espeak-ng\n", + "import os\n", + "if not os.path.exists(\"/content/piper/src/python/lng\"):\n", + " !cp -r \"/content/piper/notebooks/lng\" /content/piper/src/python/lng\n", + "import sys\n", + "sys.path.append('/content/piper/notebooks')\n", + "from translator import *\n", + "lan = Translator()\n", + "\n", + "if enhanced_accessibility:\n", + " playaudio(\"installed\")\n", + "print(\"Success!\")" + ], + "metadata": { + "cellView": "form", + "id": "v8b_PEtXb8co" + }, + "execution_count": null, + "outputs": [] + }, + { + "cell_type": "code", + "source": [ + "#@title Download your exported model\n", + "%cd /content/piper/src/python\n", + "import os\n", + "#@markdown #### ID or link of the voice package (tar.gz format):\n", + "package_url_or_id = \"https://github.com/rhasspy/piper/releases/download/v0.0.2/voice-de-thorsten-low.tar.gz\" #@param {type:\"string\"}\n", + "#@markdown ---\n", + "print(\"Downloading voice package...\")\n", + "if enhanced_accessibility:\n", + " playaudio(\"downloading\")\n", + "if package_url_or_id.startswith(\"1\"):\n", + " !gdown -q \"{package_url_or_id}\" -O \"voice_package.tar.gz\"\n", + "elif package_url_or_id.startswith(\"https://drive.google.com/file/d/\"):\n", + " !gdown -q \"{package_url_or_id}\" -O \"voice_package.tar.gz\" --fuzzy\n", + "else:\n", + " !wget -q \"{package_url_or_id}\" -O \"voice_package.tar.gz\"\n", + "if os.path.exists(\"/content/piper/src/python/voice_package.tar.gz\"):\n", + " !tar -xf voice_package.tar.gz\n", + " print(\"Voice package downloaded!\")\n", + " if enhanced_accessibility:\n", + " playaudio(\"downloaded\")\n", + "else:\n", + " if enhanced_accessibility:\n", + " playaudio(\"dwnerror\")\n", + " raise Exception(lan.translate(lang, \"Model failed to download!\"))" + ], + "metadata": { + "cellView": "form", + "id": "ykIYmVXccg6s" + }, + "execution_count": null, + "outputs": [] + }, + { + "cell_type": "markdown", + "source": [ + "# inferencing" + ], + "metadata": { + "id": "MRvkYJF6g5FT" + } + }, + { + "cell_type": "code", + "source": [ + "#@title run inference\n", + "#@markdown #### before you enjoy... Some notes!\n", + "#@markdown * You can run the cell to download voice packs and download voices you want at any time, even if you run this cell!\n", + "#@markdown * When you download a new voice, run this cell again and you will now be able to toggle between all the ones you download. Incredible, right?\n", + "\n", + "#@markdown Enjoy!!\n", + "\n", + "%cd /content/piper/src/python\n", + "# original: infer_onnx.py\n", + "import json\n", + "import logging\n", + "import math\n", + "import sys\n", + "from pathlib import Path\n", + "\n", + "import numpy as np\n", + "import onnxruntime\n", + "from piper_train.vits.utils import audio_float_to_int16\n", + "import glob\n", + "import ipywidgets as widgets\n", + "from IPython.display import display, Audio, Markdown, clear_output\n", + "from espeak_phonemizer import Phonemizer\n", + "from piper_train import phonemize\n", + "\n", + "_LOGGER = logging.getLogger(\"piper_train.infer_onnx\")\n", + "\n", + "def detect_onnx_models(path):\n", + " onnx_models = glob.glob(path + '/*.onnx')\n", + " if len(onnx_models) > 1:\n", + " return onnx_models\n", + " elif len(onnx_models) == 1:\n", + " return onnx_models[0]\n", + " else:\n", + " return None\n", + "\n", + "\n", + "def main():\n", + " \"\"\"Main entry point\"\"\"\n", + " models_path = \"/content/piper/src/python\"\n", + " logging.basicConfig(level=logging.DEBUG)\n", + " sess_options = onnxruntime.SessionOptions()\n", + " onnx_models = detect_onnx_models(models_path)\n", + " speaker_selection = widgets.Dropdown(\n", + " options=[],\n", + " description=f'{lan.translate(lang, \"Select speaker\")}:',\n", + " layout={'visibility': 'hidden'}\n", + " )\n", + " display(speaker_selection)\n", + " if onnx_models is None:\n", + " if enhanced_accessibility:\n", + " playaudio(\"novoices\")\n", + " raise Exception(lan.translate(lang, \"No downloaded voice packages!\"))\n", + " elif isinstance(onnx_models, str):\n", + " onnx_model = onnx_models\n", + " model, config = load_onnx(onnx_model, sess_options)\n", + " if config[\"num_speakers\"] > 1:\n", + " speaker_selection.options = config[\"speaker_id_map\"].values()\n", + " speaker_selection.layout.visibility = 'visible'\n", + " if enhanced_accessibility:\n", + " playaudio(\"multispeaker\")\n", + " else:\n", + " speaker_selection.layout.visibility = 'hidden'\n", + " else:\n", + " voice_model_names = []\n", + " for current in onnx_models:\n", + " voice_struct = current.split(\"/\")[5]\n", + " voice_model_names.append(voice_struct)\n", + "\n", + " selection = widgets.Dropdown(\n", + " options=voice_model_names,\n", + " description=f'{lan.translate(lang, \"Select voice package\")}:',\n", + " )\n", + " load_btn = widgets.Button(\n", + " description=lan.translate(lang, \"Load it!\")\n", + " )\n", + " config = None\n", + " def load_model(button):\n", + " nonlocal config \n", + " global onnx_model\n", + " nonlocal model\n", + " nonlocal models_path\n", + " selected_voice = selection.value\n", + " onnx_model = f\"{models_path}/{selected_voice}\"\n", + " model, config = load_onnx(onnx_model, sess_options)\n", + " if config[\"num_speakers\"] > 1:\n", + " speaker_selection.options = config[\"speaker_id_map\"].values()\n", + " speaker_selection.layout.visibility = 'visible'\n", + " if enhanced_accessibility:\n", + " playaudio(\"multispeaker\")\n", + " else:\n", + " speaker_selection.layout.visibility = 'hidden'\n", + "\n", + " load_btn.on_click(load_model)\n", + " display(selection, load_btn)\n", + " speed_slider = widgets.FloatSlider(\n", + " value=1,\n", + " min=0.25,\n", + " max=4,\n", + " step=0.1,\n", + " description=lan.translate(lang, \"Rate scale\"),\n", + " orientation='horizontal',\n", + " )\n", + " noise_scale_slider = widgets.FloatSlider(\n", + " value=0.667,\n", + " min=0.25,\n", + " max=4,\n", + " step=0.1,\n", + " description=lan.translate(lang, \"Phoneme noise scale\"),\n", + " orientation='horizontal',\n", + " )\n", + " noise_scale_w_slider = widgets.FloatSlider(\n", + " value=1,\n", + " min=0.25,\n", + " max=4,\n", + " step=0.1,\n", + " description=lan.translate(lang, \"Phoneme stressing scale\"),\n", + " orientation='horizontal',\n", + " )\n", + " play = widgets.Checkbox(\n", + " value=True,\n", + " description=lan.translate(lang, \"Auto-play\"),\n", + " disabled=False\n", + " )\n", + " text_input = widgets.Text(\n", + " value='',\n", + " placeholder=f'{lan.translate(lang, \"Enter your text here\")}:',\n", + " description=lan.translate(lang, \"Text to synthesize\"),\n", + " layout=widgets.Layout(width='80%')\n", + " )\n", + " synthesize_button = widgets.Button(\n", + " description=lan.translate(lang, \"Synthesize\"),\n", + " button_style='success', # 'success', 'info', 'warning', 'danger' or ''\n", + " tooltip=lan.translate(lang, \"Click here to synthesize the text.\"),\n", + " icon='check'\n", + " )\n", + " close_button = widgets.Button(\n", + " description=lan.translate(lang, \"Exit\"),\n", + " tooltip=lan.translate(lang, \"Closes this GUI.\"),\n", + " icon='check'\n", + " )\n", + "\n", + " def on_synthesize_button_clicked(b):\n", + " if model is None:\n", + " if enhanced_accessibility:\n", + " playaudio(\"nomodel\")\n", + " raise Exception(lan.translate(lang, \"You have not loaded any model from the list!\"))\n", + " text = text_input.value\n", + " if config[\"num_speakers\"] > 1:\n", + " sid = speaker_selection.value\n", + " else:\n", + " sid = None\n", + " rate = speed_slider.value\n", + " noise_scale = noise_scale_slider.value\n", + " noise_scale_w = noise_scale_w_slider.value\n", + " auto_play = play.value\n", + " inferencing(model, config, sid, text, rate, noise_scale, noise_scale_w, auto_play)\n", + " \n", + " def on_close_button_clicked(b):\n", + " clear_output()\n", + " if enhanced_accessibility:\n", + " playaudio(\"exit\")\n", + " \n", + " synthesize_button.on_click(on_synthesize_button_clicked)\n", + " close_button.on_click(on_close_button_clicked)\n", + " display(text_input)\n", + " display(speed_slider)\n", + " display(noise_scale_slider)\n", + " display(noise_scale_w_slider)\n", + " display(play)\n", + " display(synthesize_button)\n", + " display(close_button)\n", + "\n", + "def load_onnx(model, sess_options):\n", + " _LOGGER.debug(\"Loading model from %s\", model)\n", + " config = load_config(model)\n", + " model = onnxruntime.InferenceSession(str(model), sess_options=sess_options)\n", + " _LOGGER.info(\"Loaded model from %s\", model)\n", + " return model, config\n", + "\n", + "def load_config(model):\n", + " with open(f\"{model}.json\", \"r\") as file:\n", + " config = json.load(file)\n", + " return config\n", + "\n", + "def inferencing(model, config, sid, line, length_scale = 1, noise_scale = 0.667, noise_scale_w = 0.8, auto_play=True):\n", + " espeak_voice = config[\"espeak\"][\"voice\"]\n", + " phonemizer = Phonemizer(default_voice=espeak_voice)\n", + " phonemes = phonemize.phonemize(line, phonemizer)\n", + " ids = phonemize.phonemes_to_ids(phonemes)\n", + " phoneme_ids = ids\n", + " num_speakers = config[\"num_speakers\"]\n", + " if num_speakers == 1:\n", + " speaker_id = None # for now\n", + " else:\n", + " speaker_id = sid\n", + " text = np.expand_dims(np.array(phoneme_ids, dtype=np.int64), 0)\n", + " text_lengths = np.array([text.shape[1]], dtype=np.int64)\n", + " scales = np.array(\n", + " [noise_scale, length_scale, noise_scale_w],\n", + " dtype=np.float32,\n", + " )\n", + " sid = None\n", + " if speaker_id is not None:\n", + " sid = np.array([speaker_id], dtype=np.int64)\n", + " audio = model.run(\n", + " None,\n", + " {\n", + " \"input\": text,\n", + " \"input_lengths\": text_lengths,\n", + " \"scales\": scales,\n", + " \"sid\": sid,\n", + " },\n", + " )[0].squeeze((0, 1))\n", + " audio = audio_float_to_int16(audio.squeeze())\n", + " sample_rate = config[\"audio\"][\"sample_rate\"]\n", + " display(Markdown(f\"{line}\"))\n", + " display(Audio(audio, rate=sample_rate, autoplay=auto_play))\n", + "\n", + "def denoise(\n", + " audio: np.ndarray, bias_spec: np.ndarray, denoiser_strength: float\n", + ") -> np.ndarray:\n", + " audio_spec, audio_angles = transform(audio)\n", + "\n", + " a = bias_spec.shape[-1]\n", + " b = audio_spec.shape[-1]\n", + " repeats = max(1, math.ceil(b / a))\n", + " bias_spec_repeat = np.repeat(bias_spec, repeats, axis=-1)[..., :b]\n", + "\n", + " audio_spec_denoised = audio_spec - (bias_spec_repeat * denoiser_strength)\n", + " audio_spec_denoised = np.clip(audio_spec_denoised, a_min=0.0, a_max=None)\n", + " audio_denoised = inverse(audio_spec_denoised, audio_angles)\n", + "\n", + " return audio_denoised\n", + "\n", + "\n", + "def stft(x, fft_size, hopsamp):\n", + " \"\"\"Compute and return the STFT of the supplied time domain signal x.\n", + " Args:\n", + " x (1-dim Numpy array): A time domain signal.\n", + " fft_size (int): FFT size. Should be a power of 2, otherwise DFT will be used.\n", + " hopsamp (int):\n", + " Returns:\n", + " The STFT. The rows are the time slices and columns are the frequency bins.\n", + " \"\"\"\n", + " window = np.hanning(fft_size)\n", + " fft_size = int(fft_size)\n", + " hopsamp = int(hopsamp)\n", + " return np.array(\n", + " [\n", + " np.fft.rfft(window * x[i : i + fft_size])\n", + " for i in range(0, len(x) - fft_size, hopsamp)\n", + " ]\n", + " )\n", + "\n", + "\n", + "def istft(X, fft_size, hopsamp):\n", + " \"\"\"Invert a STFT into a time domain signal.\n", + " Args:\n", + " X (2-dim Numpy array): Input spectrogram. The rows are the time slices and columns are the frequency bins.\n", + " fft_size (int):\n", + " hopsamp (int): The hop size, in samples.\n", + " Returns:\n", + " The inverse STFT.\n", + " \"\"\"\n", + " fft_size = int(fft_size)\n", + " hopsamp = int(hopsamp)\n", + " window = np.hanning(fft_size)\n", + " time_slices = X.shape[0]\n", + " len_samples = int(time_slices * hopsamp + fft_size)\n", + " x = np.zeros(len_samples)\n", + " for n, i in enumerate(range(0, len(x) - fft_size, hopsamp)):\n", + " x[i : i + fft_size] += window * np.real(np.fft.irfft(X[n]))\n", + " return x\n", + "\n", + "\n", + "def inverse(magnitude, phase):\n", + " recombine_magnitude_phase = np.concatenate(\n", + " [magnitude * np.cos(phase), magnitude * np.sin(phase)], axis=1\n", + " )\n", + "\n", + " x_org = recombine_magnitude_phase\n", + " n_b, n_f, n_t = x_org.shape # pylint: disable=unpacking-non-sequence\n", + " x = np.empty([n_b, n_f // 2, n_t], dtype=np.complex64)\n", + " x.real = x_org[:, : n_f // 2]\n", + " x.imag = x_org[:, n_f // 2 :]\n", + " inverse_transform = []\n", + " for y in x:\n", + " y_ = istft(y.T, fft_size=1024, hopsamp=256)\n", + " inverse_transform.append(y_[None, :])\n", + "\n", + " inverse_transform = np.concatenate(inverse_transform, 0)\n", + "\n", + " return inverse_transform\n", + "\n", + "\n", + "def transform(input_data):\n", + " x = input_data\n", + " real_part = []\n", + " imag_part = []\n", + " for y in x:\n", + " y_ = stft(y, fft_size=1024, hopsamp=256).T\n", + " real_part.append(y_.real[None, :, :]) # pylint: disable=unsubscriptable-object\n", + " imag_part.append(y_.imag[None, :, :]) # pylint: disable=unsubscriptable-object\n", + " real_part = np.concatenate(real_part, 0)\n", + " imag_part = np.concatenate(imag_part, 0)\n", + "\n", + " magnitude = np.sqrt(real_part**2 + imag_part**2)\n", + " phase = np.arctan2(imag_part.data, real_part.data)\n", + "\n", + " return magnitude, phase\n", + "\n", + "main()" + ], + "metadata": { + "id": "hcKk8M2ug8kM", + "cellView": "form" + }, + "execution_count": null, + "outputs": [] + } + ] +} \ No newline at end of file