2019-03-11 19:21:20 +00:00
|
|
|
"""Test the cloud.iot module."""
|
2020-11-06 11:12:18 +00:00
|
|
|
from datetime import timedelta
|
2021-01-01 21:31:56 +00:00
|
|
|
from unittest.mock import AsyncMock, MagicMock, Mock, patch
|
2020-11-06 11:12:18 +00:00
|
|
|
|
|
|
|
import aiohttp
|
2019-03-11 19:21:20 +00:00
|
|
|
from aiohttp import web
|
|
|
|
import pytest
|
|
|
|
|
2019-06-29 03:43:57 +00:00
|
|
|
from homeassistant.components.cloud import DOMAIN
|
2019-11-28 13:23:59 +00:00
|
|
|
from homeassistant.components.cloud.client import CloudClient
|
2022-01-18 07:01:43 +00:00
|
|
|
from homeassistant.components.cloud.const import (
|
|
|
|
PREF_ALEXA_REPORT_STATE,
|
|
|
|
PREF_ENABLE_ALEXA,
|
|
|
|
PREF_ENABLE_GOOGLE,
|
|
|
|
)
|
2023-04-06 17:09:45 +00:00
|
|
|
from homeassistant.components.homeassistant.exposed_entities import (
|
|
|
|
DATA_EXPOSED_ENTITIES,
|
|
|
|
ExposedEntities,
|
|
|
|
)
|
2020-09-23 18:21:55 +00:00
|
|
|
from homeassistant.const import CONTENT_TYPE_JSON
|
2023-02-08 17:08:43 +00:00
|
|
|
from homeassistant.core import HomeAssistant, State
|
2023-04-06 17:09:45 +00:00
|
|
|
from homeassistant.helpers import entity_registry as er
|
2019-12-08 17:01:12 +00:00
|
|
|
from homeassistant.setup import async_setup_component
|
2020-11-06 11:12:18 +00:00
|
|
|
from homeassistant.util import dt as dt_util
|
2019-12-08 17:01:12 +00:00
|
|
|
|
|
|
|
from . import mock_cloud, mock_cloud_prefs
|
2019-03-11 19:21:20 +00:00
|
|
|
|
2020-11-06 11:12:18 +00:00
|
|
|
from tests.common import async_fire_time_changed
|
2019-12-08 17:01:12 +00:00
|
|
|
from tests.components.alexa import test_smart_home as test_alexa
|
2019-03-11 19:21:20 +00:00
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture
|
2019-06-21 09:17:21 +00:00
|
|
|
def mock_cloud_inst():
|
2019-03-11 19:21:20 +00:00
|
|
|
"""Mock cloud class."""
|
|
|
|
return MagicMock(subscription_expired=False)
|
|
|
|
|
|
|
|
|
2023-02-08 17:08:43 +00:00
|
|
|
async def test_handler_alexa(hass: HomeAssistant) -> None:
|
2019-03-11 19:21:20 +00:00
|
|
|
"""Test handler Alexa."""
|
2019-07-31 19:25:30 +00:00
|
|
|
hass.states.async_set("switch.test", "on", {"friendly_name": "Test switch"})
|
|
|
|
hass.states.async_set("switch.test2", "on", {"friendly_name": "Test switch 2"})
|
|
|
|
|
|
|
|
await mock_cloud(
|
|
|
|
hass,
|
|
|
|
{
|
|
|
|
"alexa": {
|
|
|
|
"filter": {"exclude_entities": "switch.test2"},
|
|
|
|
"entity_config": {
|
|
|
|
"switch.test": {
|
|
|
|
"name": "Config name",
|
|
|
|
"description": "Config description",
|
|
|
|
"display_categories": "LIGHT",
|
|
|
|
}
|
|
|
|
},
|
2019-03-11 19:21:20 +00:00
|
|
|
}
|
2019-07-31 19:25:30 +00:00
|
|
|
},
|
|
|
|
)
|
2019-03-11 19:21:20 +00:00
|
|
|
|
2022-01-18 07:01:43 +00:00
|
|
|
mock_cloud_prefs(hass, {PREF_ALEXA_REPORT_STATE: False})
|
2019-07-31 19:25:30 +00:00
|
|
|
cloud = hass.data["cloud"]
|
2019-03-11 19:21:20 +00:00
|
|
|
|
|
|
|
resp = await cloud.client.async_alexa_message(
|
2019-07-31 19:25:30 +00:00
|
|
|
test_alexa.get_new_request("Alexa.Discovery", "Discover")
|
|
|
|
)
|
2019-03-11 19:21:20 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
endpoints = resp["event"]["payload"]["endpoints"]
|
2019-03-11 19:21:20 +00:00
|
|
|
|
|
|
|
assert len(endpoints) == 1
|
|
|
|
device = endpoints[0]
|
|
|
|
|
2019-09-27 19:51:46 +00:00
|
|
|
assert device["description"] == "Config description via Home Assistant"
|
2019-07-31 19:25:30 +00:00
|
|
|
assert device["friendlyName"] == "Config name"
|
|
|
|
assert device["displayCategories"] == ["LIGHT"]
|
|
|
|
assert device["manufacturerName"] == "Home Assistant"
|
2019-03-11 19:21:20 +00:00
|
|
|
|
|
|
|
|
2023-02-10 15:05:01 +00:00
|
|
|
async def test_handler_alexa_disabled(hass: HomeAssistant, mock_cloud_fixture) -> None:
|
2019-03-11 19:21:20 +00:00
|
|
|
"""Test handler Alexa when user has disabled it."""
|
2019-10-13 21:16:27 +00:00
|
|
|
mock_cloud_fixture._prefs[PREF_ENABLE_ALEXA] = False
|
2019-07-31 19:25:30 +00:00
|
|
|
cloud = hass.data["cloud"]
|
2019-03-11 19:21:20 +00:00
|
|
|
|
|
|
|
resp = await cloud.client.async_alexa_message(
|
2019-07-31 19:25:30 +00:00
|
|
|
test_alexa.get_new_request("Alexa.Discovery", "Discover")
|
|
|
|
)
|
2019-03-11 19:21:20 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
assert resp["event"]["header"]["namespace"] == "Alexa"
|
|
|
|
assert resp["event"]["header"]["name"] == "ErrorResponse"
|
|
|
|
assert resp["event"]["payload"]["type"] == "BRIDGE_UNREACHABLE"
|
2019-03-11 19:21:20 +00:00
|
|
|
|
|
|
|
|
2023-02-08 17:08:43 +00:00
|
|
|
async def test_handler_google_actions(hass: HomeAssistant) -> None:
|
2019-03-11 19:21:20 +00:00
|
|
|
"""Test handler Google Actions."""
|
2019-07-31 19:25:30 +00:00
|
|
|
hass.states.async_set("switch.test", "on", {"friendly_name": "Test switch"})
|
|
|
|
hass.states.async_set("switch.test2", "on", {"friendly_name": "Test switch 2"})
|
|
|
|
hass.states.async_set("group.all_locks", "on", {"friendly_name": "Evil locks"})
|
|
|
|
|
|
|
|
await mock_cloud(
|
|
|
|
hass,
|
|
|
|
{
|
|
|
|
"google_actions": {
|
|
|
|
"filter": {"exclude_entities": "switch.test2"},
|
|
|
|
"entity_config": {
|
|
|
|
"switch.test": {
|
|
|
|
"name": "Config name",
|
|
|
|
"aliases": "Config alias",
|
|
|
|
"room": "living room",
|
|
|
|
}
|
|
|
|
},
|
2019-03-11 19:21:20 +00:00
|
|
|
}
|
2019-07-31 19:25:30 +00:00
|
|
|
},
|
|
|
|
)
|
2019-03-11 19:21:20 +00:00
|
|
|
|
|
|
|
mock_cloud_prefs(hass)
|
2019-07-31 19:25:30 +00:00
|
|
|
cloud = hass.data["cloud"]
|
2019-03-11 19:21:20 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
reqid = "5711642932632160983"
|
|
|
|
data = {"requestId": reqid, "inputs": [{"intent": "action.devices.SYNC"}]}
|
2019-03-11 19:21:20 +00:00
|
|
|
|
2020-01-06 21:00:39 +00:00
|
|
|
with patch(
|
|
|
|
"hass_nabucasa.Cloud._decode_claims",
|
|
|
|
return_value={"cognito:username": "myUserName"},
|
|
|
|
):
|
|
|
|
await cloud.client.get_google_config()
|
|
|
|
resp = await cloud.client.async_google_message(data)
|
2019-03-11 19:21:20 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
assert resp["requestId"] == reqid
|
|
|
|
payload = resp["payload"]
|
2019-03-11 19:21:20 +00:00
|
|
|
|
2020-01-06 21:00:39 +00:00
|
|
|
assert payload["agentUserId"] == "myUserName"
|
2019-03-11 19:21:20 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
devices = payload["devices"]
|
2019-03-11 19:21:20 +00:00
|
|
|
assert len(devices) == 1
|
|
|
|
|
|
|
|
device = devices[0]
|
2019-07-31 19:25:30 +00:00
|
|
|
assert device["id"] == "switch.test"
|
|
|
|
assert device["name"]["name"] == "Config name"
|
2020-02-02 22:48:13 +00:00
|
|
|
assert device["name"]["nicknames"] == ["Config name", "Config alias"]
|
2019-07-31 19:25:30 +00:00
|
|
|
assert device["type"] == "action.devices.types.SWITCH"
|
|
|
|
assert device["roomHint"] == "living room"
|
2019-03-11 19:21:20 +00:00
|
|
|
|
|
|
|
|
2022-06-23 09:41:34 +00:00
|
|
|
@pytest.mark.parametrize(
|
2023-02-15 13:09:50 +00:00
|
|
|
("intent", "response_payload"),
|
2022-06-23 09:41:34 +00:00
|
|
|
[
|
|
|
|
("action.devices.SYNC", {"agentUserId": "myUserName", "devices": []}),
|
|
|
|
("action.devices.QUERY", {"errorCode": "deviceTurnedOff"}),
|
|
|
|
],
|
|
|
|
)
|
|
|
|
async def test_handler_google_actions_disabled(
|
2023-02-10 15:05:01 +00:00
|
|
|
hass: HomeAssistant, mock_cloud_fixture, intent, response_payload
|
|
|
|
) -> None:
|
2019-03-11 19:21:20 +00:00
|
|
|
"""Test handler Google Actions when user has disabled it."""
|
2019-10-13 21:16:27 +00:00
|
|
|
mock_cloud_fixture._prefs[PREF_ENABLE_GOOGLE] = False
|
2019-03-11 19:21:20 +00:00
|
|
|
|
2021-09-01 16:54:54 +00:00
|
|
|
with patch("hass_nabucasa.Cloud.initialize"):
|
2019-07-31 19:25:30 +00:00
|
|
|
assert await async_setup_component(hass, "cloud", {})
|
2019-03-11 19:21:20 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
reqid = "5711642932632160983"
|
2022-06-23 09:41:34 +00:00
|
|
|
data = {"requestId": reqid, "inputs": [{"intent": intent}]}
|
2019-03-11 19:21:20 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
cloud = hass.data["cloud"]
|
2022-06-23 09:41:34 +00:00
|
|
|
with patch(
|
|
|
|
"hass_nabucasa.Cloud._decode_claims",
|
|
|
|
return_value={"cognito:username": "myUserName"},
|
|
|
|
):
|
|
|
|
resp = await cloud.client.async_google_message(data)
|
2019-03-11 19:21:20 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
assert resp["requestId"] == reqid
|
2022-06-23 09:41:34 +00:00
|
|
|
assert resp["payload"] == response_payload
|
2019-03-11 19:21:20 +00:00
|
|
|
|
|
|
|
|
2023-02-08 17:08:43 +00:00
|
|
|
async def test_webhook_msg(
|
|
|
|
hass: HomeAssistant, caplog: pytest.LogCaptureFixture
|
|
|
|
) -> None:
|
2019-03-11 19:21:20 +00:00
|
|
|
"""Test webhook msg."""
|
2021-09-01 16:54:54 +00:00
|
|
|
with patch("hass_nabucasa.Cloud.initialize"):
|
2019-07-31 19:25:30 +00:00
|
|
|
setup = await async_setup_component(hass, "cloud", {"cloud": {}})
|
2019-03-11 19:21:20 +00:00
|
|
|
assert setup
|
2019-07-31 19:25:30 +00:00
|
|
|
cloud = hass.data["cloud"]
|
2019-03-11 19:21:20 +00:00
|
|
|
|
|
|
|
await cloud.client.prefs.async_initialize()
|
2019-07-31 19:25:30 +00:00
|
|
|
await cloud.client.prefs.async_update(
|
|
|
|
cloudhooks={
|
2020-06-15 23:30:40 +00:00
|
|
|
"mock-webhook-id": {
|
|
|
|
"webhook_id": "mock-webhook-id",
|
|
|
|
"cloudhook_id": "mock-cloud-id",
|
|
|
|
},
|
|
|
|
"no-longere-existing": {
|
|
|
|
"webhook_id": "no-longere-existing",
|
|
|
|
"cloudhook_id": "mock-nonexisting-id",
|
|
|
|
},
|
2019-03-11 19:21:20 +00:00
|
|
|
}
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2019-03-11 19:21:20 +00:00
|
|
|
|
|
|
|
received = []
|
|
|
|
|
|
|
|
async def handler(hass, webhook_id, request):
|
|
|
|
"""Handle a webhook."""
|
|
|
|
received.append(request)
|
2019-07-31 19:25:30 +00:00
|
|
|
return web.json_response({"from": "handler"})
|
2019-03-11 19:21:20 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
hass.components.webhook.async_register("test", "Test", "mock-webhook-id", handler)
|
2019-03-11 19:21:20 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
response = await cloud.client.async_webhook_message(
|
|
|
|
{
|
|
|
|
"cloudhook_id": "mock-cloud-id",
|
|
|
|
"body": '{"hello": "world"}',
|
2020-09-23 18:21:55 +00:00
|
|
|
"headers": {"content-type": CONTENT_TYPE_JSON},
|
2019-07-31 19:25:30 +00:00
|
|
|
"method": "POST",
|
|
|
|
"query": None,
|
|
|
|
}
|
|
|
|
)
|
2019-03-11 19:21:20 +00:00
|
|
|
|
|
|
|
assert response == {
|
2019-07-31 19:25:30 +00:00
|
|
|
"status": 200,
|
|
|
|
"body": '{"from": "handler"}',
|
2020-09-23 18:21:55 +00:00
|
|
|
"headers": {"Content-Type": CONTENT_TYPE_JSON},
|
2019-03-11 19:21:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
assert len(received) == 1
|
2019-07-31 19:25:30 +00:00
|
|
|
assert await received[0].json() == {"hello": "world"}
|
2019-05-29 15:39:12 +00:00
|
|
|
|
2020-06-15 23:30:40 +00:00
|
|
|
# Non existing webhook
|
|
|
|
caplog.clear()
|
|
|
|
|
|
|
|
response = await cloud.client.async_webhook_message(
|
|
|
|
{
|
|
|
|
"cloudhook_id": "mock-nonexisting-id",
|
|
|
|
"body": '{"nonexisting": "payload"}',
|
2020-09-23 18:21:55 +00:00
|
|
|
"headers": {"content-type": CONTENT_TYPE_JSON},
|
2020-06-15 23:30:40 +00:00
|
|
|
"method": "POST",
|
|
|
|
"query": None,
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
assert response == {
|
|
|
|
"status": 200,
|
|
|
|
"body": None,
|
|
|
|
"headers": {"Content-Type": "application/octet-stream"},
|
|
|
|
}
|
|
|
|
|
|
|
|
assert (
|
|
|
|
"Received message for unregistered webhook no-longere-existing from cloud"
|
|
|
|
in caplog.text
|
|
|
|
)
|
|
|
|
assert '{"nonexisting": "payload"}' in caplog.text
|
|
|
|
|
2019-05-29 15:39:12 +00:00
|
|
|
|
2023-02-10 15:05:01 +00:00
|
|
|
async def test_google_config_expose_entity(
|
|
|
|
hass: HomeAssistant, mock_cloud_setup, mock_cloud_login
|
|
|
|
) -> None:
|
2019-05-29 15:39:12 +00:00
|
|
|
"""Test Google config exposing entity method uses latest config."""
|
2023-04-06 17:09:45 +00:00
|
|
|
entity_registry = er.async_get(hass)
|
|
|
|
|
|
|
|
# Enable exposing new entities to Google
|
|
|
|
exposed_entities: ExposedEntities = hass.data[DATA_EXPOSED_ENTITIES]
|
|
|
|
exposed_entities.async_set_expose_new_entities("cloud.google_assistant", True)
|
|
|
|
|
|
|
|
# Register a light entity
|
|
|
|
entity_entry = entity_registry.async_get_or_create(
|
|
|
|
"light", "test", "unique", suggested_object_id="kitchen"
|
|
|
|
)
|
|
|
|
|
2019-05-29 15:39:12 +00:00
|
|
|
cloud_client = hass.data[DOMAIN].client
|
2023-04-06 17:09:45 +00:00
|
|
|
state = State(entity_entry.entity_id, "on")
|
2019-11-28 13:23:59 +00:00
|
|
|
gconf = await cloud_client.get_google_config()
|
2019-05-29 15:39:12 +00:00
|
|
|
|
2023-05-02 20:08:09 +00:00
|
|
|
assert await gconf.should_expose(state)
|
2019-05-29 15:39:12 +00:00
|
|
|
|
2023-05-02 20:08:09 +00:00
|
|
|
await exposed_entities.async_expose_entity(
|
2023-04-06 17:09:45 +00:00
|
|
|
"cloud.google_assistant", entity_entry.entity_id, False
|
2019-05-29 15:39:12 +00:00
|
|
|
)
|
|
|
|
|
2023-05-02 20:08:09 +00:00
|
|
|
assert not await gconf.should_expose(state)
|
2019-05-29 15:39:12 +00:00
|
|
|
|
|
|
|
|
2023-02-10 15:05:01 +00:00
|
|
|
async def test_google_config_should_2fa(
|
|
|
|
hass: HomeAssistant, mock_cloud_setup, mock_cloud_login
|
|
|
|
) -> None:
|
2019-05-29 15:39:12 +00:00
|
|
|
"""Test Google config disabling 2FA method uses latest config."""
|
2023-04-06 17:09:45 +00:00
|
|
|
entity_registry = er.async_get(hass)
|
|
|
|
|
|
|
|
# Register a light entity
|
|
|
|
entity_entry = entity_registry.async_get_or_create(
|
|
|
|
"light", "test", "unique", suggested_object_id="kitchen"
|
|
|
|
)
|
|
|
|
|
2019-05-29 15:39:12 +00:00
|
|
|
cloud_client = hass.data[DOMAIN].client
|
2019-11-28 13:23:59 +00:00
|
|
|
gconf = await cloud_client.get_google_config()
|
2023-04-06 17:09:45 +00:00
|
|
|
state = State(entity_entry.entity_id, "on")
|
2019-05-29 15:39:12 +00:00
|
|
|
|
2019-11-28 13:23:59 +00:00
|
|
|
assert gconf.should_2fa(state)
|
2019-05-29 15:39:12 +00:00
|
|
|
|
2023-04-06 17:09:45 +00:00
|
|
|
entity_registry.async_update_entity_options(
|
|
|
|
entity_entry.entity_id, "cloud.google_assistant", {"disable_2fa": True}
|
2019-05-29 15:39:12 +00:00
|
|
|
)
|
|
|
|
|
2019-11-28 13:23:59 +00:00
|
|
|
assert not gconf.should_2fa(state)
|
|
|
|
|
|
|
|
|
2023-04-26 10:53:58 +00:00
|
|
|
async def test_set_username(hass: HomeAssistant) -> None:
|
2020-01-31 16:33:00 +00:00
|
|
|
"""Test we set username during login."""
|
2019-11-28 13:23:59 +00:00
|
|
|
prefs = MagicMock(
|
|
|
|
alexa_enabled=False,
|
|
|
|
google_enabled=False,
|
2020-04-30 20:29:50 +00:00
|
|
|
async_set_username=AsyncMock(return_value=None),
|
2019-11-28 13:23:59 +00:00
|
|
|
)
|
2023-04-26 10:53:58 +00:00
|
|
|
client = CloudClient(hass, prefs, None, {}, {})
|
2019-11-28 13:23:59 +00:00
|
|
|
client.cloud = MagicMock(is_logged_in=True, username="mock-username")
|
2023-04-26 01:40:01 +00:00
|
|
|
await client.on_cloud_connected()
|
2019-11-28 13:23:59 +00:00
|
|
|
|
|
|
|
assert len(prefs.async_set_username.mock_calls) == 1
|
|
|
|
assert prefs.async_set_username.mock_calls[0][1][0] == "mock-username"
|
2020-11-06 11:12:18 +00:00
|
|
|
|
|
|
|
|
2023-02-08 17:08:43 +00:00
|
|
|
async def test_login_recovers_bad_internet(
|
2023-04-26 10:53:58 +00:00
|
|
|
hass: HomeAssistant, caplog: pytest.LogCaptureFixture
|
2023-02-08 17:08:43 +00:00
|
|
|
) -> None:
|
2020-11-06 11:12:18 +00:00
|
|
|
"""Test Alexa can recover bad auth."""
|
|
|
|
prefs = Mock(
|
|
|
|
alexa_enabled=True,
|
|
|
|
google_enabled=False,
|
|
|
|
async_set_username=AsyncMock(return_value=None),
|
|
|
|
)
|
2023-04-26 10:53:58 +00:00
|
|
|
client = CloudClient(hass, prefs, None, {}, {})
|
2020-11-06 11:12:18 +00:00
|
|
|
client.cloud = Mock()
|
|
|
|
client._alexa_config = Mock(
|
|
|
|
async_enable_proactive_mode=Mock(side_effect=aiohttp.ClientError)
|
|
|
|
)
|
2023-04-26 01:40:01 +00:00
|
|
|
await client.on_cloud_connected()
|
2020-11-06 11:12:18 +00:00
|
|
|
assert len(client._alexa_config.async_enable_proactive_mode.mock_calls) == 1
|
|
|
|
assert "Unable to activate Alexa Report State" in caplog.text
|
|
|
|
|
|
|
|
async_fire_time_changed(hass, dt_util.utcnow() + timedelta(seconds=30))
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
|
|
assert len(client._alexa_config.async_enable_proactive_mode.mock_calls) == 2
|
2023-03-28 15:09:59 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def test_system_msg(hass: HomeAssistant) -> None:
|
|
|
|
"""Test system msg."""
|
|
|
|
with patch("hass_nabucasa.Cloud.initialize"):
|
|
|
|
setup = await async_setup_component(hass, "cloud", {"cloud": {}})
|
|
|
|
assert setup
|
|
|
|
cloud = hass.data["cloud"]
|
|
|
|
|
|
|
|
assert cloud.client.relayer_region is None
|
|
|
|
|
|
|
|
response = await cloud.client.async_system_message(
|
|
|
|
{
|
|
|
|
"region": "xx-earth-616",
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
assert response is None
|
|
|
|
assert cloud.client.relayer_region == "xx-earth-616"
|