2018-06-25 16:53:49 +00:00
|
|
|
"""Tests for the storage helper."""
|
2018-06-29 02:14:26 +00:00
|
|
|
import asyncio
|
2018-06-25 16:53:49 +00:00
|
|
|
from datetime import timedelta
|
2018-11-28 12:16:43 +00:00
|
|
|
import json
|
2019-12-09 15:52:24 +00:00
|
|
|
from unittest.mock import Mock, patch
|
2018-06-25 16:53:49 +00:00
|
|
|
|
|
|
|
import pytest
|
|
|
|
|
|
|
|
from homeassistant.const import EVENT_HOMEASSISTANT_STOP
|
|
|
|
from homeassistant.helpers import storage
|
|
|
|
from homeassistant.util import dt
|
|
|
|
|
|
|
|
from tests.common import async_fire_time_changed, mock_coro
|
|
|
|
|
|
|
|
MOCK_VERSION = 1
|
2019-07-31 19:25:30 +00:00
|
|
|
MOCK_KEY = "storage-test"
|
|
|
|
MOCK_DATA = {"hello": "world"}
|
|
|
|
MOCK_DATA2 = {"goodbye": "cruel world"}
|
2018-06-25 16:53:49 +00:00
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture
|
|
|
|
def store(hass):
|
2020-01-05 12:09:17 +00:00
|
|
|
"""Fixture of a store that prevents writing on Home Assistant stop."""
|
2018-06-29 02:14:26 +00:00
|
|
|
yield storage.Store(hass, MOCK_VERSION, MOCK_KEY)
|
2018-06-25 16:53:49 +00:00
|
|
|
|
|
|
|
|
2018-06-29 02:14:26 +00:00
|
|
|
async def test_loading(hass, store):
|
2018-06-25 16:53:49 +00:00
|
|
|
"""Test we can save and load data."""
|
|
|
|
await store.async_save(MOCK_DATA)
|
|
|
|
data = await store.async_load()
|
|
|
|
assert data == MOCK_DATA
|
|
|
|
|
|
|
|
|
2018-11-28 12:16:43 +00:00
|
|
|
async def test_custom_encoder(hass):
|
|
|
|
"""Test we can save and load data."""
|
2019-07-31 19:25:30 +00:00
|
|
|
|
2018-11-28 12:16:43 +00:00
|
|
|
class JSONEncoder(json.JSONEncoder):
|
|
|
|
"""Mock JSON encoder."""
|
|
|
|
|
|
|
|
def default(self, o):
|
|
|
|
"""Mock JSON encode method."""
|
|
|
|
return "9"
|
|
|
|
|
|
|
|
store = storage.Store(hass, MOCK_VERSION, MOCK_KEY, encoder=JSONEncoder)
|
|
|
|
await store.async_save(Mock())
|
|
|
|
data = await store.async_load()
|
|
|
|
assert data == "9"
|
|
|
|
|
|
|
|
|
2018-06-25 16:53:49 +00:00
|
|
|
async def test_loading_non_existing(hass, store):
|
|
|
|
"""Test we can save and load data."""
|
2019-07-31 19:25:30 +00:00
|
|
|
with patch("homeassistant.util.json.open", side_effect=FileNotFoundError):
|
2018-06-25 16:53:49 +00:00
|
|
|
data = await store.async_load()
|
2018-06-25 21:21:38 +00:00
|
|
|
assert data is None
|
2018-06-25 16:53:49 +00:00
|
|
|
|
|
|
|
|
2018-06-29 02:14:26 +00:00
|
|
|
async def test_loading_parallel(hass, store, hass_storage, caplog):
|
|
|
|
"""Test we can save and load data."""
|
2019-07-31 19:25:30 +00:00
|
|
|
hass_storage[store.key] = {"version": MOCK_VERSION, "data": MOCK_DATA}
|
2018-06-29 02:14:26 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
results = await asyncio.gather(store.async_load(), store.async_load())
|
2018-06-29 02:14:26 +00:00
|
|
|
|
|
|
|
assert results[0] is MOCK_DATA
|
|
|
|
assert results[1] is MOCK_DATA
|
2020-01-03 13:47:06 +00:00
|
|
|
assert caplog.text.count(f"Loading data for {store.key}")
|
2018-06-29 02:14:26 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def test_saving_with_delay(hass, store, hass_storage):
|
2018-06-25 16:53:49 +00:00
|
|
|
"""Test saving data after a delay."""
|
2018-08-17 18:18:21 +00:00
|
|
|
store.async_delay_save(lambda: MOCK_DATA, 1)
|
2018-06-29 02:14:26 +00:00
|
|
|
assert store.key not in hass_storage
|
2018-06-25 16:53:49 +00:00
|
|
|
|
|
|
|
async_fire_time_changed(hass, dt.utcnow() + timedelta(seconds=1))
|
|
|
|
await hass.async_block_till_done()
|
2018-06-29 02:14:26 +00:00
|
|
|
assert hass_storage[store.key] == {
|
2019-07-31 19:25:30 +00:00
|
|
|
"version": MOCK_VERSION,
|
|
|
|
"key": MOCK_KEY,
|
|
|
|
"data": MOCK_DATA,
|
2018-06-29 02:14:26 +00:00
|
|
|
}
|
2018-06-25 16:53:49 +00:00
|
|
|
|
|
|
|
|
2018-06-29 02:14:26 +00:00
|
|
|
async def test_saving_on_stop(hass, hass_storage):
|
2018-06-25 16:53:49 +00:00
|
|
|
"""Test delayed saves trigger when we quit Home Assistant."""
|
|
|
|
store = storage.Store(hass, MOCK_VERSION, MOCK_KEY)
|
2018-08-17 18:18:21 +00:00
|
|
|
store.async_delay_save(lambda: MOCK_DATA, 1)
|
2018-06-29 02:14:26 +00:00
|
|
|
assert store.key not in hass_storage
|
2018-06-25 16:53:49 +00:00
|
|
|
|
|
|
|
hass.bus.async_fire(EVENT_HOMEASSISTANT_STOP)
|
|
|
|
await hass.async_block_till_done()
|
2018-06-29 02:14:26 +00:00
|
|
|
assert hass_storage[store.key] == {
|
2019-07-31 19:25:30 +00:00
|
|
|
"version": MOCK_VERSION,
|
|
|
|
"key": MOCK_KEY,
|
|
|
|
"data": MOCK_DATA,
|
2018-06-29 02:14:26 +00:00
|
|
|
}
|
2018-06-25 16:53:49 +00:00
|
|
|
|
|
|
|
|
2018-06-29 02:14:26 +00:00
|
|
|
async def test_loading_while_delay(hass, store, hass_storage):
|
2018-06-25 16:53:49 +00:00
|
|
|
"""Test we load new data even if not written yet."""
|
2019-07-31 19:25:30 +00:00
|
|
|
await store.async_save({"delay": "no"})
|
2018-06-29 02:14:26 +00:00
|
|
|
assert hass_storage[store.key] == {
|
2019-07-31 19:25:30 +00:00
|
|
|
"version": MOCK_VERSION,
|
|
|
|
"key": MOCK_KEY,
|
|
|
|
"data": {"delay": "no"},
|
2018-06-29 02:14:26 +00:00
|
|
|
}
|
2018-06-25 16:53:49 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
store.async_delay_save(lambda: {"delay": "yes"}, 1)
|
2018-06-29 02:14:26 +00:00
|
|
|
assert hass_storage[store.key] == {
|
2019-07-31 19:25:30 +00:00
|
|
|
"version": MOCK_VERSION,
|
|
|
|
"key": MOCK_KEY,
|
|
|
|
"data": {"delay": "no"},
|
2018-06-29 02:14:26 +00:00
|
|
|
}
|
2018-06-25 16:53:49 +00:00
|
|
|
|
|
|
|
data = await store.async_load()
|
2019-07-31 19:25:30 +00:00
|
|
|
assert data == {"delay": "yes"}
|
2018-06-25 16:53:49 +00:00
|
|
|
|
|
|
|
|
2018-06-29 02:14:26 +00:00
|
|
|
async def test_writing_while_writing_delay(hass, store, hass_storage):
|
2018-06-25 16:53:49 +00:00
|
|
|
"""Test a write while a write with delay is active."""
|
2019-07-31 19:25:30 +00:00
|
|
|
store.async_delay_save(lambda: {"delay": "yes"}, 1)
|
2018-06-29 02:14:26 +00:00
|
|
|
assert store.key not in hass_storage
|
2019-07-31 19:25:30 +00:00
|
|
|
await store.async_save({"delay": "no"})
|
2018-06-29 02:14:26 +00:00
|
|
|
assert hass_storage[store.key] == {
|
2019-07-31 19:25:30 +00:00
|
|
|
"version": MOCK_VERSION,
|
|
|
|
"key": MOCK_KEY,
|
|
|
|
"data": {"delay": "no"},
|
2018-06-29 02:14:26 +00:00
|
|
|
}
|
2018-06-25 16:53:49 +00:00
|
|
|
|
|
|
|
async_fire_time_changed(hass, dt.utcnow() + timedelta(seconds=1))
|
|
|
|
await hass.async_block_till_done()
|
2018-06-29 02:14:26 +00:00
|
|
|
assert hass_storage[store.key] == {
|
2019-07-31 19:25:30 +00:00
|
|
|
"version": MOCK_VERSION,
|
|
|
|
"key": MOCK_KEY,
|
|
|
|
"data": {"delay": "no"},
|
2018-06-29 02:14:26 +00:00
|
|
|
}
|
2018-06-25 16:53:49 +00:00
|
|
|
|
|
|
|
data = await store.async_load()
|
2019-07-31 19:25:30 +00:00
|
|
|
assert data == {"delay": "no"}
|
2018-06-25 16:53:49 +00:00
|
|
|
|
|
|
|
|
2018-06-29 02:14:26 +00:00
|
|
|
async def test_migrator_no_existing_config(hass, store, hass_storage):
|
2018-06-25 16:53:49 +00:00
|
|
|
"""Test migrator with no existing config."""
|
2019-07-31 19:25:30 +00:00
|
|
|
with patch("os.path.isfile", return_value=False), patch.object(
|
|
|
|
store, "async_load", return_value=mock_coro({"cur": "config"})
|
|
|
|
):
|
|
|
|
data = await storage.async_migrator(hass, "old-path", store)
|
2018-06-25 16:53:49 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
assert data == {"cur": "config"}
|
2018-06-29 02:14:26 +00:00
|
|
|
assert store.key not in hass_storage
|
2018-06-25 16:53:49 +00:00
|
|
|
|
|
|
|
|
2018-06-29 02:14:26 +00:00
|
|
|
async def test_migrator_existing_config(hass, store, hass_storage):
|
2018-06-25 16:53:49 +00:00
|
|
|
"""Test migrating existing config."""
|
2019-07-31 19:25:30 +00:00
|
|
|
with patch("os.path.isfile", return_value=True), patch("os.remove") as mock_remove:
|
2018-06-25 16:53:49 +00:00
|
|
|
data = await storage.async_migrator(
|
2019-07-31 19:25:30 +00:00
|
|
|
hass, "old-path", store, old_conf_load_func=lambda _: {"old": "config"}
|
|
|
|
)
|
2018-06-25 16:53:49 +00:00
|
|
|
|
|
|
|
assert len(mock_remove.mock_calls) == 1
|
2019-07-31 19:25:30 +00:00
|
|
|
assert data == {"old": "config"}
|
2018-06-29 02:14:26 +00:00
|
|
|
assert hass_storage[store.key] == {
|
2019-07-31 19:25:30 +00:00
|
|
|
"key": MOCK_KEY,
|
|
|
|
"version": MOCK_VERSION,
|
|
|
|
"data": data,
|
2018-06-25 16:53:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-06-29 02:14:26 +00:00
|
|
|
async def test_migrator_transforming_config(hass, store, hass_storage):
|
2018-06-25 16:53:49 +00:00
|
|
|
"""Test migrating config to new format."""
|
2019-07-31 19:25:30 +00:00
|
|
|
|
2018-06-25 16:53:49 +00:00
|
|
|
async def old_conf_migrate_func(old_config):
|
|
|
|
"""Migrate old config to new format."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return {"new": old_config["old"]}
|
2018-06-25 16:53:49 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
with patch("os.path.isfile", return_value=True), patch("os.remove") as mock_remove:
|
2018-06-25 16:53:49 +00:00
|
|
|
data = await storage.async_migrator(
|
2019-07-31 19:25:30 +00:00
|
|
|
hass,
|
|
|
|
"old-path",
|
|
|
|
store,
|
2018-08-18 11:34:33 +00:00
|
|
|
old_conf_migrate_func=old_conf_migrate_func,
|
2019-07-31 19:25:30 +00:00
|
|
|
old_conf_load_func=lambda _: {"old": "config"},
|
|
|
|
)
|
2018-06-25 16:53:49 +00:00
|
|
|
|
|
|
|
assert len(mock_remove.mock_calls) == 1
|
2019-07-31 19:25:30 +00:00
|
|
|
assert data == {"new": "config"}
|
2018-06-29 02:14:26 +00:00
|
|
|
assert hass_storage[store.key] == {
|
2019-07-31 19:25:30 +00:00
|
|
|
"key": MOCK_KEY,
|
|
|
|
"version": MOCK_VERSION,
|
|
|
|
"data": data,
|
2018-06-25 16:53:49 +00:00
|
|
|
}
|