508 lines
16 KiB
Python
508 lines
16 KiB
Python
"""Test Google Assistant helpers."""
|
|
|
|
from datetime import timedelta
|
|
from http import HTTPStatus
|
|
from unittest.mock import Mock, call, patch
|
|
|
|
import pytest
|
|
|
|
from homeassistant.components.google_assistant import helpers
|
|
from homeassistant.components.google_assistant.const import (
|
|
EVENT_COMMAND_RECEIVED,
|
|
NOT_EXPOSE_LOCAL,
|
|
SOURCE_CLOUD,
|
|
SOURCE_LOCAL,
|
|
STORE_GOOGLE_LOCAL_WEBHOOK_ID,
|
|
)
|
|
from homeassistant.components.matter.models import MatterDeviceInfo
|
|
from homeassistant.config import async_process_ha_core_config
|
|
from homeassistant.core import HomeAssistant, State
|
|
from homeassistant.helpers import device_registry as dr, entity_registry as er
|
|
from homeassistant.setup import async_setup_component
|
|
from homeassistant.util import dt as dt_util
|
|
|
|
from . import MockConfig
|
|
|
|
from tests.common import MockConfigEntry, async_capture_events, async_mock_service
|
|
from tests.typing import ClientSessionGenerator
|
|
|
|
|
|
async def test_google_entity_sync_serialize_with_local_sdk(hass: HomeAssistant) -> None:
|
|
"""Test sync serialize attributes of a GoogleEntity."""
|
|
hass.states.async_set("light.ceiling_lights", "off")
|
|
hass.config.api = Mock(port=1234, local_ip="192.168.123.123", use_ssl=False)
|
|
await async_process_ha_core_config(
|
|
hass,
|
|
{"external_url": "https://hostname:1234"},
|
|
)
|
|
|
|
hass.http = Mock(server_port=1234)
|
|
config = MockConfig(
|
|
hass=hass,
|
|
agent_user_ids={
|
|
"mock-user-id": {
|
|
STORE_GOOGLE_LOCAL_WEBHOOK_ID: "mock-webhook-id",
|
|
},
|
|
},
|
|
)
|
|
entity = helpers.GoogleEntity(hass, config, hass.states.get("light.ceiling_lights"))
|
|
|
|
serialized = entity.sync_serialize(None, "mock-uuid")
|
|
assert "otherDeviceIds" not in serialized
|
|
assert "customData" not in serialized
|
|
|
|
config.async_enable_local_sdk()
|
|
|
|
serialized = entity.sync_serialize("mock-user-id", "abcdef")
|
|
assert serialized["otherDeviceIds"] == [{"deviceId": "light.ceiling_lights"}]
|
|
assert serialized["customData"] == {
|
|
"httpPort": 1234,
|
|
"webhookId": "mock-webhook-id",
|
|
"uuid": "abcdef",
|
|
}
|
|
|
|
for device_type in NOT_EXPOSE_LOCAL:
|
|
with patch(
|
|
"homeassistant.components.google_assistant.helpers.get_google_type",
|
|
return_value=device_type,
|
|
):
|
|
serialized = entity.sync_serialize(None, "mock-uuid")
|
|
assert "otherDeviceIds" not in serialized
|
|
assert "customData" not in serialized
|
|
|
|
|
|
async def test_google_entity_sync_serialize_with_matter(
|
|
hass: HomeAssistant,
|
|
device_registry: dr.DeviceRegistry,
|
|
entity_registry: er.EntityRegistry,
|
|
) -> None:
|
|
"""Test sync serialize attributes of a GoogleEntity that is also a Matter device."""
|
|
entry = MockConfigEntry()
|
|
entry.add_to_hass(hass)
|
|
device = device_registry.async_get_or_create(
|
|
config_entry_id=entry.entry_id,
|
|
manufacturer="Someone",
|
|
model="Some model",
|
|
sw_version="Some Version",
|
|
identifiers={("matter", "12345678")},
|
|
connections={(dr.CONNECTION_NETWORK_MAC, "12:34:56:AB:CD:EF")},
|
|
)
|
|
entity = entity_registry.async_get_or_create(
|
|
"light",
|
|
"test",
|
|
"1235",
|
|
suggested_object_id="ceiling_lights",
|
|
device_id=device.id,
|
|
)
|
|
hass.states.async_set("light.ceiling_lights", "off")
|
|
|
|
entity = helpers.GoogleEntity(
|
|
hass, MockConfig(hass=hass), hass.states.get("light.ceiling_lights")
|
|
)
|
|
|
|
serialized = entity.sync_serialize(None, "mock-uuid")
|
|
assert "matterUniqueId" not in serialized
|
|
assert "matterOriginalVendorId" not in serialized
|
|
assert "matterOriginalProductId" not in serialized
|
|
|
|
hass.config.components.add("matter")
|
|
|
|
with patch(
|
|
"homeassistant.components.matter.get_matter_device_info",
|
|
return_value=MatterDeviceInfo(
|
|
unique_id="mock-unique-id",
|
|
vendor_id="mock-vendor-id",
|
|
product_id="mock-product-id",
|
|
),
|
|
):
|
|
serialized = entity.sync_serialize("mock-user-id", "abcdef")
|
|
|
|
assert serialized["matterUniqueId"] == "mock-unique-id"
|
|
assert serialized["matterOriginalVendorId"] == "mock-vendor-id"
|
|
assert serialized["matterOriginalProductId"] == "mock-product-id"
|
|
|
|
|
|
async def test_config_local_sdk(
|
|
hass: HomeAssistant, hass_client: ClientSessionGenerator
|
|
) -> None:
|
|
"""Test the local SDK."""
|
|
command_events = async_capture_events(hass, EVENT_COMMAND_RECEIVED)
|
|
turn_on_calls = async_mock_service(hass, "light", "turn_on")
|
|
hass.states.async_set("light.ceiling_lights", "off")
|
|
|
|
assert await async_setup_component(hass, "webhook", {})
|
|
|
|
config = MockConfig(
|
|
hass=hass,
|
|
agent_user_ids={
|
|
"mock-user-id": {
|
|
STORE_GOOGLE_LOCAL_WEBHOOK_ID: "mock-webhook-id",
|
|
},
|
|
},
|
|
)
|
|
|
|
client = await hass_client()
|
|
|
|
assert config.is_local_connected is False
|
|
config.async_enable_local_sdk()
|
|
assert config.is_local_connected is False
|
|
|
|
resp = await client.post(
|
|
"/api/webhook/mock-webhook-id",
|
|
json={
|
|
"inputs": [
|
|
{
|
|
"context": {"locale_country": "US", "locale_language": "en"},
|
|
"intent": "action.devices.EXECUTE",
|
|
"payload": {
|
|
"commands": [
|
|
{
|
|
"devices": [{"id": "light.ceiling_lights"}],
|
|
"execution": [
|
|
{
|
|
"command": "action.devices.commands.OnOff",
|
|
"params": {"on": True},
|
|
}
|
|
],
|
|
}
|
|
],
|
|
"structureData": {},
|
|
},
|
|
}
|
|
],
|
|
"requestId": "mock-req-id",
|
|
},
|
|
)
|
|
|
|
assert config.is_local_connected is True
|
|
with patch(
|
|
"homeassistant.components.google_assistant.helpers.utcnow",
|
|
return_value=dt_util.utcnow() + timedelta(seconds=90),
|
|
):
|
|
assert config.is_local_connected is False
|
|
|
|
assert resp.status == HTTPStatus.OK
|
|
result = await resp.json()
|
|
assert result["requestId"] == "mock-req-id"
|
|
|
|
assert len(command_events) == 1
|
|
assert command_events[0].context.user_id == "mock-user-id"
|
|
|
|
assert len(turn_on_calls) == 1
|
|
assert turn_on_calls[0].context is command_events[0].context
|
|
|
|
config.async_disable_local_sdk()
|
|
|
|
# Webhook is no longer active
|
|
resp = await client.post("/api/webhook/mock-webhook-id")
|
|
assert resp.status == HTTPStatus.OK
|
|
assert await resp.read() == b""
|
|
|
|
|
|
async def test_config_local_sdk_if_disabled(
|
|
hass: HomeAssistant, hass_client: ClientSessionGenerator
|
|
) -> None:
|
|
"""Test the local SDK."""
|
|
assert await async_setup_component(hass, "webhook", {})
|
|
|
|
config = MockConfig(
|
|
hass=hass,
|
|
agent_user_ids={
|
|
"mock-user-id": {
|
|
STORE_GOOGLE_LOCAL_WEBHOOK_ID: "mock-webhook-id",
|
|
},
|
|
},
|
|
enabled=False,
|
|
)
|
|
assert not config.is_local_sdk_active
|
|
|
|
client = await hass_client()
|
|
|
|
config.async_enable_local_sdk()
|
|
assert config.is_local_sdk_active
|
|
|
|
resp = await client.post(
|
|
"/api/webhook/mock-webhook-id", json={"requestId": "mock-req-id"}
|
|
)
|
|
assert resp.status == HTTPStatus.OK
|
|
result = await resp.json()
|
|
assert result == {
|
|
"payload": {"errorCode": "deviceTurnedOff"},
|
|
"requestId": "mock-req-id",
|
|
}
|
|
|
|
config.async_disable_local_sdk()
|
|
assert not config.is_local_sdk_active
|
|
|
|
# Webhook is no longer active
|
|
resp = await client.post("/api/webhook/mock-webhook-id")
|
|
assert resp.status == HTTPStatus.OK
|
|
assert await resp.read() == b""
|
|
|
|
|
|
async def test_config_local_sdk_if_ssl_enabled(
|
|
hass: HomeAssistant, hass_client: ClientSessionGenerator
|
|
) -> None:
|
|
"""Test the local SDK is not enabled when SSL is enabled."""
|
|
assert await async_setup_component(hass, "webhook", {})
|
|
hass.config.api.use_ssl = True
|
|
|
|
config = MockConfig(
|
|
hass=hass,
|
|
agent_user_ids={
|
|
"mock-user-id": {
|
|
STORE_GOOGLE_LOCAL_WEBHOOK_ID: "mock-webhook-id",
|
|
},
|
|
},
|
|
enabled=False,
|
|
)
|
|
assert not config.is_local_sdk_active
|
|
|
|
client = await hass_client()
|
|
|
|
config.async_enable_local_sdk()
|
|
assert not config.is_local_sdk_active
|
|
|
|
# Webhook should not be activated
|
|
resp = await client.post("/api/webhook/mock-webhook-id")
|
|
assert resp.status == HTTPStatus.OK
|
|
assert await resp.read() == b""
|
|
|
|
|
|
async def test_agent_user_id_connect() -> None:
|
|
"""Test the connection and disconnection of users."""
|
|
config = MockConfig()
|
|
store = config._store
|
|
|
|
await config.async_connect_agent_user("agent_2")
|
|
assert store.add_agent_user_id.call_args == call("agent_2")
|
|
|
|
await config.async_connect_agent_user("agent_1")
|
|
assert store.add_agent_user_id.call_args == call("agent_1")
|
|
|
|
await config.async_disconnect_agent_user("agent_2")
|
|
assert store.pop_agent_user_id.call_args == call("agent_2")
|
|
|
|
await config.async_disconnect_agent_user("agent_1")
|
|
assert store.pop_agent_user_id.call_args == call("agent_1")
|
|
|
|
|
|
@pytest.mark.parametrize("agents", [{}, {"1"}, {"1", "2"}])
|
|
async def test_report_state_all(agents) -> None:
|
|
"""Test sync of all states."""
|
|
config = MockConfig(agent_user_ids=agents)
|
|
data = {}
|
|
with patch.object(config, "async_report_state") as mock:
|
|
await config.async_report_state_all(data)
|
|
assert sorted(mock.mock_calls) == sorted(call(data, agent) for agent in agents)
|
|
|
|
|
|
@pytest.mark.parametrize("agents", [{}, {"1"}, {"1", "2"}])
|
|
async def test_sync_entities(agents) -> None:
|
|
"""Test sync of all entities."""
|
|
config = MockConfig(agent_user_ids=agents)
|
|
with patch.object(
|
|
config, "async_sync_entities", return_value=HTTPStatus.NO_CONTENT
|
|
) as mock:
|
|
await config.async_sync_entities_all()
|
|
assert sorted(mock.mock_calls) == sorted(call(agent) for agent in agents)
|
|
|
|
|
|
@pytest.mark.parametrize("agents", [{}, {"1"}, {"1", "2"}])
|
|
async def test_sync_notifications(agents) -> None:
|
|
"""Test sync of notifications."""
|
|
config = MockConfig(agent_user_ids=agents)
|
|
with patch.object(
|
|
config, "async_sync_notification", return_value=HTTPStatus.NO_CONTENT
|
|
) as mock:
|
|
await config.async_sync_notification_all("1234", {})
|
|
assert not agents or bool(mock.mock_calls) and agents
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
("agents", "result"),
|
|
[({}, 204), ({"1": 200}, 200), ({"1": 200, "2": 300}, 300)],
|
|
)
|
|
async def test_sync_entities_all(agents, result) -> None:
|
|
"""Test sync entities ."""
|
|
config = MockConfig(agent_user_ids=set(agents.keys()))
|
|
with patch.object(
|
|
config,
|
|
"async_sync_entities",
|
|
side_effect=lambda agent_user_id: agents[agent_user_id],
|
|
) as mock:
|
|
res = await config.async_sync_entities_all()
|
|
assert sorted(mock.mock_calls) == sorted(call(agent) for agent in agents)
|
|
assert res == result
|
|
|
|
|
|
def test_supported_features_string(caplog: pytest.LogCaptureFixture) -> None:
|
|
"""Test bad supported features."""
|
|
entity = helpers.GoogleEntity(
|
|
None,
|
|
MockConfig(),
|
|
State("test.entity_id", "on", {"supported_features": "invalid"}),
|
|
)
|
|
assert entity.is_supported() is False
|
|
assert (
|
|
"Entity test.entity_id contains invalid supported_features value invalid"
|
|
in caplog.text
|
|
)
|
|
|
|
|
|
def test_request_data() -> None:
|
|
"""Test request data properties."""
|
|
config = MockConfig()
|
|
data = helpers.RequestData(
|
|
config, "test_user", SOURCE_LOCAL, "test_request_id", None
|
|
)
|
|
assert data.is_local_request is True
|
|
|
|
data = helpers.RequestData(
|
|
config, "test_user", SOURCE_CLOUD, "test_request_id", None
|
|
)
|
|
assert data.is_local_request is False
|
|
|
|
|
|
async def test_config_local_sdk_allow_min_version(
|
|
hass: HomeAssistant,
|
|
hass_client: ClientSessionGenerator,
|
|
caplog: pytest.LogCaptureFixture,
|
|
) -> None:
|
|
"""Test the local SDK."""
|
|
version = str(helpers.LOCAL_SDK_MIN_VERSION)
|
|
assert await async_setup_component(hass, "webhook", {})
|
|
|
|
config = MockConfig(
|
|
hass=hass,
|
|
agent_user_ids={
|
|
"mock-user-id": {
|
|
STORE_GOOGLE_LOCAL_WEBHOOK_ID: "mock-webhook-id",
|
|
},
|
|
},
|
|
)
|
|
|
|
client = await hass_client()
|
|
|
|
assert config._local_sdk_version_warn is False
|
|
config.async_enable_local_sdk()
|
|
|
|
await client.post(
|
|
"/api/webhook/mock-webhook-id",
|
|
headers={helpers.LOCAL_SDK_VERSION_HEADER: version},
|
|
json={
|
|
"inputs": [
|
|
{
|
|
"context": {"locale_country": "US", "locale_language": "en"},
|
|
"intent": "action.devices.SYNC",
|
|
}
|
|
],
|
|
"requestId": "mock-req-id",
|
|
},
|
|
)
|
|
assert config._local_sdk_version_warn is False
|
|
assert (
|
|
f"Local SDK version is too old ({version}), check documentation on how "
|
|
"to update to the latest version"
|
|
) not in caplog.text
|
|
|
|
|
|
@pytest.mark.parametrize("version", [None, "2.1.4"])
|
|
async def test_config_local_sdk_warn_version(
|
|
hass: HomeAssistant,
|
|
hass_client: ClientSessionGenerator,
|
|
caplog: pytest.LogCaptureFixture,
|
|
version,
|
|
) -> None:
|
|
"""Test the local SDK."""
|
|
assert await async_setup_component(hass, "webhook", {})
|
|
|
|
config = MockConfig(
|
|
hass=hass,
|
|
agent_user_ids={
|
|
"mock-user-id": {
|
|
STORE_GOOGLE_LOCAL_WEBHOOK_ID: "mock-webhook-id",
|
|
},
|
|
},
|
|
)
|
|
|
|
client = await hass_client()
|
|
|
|
assert config._local_sdk_version_warn is False
|
|
config.async_enable_local_sdk()
|
|
|
|
headers = {}
|
|
if version:
|
|
headers[helpers.LOCAL_SDK_VERSION_HEADER] = version
|
|
|
|
await client.post(
|
|
"/api/webhook/mock-webhook-id",
|
|
headers=headers,
|
|
json={
|
|
"inputs": [
|
|
{
|
|
"context": {"locale_country": "US", "locale_language": "en"},
|
|
"intent": "action.devices.SYNC",
|
|
}
|
|
],
|
|
"requestId": "mock-req-id",
|
|
},
|
|
)
|
|
assert config._local_sdk_version_warn is True
|
|
assert (
|
|
f"Local SDK version is too old ({version}), check documentation on how "
|
|
"to update to the latest version"
|
|
) in caplog.text
|
|
|
|
|
|
def test_async_get_entities_cached(hass: HomeAssistant) -> None:
|
|
"""Test async_get_entities is cached."""
|
|
config = MockConfig()
|
|
|
|
hass.states.async_set("light.ceiling_lights", "off")
|
|
hass.states.async_set("light.bed_light", "off")
|
|
hass.states.async_set("not_supported.not_supported", "off")
|
|
|
|
google_entities = helpers.async_get_entities(hass, config)
|
|
assert len(google_entities) == 2
|
|
assert config.is_supported_cache == {
|
|
"light.bed_light": (None, True),
|
|
"light.ceiling_lights": (None, True),
|
|
"not_supported.not_supported": (None, False),
|
|
}
|
|
|
|
with patch(
|
|
"homeassistant.components.google_assistant.helpers.GoogleEntity.traits",
|
|
return_value=RuntimeError("Should not be called"),
|
|
):
|
|
google_entities = helpers.async_get_entities(hass, config)
|
|
|
|
assert len(google_entities) == 2
|
|
assert config.is_supported_cache == {
|
|
"light.bed_light": (None, True),
|
|
"light.ceiling_lights": (None, True),
|
|
"not_supported.not_supported": (None, False),
|
|
}
|
|
|
|
hass.states.async_set("light.new", "on")
|
|
google_entities = helpers.async_get_entities(hass, config)
|
|
|
|
assert len(google_entities) == 3
|
|
assert config.is_supported_cache == {
|
|
"light.bed_light": (None, True),
|
|
"light.new": (None, True),
|
|
"light.ceiling_lights": (None, True),
|
|
"not_supported.not_supported": (None, False),
|
|
}
|
|
|
|
hass.states.async_set("light.new", "on", {"supported_features": 1})
|
|
google_entities = helpers.async_get_entities(hass, config)
|
|
|
|
assert len(google_entities) == 3
|
|
assert config.is_supported_cache == {
|
|
"light.bed_light": (None, True),
|
|
"light.new": (1, True),
|
|
"light.ceiling_lights": (None, True),
|
|
"not_supported.not_supported": (None, False),
|
|
}
|