Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(88)

Side by Side Diff: appengine/findit/libs/test/cache_decorator_test.py

Issue 2644543006: [Culprit-Finder] Add generator cache decorator. (Closed)
Patch Set: Small improvement. Created 3 years, 11 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
OLDNEW
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
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, {})
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698