2019-04-03 15:40:03 +00:00
|
|
|
"""Support for the Microsoft Cognitive Services text-to-speech service."""
|
2017-10-25 17:43:21 +00:00
|
|
|
from http.client import HTTPException
|
2019-03-21 05:56:46 +00:00
|
|
|
import logging
|
2017-10-25 17:43:21 +00:00
|
|
|
|
2019-12-04 10:16:35 +00:00
|
|
|
from pycsspeechtts import pycsspeechtts
|
2017-10-25 17:43:21 +00:00
|
|
|
import voluptuous as vol
|
|
|
|
|
2019-03-29 18:45:02 +00:00
|
|
|
from homeassistant.components.tts import CONF_LANG, PLATFORM_SCHEMA, Provider
|
2021-02-11 09:25:43 +00:00
|
|
|
from homeassistant.const import CONF_API_KEY, CONF_REGION, CONF_TYPE, PERCENTAGE
|
2017-10-25 17:43:21 +00:00
|
|
|
import homeassistant.helpers.config_validation as cv
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
CONF_GENDER = "gender"
|
|
|
|
CONF_OUTPUT = "output"
|
|
|
|
CONF_RATE = "rate"
|
|
|
|
CONF_VOLUME = "volume"
|
|
|
|
CONF_PITCH = "pitch"
|
|
|
|
CONF_CONTOUR = "contour"
|
2017-10-25 17:43:21 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
SUPPORTED_LANGUAGES = [
|
2019-07-31 19:25:30 +00:00
|
|
|
"ar-eg",
|
|
|
|
"ar-sa",
|
2021-06-14 18:10:21 +00:00
|
|
|
"bg-bg",
|
2019-07-31 19:25:30 +00:00
|
|
|
"ca-es",
|
|
|
|
"cs-cz",
|
2021-06-14 18:10:21 +00:00
|
|
|
"cy-gb",
|
2019-07-31 19:25:30 +00:00
|
|
|
"da-dk",
|
|
|
|
"de-at",
|
|
|
|
"de-ch",
|
|
|
|
"de-de",
|
|
|
|
"el-gr",
|
|
|
|
"en-au",
|
|
|
|
"en-ca",
|
|
|
|
"en-gb",
|
2021-06-14 18:10:21 +00:00
|
|
|
"en-hk",
|
2019-07-31 19:25:30 +00:00
|
|
|
"en-ie",
|
|
|
|
"en-in",
|
2021-06-14 18:10:21 +00:00
|
|
|
"en-nz",
|
|
|
|
"en-ph",
|
|
|
|
"en-sg",
|
2019-07-31 19:25:30 +00:00
|
|
|
"en-us",
|
2021-06-14 18:10:21 +00:00
|
|
|
"en-za",
|
|
|
|
"es-ar",
|
|
|
|
"es-co",
|
2019-07-31 19:25:30 +00:00
|
|
|
"es-es",
|
|
|
|
"es-mx",
|
2021-06-14 18:10:21 +00:00
|
|
|
"es-us",
|
|
|
|
"et-ee",
|
2019-07-31 19:25:30 +00:00
|
|
|
"fi-fi",
|
2021-06-14 18:10:21 +00:00
|
|
|
"fr-be",
|
2019-07-31 19:25:30 +00:00
|
|
|
"fr-ca",
|
|
|
|
"fr-ch",
|
|
|
|
"fr-fr",
|
2021-06-14 18:10:21 +00:00
|
|
|
"ga-ie",
|
|
|
|
"gu-in",
|
2019-07-31 19:25:30 +00:00
|
|
|
"he-il",
|
|
|
|
"hi-in",
|
2021-06-14 18:10:21 +00:00
|
|
|
"hr-hr",
|
2019-07-31 19:25:30 +00:00
|
|
|
"hu-hu",
|
|
|
|
"id-id",
|
2022-02-11 11:53:09 +00:00
|
|
|
"is-is",
|
2019-07-31 19:25:30 +00:00
|
|
|
"it-it",
|
|
|
|
"ja-jp",
|
|
|
|
"ko-kr",
|
2021-06-14 18:10:21 +00:00
|
|
|
"lt-lt",
|
|
|
|
"lv-lv",
|
|
|
|
"mr-in",
|
|
|
|
"ms-my",
|
|
|
|
"mt-mt",
|
2019-07-31 19:25:30 +00:00
|
|
|
"nb-no",
|
2021-06-14 18:10:21 +00:00
|
|
|
"nl-be",
|
2019-07-31 19:25:30 +00:00
|
|
|
"nl-nl",
|
|
|
|
"pl-pl",
|
|
|
|
"pt-br",
|
|
|
|
"pt-pt",
|
|
|
|
"ro-ro",
|
|
|
|
"ru-ru",
|
|
|
|
"sk-sk",
|
2021-02-18 16:16:45 +00:00
|
|
|
"sl-si",
|
2019-07-31 19:25:30 +00:00
|
|
|
"sv-se",
|
2021-06-14 18:10:21 +00:00
|
|
|
"sw-ke",
|
|
|
|
"ta-in",
|
|
|
|
"te-in",
|
2019-07-31 19:25:30 +00:00
|
|
|
"th-th",
|
|
|
|
"tr-tr",
|
2021-06-14 18:10:21 +00:00
|
|
|
"uk-ua",
|
|
|
|
"ur-pk",
|
|
|
|
"vi-vn",
|
2019-07-31 19:25:30 +00:00
|
|
|
"zh-cn",
|
|
|
|
"zh-hk",
|
|
|
|
"zh-tw",
|
2017-10-25 17:43:21 +00:00
|
|
|
]
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
GENDERS = ["Female", "Male"]
|
2017-10-25 17:43:21 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
DEFAULT_LANG = "en-us"
|
|
|
|
DEFAULT_GENDER = "Female"
|
2021-10-27 07:54:12 +00:00
|
|
|
DEFAULT_TYPE = "JennyNeural"
|
2019-07-31 19:25:30 +00:00
|
|
|
DEFAULT_OUTPUT = "audio-16khz-128kbitrate-mono-mp3"
|
2017-12-01 11:38:20 +00:00
|
|
|
DEFAULT_RATE = 0
|
|
|
|
DEFAULT_VOLUME = 0
|
|
|
|
DEFAULT_PITCH = "default"
|
|
|
|
DEFAULT_CONTOUR = ""
|
2019-10-25 17:42:23 +00:00
|
|
|
DEFAULT_REGION = "eastus"
|
2017-10-25 17:43:21 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend(
|
|
|
|
{
|
|
|
|
vol.Required(CONF_API_KEY): cv.string,
|
|
|
|
vol.Optional(CONF_LANG, default=DEFAULT_LANG): vol.In(SUPPORTED_LANGUAGES),
|
|
|
|
vol.Optional(CONF_GENDER, default=DEFAULT_GENDER): vol.In(GENDERS),
|
|
|
|
vol.Optional(CONF_TYPE, default=DEFAULT_TYPE): cv.string,
|
|
|
|
vol.Optional(CONF_RATE, default=DEFAULT_RATE): vol.All(
|
|
|
|
vol.Coerce(int), vol.Range(-100, 100)
|
|
|
|
),
|
|
|
|
vol.Optional(CONF_VOLUME, default=DEFAULT_VOLUME): vol.All(
|
|
|
|
vol.Coerce(int), vol.Range(-100, 100)
|
|
|
|
),
|
|
|
|
vol.Optional(CONF_PITCH, default=DEFAULT_PITCH): cv.string,
|
|
|
|
vol.Optional(CONF_CONTOUR, default=DEFAULT_CONTOUR): cv.string,
|
2019-10-25 17:42:23 +00:00
|
|
|
vol.Optional(CONF_REGION, default=DEFAULT_REGION): cv.string,
|
2019-07-31 19:25:30 +00:00
|
|
|
}
|
|
|
|
)
|
2017-10-25 17:43:21 +00:00
|
|
|
|
|
|
|
|
2019-11-05 21:39:15 +00:00
|
|
|
def get_engine(hass, config, discovery_info=None):
|
2017-10-25 17:43:21 +00:00
|
|
|
"""Set up Microsoft speech component."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return MicrosoftProvider(
|
|
|
|
config[CONF_API_KEY],
|
|
|
|
config[CONF_LANG],
|
|
|
|
config[CONF_GENDER],
|
|
|
|
config[CONF_TYPE],
|
|
|
|
config[CONF_RATE],
|
|
|
|
config[CONF_VOLUME],
|
|
|
|
config[CONF_PITCH],
|
|
|
|
config[CONF_CONTOUR],
|
2019-10-25 17:42:23 +00:00
|
|
|
config[CONF_REGION],
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2017-10-25 17:43:21 +00:00
|
|
|
|
|
|
|
|
|
|
|
class MicrosoftProvider(Provider):
|
|
|
|
"""The Microsoft speech API provider."""
|
|
|
|
|
2019-10-25 17:42:23 +00:00
|
|
|
def __init__(
|
|
|
|
self, apikey, lang, gender, ttype, rate, volume, pitch, contour, region
|
|
|
|
):
|
2017-10-25 17:43:21 +00:00
|
|
|
"""Init Microsoft TTS service."""
|
|
|
|
self._apikey = apikey
|
|
|
|
self._lang = lang
|
|
|
|
self._gender = gender
|
|
|
|
self._type = ttype
|
|
|
|
self._output = DEFAULT_OUTPUT
|
2020-09-05 19:09:14 +00:00
|
|
|
self._rate = f"{rate}{PERCENTAGE}"
|
|
|
|
self._volume = f"{volume}{PERCENTAGE}"
|
2017-12-01 11:38:20 +00:00
|
|
|
self._pitch = pitch
|
|
|
|
self._contour = contour
|
2019-10-25 17:42:23 +00:00
|
|
|
self._region = region
|
2019-07-31 19:25:30 +00:00
|
|
|
self.name = "Microsoft"
|
2017-10-25 17:43:21 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def default_language(self):
|
|
|
|
"""Return the default language."""
|
|
|
|
return self._lang
|
|
|
|
|
|
|
|
@property
|
|
|
|
def supported_languages(self):
|
|
|
|
"""Return list of supported languages."""
|
|
|
|
return SUPPORTED_LANGUAGES
|
|
|
|
|
|
|
|
def get_tts_audio(self, message, language, options=None):
|
|
|
|
"""Load TTS from Microsoft."""
|
|
|
|
if language is None:
|
|
|
|
language = self._lang
|
2019-07-31 19:25:30 +00:00
|
|
|
|
2017-10-25 17:43:21 +00:00
|
|
|
try:
|
2019-10-25 17:42:23 +00:00
|
|
|
trans = pycsspeechtts.TTSTranslator(self._apikey, self._region)
|
2019-07-31 19:25:30 +00:00
|
|
|
data = trans.speak(
|
|
|
|
language=language,
|
|
|
|
gender=self._gender,
|
|
|
|
voiceType=self._type,
|
|
|
|
output=self._output,
|
|
|
|
rate=self._rate,
|
|
|
|
volume=self._volume,
|
|
|
|
pitch=self._pitch,
|
|
|
|
contour=self._contour,
|
|
|
|
text=message,
|
|
|
|
)
|
2017-10-25 17:43:21 +00:00
|
|
|
except HTTPException as ex:
|
|
|
|
_LOGGER.error("Error occurred for Microsoft TTS: %s", ex)
|
2019-07-31 19:25:30 +00:00
|
|
|
return (None, None)
|
2017-10-25 17:43:21 +00:00
|
|
|
return ("mp3", data)
|