2016-03-09 09:25:50 +00:00
|
|
|
"""The tests for the Recorder component."""
|
2016-07-23 18:25:17 +00:00
|
|
|
# pylint: disable=protected-access
|
2020-04-16 22:11:36 +00:00
|
|
|
from datetime import datetime, timedelta
|
2021-04-27 05:20:31 +00:00
|
|
|
import sqlite3
|
2021-01-01 21:31:56 +00:00
|
|
|
from unittest.mock import patch
|
2017-02-10 02:17:17 +00:00
|
|
|
|
2021-05-20 11:05:15 +00:00
|
|
|
import pytest
|
2021-04-27 05:20:31 +00:00
|
|
|
from sqlalchemy.exc import DatabaseError, OperationalError, SQLAlchemyError
|
2020-10-08 07:15:25 +00:00
|
|
|
|
2021-04-12 06:43:54 +00:00
|
|
|
from homeassistant.components import recorder
|
2020-06-19 17:03:06 +00:00
|
|
|
from homeassistant.components.recorder import (
|
2021-05-17 21:27:51 +00:00
|
|
|
CONF_AUTO_PURGE,
|
2021-02-20 07:26:24 +00:00
|
|
|
CONF_DB_URL,
|
2020-06-24 01:02:29 +00:00
|
|
|
CONFIG_SCHEMA,
|
|
|
|
DOMAIN,
|
2021-04-12 06:43:54 +00:00
|
|
|
KEEPALIVE_TIME,
|
2021-02-24 13:26:05 +00:00
|
|
|
SERVICE_DISABLE,
|
|
|
|
SERVICE_ENABLE,
|
|
|
|
SERVICE_PURGE,
|
|
|
|
SQLITE_URL_PREFIX,
|
2020-06-19 17:03:06 +00:00
|
|
|
Recorder,
|
|
|
|
run_information,
|
|
|
|
run_information_from_instance,
|
|
|
|
run_information_with_session,
|
|
|
|
)
|
2021-04-12 06:43:54 +00:00
|
|
|
from homeassistant.components.recorder.const import DATA_INSTANCE
|
2020-06-19 17:03:06 +00:00
|
|
|
from homeassistant.components.recorder.models import Events, RecorderRuns, States
|
2017-02-26 22:38:06 +00:00
|
|
|
from homeassistant.components.recorder.util import session_scope
|
2021-02-20 06:18:21 +00:00
|
|
|
from homeassistant.const import (
|
2021-04-19 09:24:17 +00:00
|
|
|
EVENT_HOMEASSISTANT_FINAL_WRITE,
|
2021-04-12 06:43:54 +00:00
|
|
|
EVENT_HOMEASSISTANT_STARTED,
|
2021-02-20 06:18:21 +00:00
|
|
|
EVENT_HOMEASSISTANT_STOP,
|
|
|
|
MATCH_ALL,
|
|
|
|
STATE_LOCKED,
|
|
|
|
STATE_UNLOCKED,
|
|
|
|
)
|
2021-04-12 06:43:54 +00:00
|
|
|
from homeassistant.core import Context, CoreState, HomeAssistant, callback
|
2021-02-24 13:26:05 +00:00
|
|
|
from homeassistant.setup import async_setup_component, setup_component
|
2020-04-16 22:11:36 +00:00
|
|
|
from homeassistant.util import dt as dt_util
|
2017-03-03 06:44:52 +00:00
|
|
|
|
2021-03-11 17:52:07 +00:00
|
|
|
from .common import (
|
|
|
|
async_wait_recording_done,
|
|
|
|
async_wait_recording_done_without_instance,
|
|
|
|
corrupt_db_file,
|
|
|
|
wait_recording_done,
|
|
|
|
)
|
|
|
|
from .conftest import SetupRecorderInstanceT
|
2020-03-10 00:43:26 +00:00
|
|
|
|
2021-02-20 06:18:21 +00:00
|
|
|
from tests.common import (
|
2021-04-12 06:43:54 +00:00
|
|
|
async_fire_time_changed,
|
2021-02-20 06:18:21 +00:00
|
|
|
async_init_recorder_component,
|
|
|
|
fire_time_changed,
|
|
|
|
get_test_home_assistant,
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2021-04-12 06:43:54 +00:00
|
|
|
def _default_recorder(hass):
|
|
|
|
"""Return a recorder with reasonable defaults."""
|
|
|
|
return Recorder(
|
|
|
|
hass,
|
|
|
|
auto_purge=True,
|
|
|
|
keep_days=7,
|
|
|
|
commit_interval=1,
|
|
|
|
uri="sqlite://",
|
|
|
|
db_max_retries=10,
|
|
|
|
db_retry_wait=3,
|
|
|
|
entity_filter=CONFIG_SCHEMA({DOMAIN: {}}),
|
|
|
|
exclude_t=[],
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2021-02-20 06:18:21 +00:00
|
|
|
async def test_shutdown_before_startup_finishes(hass):
|
|
|
|
"""Test shutdown before recorder starts is clean."""
|
|
|
|
|
|
|
|
hass.state = CoreState.not_running
|
|
|
|
|
|
|
|
await async_init_recorder_component(hass)
|
2021-04-12 06:43:54 +00:00
|
|
|
await hass.data[DATA_INSTANCE].async_db_ready
|
2021-02-20 06:18:21 +00:00
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
|
|
session = await hass.async_add_executor_job(hass.data[DATA_INSTANCE].get_session)
|
|
|
|
|
|
|
|
with patch.object(hass.data[DATA_INSTANCE], "engine"):
|
|
|
|
hass.bus.async_fire(EVENT_HOMEASSISTANT_STOP)
|
|
|
|
await hass.async_block_till_done()
|
2021-02-21 08:52:41 +00:00
|
|
|
await hass.async_stop()
|
2021-02-20 06:18:21 +00:00
|
|
|
|
|
|
|
run_info = await hass.async_add_executor_job(run_information_with_session, session)
|
|
|
|
|
|
|
|
assert run_info.run_id == 1
|
|
|
|
assert run_info.start is not None
|
|
|
|
assert run_info.end is not None
|
2015-04-30 06:21:31 +00:00
|
|
|
|
|
|
|
|
2021-04-12 06:43:54 +00:00
|
|
|
async def test_state_gets_saved_when_set_before_start_event(
|
|
|
|
hass: HomeAssistant, async_setup_recorder_instance: SetupRecorderInstanceT
|
|
|
|
):
|
|
|
|
"""Test we can record an event when starting with not running."""
|
|
|
|
|
|
|
|
hass.state = CoreState.not_running
|
|
|
|
|
|
|
|
await async_init_recorder_component(hass)
|
|
|
|
|
|
|
|
entity_id = "test.recorder"
|
|
|
|
state = "restoring_from_db"
|
|
|
|
attributes = {"test_attr": 5, "test_attr_10": "nice"}
|
|
|
|
|
|
|
|
hass.states.async_set(entity_id, state, attributes)
|
|
|
|
|
|
|
|
hass.bus.async_fire(EVENT_HOMEASSISTANT_STARTED)
|
|
|
|
|
|
|
|
await async_wait_recording_done_without_instance(hass)
|
|
|
|
|
|
|
|
with session_scope(hass=hass) as session:
|
|
|
|
db_states = list(session.query(States))
|
|
|
|
assert len(db_states) == 1
|
|
|
|
assert db_states[0].event_id > 0
|
|
|
|
|
|
|
|
|
2021-03-11 17:52:07 +00:00
|
|
|
async def test_saving_state(
|
2021-04-22 17:58:02 +00:00
|
|
|
hass: HomeAssistant, async_setup_recorder_instance: SetupRecorderInstanceT
|
2021-03-11 17:52:07 +00:00
|
|
|
):
|
2020-10-06 19:24:13 +00:00
|
|
|
"""Test saving and restoring a state."""
|
2021-03-11 17:52:07 +00:00
|
|
|
instance = await async_setup_recorder_instance(hass)
|
2017-02-10 02:17:17 +00:00
|
|
|
|
2020-10-06 19:24:13 +00:00
|
|
|
entity_id = "test.recorder"
|
|
|
|
state = "restoring_from_db"
|
|
|
|
attributes = {"test_attr": 5, "test_attr_10": "nice"}
|
2015-04-30 06:21:31 +00:00
|
|
|
|
2021-03-11 17:52:07 +00:00
|
|
|
hass.states.async_set(entity_id, state, attributes)
|
2016-07-11 07:46:56 +00:00
|
|
|
|
2021-03-11 17:52:07 +00:00
|
|
|
await async_wait_recording_done(hass, instance)
|
2015-04-30 06:21:31 +00:00
|
|
|
|
2020-10-06 19:24:13 +00:00
|
|
|
with session_scope(hass=hass) as session:
|
|
|
|
db_states = list(session.query(States))
|
|
|
|
assert len(db_states) == 1
|
|
|
|
assert db_states[0].event_id > 0
|
|
|
|
state = db_states[0].to_native()
|
2015-04-30 06:21:31 +00:00
|
|
|
|
2020-10-06 19:24:13 +00:00
|
|
|
assert state == _state_empty_context(hass, entity_id)
|
2015-04-30 06:21:31 +00:00
|
|
|
|
|
|
|
|
2021-04-12 06:43:54 +00:00
|
|
|
async def test_saving_many_states(
|
2021-04-22 17:58:02 +00:00
|
|
|
hass: HomeAssistant, async_setup_recorder_instance: SetupRecorderInstanceT
|
2021-04-12 06:43:54 +00:00
|
|
|
):
|
|
|
|
"""Test we expire after many commits."""
|
|
|
|
instance = await async_setup_recorder_instance(hass)
|
|
|
|
|
|
|
|
entity_id = "test.recorder"
|
|
|
|
attributes = {"test_attr": 5, "test_attr_10": "nice"}
|
|
|
|
|
|
|
|
with patch.object(
|
|
|
|
hass.data[DATA_INSTANCE].event_session, "expire_all"
|
|
|
|
) as expire_all, patch.object(recorder, "EXPIRE_AFTER_COMMITS", 2):
|
|
|
|
for _ in range(3):
|
|
|
|
hass.states.async_set(entity_id, "on", attributes)
|
|
|
|
await async_wait_recording_done(hass, instance)
|
|
|
|
hass.states.async_set(entity_id, "off", attributes)
|
|
|
|
await async_wait_recording_done(hass, instance)
|
|
|
|
|
|
|
|
assert expire_all.called
|
|
|
|
|
|
|
|
with session_scope(hass=hass) as session:
|
|
|
|
db_states = list(session.query(States))
|
|
|
|
assert len(db_states) == 6
|
|
|
|
assert db_states[0].event_id > 0
|
|
|
|
|
|
|
|
|
|
|
|
async def test_saving_state_with_intermixed_time_changes(
|
2021-04-22 17:58:02 +00:00
|
|
|
hass: HomeAssistant, async_setup_recorder_instance: SetupRecorderInstanceT
|
2021-04-12 06:43:54 +00:00
|
|
|
):
|
|
|
|
"""Test saving states with intermixed time changes."""
|
|
|
|
instance = await async_setup_recorder_instance(hass)
|
|
|
|
|
|
|
|
entity_id = "test.recorder"
|
|
|
|
state = "restoring_from_db"
|
|
|
|
attributes = {"test_attr": 5, "test_attr_10": "nice"}
|
|
|
|
attributes2 = {"test_attr": 10, "test_attr_10": "mean"}
|
|
|
|
|
|
|
|
for _ in range(KEEPALIVE_TIME + 1):
|
|
|
|
async_fire_time_changed(hass, dt_util.utcnow())
|
|
|
|
hass.states.async_set(entity_id, state, attributes)
|
|
|
|
for _ in range(KEEPALIVE_TIME + 1):
|
|
|
|
async_fire_time_changed(hass, dt_util.utcnow())
|
|
|
|
hass.states.async_set(entity_id, state, attributes2)
|
|
|
|
|
|
|
|
await async_wait_recording_done(hass, instance)
|
|
|
|
|
|
|
|
with session_scope(hass=hass) as session:
|
|
|
|
db_states = list(session.query(States))
|
|
|
|
assert len(db_states) == 2
|
|
|
|
assert db_states[0].event_id > 0
|
|
|
|
|
|
|
|
|
2020-10-08 07:15:25 +00:00
|
|
|
def test_saving_state_with_exception(hass, hass_recorder, caplog):
|
|
|
|
"""Test saving and restoring a state."""
|
|
|
|
hass = hass_recorder()
|
|
|
|
|
|
|
|
entity_id = "test.recorder"
|
|
|
|
state = "restoring_from_db"
|
|
|
|
attributes = {"test_attr": 5, "test_attr_10": "nice"}
|
|
|
|
|
|
|
|
def _throw_if_state_in_session(*args, **kwargs):
|
|
|
|
for obj in hass.data[DATA_INSTANCE].event_session:
|
|
|
|
if isinstance(obj, States):
|
|
|
|
raise OperationalError(
|
|
|
|
"insert the state", "fake params", "forced to fail"
|
|
|
|
)
|
|
|
|
|
|
|
|
with patch("time.sleep"), patch.object(
|
|
|
|
hass.data[DATA_INSTANCE].event_session,
|
|
|
|
"flush",
|
|
|
|
side_effect=_throw_if_state_in_session,
|
|
|
|
):
|
|
|
|
hass.states.set(entity_id, "fail", attributes)
|
|
|
|
wait_recording_done(hass)
|
|
|
|
|
|
|
|
assert "Error executing query" in caplog.text
|
|
|
|
assert "Error saving events" not in caplog.text
|
|
|
|
|
|
|
|
caplog.clear()
|
|
|
|
hass.states.set(entity_id, state, attributes)
|
|
|
|
wait_recording_done(hass)
|
|
|
|
|
|
|
|
with session_scope(hass=hass) as session:
|
|
|
|
db_states = list(session.query(States))
|
|
|
|
assert len(db_states) >= 1
|
|
|
|
|
|
|
|
assert "Error executing query" not in caplog.text
|
|
|
|
assert "Error saving events" not in caplog.text
|
|
|
|
|
|
|
|
|
2021-04-12 06:43:54 +00:00
|
|
|
def test_saving_state_with_sqlalchemy_exception(hass, hass_recorder, caplog):
|
|
|
|
"""Test saving state when there is an SQLAlchemyError."""
|
|
|
|
hass = hass_recorder()
|
|
|
|
|
|
|
|
entity_id = "test.recorder"
|
|
|
|
state = "restoring_from_db"
|
|
|
|
attributes = {"test_attr": 5, "test_attr_10": "nice"}
|
|
|
|
|
|
|
|
def _throw_if_state_in_session(*args, **kwargs):
|
|
|
|
for obj in hass.data[DATA_INSTANCE].event_session:
|
|
|
|
if isinstance(obj, States):
|
|
|
|
raise SQLAlchemyError(
|
|
|
|
"insert the state", "fake params", "forced to fail"
|
|
|
|
)
|
|
|
|
|
|
|
|
with patch("time.sleep"), patch.object(
|
|
|
|
hass.data[DATA_INSTANCE].event_session,
|
|
|
|
"flush",
|
|
|
|
side_effect=_throw_if_state_in_session,
|
|
|
|
):
|
|
|
|
hass.states.set(entity_id, "fail", attributes)
|
|
|
|
wait_recording_done(hass)
|
|
|
|
|
|
|
|
assert "SQLAlchemyError error processing event" in caplog.text
|
|
|
|
|
|
|
|
caplog.clear()
|
|
|
|
hass.states.set(entity_id, state, attributes)
|
|
|
|
wait_recording_done(hass)
|
|
|
|
|
|
|
|
with session_scope(hass=hass) as session:
|
|
|
|
db_states = list(session.query(States))
|
|
|
|
assert len(db_states) >= 1
|
|
|
|
|
|
|
|
assert "Error executing query" not in caplog.text
|
|
|
|
assert "Error saving events" not in caplog.text
|
|
|
|
assert "SQLAlchemyError error processing event" not in caplog.text
|
|
|
|
|
|
|
|
|
2021-04-19 09:24:17 +00:00
|
|
|
async def test_force_shutdown_with_queue_of_writes_that_generate_exceptions(
|
|
|
|
hass, async_setup_recorder_instance, caplog
|
|
|
|
):
|
|
|
|
"""Test forcing shutdown."""
|
|
|
|
instance = await async_setup_recorder_instance(hass)
|
|
|
|
|
|
|
|
entity_id = "test.recorder"
|
|
|
|
attributes = {"test_attr": 5, "test_attr_10": "nice"}
|
|
|
|
|
|
|
|
await async_wait_recording_done(hass, instance)
|
|
|
|
|
|
|
|
with patch.object(instance, "db_retry_wait", 0.2), patch.object(
|
|
|
|
instance.event_session,
|
|
|
|
"flush",
|
|
|
|
side_effect=OperationalError(
|
|
|
|
"insert the state", "fake params", "forced to fail"
|
|
|
|
),
|
|
|
|
):
|
|
|
|
for _ in range(100):
|
|
|
|
hass.states.async_set(entity_id, "on", attributes)
|
|
|
|
hass.states.async_set(entity_id, "off", attributes)
|
|
|
|
|
|
|
|
hass.bus.async_fire(EVENT_HOMEASSISTANT_STOP)
|
|
|
|
hass.bus.async_fire(EVENT_HOMEASSISTANT_FINAL_WRITE)
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
|
|
assert "Error executing query" in caplog.text
|
|
|
|
assert "Error saving events" not in caplog.text
|
|
|
|
|
|
|
|
|
2020-10-06 19:24:13 +00:00
|
|
|
def test_saving_event(hass, hass_recorder):
|
|
|
|
"""Test saving and restoring an event."""
|
|
|
|
hass = hass_recorder()
|
2015-04-30 06:21:31 +00:00
|
|
|
|
2020-10-06 19:24:13 +00:00
|
|
|
event_type = "EVENT_TEST"
|
|
|
|
event_data = {"test_attr": 5, "test_attr_10": "nice"}
|
2015-04-30 06:21:31 +00:00
|
|
|
|
2020-10-06 19:24:13 +00:00
|
|
|
events = []
|
2015-04-30 06:21:31 +00:00
|
|
|
|
2020-10-06 19:24:13 +00:00
|
|
|
@callback
|
|
|
|
def event_listener(event):
|
|
|
|
"""Record events from eventbus."""
|
|
|
|
if event.event_type == event_type:
|
|
|
|
events.append(event)
|
2017-02-26 22:38:06 +00:00
|
|
|
|
2020-10-06 19:24:13 +00:00
|
|
|
hass.bus.listen(MATCH_ALL, event_listener)
|
2017-02-26 22:38:06 +00:00
|
|
|
|
2020-10-06 19:24:13 +00:00
|
|
|
hass.bus.fire(event_type, event_data)
|
2016-04-16 07:55:35 +00:00
|
|
|
|
2020-10-06 19:24:13 +00:00
|
|
|
wait_recording_done(hass)
|
2016-04-16 07:55:35 +00:00
|
|
|
|
2020-10-06 19:24:13 +00:00
|
|
|
assert len(events) == 1
|
|
|
|
event = events[0]
|
2016-04-16 02:02:17 +00:00
|
|
|
|
2020-10-06 19:24:13 +00:00
|
|
|
hass.data[DATA_INSTANCE].block_till_done()
|
2017-01-09 20:53:30 +00:00
|
|
|
|
2020-10-06 19:24:13 +00:00
|
|
|
with session_scope(hass=hass) as session:
|
|
|
|
db_events = list(session.query(Events).filter_by(event_type=event_type))
|
|
|
|
assert len(db_events) == 1
|
|
|
|
db_event = db_events[0].to_native()
|
2017-01-09 20:53:30 +00:00
|
|
|
|
2020-10-06 19:24:13 +00:00
|
|
|
assert event.event_type == db_event.event_type
|
|
|
|
assert event.data == db_event.data
|
|
|
|
assert event.origin == db_event.origin
|
2017-01-09 20:53:30 +00:00
|
|
|
|
2020-10-06 19:24:13 +00:00
|
|
|
# Recorder uses SQLite and stores datetimes as integer unix timestamps
|
|
|
|
assert event.time_fired.replace(microsecond=0) == db_event.time_fired.replace(
|
|
|
|
microsecond=0
|
|
|
|
)
|
2017-01-09 20:53:30 +00:00
|
|
|
|
|
|
|
|
2021-04-12 06:43:54 +00:00
|
|
|
def test_saving_state_with_commit_interval_zero(hass_recorder):
|
|
|
|
"""Test saving a state with a commit interval of zero."""
|
|
|
|
hass = hass_recorder({"commit_interval": 0})
|
|
|
|
assert hass.data[DATA_INSTANCE].commit_interval == 0
|
|
|
|
|
|
|
|
entity_id = "test.recorder"
|
|
|
|
state = "restoring_from_db"
|
|
|
|
attributes = {"test_attr": 5, "test_attr_10": "nice"}
|
|
|
|
|
|
|
|
hass.states.set(entity_id, state, attributes)
|
|
|
|
|
|
|
|
wait_recording_done(hass)
|
|
|
|
|
|
|
|
with session_scope(hass=hass) as session:
|
|
|
|
db_states = list(session.query(States))
|
|
|
|
assert len(db_states) == 1
|
|
|
|
assert db_states[0].event_id > 0
|
|
|
|
|
|
|
|
|
2017-01-09 20:53:30 +00:00
|
|
|
def _add_entities(hass, entity_ids):
|
|
|
|
"""Add entities."""
|
2019-07-31 19:25:30 +00:00
|
|
|
attributes = {"test_attr": 5, "test_attr_10": "nice"}
|
2017-01-09 20:53:30 +00:00
|
|
|
for idx, entity_id in enumerate(entity_ids):
|
2020-04-04 22:26:08 +00:00
|
|
|
hass.states.set(entity_id, f"state{idx}", attributes)
|
2020-03-10 00:43:26 +00:00
|
|
|
wait_recording_done(hass)
|
2017-02-26 22:38:06 +00:00
|
|
|
|
|
|
|
with session_scope(hass=hass) as session:
|
|
|
|
return [st.to_native() for st in session.query(States)]
|
2017-01-09 20:53:30 +00:00
|
|
|
|
|
|
|
|
2017-05-24 22:23:52 +00:00
|
|
|
def _add_events(hass, events):
|
|
|
|
with session_scope(hass=hass) as session:
|
|
|
|
session.query(Events).delete(synchronize_session=False)
|
|
|
|
for event_type in events:
|
|
|
|
hass.bus.fire(event_type)
|
2020-03-10 00:43:26 +00:00
|
|
|
wait_recording_done(hass)
|
2017-05-24 22:23:52 +00:00
|
|
|
|
|
|
|
with session_scope(hass=hass) as session:
|
|
|
|
return [ev.to_native() for ev in session.query(Events)]
|
|
|
|
|
|
|
|
|
2020-06-23 17:57:52 +00:00
|
|
|
def _state_empty_context(hass, entity_id):
|
|
|
|
# We don't restore context unless we need it by joining the
|
|
|
|
# events table on the event_id for state_changed events
|
|
|
|
state = hass.states.get(entity_id)
|
|
|
|
state.context = Context(id=None)
|
|
|
|
return state
|
|
|
|
|
|
|
|
|
2017-01-09 20:53:30 +00:00
|
|
|
# pylint: disable=redefined-outer-name,invalid-name
|
|
|
|
def test_saving_state_include_domains(hass_recorder):
|
|
|
|
"""Test saving and restoring a state."""
|
2019-07-31 19:25:30 +00:00
|
|
|
hass = hass_recorder({"include": {"domains": "test2"}})
|
|
|
|
states = _add_entities(hass, ["test.recorder", "test2.recorder"])
|
2017-01-09 20:53:30 +00:00
|
|
|
assert len(states) == 1
|
2020-06-23 17:57:52 +00:00
|
|
|
assert _state_empty_context(hass, "test2.recorder") == states[0]
|
2017-01-09 20:53:30 +00:00
|
|
|
|
|
|
|
|
2020-06-24 01:02:29 +00:00
|
|
|
def test_saving_state_include_domains_globs(hass_recorder):
|
|
|
|
"""Test saving and restoring a state."""
|
|
|
|
hass = hass_recorder(
|
|
|
|
{"include": {"domains": "test2", "entity_globs": "*.included_*"}}
|
|
|
|
)
|
|
|
|
states = _add_entities(
|
|
|
|
hass, ["test.recorder", "test2.recorder", "test3.included_entity"]
|
|
|
|
)
|
|
|
|
assert len(states) == 2
|
|
|
|
assert _state_empty_context(hass, "test2.recorder") == states[0]
|
|
|
|
assert _state_empty_context(hass, "test3.included_entity") == states[1]
|
|
|
|
|
|
|
|
|
2017-01-09 20:53:30 +00:00
|
|
|
def test_saving_state_incl_entities(hass_recorder):
|
|
|
|
"""Test saving and restoring a state."""
|
2019-07-31 19:25:30 +00:00
|
|
|
hass = hass_recorder({"include": {"entities": "test2.recorder"}})
|
|
|
|
states = _add_entities(hass, ["test.recorder", "test2.recorder"])
|
2017-01-09 20:53:30 +00:00
|
|
|
assert len(states) == 1
|
2020-06-23 17:57:52 +00:00
|
|
|
assert _state_empty_context(hass, "test2.recorder") == states[0]
|
2017-01-09 20:53:30 +00:00
|
|
|
|
|
|
|
|
2017-05-24 22:23:52 +00:00
|
|
|
def test_saving_event_exclude_event_type(hass_recorder):
|
|
|
|
"""Test saving and restoring an event."""
|
2020-09-30 11:11:43 +00:00
|
|
|
hass = hass_recorder(
|
|
|
|
{
|
|
|
|
"exclude": {
|
|
|
|
"event_types": [
|
|
|
|
"service_registered",
|
|
|
|
"homeassistant_start",
|
|
|
|
"component_loaded",
|
|
|
|
"core_config_updated",
|
|
|
|
"homeassistant_started",
|
|
|
|
"test",
|
|
|
|
]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)
|
2019-07-31 19:25:30 +00:00
|
|
|
events = _add_events(hass, ["test", "test2"])
|
2017-05-24 22:23:52 +00:00
|
|
|
assert len(events) == 1
|
2019-07-31 19:25:30 +00:00
|
|
|
assert events[0].event_type == "test2"
|
2017-05-24 22:23:52 +00:00
|
|
|
|
|
|
|
|
2017-01-09 20:53:30 +00:00
|
|
|
def test_saving_state_exclude_domains(hass_recorder):
|
|
|
|
"""Test saving and restoring a state."""
|
2019-07-31 19:25:30 +00:00
|
|
|
hass = hass_recorder({"exclude": {"domains": "test"}})
|
|
|
|
states = _add_entities(hass, ["test.recorder", "test2.recorder"])
|
2017-01-09 20:53:30 +00:00
|
|
|
assert len(states) == 1
|
2020-06-23 17:57:52 +00:00
|
|
|
assert _state_empty_context(hass, "test2.recorder") == states[0]
|
2017-01-09 20:53:30 +00:00
|
|
|
|
|
|
|
|
2020-06-24 01:02:29 +00:00
|
|
|
def test_saving_state_exclude_domains_globs(hass_recorder):
|
|
|
|
"""Test saving and restoring a state."""
|
|
|
|
hass = hass_recorder(
|
|
|
|
{"exclude": {"domains": "test", "entity_globs": "*.excluded_*"}}
|
|
|
|
)
|
|
|
|
states = _add_entities(
|
|
|
|
hass, ["test.recorder", "test2.recorder", "test2.excluded_entity"]
|
|
|
|
)
|
|
|
|
assert len(states) == 1
|
|
|
|
assert _state_empty_context(hass, "test2.recorder") == states[0]
|
|
|
|
|
|
|
|
|
2017-01-09 20:53:30 +00:00
|
|
|
def test_saving_state_exclude_entities(hass_recorder):
|
|
|
|
"""Test saving and restoring a state."""
|
2019-07-31 19:25:30 +00:00
|
|
|
hass = hass_recorder({"exclude": {"entities": "test.recorder"}})
|
|
|
|
states = _add_entities(hass, ["test.recorder", "test2.recorder"])
|
2017-01-09 20:53:30 +00:00
|
|
|
assert len(states) == 1
|
2020-06-23 17:57:52 +00:00
|
|
|
assert _state_empty_context(hass, "test2.recorder") == states[0]
|
2017-01-09 20:53:30 +00:00
|
|
|
|
|
|
|
|
|
|
|
def test_saving_state_exclude_domain_include_entity(hass_recorder):
|
|
|
|
"""Test saving and restoring a state."""
|
2019-07-31 19:25:30 +00:00
|
|
|
hass = hass_recorder(
|
|
|
|
{"include": {"entities": "test.recorder"}, "exclude": {"domains": "test"}}
|
|
|
|
)
|
|
|
|
states = _add_entities(hass, ["test.recorder", "test2.recorder"])
|
2017-01-09 20:53:30 +00:00
|
|
|
assert len(states) == 2
|
|
|
|
|
|
|
|
|
2020-06-24 01:02:29 +00:00
|
|
|
def test_saving_state_exclude_domain_glob_include_entity(hass_recorder):
|
|
|
|
"""Test saving and restoring a state."""
|
|
|
|
hass = hass_recorder(
|
|
|
|
{
|
|
|
|
"include": {"entities": ["test.recorder", "test.excluded_entity"]},
|
|
|
|
"exclude": {"domains": "test", "entity_globs": "*._excluded_*"},
|
|
|
|
}
|
|
|
|
)
|
|
|
|
states = _add_entities(
|
|
|
|
hass, ["test.recorder", "test2.recorder", "test.excluded_entity"]
|
|
|
|
)
|
|
|
|
assert len(states) == 3
|
|
|
|
|
|
|
|
|
2017-01-09 20:53:30 +00:00
|
|
|
def test_saving_state_include_domain_exclude_entity(hass_recorder):
|
|
|
|
"""Test saving and restoring a state."""
|
2019-07-31 19:25:30 +00:00
|
|
|
hass = hass_recorder(
|
|
|
|
{"exclude": {"entities": "test.recorder"}, "include": {"domains": "test"}}
|
|
|
|
)
|
|
|
|
states = _add_entities(hass, ["test.recorder", "test2.recorder", "test.ok"])
|
2017-01-09 20:53:30 +00:00
|
|
|
assert len(states) == 1
|
2020-06-23 17:57:52 +00:00
|
|
|
assert _state_empty_context(hass, "test.ok") == states[0]
|
|
|
|
assert _state_empty_context(hass, "test.ok").state == "state2"
|
2017-03-03 06:44:52 +00:00
|
|
|
|
|
|
|
|
2020-06-24 01:02:29 +00:00
|
|
|
def test_saving_state_include_domain_glob_exclude_entity(hass_recorder):
|
|
|
|
"""Test saving and restoring a state."""
|
|
|
|
hass = hass_recorder(
|
|
|
|
{
|
|
|
|
"exclude": {"entities": ["test.recorder", "test2.included_entity"]},
|
|
|
|
"include": {"domains": "test", "entity_globs": "*._included_*"},
|
|
|
|
}
|
|
|
|
)
|
|
|
|
states = _add_entities(
|
|
|
|
hass, ["test.recorder", "test2.recorder", "test.ok", "test2.included_entity"]
|
|
|
|
)
|
|
|
|
assert len(states) == 1
|
|
|
|
assert _state_empty_context(hass, "test.ok") == states[0]
|
|
|
|
assert _state_empty_context(hass, "test.ok").state == "state2"
|
|
|
|
|
|
|
|
|
2020-07-04 06:08:46 +00:00
|
|
|
def test_saving_state_and_removing_entity(hass, hass_recorder):
|
|
|
|
"""Test saving the state of a removed entity."""
|
|
|
|
hass = hass_recorder()
|
|
|
|
entity_id = "lock.mine"
|
|
|
|
hass.states.set(entity_id, STATE_LOCKED)
|
|
|
|
hass.states.set(entity_id, STATE_UNLOCKED)
|
|
|
|
hass.states.async_remove(entity_id)
|
|
|
|
|
|
|
|
wait_recording_done(hass)
|
|
|
|
|
|
|
|
with session_scope(hass=hass) as session:
|
|
|
|
states = list(session.query(States))
|
|
|
|
assert len(states) == 3
|
|
|
|
assert states[0].entity_id == entity_id
|
|
|
|
assert states[0].state == STATE_LOCKED
|
|
|
|
assert states[1].entity_id == entity_id
|
|
|
|
assert states[1].state == STATE_UNLOCKED
|
|
|
|
assert states[2].entity_id == entity_id
|
|
|
|
assert states[2].state is None
|
|
|
|
|
|
|
|
|
2021-04-12 06:43:54 +00:00
|
|
|
def test_recorder_setup_failure(hass):
|
2017-03-03 06:44:52 +00:00
|
|
|
"""Test some exceptions."""
|
2021-04-12 06:43:54 +00:00
|
|
|
with patch.object(Recorder, "_setup_connection") as setup, patch(
|
|
|
|
"homeassistant.components.recorder.time.sleep"
|
|
|
|
):
|
|
|
|
setup.side_effect = ImportError("driver not found")
|
|
|
|
rec = _default_recorder(hass)
|
|
|
|
rec.async_initialize()
|
|
|
|
rec.start()
|
|
|
|
rec.join()
|
|
|
|
|
|
|
|
hass.stop()
|
2017-03-03 06:44:52 +00:00
|
|
|
|
2021-04-12 06:43:54 +00:00
|
|
|
|
|
|
|
def test_recorder_setup_failure_without_event_listener(hass):
|
|
|
|
"""Test recorder setup failure when the event listener is not setup."""
|
2019-07-31 19:25:30 +00:00
|
|
|
with patch.object(Recorder, "_setup_connection") as setup, patch(
|
|
|
|
"homeassistant.components.recorder.time.sleep"
|
|
|
|
):
|
2017-03-03 06:44:52 +00:00
|
|
|
setup.side_effect = ImportError("driver not found")
|
2021-04-12 06:43:54 +00:00
|
|
|
rec = _default_recorder(hass)
|
2017-03-03 06:44:52 +00:00
|
|
|
rec.start()
|
|
|
|
rec.join()
|
|
|
|
|
|
|
|
hass.stop()
|
2019-06-08 23:18:29 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def test_defaults_set(hass):
|
|
|
|
"""Test the config defaults are set."""
|
|
|
|
recorder_config = None
|
|
|
|
|
|
|
|
async def mock_setup(hass, config):
|
|
|
|
"""Mock setup."""
|
|
|
|
nonlocal recorder_config
|
2019-07-31 19:25:30 +00:00
|
|
|
recorder_config = config["recorder"]
|
2019-06-08 23:18:29 +00:00
|
|
|
return True
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
with patch("homeassistant.components.recorder.async_setup", side_effect=mock_setup):
|
|
|
|
assert await async_setup_component(hass, "history", {})
|
2019-06-08 23:18:29 +00:00
|
|
|
|
|
|
|
assert recorder_config is not None
|
2020-06-24 01:02:29 +00:00
|
|
|
# pylint: disable=unsubscriptable-object
|
2020-04-16 22:11:36 +00:00
|
|
|
assert recorder_config["auto_purge"]
|
2019-07-31 19:25:30 +00:00
|
|
|
assert recorder_config["purge_keep_days"] == 10
|
2020-04-16 22:11:36 +00:00
|
|
|
|
|
|
|
|
2020-12-31 22:04:12 +00:00
|
|
|
def run_tasks_at_time(hass, test_time):
|
|
|
|
"""Advance the clock and wait for any callbacks to finish."""
|
|
|
|
fire_time_changed(hass, test_time)
|
|
|
|
hass.block_till_done()
|
|
|
|
hass.data[DATA_INSTANCE].block_till_done()
|
|
|
|
|
|
|
|
|
2020-04-16 22:11:36 +00:00
|
|
|
def test_auto_purge(hass_recorder):
|
2021-05-16 17:23:37 +00:00
|
|
|
"""Test periodic purge scheduling."""
|
2020-04-16 22:11:36 +00:00
|
|
|
hass = hass_recorder()
|
|
|
|
|
|
|
|
original_tz = dt_util.DEFAULT_TIME_ZONE
|
|
|
|
|
|
|
|
tz = dt_util.get_time_zone("Europe/Copenhagen")
|
|
|
|
dt_util.set_default_time_zone(tz)
|
|
|
|
|
2021-05-16 17:23:37 +00:00
|
|
|
# Purging is scheduled to happen at 4:12am every day. Exercise this behavior by
|
|
|
|
# firing time changed events and advancing the clock around this time. Pick an
|
|
|
|
# arbitrary year in the future to avoid boundary conditions relative to the current
|
|
|
|
# date.
|
2020-12-31 22:04:12 +00:00
|
|
|
#
|
|
|
|
# The clock is started at 4:15am then advanced forward below
|
2020-07-21 06:18:31 +00:00
|
|
|
now = dt_util.utcnow()
|
2021-05-08 05:46:26 +00:00
|
|
|
test_time = datetime(now.year + 2, 1, 1, 4, 15, 0, tzinfo=tz)
|
2020-12-31 22:04:12 +00:00
|
|
|
run_tasks_at_time(hass, test_time)
|
2020-04-16 22:11:36 +00:00
|
|
|
|
|
|
|
with patch(
|
2020-06-26 17:27:45 +00:00
|
|
|
"homeassistant.components.recorder.purge.purge_old_data", return_value=True
|
2021-05-17 21:27:51 +00:00
|
|
|
) as purge_old_data, patch(
|
|
|
|
"homeassistant.components.recorder.perodic_db_cleanups"
|
|
|
|
) as perodic_db_cleanups:
|
2020-12-31 22:04:12 +00:00
|
|
|
# Advance one day, and the purge task should run
|
|
|
|
test_time = test_time + timedelta(days=1)
|
|
|
|
run_tasks_at_time(hass, test_time)
|
|
|
|
assert len(purge_old_data.mock_calls) == 1
|
2021-05-17 21:27:51 +00:00
|
|
|
assert len(perodic_db_cleanups.mock_calls) == 1
|
2020-12-31 22:04:12 +00:00
|
|
|
|
|
|
|
purge_old_data.reset_mock()
|
2021-05-17 21:27:51 +00:00
|
|
|
perodic_db_cleanups.reset_mock()
|
2020-12-31 22:04:12 +00:00
|
|
|
|
|
|
|
# Advance one day, and the purge task should run again
|
|
|
|
test_time = test_time + timedelta(days=1)
|
|
|
|
run_tasks_at_time(hass, test_time)
|
|
|
|
assert len(purge_old_data.mock_calls) == 1
|
2021-05-17 21:27:51 +00:00
|
|
|
assert len(perodic_db_cleanups.mock_calls) == 1
|
2020-12-31 22:04:12 +00:00
|
|
|
|
|
|
|
purge_old_data.reset_mock()
|
2021-05-17 21:27:51 +00:00
|
|
|
perodic_db_cleanups.reset_mock()
|
2020-12-31 22:04:12 +00:00
|
|
|
|
|
|
|
# Advance less than one full day. The alarm should not yet fire.
|
|
|
|
test_time = test_time + timedelta(hours=23)
|
|
|
|
run_tasks_at_time(hass, test_time)
|
|
|
|
assert len(purge_old_data.mock_calls) == 0
|
2021-05-17 21:27:51 +00:00
|
|
|
assert len(perodic_db_cleanups.mock_calls) == 0
|
2020-04-16 22:11:36 +00:00
|
|
|
|
2020-12-31 22:04:12 +00:00
|
|
|
# Advance to the next day and fire the alarm again
|
|
|
|
test_time = test_time + timedelta(hours=1)
|
|
|
|
run_tasks_at_time(hass, test_time)
|
2020-04-16 22:11:36 +00:00
|
|
|
assert len(purge_old_data.mock_calls) == 1
|
2021-05-17 21:27:51 +00:00
|
|
|
assert len(perodic_db_cleanups.mock_calls) == 1
|
|
|
|
|
|
|
|
dt_util.set_default_time_zone(original_tz)
|
|
|
|
|
|
|
|
|
|
|
|
def test_auto_purge_disabled(hass_recorder):
|
|
|
|
"""Test periodic db cleanup still run when auto purge is disabled."""
|
|
|
|
hass = hass_recorder({CONF_AUTO_PURGE: False})
|
|
|
|
|
|
|
|
original_tz = dt_util.DEFAULT_TIME_ZONE
|
|
|
|
|
|
|
|
tz = dt_util.get_time_zone("Europe/Copenhagen")
|
|
|
|
dt_util.set_default_time_zone(tz)
|
|
|
|
|
|
|
|
# Purging is scheduled to happen at 4:12am every day. We want
|
|
|
|
# to verify that when auto purge is disabled perodic db cleanups
|
|
|
|
# are still scheduled
|
|
|
|
#
|
|
|
|
# The clock is started at 4:15am then advanced forward below
|
|
|
|
now = dt_util.utcnow()
|
|
|
|
test_time = datetime(now.year + 2, 1, 1, 4, 15, 0, tzinfo=tz)
|
|
|
|
run_tasks_at_time(hass, test_time)
|
|
|
|
|
|
|
|
with patch(
|
|
|
|
"homeassistant.components.recorder.purge.purge_old_data", return_value=True
|
|
|
|
) as purge_old_data, patch(
|
|
|
|
"homeassistant.components.recorder.perodic_db_cleanups"
|
|
|
|
) as perodic_db_cleanups:
|
|
|
|
# Advance one day, and the purge task should run
|
|
|
|
test_time = test_time + timedelta(days=1)
|
|
|
|
run_tasks_at_time(hass, test_time)
|
|
|
|
assert len(purge_old_data.mock_calls) == 0
|
|
|
|
assert len(perodic_db_cleanups.mock_calls) == 1
|
|
|
|
|
|
|
|
purge_old_data.reset_mock()
|
|
|
|
perodic_db_cleanups.reset_mock()
|
2020-04-16 22:11:36 +00:00
|
|
|
|
|
|
|
dt_util.set_default_time_zone(original_tz)
|
2020-06-18 03:26:41 +00:00
|
|
|
|
|
|
|
|
2021-05-20 11:05:15 +00:00
|
|
|
@pytest.mark.parametrize("enable_statistics", [True])
|
2021-05-16 17:23:37 +00:00
|
|
|
def test_auto_statistics(hass_recorder):
|
|
|
|
"""Test periodic statistics scheduling."""
|
|
|
|
hass = hass_recorder()
|
|
|
|
|
|
|
|
original_tz = dt_util.DEFAULT_TIME_ZONE
|
|
|
|
|
|
|
|
tz = dt_util.get_time_zone("Europe/Copenhagen")
|
|
|
|
dt_util.set_default_time_zone(tz)
|
|
|
|
|
|
|
|
# Statistics is scheduled to happen at *:12am every hour. Exercise this behavior by
|
|
|
|
# firing time changed events and advancing the clock around this time. Pick an
|
|
|
|
# arbitrary year in the future to avoid boundary conditions relative to the current
|
|
|
|
# date.
|
|
|
|
#
|
|
|
|
# The clock is started at 4:15am then advanced forward below
|
|
|
|
now = dt_util.utcnow()
|
|
|
|
test_time = datetime(now.year + 2, 1, 1, 4, 15, 0, tzinfo=tz)
|
|
|
|
run_tasks_at_time(hass, test_time)
|
|
|
|
|
|
|
|
with patch(
|
|
|
|
"homeassistant.components.recorder.statistics.compile_statistics",
|
|
|
|
return_value=True,
|
|
|
|
) as compile_statistics:
|
|
|
|
# Advance one hour, and the statistics task should run
|
|
|
|
test_time = test_time + timedelta(hours=1)
|
|
|
|
run_tasks_at_time(hass, test_time)
|
|
|
|
assert len(compile_statistics.mock_calls) == 1
|
|
|
|
|
|
|
|
compile_statistics.reset_mock()
|
|
|
|
|
|
|
|
# Advance one hour, and the statistics task should run again
|
|
|
|
test_time = test_time + timedelta(hours=1)
|
|
|
|
run_tasks_at_time(hass, test_time)
|
|
|
|
assert len(compile_statistics.mock_calls) == 1
|
|
|
|
|
|
|
|
compile_statistics.reset_mock()
|
|
|
|
|
|
|
|
# Advance less than one full hour. The task should not run.
|
|
|
|
test_time = test_time + timedelta(minutes=50)
|
|
|
|
run_tasks_at_time(hass, test_time)
|
|
|
|
assert len(compile_statistics.mock_calls) == 0
|
|
|
|
|
|
|
|
# Advance to the next hour, and the statistics task should run again
|
|
|
|
test_time = test_time + timedelta(hours=1)
|
|
|
|
run_tasks_at_time(hass, test_time)
|
|
|
|
assert len(compile_statistics.mock_calls) == 1
|
|
|
|
|
|
|
|
dt_util.set_default_time_zone(original_tz)
|
|
|
|
|
|
|
|
|
2020-06-18 03:26:41 +00:00
|
|
|
def test_saving_sets_old_state(hass_recorder):
|
|
|
|
"""Test saving sets old state."""
|
|
|
|
hass = hass_recorder()
|
|
|
|
|
|
|
|
hass.states.set("test.one", "on", {})
|
|
|
|
hass.states.set("test.two", "on", {})
|
|
|
|
wait_recording_done(hass)
|
|
|
|
hass.states.set("test.one", "off", {})
|
|
|
|
hass.states.set("test.two", "off", {})
|
|
|
|
wait_recording_done(hass)
|
|
|
|
|
|
|
|
with session_scope(hass=hass) as session:
|
|
|
|
states = list(session.query(States))
|
|
|
|
assert len(states) == 4
|
|
|
|
|
|
|
|
assert states[0].entity_id == "test.one"
|
|
|
|
assert states[1].entity_id == "test.two"
|
|
|
|
assert states[2].entity_id == "test.one"
|
|
|
|
assert states[3].entity_id == "test.two"
|
|
|
|
|
|
|
|
assert states[0].old_state_id is None
|
|
|
|
assert states[1].old_state_id is None
|
|
|
|
assert states[2].old_state_id == states[0].state_id
|
|
|
|
assert states[3].old_state_id == states[1].state_id
|
2020-06-19 17:03:06 +00:00
|
|
|
|
|
|
|
|
|
|
|
def test_saving_state_with_serializable_data(hass_recorder, caplog):
|
|
|
|
"""Test saving data that cannot be serialized does not crash."""
|
|
|
|
hass = hass_recorder()
|
|
|
|
|
2021-04-12 06:43:54 +00:00
|
|
|
hass.bus.fire("bad_event", {"fail": CannotSerializeMe()})
|
2020-06-19 17:03:06 +00:00
|
|
|
hass.states.set("test.one", "on", {"fail": CannotSerializeMe()})
|
|
|
|
wait_recording_done(hass)
|
|
|
|
hass.states.set("test.two", "on", {})
|
|
|
|
wait_recording_done(hass)
|
|
|
|
hass.states.set("test.two", "off", {})
|
|
|
|
wait_recording_done(hass)
|
|
|
|
|
|
|
|
with session_scope(hass=hass) as session:
|
|
|
|
states = list(session.query(States))
|
|
|
|
assert len(states) == 2
|
|
|
|
|
|
|
|
assert states[0].entity_id == "test.two"
|
|
|
|
assert states[1].entity_id == "test.two"
|
|
|
|
assert states[0].old_state_id is None
|
|
|
|
assert states[1].old_state_id == states[0].state_id
|
|
|
|
|
|
|
|
assert "State is not JSON serializable" in caplog.text
|
|
|
|
|
|
|
|
|
|
|
|
def test_run_information(hass_recorder):
|
|
|
|
"""Ensure run_information returns expected data."""
|
|
|
|
before_start_recording = dt_util.utcnow()
|
|
|
|
hass = hass_recorder()
|
|
|
|
run_info = run_information_from_instance(hass)
|
|
|
|
assert isinstance(run_info, RecorderRuns)
|
|
|
|
assert run_info.closed_incorrect is False
|
|
|
|
|
|
|
|
with session_scope(hass=hass) as session:
|
|
|
|
run_info = run_information_with_session(session)
|
|
|
|
assert isinstance(run_info, RecorderRuns)
|
|
|
|
assert run_info.closed_incorrect is False
|
|
|
|
|
|
|
|
run_info = run_information(hass)
|
|
|
|
assert isinstance(run_info, RecorderRuns)
|
|
|
|
assert run_info.closed_incorrect is False
|
|
|
|
|
|
|
|
hass.states.set("test.two", "on", {})
|
|
|
|
wait_recording_done(hass)
|
|
|
|
run_info = run_information(hass)
|
|
|
|
assert isinstance(run_info, RecorderRuns)
|
|
|
|
assert run_info.closed_incorrect is False
|
|
|
|
|
|
|
|
run_info = run_information(hass, before_start_recording)
|
|
|
|
assert run_info is None
|
|
|
|
|
|
|
|
run_info = run_information(hass, dt_util.utcnow())
|
|
|
|
assert isinstance(run_info, RecorderRuns)
|
|
|
|
assert run_info.closed_incorrect is False
|
|
|
|
|
|
|
|
|
2021-02-24 13:26:05 +00:00
|
|
|
def test_has_services(hass_recorder):
|
|
|
|
"""Test the services exist."""
|
|
|
|
hass = hass_recorder()
|
|
|
|
|
|
|
|
assert hass.services.has_service(DOMAIN, SERVICE_DISABLE)
|
|
|
|
assert hass.services.has_service(DOMAIN, SERVICE_ENABLE)
|
|
|
|
assert hass.services.has_service(DOMAIN, SERVICE_PURGE)
|
|
|
|
|
|
|
|
|
|
|
|
def test_service_disable_events_not_recording(hass, hass_recorder):
|
|
|
|
"""Test that events are not recorded when recorder is disabled using service."""
|
|
|
|
hass = hass_recorder()
|
|
|
|
|
|
|
|
assert hass.services.call(
|
|
|
|
DOMAIN,
|
|
|
|
SERVICE_DISABLE,
|
|
|
|
{},
|
|
|
|
blocking=True,
|
|
|
|
)
|
|
|
|
|
|
|
|
event_type = "EVENT_TEST"
|
|
|
|
|
|
|
|
events = []
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def event_listener(event):
|
|
|
|
"""Record events from eventbus."""
|
|
|
|
if event.event_type == event_type:
|
|
|
|
events.append(event)
|
|
|
|
|
|
|
|
hass.bus.listen(MATCH_ALL, event_listener)
|
|
|
|
|
|
|
|
event_data1 = {"test_attr": 5, "test_attr_10": "nice"}
|
|
|
|
hass.bus.fire(event_type, event_data1)
|
|
|
|
wait_recording_done(hass)
|
|
|
|
|
|
|
|
assert len(events) == 1
|
|
|
|
event = events[0]
|
|
|
|
|
|
|
|
with session_scope(hass=hass) as session:
|
|
|
|
db_events = list(session.query(Events).filter_by(event_type=event_type))
|
|
|
|
assert len(db_events) == 0
|
|
|
|
|
|
|
|
assert hass.services.call(
|
|
|
|
DOMAIN,
|
|
|
|
SERVICE_ENABLE,
|
|
|
|
{},
|
|
|
|
blocking=True,
|
|
|
|
)
|
|
|
|
|
|
|
|
event_data2 = {"attr_one": 5, "attr_two": "nice"}
|
|
|
|
hass.bus.fire(event_type, event_data2)
|
|
|
|
wait_recording_done(hass)
|
|
|
|
|
|
|
|
assert len(events) == 2
|
|
|
|
assert events[0] != events[1]
|
|
|
|
assert events[0].data != events[1].data
|
|
|
|
|
|
|
|
with session_scope(hass=hass) as session:
|
|
|
|
db_events = list(session.query(Events).filter_by(event_type=event_type))
|
|
|
|
assert len(db_events) == 1
|
|
|
|
db_event = db_events[0].to_native()
|
|
|
|
|
|
|
|
event = events[1]
|
|
|
|
|
|
|
|
assert event.event_type == db_event.event_type
|
|
|
|
assert event.data == db_event.data
|
|
|
|
assert event.origin == db_event.origin
|
|
|
|
assert event.time_fired.replace(microsecond=0) == db_event.time_fired.replace(
|
|
|
|
microsecond=0
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
def test_service_disable_states_not_recording(hass, hass_recorder):
|
|
|
|
"""Test that state changes are not recorded when recorder is disabled using service."""
|
|
|
|
hass = hass_recorder()
|
|
|
|
|
|
|
|
assert hass.services.call(
|
|
|
|
DOMAIN,
|
|
|
|
SERVICE_DISABLE,
|
|
|
|
{},
|
|
|
|
blocking=True,
|
|
|
|
)
|
|
|
|
|
|
|
|
hass.states.set("test.one", "on", {})
|
|
|
|
wait_recording_done(hass)
|
|
|
|
|
|
|
|
with session_scope(hass=hass) as session:
|
|
|
|
assert len(list(session.query(States))) == 0
|
|
|
|
|
|
|
|
assert hass.services.call(
|
|
|
|
DOMAIN,
|
|
|
|
SERVICE_ENABLE,
|
|
|
|
{},
|
|
|
|
blocking=True,
|
|
|
|
)
|
|
|
|
|
|
|
|
hass.states.set("test.two", "off", {})
|
|
|
|
wait_recording_done(hass)
|
|
|
|
|
|
|
|
with session_scope(hass=hass) as session:
|
|
|
|
db_states = list(session.query(States))
|
|
|
|
assert len(db_states) == 1
|
|
|
|
assert db_states[0].event_id > 0
|
|
|
|
assert db_states[0].to_native() == _state_empty_context(hass, "test.two")
|
|
|
|
|
|
|
|
|
|
|
|
def test_service_disable_run_information_recorded(tmpdir):
|
|
|
|
"""Test that runs are still recorded when recorder is disabled."""
|
|
|
|
test_db_file = tmpdir.mkdir("sqlite").join("test_run_info.db")
|
|
|
|
dburl = f"{SQLITE_URL_PREFIX}//{test_db_file}"
|
|
|
|
|
|
|
|
hass = get_test_home_assistant()
|
|
|
|
setup_component(hass, DOMAIN, {DOMAIN: {CONF_DB_URL: dburl}})
|
|
|
|
hass.start()
|
|
|
|
wait_recording_done(hass)
|
|
|
|
|
|
|
|
with session_scope(hass=hass) as session:
|
|
|
|
db_run_info = list(session.query(RecorderRuns))
|
|
|
|
assert len(db_run_info) == 1
|
|
|
|
assert db_run_info[0].start is not None
|
|
|
|
assert db_run_info[0].end is None
|
|
|
|
|
|
|
|
assert hass.services.call(
|
|
|
|
DOMAIN,
|
|
|
|
SERVICE_DISABLE,
|
|
|
|
{},
|
|
|
|
blocking=True,
|
|
|
|
)
|
|
|
|
|
|
|
|
wait_recording_done(hass)
|
|
|
|
hass.stop()
|
|
|
|
|
|
|
|
hass = get_test_home_assistant()
|
|
|
|
setup_component(hass, DOMAIN, {DOMAIN: {CONF_DB_URL: dburl}})
|
|
|
|
hass.start()
|
|
|
|
wait_recording_done(hass)
|
|
|
|
|
|
|
|
with session_scope(hass=hass) as session:
|
|
|
|
db_run_info = list(session.query(RecorderRuns))
|
|
|
|
assert len(db_run_info) == 2
|
|
|
|
assert db_run_info[0].start is not None
|
|
|
|
assert db_run_info[0].end is not None
|
|
|
|
assert db_run_info[1].start is not None
|
|
|
|
assert db_run_info[1].end is None
|
|
|
|
|
|
|
|
hass.stop()
|
|
|
|
|
|
|
|
|
2020-06-19 17:03:06 +00:00
|
|
|
class CannotSerializeMe:
|
|
|
|
"""A class that the JSONEncoder cannot serialize."""
|
2021-02-20 07:26:24 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def test_database_corruption_while_running(hass, tmpdir, caplog):
|
|
|
|
"""Test we can recover from sqlite3 db corruption."""
|
|
|
|
|
|
|
|
def _create_tmpdir_for_test_db():
|
|
|
|
return tmpdir.mkdir("sqlite").join("test.db")
|
|
|
|
|
|
|
|
test_db_file = await hass.async_add_executor_job(_create_tmpdir_for_test_db)
|
|
|
|
dburl = f"{SQLITE_URL_PREFIX}//{test_db_file}"
|
|
|
|
|
|
|
|
assert await async_setup_component(hass, DOMAIN, {DOMAIN: {CONF_DB_URL: dburl}})
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
caplog.clear()
|
|
|
|
|
|
|
|
hass.states.async_set("test.lost", "on", {})
|
|
|
|
|
2021-04-27 05:20:31 +00:00
|
|
|
sqlite3_exception = DatabaseError("statement", {}, [])
|
|
|
|
sqlite3_exception.__cause__ = sqlite3.DatabaseError()
|
|
|
|
|
2021-04-12 06:43:54 +00:00
|
|
|
with patch.object(
|
|
|
|
hass.data[DATA_INSTANCE].event_session,
|
|
|
|
"close",
|
|
|
|
side_effect=OperationalError("statement", {}, []),
|
|
|
|
):
|
|
|
|
await async_wait_recording_done_without_instance(hass)
|
|
|
|
await hass.async_add_executor_job(corrupt_db_file, test_db_file)
|
|
|
|
await async_wait_recording_done_without_instance(hass)
|
|
|
|
|
2021-04-27 05:20:31 +00:00
|
|
|
with patch.object(
|
|
|
|
hass.data[DATA_INSTANCE].event_session,
|
|
|
|
"commit",
|
|
|
|
side_effect=[sqlite3_exception, None],
|
|
|
|
):
|
|
|
|
# This state will not be recorded because
|
|
|
|
# the database corruption will be discovered
|
|
|
|
# and we will have to rollback to recover
|
|
|
|
hass.states.async_set("test.one", "off", {})
|
|
|
|
await async_wait_recording_done_without_instance(hass)
|
2021-02-20 07:26:24 +00:00
|
|
|
|
|
|
|
assert "Unrecoverable sqlite3 database corruption detected" in caplog.text
|
|
|
|
assert "The system will rename the corrupt database file" in caplog.text
|
|
|
|
assert "Connected to recorder database" in caplog.text
|
|
|
|
|
|
|
|
# This state should go into the new database
|
|
|
|
hass.states.async_set("test.two", "on", {})
|
2021-03-11 17:52:07 +00:00
|
|
|
await async_wait_recording_done_without_instance(hass)
|
2021-02-20 07:26:24 +00:00
|
|
|
|
|
|
|
def _get_last_state():
|
|
|
|
with session_scope(hass=hass) as session:
|
|
|
|
db_states = list(session.query(States))
|
|
|
|
assert len(db_states) == 1
|
|
|
|
assert db_states[0].event_id > 0
|
|
|
|
return db_states[0].to_native()
|
|
|
|
|
|
|
|
state = await hass.async_add_executor_job(_get_last_state)
|
|
|
|
assert state.entity_id == "test.two"
|
|
|
|
assert state.state == "on"
|
|
|
|
|
|
|
|
hass.bus.async_fire(EVENT_HOMEASSISTANT_STOP)
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
hass.stop()
|
2021-05-02 03:30:28 +00:00
|
|
|
|
|
|
|
|
|
|
|
def test_entity_id_filter(hass_recorder):
|
|
|
|
"""Test that entity ID filtering filters string and list."""
|
|
|
|
hass = hass_recorder(
|
|
|
|
{"include": {"domains": "hello"}, "exclude": {"domains": "hidden_domain"}}
|
|
|
|
)
|
|
|
|
|
|
|
|
for idx, data in enumerate(
|
|
|
|
(
|
|
|
|
{},
|
|
|
|
{"entity_id": "hello.world"},
|
|
|
|
{"entity_id": ["hello.world"]},
|
|
|
|
{"entity_id": ["hello.world", "hidden_domain.person"]},
|
|
|
|
{"entity_id": {"unexpected": "data"}},
|
|
|
|
)
|
|
|
|
):
|
|
|
|
hass.bus.fire("hello", data)
|
|
|
|
wait_recording_done(hass)
|
|
|
|
|
|
|
|
with session_scope(hass=hass) as session:
|
|
|
|
db_events = list(session.query(Events).filter_by(event_type="hello"))
|
|
|
|
assert len(db_events) == idx + 1, data
|
|
|
|
|
|
|
|
for data in (
|
|
|
|
{"entity_id": "hidden_domain.person"},
|
|
|
|
{"entity_id": ["hidden_domain.person"]},
|
|
|
|
):
|
|
|
|
hass.bus.fire("hello", data)
|
|
|
|
wait_recording_done(hass)
|
|
|
|
|
|
|
|
with session_scope(hass=hass) as session:
|
|
|
|
db_events = list(session.query(Events).filter_by(event_type="hello"))
|
|
|
|
# Keep referring idx + 1, as no new events are being added
|
|
|
|
assert len(db_events) == idx + 1, data
|