2018-08-19 20:29:08 +00:00
|
|
|
"""Set up some common test helper things."""
|
2017-03-07 09:11:41 +00:00
|
|
|
import asyncio
|
2016-10-24 06:48:01 +00:00
|
|
|
import functools
|
|
|
|
import logging
|
2017-03-07 09:11:41 +00:00
|
|
|
import os
|
2019-02-07 21:50:59 +00:00
|
|
|
from unittest.mock import patch
|
2016-10-24 06:48:01 +00:00
|
|
|
|
|
|
|
import pytest
|
|
|
|
import requests_mock as _requests_mock
|
|
|
|
|
2018-02-11 17:17:58 +00:00
|
|
|
from homeassistant import util
|
2016-10-24 06:48:01 +00:00
|
|
|
from homeassistant.util import location
|
2018-12-02 15:32:53 +00:00
|
|
|
from homeassistant.auth.const import GROUP_ID_ADMIN, GROUP_ID_READ_ONLY
|
|
|
|
from homeassistant.auth.providers import legacy_api_password, homeassistant
|
2016-10-24 06:48:01 +00:00
|
|
|
|
2019-10-03 04:14:35 +00:00
|
|
|
pytest.register_assert_rewrite("tests.common")
|
|
|
|
from tests.common import ( # noqa: E402 module level import not at top of file
|
2019-07-31 19:25:30 +00:00
|
|
|
async_test_home_assistant,
|
|
|
|
INSTANCES,
|
|
|
|
mock_coro,
|
|
|
|
mock_storage as mock_storage,
|
|
|
|
MockUser,
|
|
|
|
CLIENT_ID,
|
|
|
|
)
|
2019-10-03 04:14:35 +00:00
|
|
|
from tests.test_util.aiohttp import (
|
|
|
|
mock_aiohttp_client,
|
|
|
|
) # noqa: E402 module level import not at top of file
|
2016-10-24 06:48:01 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
if os.environ.get("UVLOOP") == "1":
|
2017-03-07 09:11:41 +00:00
|
|
|
import uvloop
|
2019-07-31 19:25:30 +00:00
|
|
|
|
2017-03-07 09:11:41 +00:00
|
|
|
asyncio.set_event_loop_policy(uvloop.EventLoopPolicy())
|
|
|
|
|
2018-07-13 13:31:20 +00:00
|
|
|
logging.basicConfig(level=logging.DEBUG)
|
2019-07-31 19:25:30 +00:00
|
|
|
logging.getLogger("sqlalchemy.engine").setLevel(logging.INFO)
|
2016-10-24 06:48:01 +00:00
|
|
|
|
|
|
|
|
2018-05-12 21:44:53 +00:00
|
|
|
def check_real(func):
|
2016-10-24 06:48:01 +00:00
|
|
|
"""Force a function to require a keyword _test_real to be passed in."""
|
2019-07-31 19:25:30 +00:00
|
|
|
|
2016-10-24 06:48:01 +00:00
|
|
|
@functools.wraps(func)
|
|
|
|
def guard_func(*args, **kwargs):
|
2019-07-31 19:25:30 +00:00
|
|
|
real = kwargs.pop("_test_real", None)
|
2016-10-24 06:48:01 +00:00
|
|
|
|
|
|
|
if not real:
|
2019-07-31 19:25:30 +00:00
|
|
|
raise Exception(
|
|
|
|
'Forgot to mock or pass "_test_real=True" to %s', func.__name__
|
|
|
|
)
|
2016-10-24 06:48:01 +00:00
|
|
|
|
|
|
|
return func(*args, **kwargs)
|
|
|
|
|
|
|
|
return guard_func
|
|
|
|
|
2016-11-19 05:47:59 +00:00
|
|
|
|
2016-10-24 06:48:01 +00:00
|
|
|
# Guard a few functions that would make network connections
|
2019-07-31 19:25:30 +00:00
|
|
|
location.async_detect_location_info = check_real(location.async_detect_location_info)
|
|
|
|
util.get_local_ip = lambda: "127.0.0.1"
|
2016-10-24 06:48:01 +00:00
|
|
|
|
|
|
|
|
2017-02-26 22:05:18 +00:00
|
|
|
@pytest.fixture(autouse=True)
|
|
|
|
def verify_cleanup():
|
|
|
|
"""Verify that the test has cleaned up resources correctly."""
|
|
|
|
yield
|
|
|
|
|
2017-05-17 22:19:40 +00:00
|
|
|
if len(INSTANCES) >= 2:
|
|
|
|
count = len(INSTANCES)
|
|
|
|
for inst in INSTANCES:
|
|
|
|
inst.stop()
|
2019-07-31 19:25:30 +00:00
|
|
|
pytest.exit(
|
|
|
|
"Detected non stopped instances " "({}), aborting test run".format(count)
|
|
|
|
)
|
2017-02-26 22:05:18 +00:00
|
|
|
|
|
|
|
|
2016-10-24 06:48:01 +00:00
|
|
|
@pytest.fixture
|
2018-06-29 02:14:26 +00:00
|
|
|
def hass_storage():
|
|
|
|
"""Fixture to mock storage."""
|
|
|
|
with mock_storage() as stored_data:
|
|
|
|
yield stored_data
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture
|
|
|
|
def hass(loop, hass_storage):
|
2016-10-24 06:48:01 +00:00
|
|
|
"""Fixture to provide a test instance of HASS."""
|
|
|
|
hass = loop.run_until_complete(async_test_home_assistant(loop))
|
|
|
|
|
|
|
|
yield hass
|
|
|
|
|
2018-09-19 13:40:02 +00:00
|
|
|
loop.run_until_complete(hass.async_stop(force=True))
|
2016-10-24 06:48:01 +00:00
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture
|
|
|
|
def requests_mock():
|
|
|
|
"""Fixture to provide a requests mocker."""
|
|
|
|
with _requests_mock.mock() as m:
|
|
|
|
yield m
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture
|
|
|
|
def aioclient_mock():
|
|
|
|
"""Fixture to mock aioclient calls."""
|
|
|
|
with mock_aiohttp_client() as mock_session:
|
|
|
|
yield mock_session
|
2017-02-07 17:13:24 +00:00
|
|
|
|
|
|
|
|
2018-03-06 19:53:02 +00:00
|
|
|
@pytest.fixture
|
|
|
|
def mock_device_tracker_conf():
|
|
|
|
"""Prevent device tracker from reading/writing data."""
|
|
|
|
devices = []
|
|
|
|
|
|
|
|
async def mock_update_config(path, id, entity):
|
|
|
|
devices.append(entity)
|
|
|
|
|
|
|
|
with patch(
|
2019-07-31 19:25:30 +00:00
|
|
|
"homeassistant.components.device_tracker.legacy"
|
|
|
|
".DeviceTracker.async_update_config",
|
|
|
|
side_effect=mock_update_config,
|
2018-03-06 19:53:02 +00:00
|
|
|
), patch(
|
2019-07-31 19:25:30 +00:00
|
|
|
"homeassistant.components.device_tracker.legacy.async_load_config",
|
|
|
|
side_effect=lambda *args: mock_coro(devices),
|
2018-04-10 01:21:26 +00:00
|
|
|
):
|
2018-03-06 19:53:02 +00:00
|
|
|
yield devices
|
2018-12-02 15:32:53 +00:00
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture
|
|
|
|
def hass_access_token(hass, hass_admin_user):
|
|
|
|
"""Return an access token to access Home Assistant."""
|
|
|
|
refresh_token = hass.loop.run_until_complete(
|
2019-07-31 19:25:30 +00:00
|
|
|
hass.auth.async_create_refresh_token(hass_admin_user, CLIENT_ID)
|
|
|
|
)
|
2018-12-14 09:19:27 +00:00
|
|
|
return hass.auth.async_create_access_token(refresh_token)
|
2018-12-02 15:32:53 +00:00
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture
|
|
|
|
def hass_owner_user(hass, local_auth):
|
|
|
|
"""Return a Home Assistant admin user."""
|
|
|
|
return MockUser(is_owner=True).add_to_hass(hass)
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture
|
|
|
|
def hass_admin_user(hass, local_auth):
|
|
|
|
"""Return a Home Assistant admin user."""
|
2019-07-31 19:25:30 +00:00
|
|
|
admin_group = hass.loop.run_until_complete(
|
|
|
|
hass.auth.async_get_group(GROUP_ID_ADMIN)
|
|
|
|
)
|
2018-12-02 15:32:53 +00:00
|
|
|
return MockUser(groups=[admin_group]).add_to_hass(hass)
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture
|
|
|
|
def hass_read_only_user(hass, local_auth):
|
|
|
|
"""Return a Home Assistant read only user."""
|
2019-07-31 19:25:30 +00:00
|
|
|
read_only_group = hass.loop.run_until_complete(
|
|
|
|
hass.auth.async_get_group(GROUP_ID_READ_ONLY)
|
|
|
|
)
|
2018-12-02 15:32:53 +00:00
|
|
|
return MockUser(groups=[read_only_group]).add_to_hass(hass)
|
|
|
|
|
|
|
|
|
2018-12-14 09:19:27 +00:00
|
|
|
@pytest.fixture
|
|
|
|
def hass_read_only_access_token(hass, hass_read_only_user):
|
|
|
|
"""Return a Home Assistant read only user."""
|
|
|
|
refresh_token = hass.loop.run_until_complete(
|
2019-07-31 19:25:30 +00:00
|
|
|
hass.auth.async_create_refresh_token(hass_read_only_user, CLIENT_ID)
|
|
|
|
)
|
2018-12-14 09:19:27 +00:00
|
|
|
return hass.auth.async_create_access_token(refresh_token)
|
|
|
|
|
|
|
|
|
2018-12-02 15:32:53 +00:00
|
|
|
@pytest.fixture
|
|
|
|
def legacy_auth(hass):
|
|
|
|
"""Load legacy API password provider."""
|
|
|
|
prv = legacy_api_password.LegacyApiPasswordAuthProvider(
|
2019-07-31 19:25:30 +00:00
|
|
|
hass,
|
|
|
|
hass.auth._store,
|
|
|
|
{"type": "legacy_api_password", "api_password": "test-password"},
|
2018-12-02 15:32:53 +00:00
|
|
|
)
|
|
|
|
hass.auth._providers[(prv.type, prv.id)] = prv
|
2019-03-11 02:55:36 +00:00
|
|
|
return prv
|
2018-12-02 15:32:53 +00:00
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture
|
|
|
|
def local_auth(hass):
|
|
|
|
"""Load local auth provider."""
|
|
|
|
prv = homeassistant.HassAuthProvider(
|
2019-07-31 19:25:30 +00:00
|
|
|
hass, hass.auth._store, {"type": "homeassistant"}
|
2018-12-02 15:32:53 +00:00
|
|
|
)
|
|
|
|
hass.auth._providers[(prv.type, prv.id)] = prv
|
2019-03-11 02:55:36 +00:00
|
|
|
return prv
|
2018-12-02 15:32:53 +00:00
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture
|
|
|
|
def hass_client(hass, aiohttp_client, hass_access_token):
|
|
|
|
"""Return an authenticated HTTP client."""
|
2019-07-31 19:25:30 +00:00
|
|
|
|
2018-12-02 15:32:53 +00:00
|
|
|
async def auth_client():
|
|
|
|
"""Return an authenticated client."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return await aiohttp_client(
|
|
|
|
hass.http.app,
|
|
|
|
headers={"Authorization": "Bearer {}".format(hass_access_token)},
|
|
|
|
)
|
2018-12-02 15:32:53 +00:00
|
|
|
|
|
|
|
return auth_client
|