386 lines
13 KiB
Python
386 lines
13 KiB
Python
"""Test the translation helper."""
|
|
import asyncio
|
|
from os import path
|
|
import pathlib
|
|
|
|
import pytest
|
|
|
|
from homeassistant.generated import config_flows
|
|
from homeassistant.helpers import translation
|
|
from homeassistant.loader import async_get_integration
|
|
from homeassistant.setup import async_setup_component, setup_component
|
|
|
|
from tests.async_mock import Mock, patch
|
|
|
|
|
|
@pytest.fixture
|
|
def mock_config_flows():
|
|
"""Mock the config flows."""
|
|
flows = []
|
|
with patch.object(config_flows, "FLOWS", flows):
|
|
yield flows
|
|
|
|
|
|
def test_recursive_flatten():
|
|
"""Test the flatten function."""
|
|
data = {"parent1": {"child1": "data1", "child2": "data2"}, "parent2": "data3"}
|
|
|
|
flattened = translation.recursive_flatten("prefix.", data)
|
|
|
|
assert flattened == {
|
|
"prefix.parent1.child1": "data1",
|
|
"prefix.parent1.child2": "data2",
|
|
"prefix.parent2": "data3",
|
|
}
|
|
|
|
|
|
async def test_component_translation_path(hass):
|
|
"""Test the component translation file function."""
|
|
assert await async_setup_component(
|
|
hass,
|
|
"switch",
|
|
{"switch": [{"platform": "test"}, {"platform": "test_embedded"}]},
|
|
)
|
|
assert await async_setup_component(hass, "test_standalone", {"test_standalone"})
|
|
assert await async_setup_component(hass, "test_package", {"test_package"})
|
|
|
|
(
|
|
int_test,
|
|
int_test_embedded,
|
|
int_test_standalone,
|
|
int_test_package,
|
|
) = await asyncio.gather(
|
|
async_get_integration(hass, "test"),
|
|
async_get_integration(hass, "test_embedded"),
|
|
async_get_integration(hass, "test_standalone"),
|
|
async_get_integration(hass, "test_package"),
|
|
)
|
|
|
|
assert path.normpath(
|
|
translation.component_translation_path("switch.test", "en", int_test)
|
|
) == path.normpath(
|
|
hass.config.path("custom_components", "test", "translations", "switch.en.json")
|
|
)
|
|
|
|
assert path.normpath(
|
|
translation.component_translation_path(
|
|
"switch.test_embedded", "en", int_test_embedded
|
|
)
|
|
) == path.normpath(
|
|
hass.config.path(
|
|
"custom_components", "test_embedded", "translations", "switch.en.json"
|
|
)
|
|
)
|
|
|
|
assert (
|
|
translation.component_translation_path(
|
|
"test_standalone", "en", int_test_standalone
|
|
)
|
|
is None
|
|
)
|
|
|
|
assert path.normpath(
|
|
translation.component_translation_path("test_package", "en", int_test_package)
|
|
) == path.normpath(
|
|
hass.config.path("custom_components", "test_package", "translations", "en.json")
|
|
)
|
|
|
|
|
|
def test_load_translations_files(hass):
|
|
"""Test the load translation files function."""
|
|
# Test one valid and one invalid file
|
|
file1 = hass.config.path(
|
|
"custom_components", "test", "translations", "switch.en.json"
|
|
)
|
|
file2 = hass.config.path(
|
|
"custom_components", "test", "translations", "invalid.json"
|
|
)
|
|
assert translation.load_translations_files(
|
|
{"switch.test": file1, "invalid": file2}
|
|
) == {
|
|
"switch.test": {
|
|
"state": {"string1": "Value 1", "string2": "Value 2"},
|
|
"something": "else",
|
|
},
|
|
"invalid": {},
|
|
}
|
|
|
|
|
|
async def test_get_translations(hass, mock_config_flows):
|
|
"""Test the get translations helper."""
|
|
translations = await translation.async_get_translations(hass, "en", "state")
|
|
assert translations == {}
|
|
|
|
assert await async_setup_component(hass, "switch", {"switch": {"platform": "test"}})
|
|
await hass.async_block_till_done()
|
|
|
|
translations = await translation.async_get_translations(hass, "en", "state")
|
|
|
|
assert translations["component.switch.state.string1"] == "Value 1"
|
|
assert translations["component.switch.state.string2"] == "Value 2"
|
|
|
|
translations = await translation.async_get_translations(hass, "de", "state")
|
|
assert "component.switch.something" not in translations
|
|
assert translations["component.switch.state.string1"] == "German Value 1"
|
|
assert translations["component.switch.state.string2"] == "German Value 2"
|
|
|
|
# Test a partial translation
|
|
translations = await translation.async_get_translations(hass, "es", "state")
|
|
assert translations["component.switch.state.string1"] == "Spanish Value 1"
|
|
assert translations["component.switch.state.string2"] == "Value 2"
|
|
|
|
# Test that an untranslated language falls back to English.
|
|
translations = await translation.async_get_translations(
|
|
hass, "invalid-language", "state"
|
|
)
|
|
assert translations["component.switch.state.string1"] == "Value 1"
|
|
assert translations["component.switch.state.string2"] == "Value 2"
|
|
|
|
|
|
async def test_get_translations_loads_config_flows(hass, mock_config_flows):
|
|
"""Test the get translations helper loads config flow translations."""
|
|
mock_config_flows.append("component1")
|
|
integration = Mock(file_path=pathlib.Path(__file__))
|
|
integration.name = "Component 1"
|
|
|
|
with patch(
|
|
"homeassistant.helpers.translation.component_translation_path",
|
|
return_value="bla.json",
|
|
), patch(
|
|
"homeassistant.helpers.translation.load_translations_files",
|
|
return_value={"component1": {"title": "world"}},
|
|
), patch(
|
|
"homeassistant.helpers.translation.async_get_integration",
|
|
return_value=integration,
|
|
):
|
|
translations = await translation.async_get_translations(
|
|
hass, "en", "title", config_flow=True
|
|
)
|
|
translations_again = await translation.async_get_translations(
|
|
hass, "en", "title", config_flow=True
|
|
)
|
|
|
|
assert translations == translations_again
|
|
|
|
assert translations == {
|
|
"component.component1.title": "world",
|
|
}
|
|
|
|
assert "component1" not in hass.config.components
|
|
|
|
mock_config_flows.append("component2")
|
|
integration = Mock(file_path=pathlib.Path(__file__))
|
|
integration.name = "Component 2"
|
|
|
|
with patch(
|
|
"homeassistant.helpers.translation.component_translation_path",
|
|
return_value="bla.json",
|
|
), patch(
|
|
"homeassistant.helpers.translation.load_translations_files",
|
|
return_value={"component2": {"title": "world"}},
|
|
), patch(
|
|
"homeassistant.helpers.translation.async_get_integration",
|
|
return_value=integration,
|
|
):
|
|
translations = await translation.async_get_translations(
|
|
hass, "en", "title", config_flow=True
|
|
)
|
|
translations_again = await translation.async_get_translations(
|
|
hass, "en", "title", config_flow=True
|
|
)
|
|
|
|
assert translations == translations_again
|
|
|
|
assert translations == {
|
|
"component.component1.title": "world",
|
|
"component.component2.title": "world",
|
|
}
|
|
|
|
translations_all_cached = await translation.async_get_translations(
|
|
hass, "en", "title", config_flow=True
|
|
)
|
|
assert translations == translations_all_cached
|
|
|
|
assert "component1" not in hass.config.components
|
|
assert "component2" not in hass.config.components
|
|
|
|
|
|
async def test_get_translations_while_loading_components(hass):
|
|
"""Test the get translations helper loads config flow translations."""
|
|
integration = Mock(file_path=pathlib.Path(__file__))
|
|
integration.name = "Component 1"
|
|
hass.config.components.add("component1")
|
|
load_count = 0
|
|
|
|
def mock_load_translation_files(files):
|
|
"""Mock load translation files."""
|
|
nonlocal load_count
|
|
load_count += 1
|
|
# Mimic race condition by loading a component during setup
|
|
setup_component(hass, "persistent_notification", {})
|
|
return {"component1": {"title": "world"}}
|
|
|
|
with patch(
|
|
"homeassistant.helpers.translation.component_translation_path",
|
|
return_value="bla.json",
|
|
), patch(
|
|
"homeassistant.helpers.translation.load_translations_files",
|
|
mock_load_translation_files,
|
|
), patch(
|
|
"homeassistant.helpers.translation.async_get_integration",
|
|
return_value=integration,
|
|
):
|
|
tasks = [
|
|
translation.async_get_translations(hass, "en", "title") for _ in range(5)
|
|
]
|
|
all_translations = await asyncio.gather(*tasks)
|
|
|
|
assert all_translations[0] == {
|
|
"component.component1.title": "world",
|
|
}
|
|
assert load_count == 1
|
|
|
|
|
|
async def test_get_translation_categories(hass):
|
|
"""Test the get translations helper loads config flow translations."""
|
|
with patch.object(translation, "async_get_config_flows", return_value={"light"}):
|
|
translations = await translation.async_get_translations(
|
|
hass, "en", "title", None, True
|
|
)
|
|
assert "component.light.title" in translations
|
|
|
|
translations = await translation.async_get_translations(
|
|
hass, "en", "device_automation", None, True
|
|
)
|
|
assert "component.light.device_automation.action_type.turn_on" in translations
|
|
|
|
|
|
async def test_translation_merging(hass, caplog):
|
|
"""Test we merge translations of two integrations."""
|
|
hass.config.components.add("sensor.moon")
|
|
hass.config.components.add("sensor")
|
|
|
|
translations = await translation.async_get_translations(hass, "en", "state")
|
|
|
|
assert "component.sensor.state.moon__phase.first_quarter" in translations
|
|
|
|
hass.config.components.add("sensor.season")
|
|
|
|
# Patch in some bad translation data
|
|
|
|
orig_load_translations = translation.load_translations_files
|
|
|
|
def mock_load_translations_files(files):
|
|
"""Mock loading."""
|
|
result = orig_load_translations(files)
|
|
result["sensor.season"] = {"state": "bad data"}
|
|
return result
|
|
|
|
with patch(
|
|
"homeassistant.helpers.translation.load_translations_files",
|
|
side_effect=mock_load_translations_files,
|
|
):
|
|
translations = await translation.async_get_translations(hass, "en", "state")
|
|
|
|
assert "component.sensor.state.moon__phase.first_quarter" in translations
|
|
|
|
assert (
|
|
"An integration providing translations for sensor provided invalid data: bad data"
|
|
in caplog.text
|
|
)
|
|
|
|
|
|
async def test_translation_merging_loaded_apart(hass, caplog):
|
|
"""Test we merge translations of two integrations when they are not loaded at the same time."""
|
|
hass.config.components.add("sensor")
|
|
|
|
translations = await translation.async_get_translations(hass, "en", "state")
|
|
|
|
assert "component.sensor.state.moon__phase.first_quarter" not in translations
|
|
|
|
hass.config.components.add("sensor.moon")
|
|
|
|
translations = await translation.async_get_translations(hass, "en", "state")
|
|
|
|
assert "component.sensor.state.moon__phase.first_quarter" in translations
|
|
|
|
translations = await translation.async_get_translations(
|
|
hass, "en", "state", integration="sensor"
|
|
)
|
|
|
|
assert "component.sensor.state.moon__phase.first_quarter" in translations
|
|
|
|
|
|
async def test_caching(hass):
|
|
"""Test we cache data."""
|
|
hass.config.components.add("sensor")
|
|
hass.config.components.add("light")
|
|
|
|
# Patch with same method so we can count invocations
|
|
with patch(
|
|
"homeassistant.helpers.translation._merge_resources",
|
|
side_effect=translation._merge_resources,
|
|
) as mock_merge:
|
|
load1 = await translation.async_get_translations(hass, "en", "state")
|
|
assert len(mock_merge.mock_calls) == 1
|
|
|
|
load2 = await translation.async_get_translations(hass, "en", "state")
|
|
assert len(mock_merge.mock_calls) == 1
|
|
|
|
assert load1 == load2
|
|
|
|
for key in load1:
|
|
assert key.startswith("component.sensor.state.") or key.startswith(
|
|
"component.light.state."
|
|
)
|
|
|
|
load_sensor_only = await translation.async_get_translations(
|
|
hass, "en", "state", integration="sensor"
|
|
)
|
|
assert load_sensor_only
|
|
for key in load_sensor_only:
|
|
assert key.startswith("component.sensor.state.")
|
|
|
|
load_light_only = await translation.async_get_translations(
|
|
hass, "en", "state", integration="light"
|
|
)
|
|
assert load_light_only
|
|
for key in load_light_only:
|
|
assert key.startswith("component.light.state.")
|
|
|
|
hass.config.components.add("media_player")
|
|
|
|
# Patch with same method so we can count invocations
|
|
with patch(
|
|
"homeassistant.helpers.translation._build_resources",
|
|
side_effect=translation._build_resources,
|
|
) as mock_build:
|
|
load_sensor_only = await translation.async_get_translations(
|
|
hass, "en", "title", integration="sensor"
|
|
)
|
|
assert load_sensor_only
|
|
for key in load_sensor_only:
|
|
assert key == "component.sensor.title"
|
|
assert len(mock_build.mock_calls) == 0
|
|
|
|
assert await translation.async_get_translations(
|
|
hass, "en", "title", integration="sensor"
|
|
)
|
|
assert len(mock_build.mock_calls) == 0
|
|
|
|
load_light_only = await translation.async_get_translations(
|
|
hass, "en", "title", integration="media_player"
|
|
)
|
|
assert load_light_only
|
|
for key in load_light_only:
|
|
assert key == "component.media_player.title"
|
|
assert len(mock_build.mock_calls) > 1
|
|
|
|
|
|
async def test_custom_component_translations(hass):
|
|
"""Test getting translation from custom components."""
|
|
hass.config.components.add("test_standalone")
|
|
hass.config.components.add("test_embedded")
|
|
hass.config.components.add("test_package")
|
|
assert await translation.async_get_translations(hass, "en", "state") == {}
|