2017-02-21 07:40:27 +00:00
|
|
|
"""The tests for the Restore component."""
|
2021-04-17 07:03:18 +00:00
|
|
|
from datetime import datetime, timedelta
|
2021-01-01 21:31:56 +00:00
|
|
|
from unittest.mock import patch
|
2017-02-21 07:40:27 +00:00
|
|
|
|
2021-04-17 07:03:18 +00:00
|
|
|
from homeassistant.const import EVENT_HOMEASSISTANT_START, EVENT_HOMEASSISTANT_STOP
|
2018-11-28 12:16:43 +00:00
|
|
|
from homeassistant.core import CoreState, State
|
|
|
|
from homeassistant.exceptions import HomeAssistantError
|
|
|
|
from homeassistant.helpers.entity import Entity
|
2017-02-21 07:40:27 +00:00
|
|
|
from homeassistant.helpers.restore_state import (
|
2019-07-31 19:25:30 +00:00
|
|
|
DATA_RESTORE_STATE_TASK,
|
|
|
|
STORAGE_KEY,
|
2019-12-09 15:52:24 +00:00
|
|
|
RestoreEntity,
|
|
|
|
RestoreStateData,
|
|
|
|
StoredState,
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2018-11-28 12:16:43 +00:00
|
|
|
from homeassistant.util import dt as dt_util
|
2017-02-21 07:40:27 +00:00
|
|
|
|
2021-04-17 07:03:18 +00:00
|
|
|
from tests.common import async_fire_time_changed
|
|
|
|
|
2017-02-21 07:40:27 +00:00
|
|
|
|
2018-11-28 12:16:43 +00:00
|
|
|
async def test_caching_data(hass):
|
|
|
|
"""Test that we cache data."""
|
2018-11-30 07:26:19 +00:00
|
|
|
now = dt_util.utcnow()
|
|
|
|
stored_states = [
|
2019-07-31 19:25:30 +00:00
|
|
|
StoredState(State("input_boolean.b0", "on"), now),
|
|
|
|
StoredState(State("input_boolean.b1", "on"), now),
|
|
|
|
StoredState(State("input_boolean.b2", "on"), now),
|
2017-02-21 07:40:27 +00:00
|
|
|
]
|
|
|
|
|
2018-11-28 12:16:43 +00:00
|
|
|
data = await RestoreStateData.async_get_instance(hass)
|
2020-08-12 20:35:24 +00:00
|
|
|
await hass.async_block_till_done()
|
2018-11-30 07:26:19 +00:00
|
|
|
await data.store.async_save([state.as_dict() for state in stored_states])
|
2017-02-21 07:40:27 +00:00
|
|
|
|
2018-11-28 12:16:43 +00:00
|
|
|
# Emulate a fresh load
|
2021-09-11 19:02:01 +00:00
|
|
|
hass.data.pop(DATA_RESTORE_STATE_TASK)
|
2018-11-28 12:16:43 +00:00
|
|
|
|
|
|
|
entity = RestoreEntity()
|
|
|
|
entity.hass = hass
|
2019-07-31 19:25:30 +00:00
|
|
|
entity.entity_id = "input_boolean.b1"
|
2018-11-28 12:16:43 +00:00
|
|
|
|
|
|
|
# Mock that only b1 is present this run
|
2019-07-31 19:25:30 +00:00
|
|
|
with patch(
|
|
|
|
"homeassistant.helpers.restore_state.Store.async_save"
|
|
|
|
) as mock_write_data:
|
2018-11-28 12:16:43 +00:00
|
|
|
state = await entity.async_get_last_state()
|
2020-08-12 20:35:24 +00:00
|
|
|
await hass.async_block_till_done()
|
2017-02-21 07:40:27 +00:00
|
|
|
|
|
|
|
assert state is not None
|
2019-07-31 19:25:30 +00:00
|
|
|
assert state.entity_id == "input_boolean.b1"
|
|
|
|
assert state.state == "on"
|
2017-02-21 07:40:27 +00:00
|
|
|
|
2018-11-28 12:16:43 +00:00
|
|
|
assert mock_write_data.called
|
2017-02-21 07:40:27 +00:00
|
|
|
|
2017-02-22 08:15:48 +00:00
|
|
|
|
2021-04-17 07:03:18 +00:00
|
|
|
async def test_periodic_write(hass):
|
|
|
|
"""Test that we write periodiclly but not after stop."""
|
|
|
|
data = await RestoreStateData.async_get_instance(hass)
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
await data.store.async_save([])
|
|
|
|
|
|
|
|
# Emulate a fresh load
|
2021-09-11 19:02:01 +00:00
|
|
|
hass.data.pop(DATA_RESTORE_STATE_TASK)
|
2021-04-17 07:03:18 +00:00
|
|
|
|
|
|
|
entity = RestoreEntity()
|
|
|
|
entity.hass = hass
|
|
|
|
entity.entity_id = "input_boolean.b1"
|
|
|
|
|
|
|
|
with patch(
|
|
|
|
"homeassistant.helpers.restore_state.Store.async_save"
|
|
|
|
) as mock_write_data:
|
|
|
|
await entity.async_get_last_state()
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
|
|
assert mock_write_data.called
|
|
|
|
|
|
|
|
with patch(
|
|
|
|
"homeassistant.helpers.restore_state.Store.async_save"
|
|
|
|
) as mock_write_data:
|
|
|
|
async_fire_time_changed(hass, dt_util.utcnow() + timedelta(minutes=15))
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
|
|
assert mock_write_data.called
|
|
|
|
|
|
|
|
with patch(
|
|
|
|
"homeassistant.helpers.restore_state.Store.async_save"
|
|
|
|
) as mock_write_data:
|
|
|
|
hass.bus.async_fire(EVENT_HOMEASSISTANT_STOP)
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
|
|
assert mock_write_data.called
|
|
|
|
|
|
|
|
with patch(
|
|
|
|
"homeassistant.helpers.restore_state.Store.async_save"
|
|
|
|
) as mock_write_data:
|
|
|
|
async_fire_time_changed(hass, dt_util.utcnow() + timedelta(minutes=30))
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
|
|
assert not mock_write_data.called
|
|
|
|
|
|
|
|
|
2021-08-09 22:38:56 +00:00
|
|
|
async def test_save_persistent_states(hass):
|
|
|
|
"""Test that we cancel the currently running job, save the data, and verify the perdiodic job continues."""
|
|
|
|
data = await RestoreStateData.async_get_instance(hass)
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
await data.store.async_save([])
|
|
|
|
|
|
|
|
# Emulate a fresh load
|
2021-09-11 19:02:01 +00:00
|
|
|
hass.data.pop(DATA_RESTORE_STATE_TASK)
|
2021-08-09 22:38:56 +00:00
|
|
|
|
|
|
|
entity = RestoreEntity()
|
|
|
|
entity.hass = hass
|
|
|
|
entity.entity_id = "input_boolean.b1"
|
|
|
|
|
|
|
|
with patch(
|
|
|
|
"homeassistant.helpers.restore_state.Store.async_save"
|
|
|
|
) as mock_write_data:
|
|
|
|
await entity.async_get_last_state()
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
|
|
# Startup Save
|
|
|
|
assert mock_write_data.called
|
|
|
|
|
|
|
|
with patch(
|
|
|
|
"homeassistant.helpers.restore_state.Store.async_save"
|
|
|
|
) as mock_write_data:
|
|
|
|
async_fire_time_changed(hass, dt_util.utcnow() + timedelta(minutes=10))
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
|
|
# Not quite the first interval
|
|
|
|
assert not mock_write_data.called
|
|
|
|
|
|
|
|
with patch(
|
|
|
|
"homeassistant.helpers.restore_state.Store.async_save"
|
|
|
|
) as mock_write_data:
|
|
|
|
await RestoreStateData.async_save_persistent_states(hass)
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
|
|
assert mock_write_data.called
|
|
|
|
|
|
|
|
with patch(
|
|
|
|
"homeassistant.helpers.restore_state.Store.async_save"
|
|
|
|
) as mock_write_data:
|
|
|
|
async_fire_time_changed(hass, dt_util.utcnow() + timedelta(minutes=20))
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
# Verify still saving
|
|
|
|
assert mock_write_data.called
|
|
|
|
|
|
|
|
with patch(
|
|
|
|
"homeassistant.helpers.restore_state.Store.async_save"
|
|
|
|
) as mock_write_data:
|
|
|
|
hass.bus.async_fire(EVENT_HOMEASSISTANT_STOP)
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
# Verify normal shutdown
|
|
|
|
assert mock_write_data.called
|
|
|
|
|
|
|
|
|
2018-11-28 12:16:43 +00:00
|
|
|
async def test_hass_starting(hass):
|
|
|
|
"""Test that we cache data."""
|
|
|
|
hass.state = CoreState.starting
|
2017-10-03 03:25:04 +00:00
|
|
|
|
2018-11-30 07:26:19 +00:00
|
|
|
now = dt_util.utcnow()
|
|
|
|
stored_states = [
|
2019-07-31 19:25:30 +00:00
|
|
|
StoredState(State("input_boolean.b0", "on"), now),
|
|
|
|
StoredState(State("input_boolean.b1", "on"), now),
|
|
|
|
StoredState(State("input_boolean.b2", "on"), now),
|
2017-10-03 03:25:04 +00:00
|
|
|
]
|
|
|
|
|
2018-11-28 12:16:43 +00:00
|
|
|
data = await RestoreStateData.async_get_instance(hass)
|
2020-08-12 20:35:24 +00:00
|
|
|
await hass.async_block_till_done()
|
2018-11-30 07:26:19 +00:00
|
|
|
await data.store.async_save([state.as_dict() for state in stored_states])
|
2017-10-03 03:25:04 +00:00
|
|
|
|
2018-11-28 12:16:43 +00:00
|
|
|
# Emulate a fresh load
|
2021-09-11 19:02:01 +00:00
|
|
|
hass.state = CoreState.not_running
|
|
|
|
hass.data.pop(DATA_RESTORE_STATE_TASK)
|
2017-10-03 03:25:04 +00:00
|
|
|
|
2018-11-28 12:16:43 +00:00
|
|
|
entity = RestoreEntity()
|
|
|
|
entity.hass = hass
|
2019-07-31 19:25:30 +00:00
|
|
|
entity.entity_id = "input_boolean.b1"
|
2017-10-03 03:25:04 +00:00
|
|
|
|
2018-11-28 12:16:43 +00:00
|
|
|
# Mock that only b1 is present this run
|
2019-07-31 19:25:30 +00:00
|
|
|
states = [State("input_boolean.b1", "on")]
|
|
|
|
with patch(
|
|
|
|
"homeassistant.helpers.restore_state.Store.async_save"
|
|
|
|
) as mock_write_data, patch.object(hass.states, "async_all", return_value=states):
|
2018-11-28 12:16:43 +00:00
|
|
|
state = await entity.async_get_last_state()
|
2020-08-12 20:35:24 +00:00
|
|
|
await hass.async_block_till_done()
|
2017-10-03 03:25:04 +00:00
|
|
|
|
2018-11-28 12:16:43 +00:00
|
|
|
assert state is not None
|
2019-07-31 19:25:30 +00:00
|
|
|
assert state.entity_id == "input_boolean.b1"
|
|
|
|
assert state.state == "on"
|
2017-10-03 03:25:04 +00:00
|
|
|
|
2018-11-28 12:16:43 +00:00
|
|
|
# Assert that no data was written yet, since hass is still starting.
|
|
|
|
assert not mock_write_data.called
|
2017-10-03 03:25:04 +00:00
|
|
|
|
2018-11-28 12:16:43 +00:00
|
|
|
# Finish hass startup
|
2019-07-31 19:25:30 +00:00
|
|
|
with patch(
|
|
|
|
"homeassistant.helpers.restore_state.Store.async_save"
|
|
|
|
) as mock_write_data:
|
2018-11-28 12:16:43 +00:00
|
|
|
hass.bus.async_fire(EVENT_HOMEASSISTANT_START)
|
|
|
|
await hass.async_block_till_done()
|
2017-10-03 03:25:04 +00:00
|
|
|
|
2018-11-28 12:16:43 +00:00
|
|
|
# Assert that this session states were written
|
|
|
|
assert mock_write_data.called
|
2017-10-03 03:25:04 +00:00
|
|
|
|
|
|
|
|
2018-11-28 12:16:43 +00:00
|
|
|
async def test_dump_data(hass):
|
|
|
|
"""Test that we cache data."""
|
|
|
|
states = [
|
2019-07-31 19:25:30 +00:00
|
|
|
State("input_boolean.b0", "on"),
|
|
|
|
State("input_boolean.b1", "on"),
|
|
|
|
State("input_boolean.b2", "on"),
|
2020-01-05 10:58:59 +00:00
|
|
|
State("input_boolean.b5", "unavailable", {"restored": True}),
|
2018-11-28 12:16:43 +00:00
|
|
|
]
|
2017-10-03 03:25:04 +00:00
|
|
|
|
2018-11-28 12:16:43 +00:00
|
|
|
entity = Entity()
|
|
|
|
entity.hass = hass
|
2019-07-31 19:25:30 +00:00
|
|
|
entity.entity_id = "input_boolean.b0"
|
2019-06-26 16:22:51 +00:00
|
|
|
await entity.async_internal_added_to_hass()
|
2018-11-28 12:16:43 +00:00
|
|
|
|
|
|
|
entity = RestoreEntity()
|
|
|
|
entity.hass = hass
|
2019-07-31 19:25:30 +00:00
|
|
|
entity.entity_id = "input_boolean.b1"
|
2019-06-26 16:22:51 +00:00
|
|
|
await entity.async_internal_added_to_hass()
|
2018-11-28 12:16:43 +00:00
|
|
|
|
|
|
|
data = await RestoreStateData.async_get_instance(hass)
|
2018-11-30 07:26:19 +00:00
|
|
|
now = dt_util.utcnow()
|
2018-11-28 12:16:43 +00:00
|
|
|
data.last_states = {
|
2019-07-31 19:25:30 +00:00
|
|
|
"input_boolean.b0": StoredState(State("input_boolean.b0", "off"), now),
|
|
|
|
"input_boolean.b1": StoredState(State("input_boolean.b1", "off"), now),
|
|
|
|
"input_boolean.b2": StoredState(State("input_boolean.b2", "off"), now),
|
|
|
|
"input_boolean.b3": StoredState(State("input_boolean.b3", "off"), now),
|
|
|
|
"input_boolean.b4": StoredState(
|
|
|
|
State("input_boolean.b4", "off"),
|
|
|
|
datetime(1985, 10, 26, 1, 22, tzinfo=dt_util.UTC),
|
|
|
|
),
|
2020-01-05 10:58:59 +00:00
|
|
|
"input_boolean.b5": StoredState(State("input_boolean.b5", "off"), now),
|
2018-11-28 12:16:43 +00:00
|
|
|
}
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
with patch(
|
|
|
|
"homeassistant.helpers.restore_state.Store.async_save"
|
|
|
|
) as mock_write_data, patch.object(hass.states, "async_all", return_value=states):
|
2018-11-28 12:16:43 +00:00
|
|
|
await data.async_dump_states()
|
|
|
|
|
|
|
|
assert mock_write_data.called
|
|
|
|
args = mock_write_data.mock_calls[0][1]
|
|
|
|
written_states = args[0]
|
|
|
|
|
|
|
|
# b0 should not be written, since it didn't extend RestoreEntity
|
|
|
|
# b1 should be written, since it is present in the current run
|
|
|
|
# b2 should not be written, since it is not registered with the helper
|
|
|
|
# b3 should be written, since it is still not expired
|
|
|
|
# b4 should not be written, since it is now expired
|
2020-01-05 10:58:59 +00:00
|
|
|
# b5 should be written, since current state is restored by entity registry
|
|
|
|
assert len(written_states) == 3
|
2019-07-31 19:25:30 +00:00
|
|
|
assert written_states[0]["state"]["entity_id"] == "input_boolean.b1"
|
|
|
|
assert written_states[0]["state"]["state"] == "on"
|
|
|
|
assert written_states[1]["state"]["entity_id"] == "input_boolean.b3"
|
|
|
|
assert written_states[1]["state"]["state"] == "off"
|
2020-01-05 10:58:59 +00:00
|
|
|
assert written_states[2]["state"]["entity_id"] == "input_boolean.b5"
|
|
|
|
assert written_states[2]["state"]["state"] == "off"
|
2018-11-28 12:16:43 +00:00
|
|
|
|
|
|
|
# Test that removed entities are not persisted
|
2019-06-26 16:22:51 +00:00
|
|
|
await entity.async_remove()
|
2018-11-28 12:16:43 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
with patch(
|
|
|
|
"homeassistant.helpers.restore_state.Store.async_save"
|
|
|
|
) as mock_write_data, patch.object(hass.states, "async_all", return_value=states):
|
2018-11-28 12:16:43 +00:00
|
|
|
await data.async_dump_states()
|
|
|
|
|
|
|
|
assert mock_write_data.called
|
|
|
|
args = mock_write_data.mock_calls[0][1]
|
|
|
|
written_states = args[0]
|
2020-01-05 10:58:59 +00:00
|
|
|
assert len(written_states) == 2
|
2019-07-31 19:25:30 +00:00
|
|
|
assert written_states[0]["state"]["entity_id"] == "input_boolean.b3"
|
|
|
|
assert written_states[0]["state"]["state"] == "off"
|
2020-01-05 10:58:59 +00:00
|
|
|
assert written_states[1]["state"]["entity_id"] == "input_boolean.b5"
|
|
|
|
assert written_states[1]["state"]["state"] == "off"
|
2018-11-28 12:16:43 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def test_dump_error(hass):
|
|
|
|
"""Test that we cache data."""
|
|
|
|
states = [
|
2019-07-31 19:25:30 +00:00
|
|
|
State("input_boolean.b0", "on"),
|
|
|
|
State("input_boolean.b1", "on"),
|
|
|
|
State("input_boolean.b2", "on"),
|
2018-11-28 12:16:43 +00:00
|
|
|
]
|
2017-10-03 03:25:04 +00:00
|
|
|
|
2018-11-28 12:16:43 +00:00
|
|
|
entity = Entity()
|
|
|
|
entity.hass = hass
|
2019-07-31 19:25:30 +00:00
|
|
|
entity.entity_id = "input_boolean.b0"
|
2019-06-26 16:22:51 +00:00
|
|
|
await entity.async_internal_added_to_hass()
|
2017-10-03 03:25:04 +00:00
|
|
|
|
2018-11-28 12:16:43 +00:00
|
|
|
entity = RestoreEntity()
|
|
|
|
entity.hass = hass
|
2019-07-31 19:25:30 +00:00
|
|
|
entity.entity_id = "input_boolean.b1"
|
2019-06-26 16:22:51 +00:00
|
|
|
await entity.async_internal_added_to_hass()
|
2017-02-22 08:15:48 +00:00
|
|
|
|
2018-11-28 12:16:43 +00:00
|
|
|
data = await RestoreStateData.async_get_instance(hass)
|
2017-02-22 08:15:48 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
with patch(
|
|
|
|
"homeassistant.helpers.restore_state.Store.async_save",
|
2020-04-30 20:29:50 +00:00
|
|
|
side_effect=HomeAssistantError,
|
2019-07-31 19:25:30 +00:00
|
|
|
) as mock_write_data, patch.object(hass.states, "async_all", return_value=states):
|
2018-11-28 12:16:43 +00:00
|
|
|
await data.async_dump_states()
|
2017-02-22 08:15:48 +00:00
|
|
|
|
2018-11-28 12:16:43 +00:00
|
|
|
assert mock_write_data.called
|
2017-02-22 08:15:48 +00:00
|
|
|
|
|
|
|
|
2018-11-28 12:16:43 +00:00
|
|
|
async def test_load_error(hass):
|
|
|
|
"""Test that we cache data."""
|
|
|
|
entity = RestoreEntity()
|
|
|
|
entity.hass = hass
|
2019-07-31 19:25:30 +00:00
|
|
|
entity.entity_id = "input_boolean.b1"
|
2017-02-22 08:15:48 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
with patch(
|
|
|
|
"homeassistant.helpers.storage.Store.async_load",
|
2020-04-30 20:29:50 +00:00
|
|
|
side_effect=HomeAssistantError,
|
2019-07-31 19:25:30 +00:00
|
|
|
):
|
2018-11-28 12:16:43 +00:00
|
|
|
state = await entity.async_get_last_state()
|
2017-02-22 08:15:48 +00:00
|
|
|
|
2018-11-28 12:16:43 +00:00
|
|
|
assert state is None
|
2018-12-02 09:51:15 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def test_state_saved_on_remove(hass):
|
|
|
|
"""Test that we save entity state on removal."""
|
|
|
|
entity = RestoreEntity()
|
|
|
|
entity.hass = hass
|
2019-07-31 19:25:30 +00:00
|
|
|
entity.entity_id = "input_boolean.b0"
|
2019-06-26 16:22:51 +00:00
|
|
|
await entity.async_internal_added_to_hass()
|
2018-12-02 09:51:15 +00:00
|
|
|
|
2019-07-11 03:41:03 +00:00
|
|
|
now = dt_util.utcnow()
|
2019-07-31 19:25:30 +00:00
|
|
|
hass.states.async_set(
|
|
|
|
"input_boolean.b0", "on", {"complicated": {"value": {1, 2, now}}}
|
|
|
|
)
|
2018-12-02 09:51:15 +00:00
|
|
|
|
|
|
|
data = await RestoreStateData.async_get_instance(hass)
|
|
|
|
|
|
|
|
# No last states should currently be saved
|
|
|
|
assert not data.last_states
|
|
|
|
|
2019-06-26 16:22:51 +00:00
|
|
|
await entity.async_remove()
|
2018-12-02 09:51:15 +00:00
|
|
|
|
|
|
|
# We should store the input boolean state when it is removed
|
2019-07-31 19:25:30 +00:00
|
|
|
state = data.last_states["input_boolean.b0"].state
|
|
|
|
assert state.state == "on"
|
|
|
|
assert isinstance(state.attributes["complicated"]["value"], list)
|
|
|
|
assert set(state.attributes["complicated"]["value"]) == {1, 2, now.isoformat()}
|
2019-01-24 05:12:38 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def test_restoring_invalid_entity_id(hass, hass_storage):
|
|
|
|
"""Test restoring invalid entity IDs."""
|
|
|
|
entity = RestoreEntity()
|
|
|
|
entity.hass = hass
|
2019-07-31 19:25:30 +00:00
|
|
|
entity.entity_id = "test.invalid__entity_id"
|
2019-01-24 05:12:38 +00:00
|
|
|
now = dt_util.utcnow().isoformat()
|
|
|
|
hass_storage[STORAGE_KEY] = {
|
2019-07-31 19:25:30 +00:00
|
|
|
"version": 1,
|
|
|
|
"key": STORAGE_KEY,
|
|
|
|
"data": [
|
2019-01-24 05:12:38 +00:00
|
|
|
{
|
2019-07-31 19:25:30 +00:00
|
|
|
"state": {
|
|
|
|
"entity_id": "test.invalid__entity_id",
|
|
|
|
"state": "off",
|
|
|
|
"attributes": {},
|
|
|
|
"last_changed": now,
|
|
|
|
"last_updated": now,
|
|
|
|
"context": {
|
|
|
|
"id": "3c2243ff5f30447eb12e7348cfd5b8ff",
|
|
|
|
"user_id": None,
|
|
|
|
},
|
2019-01-24 05:12:38 +00:00
|
|
|
},
|
2019-07-31 19:25:30 +00:00
|
|
|
"last_seen": dt_util.utcnow().isoformat(),
|
2019-01-24 05:12:38 +00:00
|
|
|
}
|
2019-07-31 19:25:30 +00:00
|
|
|
],
|
2019-01-24 05:12:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
state = await entity.async_get_last_state()
|
|
|
|
assert state is None
|