123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238 |
- # -*- coding: utf-8 -*-
- """
- unit tests for salt.cache
- """
- # Import Python libs
- from __future__ import absolute_import, print_function, unicode_literals
- import salt.cache
- # Import Salt libs
- import salt.payload
- from tests.support.mock import patch
- # Import Salt Testing libs
- # import integration
- from tests.support.unit import TestCase
- class CacheFunctionsTest(TestCase):
- """
- Validate the cache package functions.
- """
- def setUp(self):
- self.opts = {
- "cache": "localfs",
- "memcache_expire_seconds": 0,
- "memcache_max_items": 0,
- "memcache_full_cleanup": False,
- "memcache_debug": False,
- }
- def test_factory_cache(self):
- ret = salt.cache.factory(self.opts)
- self.assertIsInstance(ret, salt.cache.Cache)
- def test_factory_memcache(self):
- self.opts["memcache_expire_seconds"] = 10
- ret = salt.cache.factory(self.opts)
- self.assertIsInstance(ret, salt.cache.MemCache)
- class MemCacheTest(TestCase):
- """
- Validate Cache class methods
- """
- @patch("salt.payload.Serial")
- def setUp(self, serial_mock): # pylint: disable=W0221
- salt.cache.MemCache.data = {}
- self.opts = {
- "cache": "fake_driver",
- "memcache_expire_seconds": 10,
- "memcache_max_items": 3,
- "memcache_full_cleanup": False,
- "memcache_debug": False,
- }
- self.cache = salt.cache.factory(self.opts)
- @patch("salt.cache.Cache.fetch", return_value="fake_data")
- @patch("salt.loader.cache", return_value={})
- def test_fetch(self, loader_mock, cache_fetch_mock):
- # Fetch value, it will be kept in cache.
- with patch("time.time", return_value=0):
- ret = self.cache.fetch("bank", "key")
- self.assertEqual(ret, "fake_data")
- self.assertDictEqual(
- salt.cache.MemCache.data,
- {"fake_driver": {("bank", "key"): [0, "fake_data"]}},
- )
- cache_fetch_mock.assert_called_once_with("bank", "key")
- cache_fetch_mock.reset_mock()
- # Fetch again, cached value is used, time updated.
- with patch("time.time", return_value=1):
- ret = self.cache.fetch("bank", "key")
- self.assertEqual(ret, "fake_data")
- self.assertDictEqual(
- salt.cache.MemCache.data,
- {"fake_driver": {("bank", "key"): [1, "fake_data"]}},
- )
- cache_fetch_mock.assert_not_called()
- # Fetch after expire
- with patch("time.time", return_value=12):
- ret = self.cache.fetch("bank", "key")
- self.assertEqual(ret, "fake_data")
- self.assertDictEqual(
- salt.cache.MemCache.data,
- {"fake_driver": {("bank", "key"): [12, "fake_data"]}},
- )
- cache_fetch_mock.assert_called_once_with("bank", "key")
- cache_fetch_mock.reset_mock()
- @patch("salt.cache.Cache.store")
- @patch("salt.loader.cache", return_value={})
- def test_store(self, loader_mock, cache_store_mock):
- # Fetch value, it will be kept in cache.
- with patch("time.time", return_value=0):
- self.cache.store("bank", "key", "fake_data")
- self.assertDictEqual(
- salt.cache.MemCache.data,
- {"fake_driver": {("bank", "key"): [0, "fake_data"]}},
- )
- cache_store_mock.assert_called_once_with("bank", "key", "fake_data")
- cache_store_mock.reset_mock()
- # Store another value.
- with patch("time.time", return_value=1):
- self.cache.store("bank", "key2", "fake_data2")
- self.assertDictEqual(
- salt.cache.MemCache.data,
- {
- "fake_driver": {
- ("bank", "key"): [0, "fake_data"],
- ("bank", "key2"): [1, "fake_data2"],
- }
- },
- )
- cache_store_mock.assert_called_once_with("bank", "key2", "fake_data2")
- @patch("salt.cache.Cache.store")
- @patch("salt.cache.Cache.flush")
- @patch("salt.loader.cache", return_value={})
- def test_flush(self, loader_mock, cache_flush_mock, cache_store_mock):
- # Flush non-existing bank
- self.cache.flush("bank")
- self.assertDictEqual(salt.cache.MemCache.data, {"fake_driver": {}})
- cache_flush_mock.assert_called_once_with("bank", None)
- cache_flush_mock.reset_mock()
- # Flush non-existing key
- self.cache.flush("bank", "key")
- self.assertDictEqual(salt.cache.MemCache.data, {"fake_driver": {}})
- cache_flush_mock.assert_called_once_with("bank", "key")
- cache_flush_mock.reset_mock()
- # Flush existing key
- with patch("time.time", return_value=0):
- self.cache.store("bank", "key", "fake_data")
- self.assertEqual(
- salt.cache.MemCache.data["fake_driver"][("bank", "key")], [0, "fake_data"]
- )
- self.assertDictEqual(
- salt.cache.MemCache.data,
- {"fake_driver": {("bank", "key"): [0, "fake_data"]}},
- )
- self.cache.flush("bank", "key")
- self.assertDictEqual(salt.cache.MemCache.data, {"fake_driver": {}})
- cache_flush_mock.assert_called_once_with("bank", "key")
- cache_flush_mock.reset_mock()
- @patch("salt.cache.Cache.store")
- @patch("salt.loader.cache", return_value={})
- def test_max_items(self, loader_mock, cache_store_mock):
- # Put MAX=3 values
- with patch("time.time", return_value=0):
- self.cache.store("bank1", "key1", "fake_data11")
- with patch("time.time", return_value=1):
- self.cache.store("bank1", "key2", "fake_data12")
- with patch("time.time", return_value=2):
- self.cache.store("bank2", "key1", "fake_data21")
- self.assertDictEqual(
- salt.cache.MemCache.data["fake_driver"],
- {
- ("bank1", "key1"): [0, "fake_data11"],
- ("bank1", "key2"): [1, "fake_data12"],
- ("bank2", "key1"): [2, "fake_data21"],
- },
- )
- # Put one more and check the oldest was removed
- with patch("time.time", return_value=3):
- self.cache.store("bank2", "key2", "fake_data22")
- self.assertDictEqual(
- salt.cache.MemCache.data["fake_driver"],
- {
- ("bank1", "key2"): [1, "fake_data12"],
- ("bank2", "key1"): [2, "fake_data21"],
- ("bank2", "key2"): [3, "fake_data22"],
- },
- )
- @patch("salt.cache.Cache.store")
- @patch("salt.loader.cache", return_value={})
- def test_full_cleanup(self, loader_mock, cache_store_mock):
- # Enable full cleanup
- self.cache.cleanup = True
- # Put MAX=3 values
- with patch("time.time", return_value=0):
- self.cache.store("bank1", "key1", "fake_data11")
- with patch("time.time", return_value=1):
- self.cache.store("bank1", "key2", "fake_data12")
- with patch("time.time", return_value=2):
- self.cache.store("bank2", "key1", "fake_data21")
- self.assertDictEqual(
- salt.cache.MemCache.data["fake_driver"],
- {
- ("bank1", "key1"): [0, "fake_data11"],
- ("bank1", "key2"): [1, "fake_data12"],
- ("bank2", "key1"): [2, "fake_data21"],
- },
- )
- # Put one more and check all expired was removed
- with patch("time.time", return_value=12):
- self.cache.store("bank2", "key2", "fake_data22")
- self.assertDictEqual(
- salt.cache.MemCache.data["fake_driver"],
- {
- ("bank2", "key1"): [2, "fake_data21"],
- ("bank2", "key2"): [12, "fake_data22"],
- },
- )
- @patch("salt.cache.Cache.fetch", return_value="fake_data")
- @patch("salt.loader.cache", return_value={})
- def test_fetch_debug(self, loader_mock, cache_fetch_mock):
- # Recreate cache with debug enabled
- self.opts["memcache_debug"] = True
- self.cache = salt.cache.factory(self.opts)
- # Fetch 2 values (no cache hit)
- with patch("time.time", return_value=0):
- ret = self.cache.fetch("bank", "key1")
- with patch("time.time", return_value=1):
- ret = self.cache.fetch("bank", "key2")
- # Fetch 3 times (cache hit)
- with patch("time.time", return_value=2):
- ret = self.cache.fetch("bank", "key2")
- with patch("time.time", return_value=3):
- ret = self.cache.fetch("bank", "key1")
- with patch("time.time", return_value=4):
- ret = self.cache.fetch("bank", "key1")
- # Fetch an expired value (no cache hit)
- with patch("time.time", return_value=13):
- ret = self.cache.fetch("bank", "key2")
- # Check debug data
- self.assertEqual(self.cache.call, 6)
- self.assertEqual(self.cache.hit, 3)
|