core/tests/components/tts/test_init.py

911 lines
29 KiB
Python
Raw Normal View History

"""The tests for the TTS component."""
from http import HTTPStatus
2023-03-27 12:01:17 +00:00
from typing import Any
2021-01-01 21:31:56 +00:00
import pytest
import voluptuous as vol
from homeassistant.components import media_source, tts
from homeassistant.components.media_player import (
ATTR_MEDIA_ANNOUNCE,
2019-07-31 19:25:30 +00:00
ATTR_MEDIA_CONTENT_ID,
ATTR_MEDIA_CONTENT_TYPE,
DOMAIN as DOMAIN_MP,
SERVICE_PLAY_MEDIA,
MediaType,
2019-07-31 19:25:30 +00:00
)
from homeassistant.config import async_process_ha_core_config
from homeassistant.core import HomeAssistant
from homeassistant.exceptions import HomeAssistantError
2023-03-27 12:01:17 +00:00
from homeassistant.helpers.typing import ConfigType, DiscoveryInfoType
from homeassistant.setup import async_setup_component
from homeassistant.util.network import normalize_url
2023-03-27 12:01:17 +00:00
from tests.common import (
MockModule,
assert_setup_component,
async_mock_service,
mock_integration,
mock_platform,
)
from tests.typing import ClientSessionGenerator
2022-02-14 16:54:12 +00:00
ORIG_WRITE_TAGS = tts.SpeechManager.write_tags
async def get_media_source_url(hass, media_content_id):
"""Get the media source url."""
if media_source.DOMAIN not in hass.config.components:
assert await async_setup_component(hass, media_source.DOMAIN, {})
resolved = await media_source.async_resolve_media(hass, media_content_id, None)
return resolved.url
2023-03-27 12:01:17 +00:00
SUPPORT_LANGUAGES = ["de", "en", "en_US"]
DEFAULT_LANG = "en"
class MockProvider(tts.Provider):
"""Test speech API provider."""
def __init__(self, lang: str) -> None:
"""Initialize test provider."""
self._lang = lang
self.name = "Test"
@property
def default_language(self) -> str:
"""Return the default language."""
return self._lang
@property
def supported_languages(self) -> list[str]:
"""Return list of supported languages."""
return SUPPORT_LANGUAGES
@property
def supported_options(self) -> list[str]:
"""Return list of supported options like voice, emotions."""
return ["voice", "age"]
def get_tts_audio(
self, message: str, language: str, options: dict[str, Any] | None = None
) -> tts.TtsAudioType:
"""Load TTS dat."""
return ("mp3", b"")
class MockTTS:
"""A mock TTS platform."""
PLATFORM_SCHEMA = tts.PLATFORM_SCHEMA.extend(
{vol.Optional(tts.CONF_LANG, default=DEFAULT_LANG): vol.In(SUPPORT_LANGUAGES)}
)
def __init__(self, provider=None) -> None:
"""Initialize."""
if provider is None:
provider = MockProvider
self._provider = provider
async def async_get_engine(
self,
hass: HomeAssistant,
config: ConfigType,
discovery_info: DiscoveryInfoType | None = None,
) -> tts.Provider:
"""Set up a mock speech component."""
return self._provider(config.get(tts.CONF_LANG, DEFAULT_LANG))
@pytest.fixture
2023-03-27 12:01:17 +00:00
def test_provider():
"""Test TTS provider."""
return MockProvider("en")
@pytest.fixture(autouse=True)
async def internal_url_mock(hass):
"""Mock internal URL of the instance."""
await async_process_ha_core_config(
2020-08-27 11:56:20 +00:00
hass,
{"internal_url": "http://example.local:8123"},
)
@pytest.fixture
2023-03-27 12:01:17 +00:00
async def mock_tts(hass):
"""Mock TTS."""
2023-03-27 12:01:17 +00:00
mock_integration(hass, MockModule(domain="test"))
mock_platform(hass, "test.tts", MockTTS())
@pytest.fixture
async def setup_tts(hass, mock_tts):
"""Mock TTS."""
assert await async_setup_component(hass, tts.DOMAIN, {"tts": {"platform": "test"}})
2023-03-27 12:01:17 +00:00
async def test_setup_component(hass: HomeAssistant, setup_tts) -> None:
"""Set up a TTS platform with defaults."""
assert hass.services.has_service(tts.DOMAIN, "test_say")
assert hass.services.has_service(tts.DOMAIN, "clear_cache")
2023-03-27 12:01:17 +00:00
assert f"{tts.DOMAIN}.test" in hass.config.components
2023-03-27 12:01:17 +00:00
async def test_setup_component_no_access_cache_folder(
hass: HomeAssistant, mock_init_cache_dir, mock_tts
) -> None:
2023-03-27 12:01:17 +00:00
"""Set up a TTS platform with defaults."""
config = {tts.DOMAIN: {"platform": "test"}}
mock_init_cache_dir.side_effect = OSError(2, "No access")
assert not await async_setup_component(hass, tts.DOMAIN, config)
2023-03-27 12:01:17 +00:00
assert not hass.services.has_service(tts.DOMAIN, "test_say")
assert not hass.services.has_service(tts.DOMAIN, "clear_cache")
async def test_setup_component_and_test_service(
2023-03-27 12:01:17 +00:00
hass: HomeAssistant, empty_cache_dir, mock_tts
) -> None:
2023-03-27 12:01:17 +00:00
"""Set up a TTS platform and call service."""
calls = async_mock_service(hass, DOMAIN_MP, SERVICE_PLAY_MEDIA)
2023-03-27 12:01:17 +00:00
config = {tts.DOMAIN: {"platform": "test"}}
with assert_setup_component(1, tts.DOMAIN):
assert await async_setup_component(hass, tts.DOMAIN, config)
await hass.services.async_call(
tts.DOMAIN,
2023-03-27 12:01:17 +00:00
"test_say",
{
"entity_id": "media_player.something",
tts.ATTR_MESSAGE: "There is someone at the door.",
},
blocking=True,
)
assert len(calls) == 1
assert calls[0].data[ATTR_MEDIA_ANNOUNCE] is True
assert calls[0].data[ATTR_MEDIA_CONTENT_TYPE] == MediaType.MUSIC
assert (
await get_media_source_url(hass, calls[0].data[ATTR_MEDIA_CONTENT_ID])
2023-03-27 12:01:17 +00:00
== "/api/tts_proxy/42f18378fd4393d18c8dd11d03fa9563c1e54491_en_-_test.mp3"
)
await hass.async_block_till_done()
assert (
2023-03-27 12:01:17 +00:00
empty_cache_dir / "42f18378fd4393d18c8dd11d03fa9563c1e54491_en_-_test.mp3"
).is_file()
async def test_setup_component_and_test_service_with_config_language(
2023-03-27 12:01:17 +00:00
hass: HomeAssistant, empty_cache_dir, mock_tts
) -> None:
2023-03-27 12:01:17 +00:00
"""Set up a TTS platform and call service."""
calls = async_mock_service(hass, DOMAIN_MP, SERVICE_PLAY_MEDIA)
2023-03-27 12:01:17 +00:00
config = {tts.DOMAIN: {"platform": "test", "language": "de"}}
with assert_setup_component(1, tts.DOMAIN):
assert await async_setup_component(hass, tts.DOMAIN, config)
await hass.services.async_call(
tts.DOMAIN,
2023-03-27 12:01:17 +00:00
"test_say",
{
"entity_id": "media_player.something",
tts.ATTR_MESSAGE: "There is someone at the door.",
},
blocking=True,
)
assert len(calls) == 1
assert calls[0].data[ATTR_MEDIA_CONTENT_TYPE] == MediaType.MUSIC
assert (
await get_media_source_url(hass, calls[0].data[ATTR_MEDIA_CONTENT_ID])
2023-03-27 12:01:17 +00:00
== "/api/tts_proxy/42f18378fd4393d18c8dd11d03fa9563c1e54491_de_-_test.mp3"
)
await hass.async_block_till_done()
assert (
2023-03-27 12:01:17 +00:00
empty_cache_dir / "42f18378fd4393d18c8dd11d03fa9563c1e54491_de_-_test.mp3"
).is_file()
2020-08-25 12:23:21 +00:00
async def test_setup_component_and_test_service_with_config_language_special(
2023-03-27 12:01:17 +00:00
hass: HomeAssistant, empty_cache_dir, mock_tts
) -> None:
2023-03-27 12:01:17 +00:00
"""Set up a TTS platform and call service with extend language."""
2020-08-25 12:23:21 +00:00
calls = async_mock_service(hass, DOMAIN_MP, SERVICE_PLAY_MEDIA)
2023-03-27 12:01:17 +00:00
config = {tts.DOMAIN: {"platform": "test", "language": "en_US"}}
2020-08-25 12:23:21 +00:00
with assert_setup_component(1, tts.DOMAIN):
assert await async_setup_component(hass, tts.DOMAIN, config)
await hass.services.async_call(
tts.DOMAIN,
2023-03-27 12:01:17 +00:00
"test_say",
2020-08-25 12:23:21 +00:00
{
"entity_id": "media_player.something",
tts.ATTR_MESSAGE: "There is someone at the door.",
},
blocking=True,
)
assert len(calls) == 1
assert calls[0].data[ATTR_MEDIA_CONTENT_TYPE] == MediaType.MUSIC
2020-08-25 12:23:21 +00:00
assert (
await get_media_source_url(hass, calls[0].data[ATTR_MEDIA_CONTENT_ID])
2023-03-27 12:01:17 +00:00
== "/api/tts_proxy/42f18378fd4393d18c8dd11d03fa9563c1e54491_en-us_-_test.mp3"
2020-08-25 12:23:21 +00:00
)
await hass.async_block_till_done()
assert (
2023-03-27 12:01:17 +00:00
empty_cache_dir / "42f18378fd4393d18c8dd11d03fa9563c1e54491_en-us_-_test.mp3"
2020-08-25 12:23:21 +00:00
).is_file()
async def test_setup_component_and_test_service_with_wrong_conf_language(
2023-03-27 12:01:17 +00:00
hass: HomeAssistant, mock_tts
) -> None:
2023-03-27 12:01:17 +00:00
"""Set up a TTS platform and call service with wrong config."""
config = {tts.DOMAIN: {"platform": "test", "language": "ru"}}
with assert_setup_component(0, tts.DOMAIN):
assert await async_setup_component(hass, tts.DOMAIN, config)
async def test_setup_component_and_test_service_with_service_language(
2023-03-27 12:01:17 +00:00
hass: HomeAssistant, empty_cache_dir, mock_tts
) -> None:
2023-03-27 12:01:17 +00:00
"""Set up a TTS platform and call service."""
calls = async_mock_service(hass, DOMAIN_MP, SERVICE_PLAY_MEDIA)
2023-03-27 12:01:17 +00:00
config = {tts.DOMAIN: {"platform": "test"}}
with assert_setup_component(1, tts.DOMAIN):
assert await async_setup_component(hass, tts.DOMAIN, config)
await hass.services.async_call(
tts.DOMAIN,
2023-03-27 12:01:17 +00:00
"test_say",
{
"entity_id": "media_player.something",
tts.ATTR_MESSAGE: "There is someone at the door.",
tts.ATTR_LANGUAGE: "de",
},
blocking=True,
)
assert len(calls) == 1
assert calls[0].data[ATTR_MEDIA_CONTENT_TYPE] == MediaType.MUSIC
assert (
await get_media_source_url(hass, calls[0].data[ATTR_MEDIA_CONTENT_ID])
2023-03-27 12:01:17 +00:00
== "/api/tts_proxy/42f18378fd4393d18c8dd11d03fa9563c1e54491_de_-_test.mp3"
)
await hass.async_block_till_done()
assert (
2023-03-27 12:01:17 +00:00
empty_cache_dir / "42f18378fd4393d18c8dd11d03fa9563c1e54491_de_-_test.mp3"
).is_file()
async def test_setup_component_test_service_with_wrong_service_language(
2023-03-27 12:01:17 +00:00
hass: HomeAssistant, empty_cache_dir, mock_tts
) -> None:
2023-03-27 12:01:17 +00:00
"""Set up a TTS platform and call service."""
calls = async_mock_service(hass, DOMAIN_MP, SERVICE_PLAY_MEDIA)
2023-03-27 12:01:17 +00:00
config = {tts.DOMAIN: {"platform": "test"}}
with assert_setup_component(1, tts.DOMAIN):
assert await async_setup_component(hass, tts.DOMAIN, config)
with pytest.raises(HomeAssistantError):
await hass.services.async_call(
tts.DOMAIN,
2023-03-27 12:01:17 +00:00
"test_say",
{
"entity_id": "media_player.something",
tts.ATTR_MESSAGE: "There is someone at the door.",
tts.ATTR_LANGUAGE: "lang",
},
blocking=True,
)
assert len(calls) == 0
assert not (
2023-03-27 12:01:17 +00:00
empty_cache_dir / "42f18378fd4393d18c8dd11d03fa9563c1e54491_lang_-_test.mp3"
).is_file()
async def test_setup_component_and_test_service_with_service_options(
2023-03-27 12:01:17 +00:00
hass: HomeAssistant, empty_cache_dir, mock_tts
) -> None:
2023-03-27 12:01:17 +00:00
"""Set up a TTS platform and call service with options."""
calls = async_mock_service(hass, DOMAIN_MP, SERVICE_PLAY_MEDIA)
2023-03-27 12:01:17 +00:00
config = {tts.DOMAIN: {"platform": "test"}}
2019-07-31 19:25:30 +00:00
with assert_setup_component(1, tts.DOMAIN):
assert await async_setup_component(hass, tts.DOMAIN, config)
await hass.services.async_call(
tts.DOMAIN,
2023-03-27 12:01:17 +00:00
"test_say",
{
"entity_id": "media_player.something",
tts.ATTR_MESSAGE: "There is someone at the door.",
tts.ATTR_LANGUAGE: "de",
tts.ATTR_OPTIONS: {"voice": "alex", "age": 5},
},
blocking=True,
2019-07-31 19:25:30 +00:00
)
opt_hash = tts._hash_options({"voice": "alex", "age": 5})
assert len(calls) == 1
assert calls[0].data[ATTR_MEDIA_CONTENT_TYPE] == MediaType.MUSIC
assert (
await get_media_source_url(hass, calls[0].data[ATTR_MEDIA_CONTENT_ID])
2023-03-27 12:01:17 +00:00
== f"/api/tts_proxy/42f18378fd4393d18c8dd11d03fa9563c1e54491_de_{opt_hash}_test.mp3"
)
await hass.async_block_till_done()
assert (
empty_cache_dir
2023-03-27 12:01:17 +00:00
/ f"42f18378fd4393d18c8dd11d03fa9563c1e54491_de_{opt_hash}_test.mp3"
).is_file()
async def test_setup_component_and_test_with_service_options_def(
hass: HomeAssistant, empty_cache_dir
) -> None:
2023-03-27 12:01:17 +00:00
"""Set up a TTS platform and call service with default options."""
calls = async_mock_service(hass, DOMAIN_MP, SERVICE_PLAY_MEDIA)
2023-03-27 12:01:17 +00:00
config = {tts.DOMAIN: {"platform": "test"}}
class MockProviderWithDefaults(MockProvider):
@property
def default_options(self):
return {"voice": "alex"}
2023-03-27 12:01:17 +00:00
mock_integration(hass, MockModule(domain="test"))
mock_platform(hass, "test.tts", MockTTS(MockProviderWithDefaults))
with assert_setup_component(1, tts.DOMAIN):
assert await async_setup_component(hass, tts.DOMAIN, config)
await hass.services.async_call(
2019-07-31 19:25:30 +00:00
tts.DOMAIN,
2023-03-27 12:01:17 +00:00
"test_say",
2019-07-31 19:25:30 +00:00
{
"entity_id": "media_player.something",
tts.ATTR_MESSAGE: "There is someone at the door.",
2019-07-31 19:25:30 +00:00
tts.ATTR_LANGUAGE: "de",
},
blocking=True,
2019-07-31 19:25:30 +00:00
)
opt_hash = tts._hash_options({"voice": "alex"})
assert len(calls) == 1
assert calls[0].data[ATTR_MEDIA_CONTENT_TYPE] == MediaType.MUSIC
assert (
await get_media_source_url(hass, calls[0].data[ATTR_MEDIA_CONTENT_ID])
2023-03-27 12:01:17 +00:00
== f"/api/tts_proxy/42f18378fd4393d18c8dd11d03fa9563c1e54491_de_{opt_hash}_test.mp3"
2019-07-31 19:25:30 +00:00
)
2020-04-09 00:18:09 +00:00
await hass.async_block_till_done()
assert (
empty_cache_dir
2023-03-27 12:01:17 +00:00
/ f"42f18378fd4393d18c8dd11d03fa9563c1e54491_de_{opt_hash}_test.mp3"
2020-04-09 00:18:09 +00:00
).is_file()
async def test_setup_component_and_test_service_with_service_options_wrong(
2023-03-27 12:01:17 +00:00
hass: HomeAssistant, empty_cache_dir, mock_tts
) -> None:
2023-03-27 12:01:17 +00:00
"""Set up a TTS platform and call service with wrong options."""
calls = async_mock_service(hass, DOMAIN_MP, SERVICE_PLAY_MEDIA)
2023-03-27 12:01:17 +00:00
config = {tts.DOMAIN: {"platform": "test"}}
with assert_setup_component(1, tts.DOMAIN):
assert await async_setup_component(hass, tts.DOMAIN, config)
with pytest.raises(HomeAssistantError):
await hass.services.async_call(
tts.DOMAIN,
2023-03-27 12:01:17 +00:00
"test_say",
{
"entity_id": "media_player.something",
tts.ATTR_MESSAGE: "There is someone at the door.",
tts.ATTR_LANGUAGE: "de",
tts.ATTR_OPTIONS: {"speed": 1},
},
blocking=True,
)
opt_hash = tts._hash_options({"speed": 1})
assert len(calls) == 0
await hass.async_block_till_done()
assert not (
empty_cache_dir
2023-03-27 12:01:17 +00:00
/ f"42f18378fd4393d18c8dd11d03fa9563c1e54491_de_{opt_hash}_test.mp3"
).is_file()
async def test_setup_component_and_test_service_with_base_url_set(
2023-03-27 12:01:17 +00:00
hass: HomeAssistant, mock_tts
) -> None:
2023-03-27 12:01:17 +00:00
"""Set up a TTS platform with ``base_url`` set and call service."""
calls = async_mock_service(hass, DOMAIN_MP, SERVICE_PLAY_MEDIA)
2023-03-27 12:01:17 +00:00
config = {tts.DOMAIN: {"platform": "test", "base_url": "http://fnord"}}
with assert_setup_component(1, tts.DOMAIN):
assert await async_setup_component(hass, tts.DOMAIN, config)
await hass.services.async_call(
tts.DOMAIN,
2023-03-27 12:01:17 +00:00
"test_say",
{
"entity_id": "media_player.something",
tts.ATTR_MESSAGE: "There is someone at the door.",
},
blocking=True,
)
assert len(calls) == 1
assert calls[0].data[ATTR_MEDIA_CONTENT_TYPE] == MediaType.MUSIC
assert (
await get_media_source_url(hass, calls[0].data[ATTR_MEDIA_CONTENT_ID])
== "http://fnord"
"/api/tts_proxy/42f18378fd4393d18c8dd11d03fa9563c1e54491"
2023-03-27 12:01:17 +00:00
"_en_-_test.mp3"
)
async def test_setup_component_and_test_service_clear_cache(
2023-03-27 12:01:17 +00:00
hass: HomeAssistant, empty_cache_dir, mock_tts
) -> None:
2023-03-27 12:01:17 +00:00
"""Set up a TTS platform and call service clear cache."""
calls = async_mock_service(hass, DOMAIN_MP, SERVICE_PLAY_MEDIA)
2023-03-27 12:01:17 +00:00
config = {tts.DOMAIN: {"platform": "test"}}
with assert_setup_component(1, tts.DOMAIN):
assert await async_setup_component(hass, tts.DOMAIN, config)
await hass.services.async_call(
tts.DOMAIN,
2023-03-27 12:01:17 +00:00
"test_say",
{
"entity_id": "media_player.something",
tts.ATTR_MESSAGE: "There is someone at the door.",
},
blocking=True,
)
# To make sure the file is persisted
assert len(calls) == 1
await get_media_source_url(hass, calls[0].data[ATTR_MEDIA_CONTENT_ID])
await hass.async_block_till_done()
assert (
2023-03-27 12:01:17 +00:00
empty_cache_dir / "42f18378fd4393d18c8dd11d03fa9563c1e54491_en_-_test.mp3"
).is_file()
await hass.services.async_call(
tts.DOMAIN, tts.SERVICE_CLEAR_CACHE, {}, blocking=True
)
assert not (
2023-03-27 12:01:17 +00:00
empty_cache_dir / "42f18378fd4393d18c8dd11d03fa9563c1e54491_en_-_test.mp3"
).is_file()
async def test_setup_component_and_test_service_with_receive_voice(
2023-03-27 12:01:17 +00:00
hass: HomeAssistant, test_provider, hass_client: ClientSessionGenerator, mock_tts
) -> None:
2023-03-27 12:01:17 +00:00
"""Set up a TTS platform and call service and receive voice."""
calls = async_mock_service(hass, DOMAIN_MP, SERVICE_PLAY_MEDIA)
2023-03-27 12:01:17 +00:00
config = {tts.DOMAIN: {"platform": "test"}}
with assert_setup_component(1, tts.DOMAIN):
assert await async_setup_component(hass, tts.DOMAIN, config)
message = "There is someone at the door."
await hass.services.async_call(
tts.DOMAIN,
2023-03-27 12:01:17 +00:00
"test_say",
{
"entity_id": "media_player.something",
tts.ATTR_MESSAGE: message,
},
blocking=True,
)
assert len(calls) == 1
url = await get_media_source_url(hass, calls[0].data[ATTR_MEDIA_CONTENT_ID])
client = await hass_client()
req = await client.get(url)
2023-03-27 12:01:17 +00:00
_, tts_data = test_provider.get_tts_audio("bla", "en")
tts_data = tts.SpeechManager.write_tags(
"42f18378fd4393d18c8dd11d03fa9563c1e54491_en_-_test.mp3",
tts_data,
test_provider,
message,
"en",
None,
)
assert req.status == HTTPStatus.OK
2023-03-27 12:01:17 +00:00
assert await req.read() == tts_data
extension, data = await tts.async_get_media_source_audio(
hass, calls[0].data[ATTR_MEDIA_CONTENT_ID]
)
assert extension == "mp3"
2023-03-27 12:01:17 +00:00
assert tts_data == data
async def test_setup_component_and_test_service_with_receive_voice_german(
2023-03-27 12:01:17 +00:00
hass: HomeAssistant, test_provider, hass_client: ClientSessionGenerator, mock_tts
) -> None:
2023-03-27 12:01:17 +00:00
"""Set up a TTS platform and call service and receive voice."""
calls = async_mock_service(hass, DOMAIN_MP, SERVICE_PLAY_MEDIA)
2023-03-27 12:01:17 +00:00
config = {tts.DOMAIN: {"platform": "test", "language": "de"}}
with assert_setup_component(1, tts.DOMAIN):
assert await async_setup_component(hass, tts.DOMAIN, config)
await hass.services.async_call(
tts.DOMAIN,
2023-03-27 12:01:17 +00:00
"test_say",
{
"entity_id": "media_player.something",
tts.ATTR_MESSAGE: "There is someone at the door.",
},
blocking=True,
)
assert len(calls) == 1
url = await get_media_source_url(hass, calls[0].data[ATTR_MEDIA_CONTENT_ID])
client = await hass_client()
req = await client.get(url)
2023-03-27 12:01:17 +00:00
_, tts_data = test_provider.get_tts_audio("bla", "de")
tts_data = tts.SpeechManager.write_tags(
"42f18378fd4393d18c8dd11d03fa9563c1e54491_de_-_test.mp3",
tts_data,
test_provider,
"There is someone at the door.",
"de",
None,
)
assert req.status == HTTPStatus.OK
2023-03-27 12:01:17 +00:00
assert await req.read() == tts_data
async def test_setup_component_and_web_view_wrong_file(
2023-03-27 12:01:17 +00:00
hass: HomeAssistant, hass_client: ClientSessionGenerator, mock_tts
) -> None:
2023-03-27 12:01:17 +00:00
"""Set up a TTS platform and receive wrong file from web."""
config = {tts.DOMAIN: {"platform": "test"}}
with assert_setup_component(1, tts.DOMAIN):
assert await async_setup_component(hass, tts.DOMAIN, config)
client = await hass_client()
2023-03-27 12:01:17 +00:00
url = "/api/tts_proxy/42f18378fd4393d18c8dd11d03fa9563c1e54491_en_-_test.mp3"
req = await client.get(url)
assert req.status == HTTPStatus.NOT_FOUND
async def test_setup_component_and_web_view_wrong_filename(
2023-03-27 12:01:17 +00:00
hass: HomeAssistant, hass_client: ClientSessionGenerator, mock_tts
) -> None:
2023-03-27 12:01:17 +00:00
"""Set up a TTS platform and receive wrong filename from web."""
config = {tts.DOMAIN: {"platform": "test"}}
with assert_setup_component(1, tts.DOMAIN):
assert await async_setup_component(hass, tts.DOMAIN, config)
client = await hass_client()
2023-03-27 12:01:17 +00:00
url = "/api/tts_proxy/265944dsk32c1b2a621be5930510bb2cd_en_-_test.mp3"
req = await client.get(url)
assert req.status == HTTPStatus.NOT_FOUND
async def test_setup_component_test_without_cache(
2023-03-27 12:01:17 +00:00
hass: HomeAssistant, empty_cache_dir, mock_tts
) -> None:
2023-03-27 12:01:17 +00:00
"""Set up a TTS platform without cache."""
calls = async_mock_service(hass, DOMAIN_MP, SERVICE_PLAY_MEDIA)
2023-03-27 12:01:17 +00:00
config = {tts.DOMAIN: {"platform": "test", "cache": False}}
with assert_setup_component(1, tts.DOMAIN):
assert await async_setup_component(hass, tts.DOMAIN, config)
await hass.services.async_call(
tts.DOMAIN,
2023-03-27 12:01:17 +00:00
"test_say",
{
"entity_id": "media_player.something",
tts.ATTR_MESSAGE: "There is someone at the door.",
},
blocking=True,
)
assert len(calls) == 1
await hass.async_block_till_done()
assert not (
2023-03-27 12:01:17 +00:00
empty_cache_dir / "42f18378fd4393d18c8dd11d03fa9563c1e54491_en_-_test.mp3"
).is_file()
async def test_setup_component_test_with_cache_call_service_without_cache(
2023-03-27 12:01:17 +00:00
hass: HomeAssistant, empty_cache_dir, mock_tts
) -> None:
2023-03-27 12:01:17 +00:00
"""Set up a TTS platform with cache and call service without cache."""
calls = async_mock_service(hass, DOMAIN_MP, SERVICE_PLAY_MEDIA)
2023-03-27 12:01:17 +00:00
config = {tts.DOMAIN: {"platform": "test", "cache": True}}
with assert_setup_component(1, tts.DOMAIN):
assert await async_setup_component(hass, tts.DOMAIN, config)
await hass.services.async_call(
tts.DOMAIN,
2023-03-27 12:01:17 +00:00
"test_say",
{
"entity_id": "media_player.something",
tts.ATTR_MESSAGE: "There is someone at the door.",
tts.ATTR_CACHE: False,
},
blocking=True,
)
assert len(calls) == 1
await hass.async_block_till_done()
assert not (
2023-03-27 12:01:17 +00:00
empty_cache_dir / "42f18378fd4393d18c8dd11d03fa9563c1e54491_en_-_test.mp3"
).is_file()
async def test_setup_component_test_with_cache_dir(
2023-03-27 12:01:17 +00:00
hass: HomeAssistant, empty_cache_dir, test_provider
) -> None:
2023-03-27 12:01:17 +00:00
"""Set up a TTS platform with cache and call service without cache."""
calls = async_mock_service(hass, DOMAIN_MP, SERVICE_PLAY_MEDIA)
2023-03-27 12:01:17 +00:00
_, tts_data = test_provider.get_tts_audio("bla", "en")
cache_file = (
2023-03-27 12:01:17 +00:00
empty_cache_dir / "42f18378fd4393d18c8dd11d03fa9563c1e54491_en_-_test.mp3"
)
with open(cache_file, "wb") as voice_file:
2023-03-27 12:01:17 +00:00
voice_file.write(tts_data)
config = {tts.DOMAIN: {"platform": "test", "cache": True}}
2023-03-27 12:01:17 +00:00
class MockProviderBoom(MockProvider):
def get_tts_audio(
self, message: str, language: str, options: dict[str, Any] | None = None
) -> tts.TtsAudioType:
"""Load TTS dat."""
# This should not be called, data should be fetched from cache
raise Exception("Boom!")
mock_integration(hass, MockModule(domain="test"))
mock_platform(hass, "test.tts", MockTTS(MockProviderBoom))
with assert_setup_component(1, tts.DOMAIN):
assert await async_setup_component(hass, tts.DOMAIN, config)
2023-03-27 12:01:17 +00:00
await hass.services.async_call(
tts.DOMAIN,
"test_say",
{
"entity_id": "media_player.something",
tts.ATTR_MESSAGE: "There is someone at the door.",
},
blocking=True,
)
assert len(calls) == 1
assert (
await get_media_source_url(hass, calls[0].data[ATTR_MEDIA_CONTENT_ID])
2023-03-27 12:01:17 +00:00
== "/api/tts_proxy/42f18378fd4393d18c8dd11d03fa9563c1e54491_en_-_test.mp3"
)
async def test_setup_component_test_with_error_on_get_tts(hass: HomeAssistant) -> None:
2023-03-27 12:01:17 +00:00
"""Set up a TTS platform with wrong get_tts_audio."""
config = {tts.DOMAIN: {"platform": "test"}}
2019-07-31 19:25:30 +00:00
2023-03-27 12:01:17 +00:00
class MockProviderEmpty(MockProvider):
def get_tts_audio(
self, message: str, language: str, options: dict[str, Any] | None = None
) -> tts.TtsAudioType:
"""Load TTS dat."""
return (None, None)
mock_integration(hass, MockModule(domain="test"))
mock_platform(hass, "test.tts", MockTTS(MockProviderEmpty))
with assert_setup_component(1, tts.DOMAIN):
assert await async_setup_component(hass, tts.DOMAIN, config)
async def test_setup_component_load_cache_retrieve_without_mem_cache(
hass: HomeAssistant,
2023-03-27 12:01:17 +00:00
test_provider,
empty_cache_dir,
hass_client: ClientSessionGenerator,
2023-03-27 12:01:17 +00:00
mock_tts,
) -> None:
"""Set up component and load cache and get without mem cache."""
2023-03-27 12:01:17 +00:00
_, tts_data = test_provider.get_tts_audio("bla", "en")
cache_file = (
2023-03-27 12:01:17 +00:00
empty_cache_dir / "42f18378fd4393d18c8dd11d03fa9563c1e54491_en_-_test.mp3"
)
with open(cache_file, "wb") as voice_file:
2023-03-27 12:01:17 +00:00
voice_file.write(tts_data)
2023-03-27 12:01:17 +00:00
config = {tts.DOMAIN: {"platform": "test", "cache": True}}
with assert_setup_component(1, tts.DOMAIN):
assert await async_setup_component(hass, tts.DOMAIN, config)
client = await hass_client()
2023-03-27 12:01:17 +00:00
url = "/api/tts_proxy/42f18378fd4393d18c8dd11d03fa9563c1e54491_en_-_test.mp3"
req = await client.get(url)
assert req.status == HTTPStatus.OK
2023-03-27 12:01:17 +00:00
assert await req.read() == tts_data
async def test_setup_component_and_web_get_url(
2023-03-27 12:01:17 +00:00
hass: HomeAssistant, hass_client: ClientSessionGenerator, mock_tts
) -> None:
2023-03-27 12:01:17 +00:00
"""Set up a TTS platform and receive file from web."""
config = {tts.DOMAIN: {"platform": "test"}}
await async_setup_component(hass, tts.DOMAIN, config)
client = await hass_client()
url = "/api/tts_get_url"
2023-03-27 12:01:17 +00:00
data = {"platform": "test", "message": "There is someone at the door."}
req = await client.post(url, json=data)
assert req.status == HTTPStatus.OK
response = await req.json()
assert response == {
2023-03-27 12:01:17 +00:00
"url": "http://example.local:8123/api/tts_proxy/42f18378fd4393d18c8dd11d03fa9563c1e54491_en_-_test.mp3",
"path": "/api/tts_proxy/42f18378fd4393d18c8dd11d03fa9563c1e54491_en_-_test.mp3",
}
async def test_setup_component_and_web_get_url_bad_config(
2023-03-27 12:01:17 +00:00
hass: HomeAssistant, hass_client: ClientSessionGenerator, mock_tts
) -> None:
2023-03-27 12:01:17 +00:00
"""Set up a TTS platform and receive wrong file from web."""
config = {tts.DOMAIN: {"platform": "test"}}
await async_setup_component(hass, tts.DOMAIN, config)
client = await hass_client()
url = "/api/tts_get_url"
data = {"message": "There is someone at the door."}
req = await client.post(url, json=data)
assert req.status == HTTPStatus.BAD_REQUEST
2023-03-27 12:01:17 +00:00
async def test_tags_with_wave(hass: HomeAssistant, test_provider) -> None:
"""Set up a TTS platform and call service and receive voice."""
# below data represents an empty wav file
2023-03-27 12:01:17 +00:00
tts_data = bytes.fromhex(
"52 49 46 46 24 00 00 00 57 41 56 45 66 6d 74 20 10 00 00 00 01 00 02 00"
+ "22 56 00 00 88 58 01 00 04 00 10 00 64 61 74 61 00 00 00 00"
)
2022-02-14 16:54:12 +00:00
tagged_data = ORIG_WRITE_TAGS(
2023-03-27 12:01:17 +00:00
"42f18378fd4393d18c8dd11d03fa9563c1e54491_en_-_test.wav",
tts_data,
test_provider,
"AI person is in front of your door.",
"en",
None,
)
2023-03-27 12:01:17 +00:00
assert tagged_data != tts_data
@pytest.mark.parametrize(
"value",
(
"http://example.local:8123",
"http://example.local",
"http://example.local:80",
"https://example.com",
"https://example.com:443",
"https://example.com:8123",
),
)
def test_valid_base_url(value) -> None:
"""Test we validate base urls."""
assert tts.valid_base_url(value) == normalize_url(value)
# Test we strip trailing `/`
assert tts.valid_base_url(value + "/") == normalize_url(value)
@pytest.mark.parametrize(
"value",
(
"http://example.local:8123/sub-path",
"http://example.local/sub-path",
"https://example.com/sub-path",
"https://example.com:8123/sub-path",
"mailto:some@email",
"http:example.com",
"http:/example.com",
"http//example.com",
"example.com",
),
)
def test_invalid_base_url(value) -> None:
"""Test we catch bad base urls."""
with pytest.raises(vol.Invalid):
tts.valid_base_url(value)
@pytest.mark.parametrize(
("engine", "language", "options", "cache", "result_engine", "result_query"),
(
2023-03-27 12:01:17 +00:00
(None, None, None, None, "test", ""),
(None, "de", None, None, "test", "language=de"),
(None, "de", {"voice": "henk"}, None, "test", "language=de&voice=henk"),
(None, "de", None, True, "test", "cache=true&language=de"),
),
)
async def test_generate_media_source_id(
hass: HomeAssistant,
setup_tts,
engine,
language,
options,
cache,
result_engine,
result_query,
) -> None:
"""Test generating a media source ID."""
media_source_id = tts.generate_media_source_id(
hass, "msg", engine, language, options, cache
)
assert media_source_id.startswith("media-source://tts/")
_, _, engine_query = media_source_id.rpartition("/")
engine, _, query = engine_query.partition("?")
assert engine == result_engine
assert query.startswith("message=msg")
assert query[12:] == result_query
@pytest.mark.parametrize(
("engine", "language", "options"),
(
("not-loaded-engine", None, None),
(None, "unsupported-language", None),
(None, None, {"option": "not-supported"}),
),
)
async def test_generate_media_source_id_invalid_options(
hass: HomeAssistant, setup_tts, engine, language, options
) -> None:
"""Test generating a media source ID."""
with pytest.raises(HomeAssistantError):
tts.generate_media_source_id(hass, "msg", engine, language, options, None)