mirror of https://github.com/home-assistant/core
314 lines
9.4 KiB
Python
314 lines
9.4 KiB
Python
"""The tests the for Locative device tracker platform."""
|
|
|
|
from http import HTTPStatus
|
|
from unittest.mock import patch
|
|
|
|
from aiohttp.test_utils import TestClient
|
|
import pytest
|
|
|
|
from homeassistant import config_entries
|
|
from homeassistant.components import locative
|
|
from homeassistant.components.device_tracker import DOMAIN as DEVICE_TRACKER_DOMAIN
|
|
from homeassistant.components.device_tracker.legacy import Device
|
|
from homeassistant.components.locative import DOMAIN, TRACKER_UPDATE
|
|
from homeassistant.core import HomeAssistant
|
|
from homeassistant.core_config import async_process_ha_core_config
|
|
from homeassistant.data_entry_flow import FlowResultType
|
|
from homeassistant.helpers.dispatcher import DATA_DISPATCHER
|
|
from homeassistant.setup import async_setup_component
|
|
|
|
from tests.typing import ClientSessionGenerator
|
|
|
|
|
|
@pytest.fixture(autouse=True)
|
|
def mock_dev_track(mock_device_tracker_conf: list[Device]) -> None:
|
|
"""Mock device tracker config loading."""
|
|
|
|
|
|
@pytest.fixture
|
|
async def locative_client(
|
|
hass: HomeAssistant, hass_client: ClientSessionGenerator
|
|
) -> TestClient:
|
|
"""Locative mock client."""
|
|
assert await async_setup_component(hass, DOMAIN, {DOMAIN: {}})
|
|
await hass.async_block_till_done()
|
|
|
|
with patch("homeassistant.components.device_tracker.legacy.update_config"):
|
|
return await hass_client()
|
|
|
|
|
|
@pytest.fixture
|
|
async def webhook_id(hass: HomeAssistant, locative_client: TestClient) -> str:
|
|
"""Initialize the Geofency component and get the webhook_id."""
|
|
await async_process_ha_core_config(
|
|
hass,
|
|
{"internal_url": "http://example.local:8123"},
|
|
)
|
|
result = await hass.config_entries.flow.async_init(
|
|
"locative", context={"source": config_entries.SOURCE_USER}
|
|
)
|
|
assert result["type"] is FlowResultType.FORM, result
|
|
|
|
result = await hass.config_entries.flow.async_configure(result["flow_id"], {})
|
|
assert result["type"] is FlowResultType.CREATE_ENTRY
|
|
await hass.async_block_till_done()
|
|
|
|
return result["result"].data["webhook_id"]
|
|
|
|
|
|
async def test_missing_data(locative_client: TestClient, webhook_id: str) -> None:
|
|
"""Test missing data."""
|
|
url = f"/api/webhook/{webhook_id}"
|
|
|
|
data = {
|
|
"latitude": 1.0,
|
|
"longitude": 1.1,
|
|
"device": "123",
|
|
"id": "Home",
|
|
"trigger": "enter",
|
|
}
|
|
|
|
# No data
|
|
req = await locative_client.post(url)
|
|
assert req.status == HTTPStatus.UNPROCESSABLE_ENTITY
|
|
|
|
# No latitude
|
|
copy = data.copy()
|
|
del copy["latitude"]
|
|
req = await locative_client.post(url, data=copy)
|
|
assert req.status == HTTPStatus.UNPROCESSABLE_ENTITY
|
|
|
|
# No device
|
|
copy = data.copy()
|
|
del copy["device"]
|
|
req = await locative_client.post(url, data=copy)
|
|
assert req.status == HTTPStatus.UNPROCESSABLE_ENTITY
|
|
|
|
# No location
|
|
copy = data.copy()
|
|
del copy["id"]
|
|
req = await locative_client.post(url, data=copy)
|
|
assert req.status == HTTPStatus.UNPROCESSABLE_ENTITY
|
|
|
|
# No trigger
|
|
copy = data.copy()
|
|
del copy["trigger"]
|
|
req = await locative_client.post(url, data=copy)
|
|
assert req.status == HTTPStatus.UNPROCESSABLE_ENTITY
|
|
|
|
# Test message
|
|
copy = data.copy()
|
|
copy["trigger"] = "test"
|
|
req = await locative_client.post(url, data=copy)
|
|
assert req.status == HTTPStatus.OK
|
|
|
|
# Test message, no location
|
|
copy = data.copy()
|
|
copy["trigger"] = "test"
|
|
del copy["id"]
|
|
req = await locative_client.post(url, data=copy)
|
|
assert req.status == HTTPStatus.OK
|
|
|
|
# Unknown trigger
|
|
copy = data.copy()
|
|
copy["trigger"] = "foobar"
|
|
req = await locative_client.post(url, data=copy)
|
|
assert req.status == HTTPStatus.UNPROCESSABLE_ENTITY
|
|
|
|
|
|
async def test_enter_and_exit(
|
|
hass: HomeAssistant, locative_client: TestClient, webhook_id: str
|
|
) -> None:
|
|
"""Test when there is a known zone."""
|
|
url = f"/api/webhook/{webhook_id}"
|
|
|
|
data = {
|
|
"latitude": 40.7855,
|
|
"longitude": -111.7367,
|
|
"device": "123",
|
|
"id": "Home",
|
|
"trigger": "enter",
|
|
}
|
|
|
|
# Enter the Home
|
|
req = await locative_client.post(url, data=data)
|
|
await hass.async_block_till_done()
|
|
assert req.status == HTTPStatus.OK
|
|
state_name = hass.states.get(f"{DEVICE_TRACKER_DOMAIN}.{data['device']}").state
|
|
assert state_name == "home"
|
|
|
|
data["id"] = "HOME"
|
|
data["trigger"] = "exit"
|
|
|
|
# Exit Home
|
|
req = await locative_client.post(url, data=data)
|
|
await hass.async_block_till_done()
|
|
assert req.status == HTTPStatus.OK
|
|
state_name = hass.states.get(f"{DEVICE_TRACKER_DOMAIN}.{data['device']}").state
|
|
assert state_name == "not_home"
|
|
|
|
data["id"] = "hOmE"
|
|
data["trigger"] = "enter"
|
|
|
|
# Enter Home again
|
|
req = await locative_client.post(url, data=data)
|
|
await hass.async_block_till_done()
|
|
assert req.status == HTTPStatus.OK
|
|
state_name = hass.states.get(f"{DEVICE_TRACKER_DOMAIN}.{data['device']}").state
|
|
assert state_name == "home"
|
|
|
|
data["trigger"] = "exit"
|
|
|
|
# Exit Home
|
|
req = await locative_client.post(url, data=data)
|
|
await hass.async_block_till_done()
|
|
assert req.status == HTTPStatus.OK
|
|
state_name = hass.states.get(f"{DEVICE_TRACKER_DOMAIN}.{data['device']}").state
|
|
assert state_name == "not_home"
|
|
|
|
data["id"] = "work"
|
|
data["trigger"] = "enter"
|
|
|
|
# Enter Work
|
|
req = await locative_client.post(url, data=data)
|
|
await hass.async_block_till_done()
|
|
assert req.status == HTTPStatus.OK
|
|
state_name = hass.states.get(f"{DEVICE_TRACKER_DOMAIN}.{data['device']}").state
|
|
assert state_name == "work"
|
|
|
|
|
|
async def test_exit_after_enter(
|
|
hass: HomeAssistant, locative_client: TestClient, webhook_id: str
|
|
) -> None:
|
|
"""Test when an exit message comes after an enter message."""
|
|
url = f"/api/webhook/{webhook_id}"
|
|
|
|
data = {
|
|
"latitude": 40.7855,
|
|
"longitude": -111.7367,
|
|
"device": "123",
|
|
"id": "Home",
|
|
"trigger": "enter",
|
|
}
|
|
|
|
# Enter Home
|
|
req = await locative_client.post(url, data=data)
|
|
await hass.async_block_till_done()
|
|
assert req.status == HTTPStatus.OK
|
|
|
|
state = hass.states.get(f"{DEVICE_TRACKER_DOMAIN}.{data['device']}")
|
|
assert state.state == "home"
|
|
|
|
data["id"] = "Work"
|
|
|
|
# Enter Work
|
|
req = await locative_client.post(url, data=data)
|
|
await hass.async_block_till_done()
|
|
assert req.status == HTTPStatus.OK
|
|
|
|
state = hass.states.get(f"{DEVICE_TRACKER_DOMAIN}.{data['device']}")
|
|
assert state.state == "work"
|
|
|
|
data["id"] = "Home"
|
|
data["trigger"] = "exit"
|
|
|
|
# Exit Home
|
|
req = await locative_client.post(url, data=data)
|
|
await hass.async_block_till_done()
|
|
assert req.status == HTTPStatus.OK
|
|
|
|
state = hass.states.get(f"{DEVICE_TRACKER_DOMAIN}.{data['device']}")
|
|
assert state.state == "work"
|
|
|
|
|
|
async def test_exit_first(
|
|
hass: HomeAssistant, locative_client: TestClient, webhook_id: str
|
|
) -> None:
|
|
"""Test when an exit message is sent first on a new device."""
|
|
url = f"/api/webhook/{webhook_id}"
|
|
|
|
data = {
|
|
"latitude": 40.7855,
|
|
"longitude": -111.7367,
|
|
"device": "new_device",
|
|
"id": "Home",
|
|
"trigger": "exit",
|
|
}
|
|
|
|
# Exit Home
|
|
req = await locative_client.post(url, data=data)
|
|
await hass.async_block_till_done()
|
|
assert req.status == HTTPStatus.OK
|
|
|
|
state = hass.states.get(f"{DEVICE_TRACKER_DOMAIN}.{data['device']}")
|
|
assert state.state == "not_home"
|
|
|
|
|
|
async def test_two_devices(
|
|
hass: HomeAssistant, locative_client: TestClient, webhook_id: str
|
|
) -> None:
|
|
"""Test updating two different devices."""
|
|
url = f"/api/webhook/{webhook_id}"
|
|
|
|
data_device_1 = {
|
|
"latitude": 40.7855,
|
|
"longitude": -111.7367,
|
|
"device": "device_1",
|
|
"id": "Home",
|
|
"trigger": "exit",
|
|
}
|
|
|
|
# Exit Home
|
|
req = await locative_client.post(url, data=data_device_1)
|
|
await hass.async_block_till_done()
|
|
assert req.status == HTTPStatus.OK
|
|
|
|
state = hass.states.get(f"{DEVICE_TRACKER_DOMAIN}.{data_device_1['device']}")
|
|
assert state.state == "not_home"
|
|
|
|
# Enter Home
|
|
data_device_2 = dict(data_device_1)
|
|
data_device_2["device"] = "device_2"
|
|
data_device_2["trigger"] = "enter"
|
|
req = await locative_client.post(url, data=data_device_2)
|
|
await hass.async_block_till_done()
|
|
assert req.status == HTTPStatus.OK
|
|
|
|
state = hass.states.get(f"{DEVICE_TRACKER_DOMAIN}.{data_device_2['device']}")
|
|
assert state.state == "home"
|
|
state = hass.states.get(f"{DEVICE_TRACKER_DOMAIN}.{data_device_1['device']}")
|
|
assert state.state == "not_home"
|
|
|
|
|
|
@pytest.mark.xfail(
|
|
reason="The device_tracker component does not support unloading yet."
|
|
)
|
|
async def test_load_unload_entry(
|
|
hass: HomeAssistant, locative_client: TestClient, webhook_id: str
|
|
) -> None:
|
|
"""Test that the appropriate dispatch signals are added and removed."""
|
|
url = f"/api/webhook/{webhook_id}"
|
|
|
|
data = {
|
|
"latitude": 40.7855,
|
|
"longitude": -111.7367,
|
|
"device": "new_device",
|
|
"id": "Home",
|
|
"trigger": "exit",
|
|
}
|
|
|
|
# Exit Home
|
|
req = await locative_client.post(url, data=data)
|
|
await hass.async_block_till_done()
|
|
assert req.status == HTTPStatus.OK
|
|
|
|
state = hass.states.get(f"{DEVICE_TRACKER_DOMAIN}.{data['device']}")
|
|
assert state.state == "not_home"
|
|
assert len(hass.data[DATA_DISPATCHER][TRACKER_UPDATE]) == 1
|
|
|
|
entry = hass.config_entries.async_entries(DOMAIN)[0]
|
|
|
|
await locative.async_unload_entry(hass, entry)
|
|
await hass.async_block_till_done()
|
|
assert not hass.data[DATA_DISPATCHER][TRACKER_UPDATE]
|