2019-04-03 15:40:03 +00:00
|
|
|
"""Support for the Amazon Polly text to speech service."""
|
2017-01-26 22:22:47 +00:00
|
|
|
import logging
|
2018-12-13 15:43:59 +00:00
|
|
|
|
2019-12-09 12:57:24 +00:00
|
|
|
import boto3
|
2017-01-26 22:22:47 +00:00
|
|
|
import voluptuous as vol
|
|
|
|
|
2019-03-29 18:45:02 +00:00
|
|
|
from homeassistant.components.tts import PLATFORM_SCHEMA, Provider
|
2017-01-26 22:22:47 +00:00
|
|
|
import homeassistant.helpers.config_validation as cv
|
|
|
|
|
2018-11-29 08:26:48 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
CONF_REGION = "region_name"
|
|
|
|
CONF_ACCESS_KEY_ID = "aws_access_key_id"
|
|
|
|
CONF_SECRET_ACCESS_KEY = "aws_secret_access_key"
|
|
|
|
CONF_PROFILE_NAME = "profile_name"
|
|
|
|
ATTR_CREDENTIALS = "credentials"
|
|
|
|
|
|
|
|
DEFAULT_REGION = "us-east-1"
|
|
|
|
SUPPORTED_REGIONS = [
|
|
|
|
"us-east-1",
|
|
|
|
"us-east-2",
|
|
|
|
"us-west-1",
|
|
|
|
"us-west-2",
|
|
|
|
"ca-central-1",
|
|
|
|
"eu-west-1",
|
|
|
|
"eu-central-1",
|
|
|
|
"eu-west-2",
|
|
|
|
"eu-west-3",
|
|
|
|
"ap-southeast-1",
|
|
|
|
"ap-southeast-2",
|
|
|
|
"ap-northeast-2",
|
|
|
|
"ap-northeast-1",
|
|
|
|
"ap-south-1",
|
|
|
|
"sa-east-1",
|
|
|
|
]
|
|
|
|
|
2019-10-02 08:25:35 +00:00
|
|
|
CONF_ENGINE = "engine"
|
2019-07-31 19:25:30 +00:00
|
|
|
CONF_VOICE = "voice"
|
|
|
|
CONF_OUTPUT_FORMAT = "output_format"
|
|
|
|
CONF_SAMPLE_RATE = "sample_rate"
|
|
|
|
CONF_TEXT_TYPE = "text_type"
|
2017-01-26 22:22:47 +00:00
|
|
|
|
2018-11-29 08:26:48 +00:00
|
|
|
SUPPORTED_VOICES = [
|
2019-07-31 19:25:30 +00:00
|
|
|
"Zhiyu", # Chinese
|
|
|
|
"Mads",
|
|
|
|
"Naja", # Danish
|
|
|
|
"Ruben",
|
|
|
|
"Lotte", # Dutch
|
|
|
|
"Russell",
|
2019-08-02 21:20:07 +00:00
|
|
|
"Nicole", # English Australian
|
2019-07-31 19:25:30 +00:00
|
|
|
"Brian",
|
|
|
|
"Amy",
|
|
|
|
"Emma", # English
|
|
|
|
"Aditi",
|
|
|
|
"Raveena", # English, Indian
|
|
|
|
"Joey",
|
|
|
|
"Justin",
|
|
|
|
"Matthew",
|
|
|
|
"Ivy",
|
|
|
|
"Joanna",
|
|
|
|
"Kendra",
|
|
|
|
"Kimberly",
|
|
|
|
"Salli", # English
|
|
|
|
"Geraint", # English Welsh
|
|
|
|
"Mathieu",
|
|
|
|
"Celine",
|
|
|
|
"Lea", # French
|
|
|
|
"Chantal", # French Canadian
|
|
|
|
"Hans",
|
|
|
|
"Marlene",
|
|
|
|
"Vicki", # German
|
|
|
|
"Aditi", # Hindi
|
|
|
|
"Karl",
|
|
|
|
"Dora", # Icelandic
|
|
|
|
"Giorgio",
|
|
|
|
"Carla",
|
|
|
|
"Bianca", # Italian
|
|
|
|
"Takumi",
|
|
|
|
"Mizuki", # Japanese
|
|
|
|
"Seoyeon", # Korean
|
|
|
|
"Liv", # Norwegian
|
|
|
|
"Jacek",
|
|
|
|
"Jan",
|
|
|
|
"Ewa",
|
|
|
|
"Maja", # Polish
|
|
|
|
"Ricardo",
|
|
|
|
"Vitoria", # Portuguese, Brazilian
|
|
|
|
"Cristiano",
|
|
|
|
"Ines", # Portuguese, European
|
|
|
|
"Carmen", # Romanian
|
|
|
|
"Maxim",
|
|
|
|
"Tatyana", # Russian
|
|
|
|
"Enrique",
|
|
|
|
"Conchita",
|
|
|
|
"Lucia", # Spanish European
|
|
|
|
"Mia", # Spanish Mexican
|
|
|
|
"Miguel",
|
|
|
|
"Penelope", # Spanish US
|
|
|
|
"Astrid", # Swedish
|
|
|
|
"Filiz", # Turkish
|
|
|
|
"Gwyneth", # Welsh
|
2018-11-29 08:26:48 +00:00
|
|
|
]
|
2017-01-26 22:22:47 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
SUPPORTED_OUTPUT_FORMATS = ["mp3", "ogg_vorbis", "pcm"]
|
2017-01-26 22:22:47 +00:00
|
|
|
|
2019-10-02 08:25:35 +00:00
|
|
|
SUPPORTED_ENGINES = ["neural", "standard"]
|
|
|
|
|
|
|
|
SUPPORTED_SAMPLE_RATES = ["8000", "16000", "22050", "24000"]
|
2017-01-26 22:22:47 +00:00
|
|
|
|
|
|
|
SUPPORTED_SAMPLE_RATES_MAP = {
|
2019-10-02 08:25:35 +00:00
|
|
|
"mp3": ["8000", "16000", "22050", "24000"],
|
2019-07-31 19:25:30 +00:00
|
|
|
"ogg_vorbis": ["8000", "16000", "22050"],
|
|
|
|
"pcm": ["8000", "16000"],
|
2017-01-26 22:22:47 +00:00
|
|
|
}
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
SUPPORTED_TEXT_TYPES = ["text", "ssml"]
|
|
|
|
|
|
|
|
CONTENT_TYPE_EXTENSIONS = {"audio/mpeg": "mp3", "audio/ogg": "ogg", "audio/pcm": "pcm"}
|
|
|
|
|
2019-10-02 08:25:35 +00:00
|
|
|
DEFAULT_ENGINE = "standard"
|
2019-07-31 19:25:30 +00:00
|
|
|
DEFAULT_VOICE = "Joanna"
|
|
|
|
DEFAULT_OUTPUT_FORMAT = "mp3"
|
|
|
|
DEFAULT_TEXT_TYPE = "text"
|
|
|
|
|
|
|
|
DEFAULT_SAMPLE_RATES = {"mp3": "22050", "ogg_vorbis": "22050", "pcm": "16000"}
|
|
|
|
|
|
|
|
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend(
|
|
|
|
{
|
|
|
|
vol.Optional(CONF_REGION, default=DEFAULT_REGION): vol.In(SUPPORTED_REGIONS),
|
|
|
|
vol.Inclusive(CONF_ACCESS_KEY_ID, ATTR_CREDENTIALS): cv.string,
|
|
|
|
vol.Inclusive(CONF_SECRET_ACCESS_KEY, ATTR_CREDENTIALS): cv.string,
|
|
|
|
vol.Exclusive(CONF_PROFILE_NAME, ATTR_CREDENTIALS): cv.string,
|
|
|
|
vol.Optional(CONF_VOICE, default=DEFAULT_VOICE): vol.In(SUPPORTED_VOICES),
|
2019-10-02 08:25:35 +00:00
|
|
|
vol.Optional(CONF_ENGINE, default=DEFAULT_ENGINE): vol.In(SUPPORTED_ENGINES),
|
2019-07-31 19:25:30 +00:00
|
|
|
vol.Optional(CONF_OUTPUT_FORMAT, default=DEFAULT_OUTPUT_FORMAT): vol.In(
|
|
|
|
SUPPORTED_OUTPUT_FORMATS
|
|
|
|
),
|
|
|
|
vol.Optional(CONF_SAMPLE_RATE): vol.All(
|
|
|
|
cv.string, vol.In(SUPPORTED_SAMPLE_RATES)
|
|
|
|
),
|
|
|
|
vol.Optional(CONF_TEXT_TYPE, default=DEFAULT_TEXT_TYPE): vol.In(
|
|
|
|
SUPPORTED_TEXT_TYPES
|
|
|
|
),
|
|
|
|
}
|
|
|
|
)
|
2017-01-26 22:22:47 +00:00
|
|
|
|
|
|
|
|
2019-11-05 21:39:15 +00:00
|
|
|
def get_engine(hass, config, discovery_info=None):
|
2017-05-02 16:18:47 +00:00
|
|
|
"""Set up Amazon Polly speech component."""
|
2020-04-08 10:59:38 +00:00
|
|
|
output_format = config[CONF_OUTPUT_FORMAT]
|
2019-07-31 19:25:30 +00:00
|
|
|
sample_rate = config.get(CONF_SAMPLE_RATE, DEFAULT_SAMPLE_RATES[output_format])
|
2017-01-26 22:22:47 +00:00
|
|
|
if sample_rate not in SUPPORTED_SAMPLE_RATES_MAP.get(output_format):
|
2019-07-31 19:25:30 +00:00
|
|
|
_LOGGER.error(
|
|
|
|
"%s is not a valid sample rate for %s", sample_rate, output_format
|
|
|
|
)
|
2017-01-26 22:22:47 +00:00
|
|
|
return None
|
|
|
|
|
|
|
|
config[CONF_SAMPLE_RATE] = sample_rate
|
|
|
|
|
|
|
|
profile = config.get(CONF_PROFILE_NAME)
|
|
|
|
|
|
|
|
if profile is not None:
|
|
|
|
boto3.setup_default_session(profile_name=profile)
|
|
|
|
|
|
|
|
aws_config = {
|
2020-04-08 10:59:38 +00:00
|
|
|
CONF_REGION: config[CONF_REGION],
|
2017-01-26 22:22:47 +00:00
|
|
|
CONF_ACCESS_KEY_ID: config.get(CONF_ACCESS_KEY_ID),
|
|
|
|
CONF_SECRET_ACCESS_KEY: config.get(CONF_SECRET_ACCESS_KEY),
|
|
|
|
}
|
|
|
|
|
|
|
|
del config[CONF_REGION]
|
|
|
|
del config[CONF_ACCESS_KEY_ID]
|
|
|
|
del config[CONF_SECRET_ACCESS_KEY]
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
polly_client = boto3.client("polly", **aws_config)
|
2017-01-26 22:22:47 +00:00
|
|
|
|
|
|
|
supported_languages = []
|
|
|
|
|
|
|
|
all_voices = {}
|
|
|
|
|
|
|
|
all_voices_req = polly_client.describe_voices()
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
for voice in all_voices_req.get("Voices"):
|
|
|
|
all_voices[voice.get("Id")] = voice
|
|
|
|
if voice.get("LanguageCode") not in supported_languages:
|
|
|
|
supported_languages.append(voice.get("LanguageCode"))
|
2017-01-26 22:22:47 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
return AmazonPollyProvider(polly_client, config, supported_languages, all_voices)
|
2017-01-26 22:22:47 +00:00
|
|
|
|
|
|
|
|
|
|
|
class AmazonPollyProvider(Provider):
|
|
|
|
"""Amazon Polly speech api provider."""
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
def __init__(self, polly_client, config, supported_languages, all_voices):
|
2017-01-26 22:22:47 +00:00
|
|
|
"""Initialize Amazon Polly provider for TTS."""
|
|
|
|
self.client = polly_client
|
|
|
|
self.config = config
|
|
|
|
self.supported_langs = supported_languages
|
|
|
|
self.all_voices = all_voices
|
2020-04-08 10:59:38 +00:00
|
|
|
self.default_voice = self.config[CONF_VOICE]
|
2019-07-31 19:25:30 +00:00
|
|
|
self.name = "Amazon Polly"
|
2017-01-26 22:22:47 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def supported_languages(self):
|
2017-05-02 20:47:20 +00:00
|
|
|
"""Return a list of supported languages."""
|
2017-01-26 22:22:47 +00:00
|
|
|
return self.supported_langs
|
|
|
|
|
|
|
|
@property
|
|
|
|
def default_language(self):
|
2017-05-02 20:47:20 +00:00
|
|
|
"""Return the default language."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return self.all_voices.get(self.default_voice).get("LanguageCode")
|
2017-01-26 22:22:47 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def default_options(self):
|
2017-05-02 20:47:20 +00:00
|
|
|
"""Return dict include default options."""
|
2017-01-26 22:22:47 +00:00
|
|
|
return {CONF_VOICE: self.default_voice}
|
|
|
|
|
|
|
|
@property
|
|
|
|
def supported_options(self):
|
2017-05-02 20:47:20 +00:00
|
|
|
"""Return a list of supported options."""
|
2017-01-26 22:22:47 +00:00
|
|
|
return [CONF_VOICE]
|
|
|
|
|
|
|
|
def get_tts_audio(self, message, language=None, options=None):
|
|
|
|
"""Request TTS file from Polly."""
|
|
|
|
voice_id = options.get(CONF_VOICE, self.default_voice)
|
|
|
|
voice_in_dict = self.all_voices.get(voice_id)
|
2019-07-31 19:25:30 +00:00
|
|
|
if language != voice_in_dict.get("LanguageCode"):
|
|
|
|
_LOGGER.error("%s does not support the %s language", voice_id, language)
|
2018-11-29 08:26:48 +00:00
|
|
|
return None, None
|
2017-01-26 22:22:47 +00:00
|
|
|
|
|
|
|
resp = self.client.synthesize_speech(
|
2019-10-02 08:25:35 +00:00
|
|
|
Engine=self.config[CONF_ENGINE],
|
2017-01-26 22:22:47 +00:00
|
|
|
OutputFormat=self.config[CONF_OUTPUT_FORMAT],
|
|
|
|
SampleRate=self.config[CONF_SAMPLE_RATE],
|
|
|
|
Text=message,
|
|
|
|
TextType=self.config[CONF_TEXT_TYPE],
|
2019-07-31 19:25:30 +00:00
|
|
|
VoiceId=voice_id,
|
2017-01-26 22:22:47 +00:00
|
|
|
)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
return (
|
|
|
|
CONTENT_TYPE_EXTENSIONS[resp.get("ContentType")],
|
|
|
|
resp.get("AudioStream").read(),
|
|
|
|
)
|