Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 # Copyright 2015 The Chromium Authors. All rights reserved. | 1 # Copyright 2015 The Chromium Authors. All rights reserved. |
| 2 # Use of this source code is governed by a BSD-style license that can be | 2 # Use of this source code is governed by a BSD-style license that can be |
| 3 # found in the LICENSE file. | 3 # found in the LICENSE file. |
| 4 | 4 |
| 5 import mock | |
| 5 import pickle | 6 import pickle |
| 7 import unittest | |
| 6 import zlib | 8 import zlib |
| 7 | 9 |
| 8 from testing_utils import testing | |
| 9 | |
| 10 from libs import cache | 10 from libs import cache |
| 11 from libs import cache_decorator | 11 from libs import cache_decorator |
| 12 | 12 |
| 13 | 13 |
| 14 class _DummyCache(cache.Cache): | 14 class _DummyCache(cache.Cache): |
| 15 def __init__(self, cached_data): | 15 def __init__(self, cached_data): |
| 16 self.cached_data = cached_data | 16 self.cached_data = cached_data |
| 17 | 17 |
| 18 def Get(self, key): | 18 def Get(self, key): |
| 19 return self.cached_data.get(key) | 19 return self.cached_data.get(key) |
| 20 | 20 |
| 21 def Set(self, key, data, expire_time=0): | 21 def Set(self, key, data, expire_time=0): |
| 22 self.cached_data[key] = data | 22 self.cached_data[key] = data |
| 23 | 23 |
| 24 | 24 |
| 25 def _DummyKeyGenerator(func, *_): | 25 def _DummyKeyGenerator(func, args, kwargs, # pylint: disable=W0613 |
| 26 return func.__name__ | 26 namespace=None): |
| 27 namespace = namespace or '%s.%s' % (func.__module__, func.__name__) | |
| 28 return namespace + '-' + func.__name__ | |
| 27 | 29 |
| 28 | 30 |
| 29 class CacheDecoratorTest(testing.AppengineTestCase): | 31 class CachedTest(unittest.TestCase): |
| 32 """Tests ``Cached`` decorator.""" | |
| 33 | |
| 30 def testDefaultKeyGenerator(self): | 34 def testDefaultKeyGenerator(self): |
| 35 namespace = 'n' | |
| 31 expected_params = { | 36 expected_params = { |
| 32 'id1': 'fi', | 37 'id1': 'fi', |
| 33 'id2': 'pi', | 38 'id2': 'pi', |
| 34 'url': 'http://url', | 39 'url': 'http://url', |
| 35 } | 40 } |
| 36 # Hexadecimal digits of MD5 digest of "pickled_params". | 41 # Hexadecimal digits of MD5 digest of "pickled_params". |
| 37 expected_key = 'f5f173c811f7c537a80d44511903a3e0' | 42 expected_key = namespace + '-f5f173c811f7c537a80d44511903a3e0' |
| 38 | 43 |
| 39 def MockPickleDumps(params): | 44 def MockPickleDumps(params): |
| 40 self.assertEqual(expected_params, params) | 45 self.assertEqual(expected_params, params) |
| 41 return 'pickled_params' | 46 return 'pickled_params' |
| 42 | 47 |
| 43 def Func(id1, id2, url=None): # Unused parameters-pylint: disable=W0613 | 48 def Func(id1, id2, url=None): # Unused parameters-pylint: disable=W0613 |
|
chanli
2017/01/21 07:11:56
nit: Func -> func?
Sharu Jiang
2017/01/23 19:15:35
Done.
| |
| 44 return 1 # pragma: no cover. | 49 return 1 # pragma: no cover. |
| 45 | 50 |
| 46 class CallableIdentifier(object): | 51 class CallableIdentifier(object): |
| 47 def identifier(self): | 52 def identifier(self): |
| 48 return 'fi' | 53 return 'fi' |
| 49 | 54 |
| 50 class PropertyIdentifier(object): | 55 class PropertyIdentifier(object): |
| 51 @property | 56 @property |
| 52 def identifier(self): | 57 def identifier(self): |
| 53 return 'pi' | 58 return 'pi' |
| 54 | 59 |
| 55 self.mock(pickle, 'dumps', MockPickleDumps) | 60 with mock.patch('pickle.dumps', MockPickleDumps): |
| 56 | 61 args = (CallableIdentifier(), PropertyIdentifier()) |
| 57 args = (CallableIdentifier(), PropertyIdentifier()) | 62 kwargs = {'url': 'http://url'} |
| 58 kwargs = {'url': 'http://url'} | 63 key = cache_decorator._DefaultKeyGenerator(Func, args, kwargs, |
| 59 key = cache_decorator._DefaultKeyGenerator(Func, args, kwargs) | 64 namespace=namespace) |
|
chanli
2017/01/21 07:11:56
Nit: move all parameters in one line?
Sharu Jiang
2017/01/23 19:15:35
Done.
| |
| 60 self.assertEqual(expected_key, key) | 65 self.assertEqual(expected_key, key) |
| 61 | 66 |
| 62 def testCachedDecoratorWhenResultIsAlreadyCached(self): | 67 def testCachedDecoratorWhenResultIsAlreadyCached(self): |
| 63 dummy_cache = _DummyCache({'n-Func': 1}) | 68 dummy_cache = _DummyCache({'n-Func': 1}) |
| 64 | 69 |
| 65 @cache_decorator.Cached( | 70 @cache_decorator.Cached( |
| 66 namespace='n', key_generator=_DummyKeyGenerator, cache=dummy_cache) | 71 namespace='n', key_generator=_DummyKeyGenerator, cache=dummy_cache) |
| 67 def Func(): | 72 def Func(): |
| 68 return 2 # pragma: no cover. | 73 return 2 # pragma: no cover. |
| 69 | 74 |
| 70 self.assertEqual(1, Func()) | 75 self.assertEqual(1, Func()) |
| (...skipping 48 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 119 return self.url + '/' + path | 124 return self.url + '/' + path |
| 120 | 125 |
| 121 a1 = A('http://test', 3) | 126 a1 = A('http://test', 3) |
| 122 self.assertEqual('http://test/p1', a1.Func('p1')) | 127 self.assertEqual('http://test/p1', a1.Func('p1')) |
| 123 self.assertEqual('http://test/p1', a1.Func('p1')) | 128 self.assertEqual('http://test/p1', a1.Func('p1')) |
| 124 self.assertEqual(1, a1.runs) | 129 self.assertEqual(1, a1.runs) |
| 125 | 130 |
| 126 a2 = A('http://test', 5) | 131 a2 = A('http://test', 5) |
| 127 self.assertEqual('http://test/p1', a2.Func('p1')) | 132 self.assertEqual('http://test/p1', a2.Func('p1')) |
| 128 self.assertEqual(0, a2.runs) | 133 self.assertEqual(0, a2.runs) |
| 134 | |
| 135 | |
| 136 class GeneratorCachedTest(unittest.TestCase): | |
| 137 """Tests ``GeneratorCached`` decorator.""" | |
| 138 | |
| 139 def testWhenResultIsAlreadyCached(self): | |
| 140 value_list = [0, 1, 2] | |
| 141 key = 'n-Func' | |
| 142 # Zero value won't be cached. | |
| 143 cached_keys = ['%s-%d' % (key, i) for i in xrange(len(value_list))] | |
| 144 cached_data = {key: value for key, value in zip(cached_keys, value_list)} | |
| 145 cached_data[key] = cached_keys | |
| 146 | |
| 147 dummy_cache = _DummyCache(cached_data) | |
| 148 | |
| 149 @cache_decorator.GeneratorCached( | |
| 150 dummy_cache, namespace='n', key_generator=_DummyKeyGenerator) | |
| 151 def Func(): # pragma: no cover | |
| 152 for value in value_list: | |
| 153 yield value | |
| 154 | |
| 155 for value, expected_value in zip(Func(), value_list): | |
| 156 self.assertEqual(value, expected_value) | |
| 157 | |
| 158 def testWhenResultIsNotCachedYet(self): | |
| 159 value_list = [0, 1, 2, 3, 4] | |
| 160 dummy_cache = _DummyCache({}) | |
| 161 | |
| 162 @cache_decorator.GeneratorCached( | |
| 163 dummy_cache, namespace='n', key_generator=_DummyKeyGenerator) | |
| 164 def Func(): # pragma: no cover | |
| 165 for value in value_list: | |
| 166 yield value | |
| 167 | |
| 168 key = _DummyKeyGenerator(Func, [], {}, namespace='n') | |
| 169 cached_keys = ['%s-%d' % (key, i) for i in xrange(len(value_list))] | |
| 170 cached_data = {key: value for key, value in zip(cached_keys, value_list)} | |
| 171 cached_data[key] = cached_keys | |
| 172 for value, expected_value in zip(Func(), value_list): | |
| 173 self.assertEqual(value, expected_value) | |
| 174 | |
| 175 self.assertDictEqual(dummy_cache.cached_data, cached_data) | |
| 176 | |
| 177 @mock.patch('libs.cache_decorator.GeneratorCached.SetCache', | |
| 178 lambda *_: False) | |
| 179 def testFailedToSetValue(self): | |
| 180 value_list = [0, 1, 2, 3, 4] | |
| 181 dummy_cache = _DummyCache({}) | |
| 182 | |
| 183 @cache_decorator.GeneratorCached( | |
| 184 dummy_cache, namespace='n', key_generator=_DummyKeyGenerator) | |
| 185 def Func(): # pragma: no cover | |
| 186 for value in value_list: | |
| 187 yield value | |
| 188 | |
| 189 key = _DummyKeyGenerator(Func, [], {}, namespace='n') | |
| 190 cached_keys = ['%s-%d' % (key, i) for i in xrange(len(value_list))] | |
| 191 cached_data = {key: value for key, value in zip(cached_keys, value_list)} | |
| 192 cached_data[key] = cached_keys | |
| 193 for value, expected_value in zip(Func(), value_list): | |
| 194 self.assertEqual(value, expected_value) | |
| 195 | |
| 196 self.assertDictEqual(dummy_cache.cached_data, {}) | |
| OLD | NEW |