1
0
mirror of https://github.com/trezor/trezor-firmware.git synced 2025-01-25 06:40:58 +00:00
trezor-firmware/core/tests/test_storage.cache.py

176 lines
5.8 KiB
Python
Raw Normal View History

2020-01-30 14:58:12 +00:00
from common import *
from mock_storage import mock_storage
from storage import cache
from trezor.messages.Initialize import Initialize
2020-08-25 12:51:06 +00:00
from trezor.messages.EndSession import EndSession
from trezor.wire import DUMMY_CONTEXT, InvalidSession
2020-01-30 14:58:12 +00:00
2020-08-25 12:51:06 +00:00
from apps.base import handle_Initialize, handle_EndSession
2020-01-30 14:58:12 +00:00
KEY = 99
class TestStorageCache(unittest.TestCase):
2020-08-25 12:51:06 +00:00
def setUp(self):
cache.clear_all()
def test_start_session(self):
session_id_a = cache.start_session()
2020-01-30 14:58:12 +00:00
self.assertIsNotNone(session_id_a)
session_id_b = cache.start_session()
self.assertNotEqual(session_id_a, session_id_b)
cache.clear_all()
with self.assertRaises(InvalidSession):
cache.set(KEY, "something")
with self.assertRaises(InvalidSession):
cache.get(KEY)
2020-08-25 12:51:06 +00:00
def test_end_session(self):
session_id = cache.start_session()
self.assertTrue(cache.is_session_started())
cache.set(KEY, "A")
cache.end_current_session()
self.assertFalse(cache.is_session_started())
self.assertRaises(InvalidSession, cache.get, KEY)
2020-08-25 12:51:06 +00:00
# ending an ended session should be a no-op
cache.end_current_session()
self.assertFalse(cache.is_session_started())
session_id_a = cache.start_session(session_id)
# original session no longer exists
self.assertNotEqual(session_id_a, session_id)
# original session data no longer exists
self.assertIsNone(cache.get(KEY))
# create a new session
session_id_b = cache.start_session()
# switch back to original session
session_id = cache.start_session(session_id_a)
self.assertEqual(session_id, session_id_a)
# end original session
cache.end_current_session()
# switch back to B
session_id = cache.start_session(session_id_b)
self.assertEqual(session_id, session_id_b)
def test_session_queue(self):
session_id = cache.start_session()
self.assertEqual(cache.start_session(session_id), session_id)
cache.set(KEY, "A")
for i in range(cache._MAX_SESSIONS_COUNT):
cache.start_session()
self.assertNotEqual(cache.start_session(session_id), session_id)
self.assertIsNone(cache.get(KEY))
2020-01-30 14:58:12 +00:00
def test_get_set(self):
session_id1 = cache.start_session()
2020-01-30 14:58:12 +00:00
cache.set(KEY, "hello")
self.assertEqual(cache.get(KEY), "hello")
session_id2 = cache.start_session()
cache.set(KEY, "world")
self.assertEqual(cache.get(KEY), "world")
2020-01-30 14:58:12 +00:00
cache.start_session(session_id2)
self.assertEqual(cache.get(KEY), "world")
cache.start_session(session_id1)
self.assertEqual(cache.get(KEY), "hello")
cache.clear_all()
with self.assertRaises(InvalidSession):
cache.get(KEY)
2020-01-30 14:58:12 +00:00
2020-04-20 09:36:28 +00:00
def test_decorator_mismatch(self):
with self.assertRaises(AssertionError):
@cache.stored(KEY)
async def async_fun():
pass
def test_decorators(self):
run_count = 0
cache.start_session()
@cache.stored(KEY)
def func():
nonlocal run_count
run_count += 1
return "foo"
# cache is empty
self.assertIsNone(cache.get(KEY))
self.assertEqual(run_count, 0)
self.assertEqual(func(), "foo")
# function was run
self.assertEqual(run_count, 1)
self.assertEqual(cache.get(KEY), "foo")
# function does not run again but returns cached value
self.assertEqual(func(), "foo")
self.assertEqual(run_count, 1)
@cache.stored_async(KEY)
async def async_func():
nonlocal run_count
run_count += 1
return "bar"
# cache is still full
self.assertEqual(await_result(async_func()), "foo")
self.assertEqual(run_count, 1)
cache.start_session()
self.assertEqual(await_result(async_func()), "bar")
self.assertEqual(run_count, 2)
# awaitable is also run only once
self.assertEqual(await_result(async_func()), "bar")
self.assertEqual(run_count, 2)
2020-01-30 14:58:12 +00:00
@mock_storage
def test_Initialize(self):
def call_Initialize(**kwargs):
msg = Initialize(**kwargs)
return await_result(handle_Initialize(DUMMY_CONTEXT, msg))
# calling Initialize without an ID allocates a new one
session_id = cache.start_session()
2020-01-30 14:58:12 +00:00
features = call_Initialize()
self.assertNotEqual(session_id, features.session_id)
2020-01-30 14:58:12 +00:00
# calling Initialize with the current ID does not allocate a new one
features = call_Initialize(session_id=session_id)
self.assertEqual(session_id, features.session_id)
2020-01-30 14:58:12 +00:00
# store "hello"
2020-01-30 14:58:12 +00:00
cache.set(KEY, "hello")
# check that it is cleared
2020-01-30 14:58:12 +00:00
features = call_Initialize()
session_id = features.session_id
2020-01-30 14:58:12 +00:00
self.assertIsNone(cache.get(KEY))
# store "hello" again
2020-01-30 14:58:12 +00:00
cache.set(KEY, "hello")
self.assertEqual(cache.get(KEY), "hello")
# supplying a different session ID starts a new cache
call_Initialize(session_id=b"A")
2020-01-30 14:58:12 +00:00
self.assertIsNone(cache.get(KEY))
# but resuming a session loads the previous one
call_Initialize(session_id=session_id)
self.assertEqual(cache.get(KEY), "hello")
2020-08-25 12:51:06 +00:00
def test_EndSession(self):
self.assertRaises(InvalidSession, cache.get, KEY)
2020-08-25 12:51:06 +00:00
session_id = cache.start_session()
self.assertTrue(cache.is_session_started())
self.assertIsNone(cache.get(KEY))
await_result(handle_EndSession(DUMMY_CONTEXT, EndSession()))
self.assertFalse(cache.is_session_started())
self.assertRaises(InvalidSession, cache.get, KEY)
2020-08-25 12:51:06 +00:00
2020-01-30 14:58:12 +00:00
if __name__ == "__main__":
unittest.main()