test_datastructures.py 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353
  1. from __future__ import absolute_import
  2. import pickle
  3. import sys
  4. from billiard.einfo import ExceptionInfo
  5. from mock import Mock, patch
  6. from time import time
  7. from celery.datastructures import (
  8. LimitedSet,
  9. AttributeDict,
  10. DictAttribute,
  11. ConfigurationView,
  12. DependencyGraph,
  13. )
  14. from celery.five import items
  15. from celery.tests.case import Case, WhateverIO, SkipTest
  16. class Object(object):
  17. pass
  18. class test_DictAttribute(Case):
  19. def test_get_set_keys_values_items(self):
  20. x = DictAttribute(Object())
  21. x['foo'] = 'The quick brown fox'
  22. self.assertEqual(x['foo'], 'The quick brown fox')
  23. self.assertEqual(x['foo'], x.obj.foo)
  24. self.assertEqual(x.get('foo'), 'The quick brown fox')
  25. self.assertIsNone(x.get('bar'))
  26. with self.assertRaises(KeyError):
  27. x['bar']
  28. x.foo = 'The quick yellow fox'
  29. self.assertEqual(x['foo'], 'The quick yellow fox')
  30. self.assertIn(
  31. ('foo', 'The quick yellow fox'),
  32. list(x.items()),
  33. )
  34. self.assertIn('foo', list(x.keys()))
  35. self.assertIn('The quick yellow fox', list(x.values()))
  36. def test_setdefault(self):
  37. x = DictAttribute(Object())
  38. self.assertEqual(x.setdefault('foo', 'NEW'), 'NEW')
  39. self.assertEqual(x.setdefault('foo', 'XYZ'), 'NEW')
  40. def test_contains(self):
  41. x = DictAttribute(Object())
  42. x['foo'] = 1
  43. self.assertIn('foo', x)
  44. self.assertNotIn('bar', x)
  45. def test_items(self):
  46. obj = Object()
  47. obj.attr1 = 1
  48. x = DictAttribute(obj)
  49. x['attr2'] = 2
  50. self.assertEqual(x['attr1'], 1)
  51. self.assertEqual(x['attr2'], 2)
  52. class test_ConfigurationView(Case):
  53. def setUp(self):
  54. self.view = ConfigurationView({'changed_key': 1,
  55. 'both': 2},
  56. [{'default_key': 1,
  57. 'both': 1}])
  58. def test_setdefault(self):
  59. self.assertEqual(self.view.setdefault('both', 36), 2)
  60. self.assertEqual(self.view.setdefault('new', 36), 36)
  61. def test_get(self):
  62. self.assertEqual(self.view.get('both'), 2)
  63. sp = object()
  64. self.assertIs(self.view.get('nonexisting', sp), sp)
  65. def test_update(self):
  66. changes = dict(self.view.changes)
  67. self.view.update(a=1, b=2, c=3)
  68. self.assertDictEqual(self.view.changes,
  69. dict(changes, a=1, b=2, c=3))
  70. def test_contains(self):
  71. self.assertIn('changed_key', self.view)
  72. self.assertIn('default_key', self.view)
  73. self.assertNotIn('new', self.view)
  74. def test_repr(self):
  75. self.assertIn('changed_key', repr(self.view))
  76. self.assertIn('default_key', repr(self.view))
  77. def test_iter(self):
  78. expected = {'changed_key': 1,
  79. 'default_key': 1,
  80. 'both': 2}
  81. self.assertDictEqual(dict(items(self.view)), expected)
  82. self.assertItemsEqual(list(iter(self.view)),
  83. list(expected.keys()))
  84. self.assertItemsEqual(list(self.view.keys()), list(expected.keys()))
  85. self.assertItemsEqual(
  86. list(self.view.values()),
  87. list(expected.values()),
  88. )
  89. self.assertIn('changed_key', list(self.view.keys()))
  90. self.assertIn(2, list(self.view.values()))
  91. self.assertIn(('both', 2), list(self.view.items()))
  92. def test_add_defaults_dict(self):
  93. defaults = {'foo': 10}
  94. self.view.add_defaults(defaults)
  95. self.assertEqual(self.view.foo, 10)
  96. def test_add_defaults_object(self):
  97. defaults = Object()
  98. defaults.foo = 10
  99. self.view.add_defaults(defaults)
  100. self.assertEqual(self.view.foo, 10)
  101. def test_clear(self):
  102. self.view.clear()
  103. self.assertEqual(self.view.both, 1)
  104. self.assertNotIn('changed_key', self.view)
  105. def test_bool(self):
  106. self.assertTrue(bool(self.view))
  107. self.view._order[:] = []
  108. self.assertFalse(bool(self.view))
  109. def test_len(self):
  110. self.assertEqual(len(self.view), 3)
  111. self.view.KEY = 33
  112. self.assertEqual(len(self.view), 4)
  113. self.view.clear()
  114. self.assertEqual(len(self.view), 2)
  115. def test_isa_mapping(self):
  116. from collections import Mapping
  117. self.assertTrue(issubclass(ConfigurationView, Mapping))
  118. def test_isa_mutable_mapping(self):
  119. from collections import MutableMapping
  120. self.assertTrue(issubclass(ConfigurationView, MutableMapping))
  121. class test_ExceptionInfo(Case):
  122. def test_exception_info(self):
  123. try:
  124. raise LookupError('The quick brown fox jumps...')
  125. except Exception:
  126. einfo = ExceptionInfo()
  127. self.assertEqual(str(einfo), einfo.traceback)
  128. self.assertIsInstance(einfo.exception, LookupError)
  129. self.assertTupleEqual(
  130. einfo.exception.args, ('The quick brown fox jumps...', ),
  131. )
  132. self.assertTrue(einfo.traceback)
  133. r = repr(einfo)
  134. self.assertTrue(r)
  135. class test_LimitedSet(Case):
  136. def test_add(self):
  137. if sys.platform == 'win32':
  138. raise SkipTest('Not working properly on Windows')
  139. s = LimitedSet(maxlen=2)
  140. s.add('foo')
  141. s.add('bar')
  142. for n in 'foo', 'bar':
  143. self.assertIn(n, s)
  144. s.add('baz')
  145. for n in 'bar', 'baz':
  146. self.assertIn(n, s)
  147. self.assertNotIn('foo', s)
  148. def test_purge(self):
  149. s = LimitedSet(maxlen=None)
  150. [s.add(i) for i in range(10)]
  151. s.maxlen = 2
  152. s.purge(1)
  153. self.assertEqual(len(s), 9)
  154. s.purge(None)
  155. self.assertEqual(len(s), 2)
  156. # expired
  157. s = LimitedSet(maxlen=None, expires=1)
  158. [s.add(i) for i in range(10)]
  159. s.maxlen = 2
  160. s.purge(1, now=lambda: time() + 100)
  161. self.assertEqual(len(s), 9)
  162. s.purge(None, now=lambda: time() + 100)
  163. self.assertEqual(len(s), 2)
  164. # not expired
  165. s = LimitedSet(maxlen=None, expires=1)
  166. [s.add(i) for i in range(10)]
  167. s.maxlen = 2
  168. s.purge(1, now=lambda: time() - 100)
  169. self.assertEqual(len(s), 10)
  170. s.purge(None, now=lambda: time() - 100)
  171. self.assertEqual(len(s), 10)
  172. s = LimitedSet(maxlen=None)
  173. [s.add(i) for i in range(10)]
  174. s.maxlen = 2
  175. with patch('celery.datastructures.heappop') as hp:
  176. hp.side_effect = IndexError()
  177. s.purge()
  178. hp.assert_called_with(s._heap)
  179. with patch('celery.datastructures.heappop') as hp:
  180. s._data = dict((i * 2, i * 2) for i in range(10))
  181. s.purge()
  182. self.assertEqual(hp.call_count, 10)
  183. def test_pickleable(self):
  184. s = LimitedSet(maxlen=2)
  185. s.add('foo')
  186. s.add('bar')
  187. self.assertEqual(pickle.loads(pickle.dumps(s)), s)
  188. def test_iter(self):
  189. s = LimitedSet(maxlen=3)
  190. items = ['foo', 'bar', 'baz', 'xaz']
  191. for item in items:
  192. s.add(item)
  193. l = list(iter(s))
  194. for item in items[1:]:
  195. self.assertIn(item, l)
  196. self.assertNotIn('foo', l)
  197. self.assertListEqual(l, items[1:], 'order by insertion time')
  198. def test_repr(self):
  199. s = LimitedSet(maxlen=2)
  200. items = 'foo', 'bar'
  201. for item in items:
  202. s.add(item)
  203. self.assertIn('LimitedSet(', repr(s))
  204. def test_discard(self):
  205. s = LimitedSet(maxlen=2)
  206. s.add('foo')
  207. s.discard('foo')
  208. self.assertNotIn('foo', s)
  209. s.discard('foo')
  210. def test_clear(self):
  211. s = LimitedSet(maxlen=2)
  212. s.add('foo')
  213. s.add('bar')
  214. self.assertEqual(len(s), 2)
  215. s.clear()
  216. self.assertFalse(s)
  217. def test_update(self):
  218. s1 = LimitedSet(maxlen=2)
  219. s1.add('foo')
  220. s1.add('bar')
  221. s2 = LimitedSet(maxlen=2)
  222. s2.update(s1)
  223. self.assertItemsEqual(list(s2), ['foo', 'bar'])
  224. s2.update(['bla'])
  225. self.assertItemsEqual(list(s2), ['bla', 'bar'])
  226. s2.update(['do', 're'])
  227. self.assertItemsEqual(list(s2), ['do', 're'])
  228. def test_as_dict(self):
  229. s = LimitedSet(maxlen=2)
  230. s.add('foo')
  231. self.assertIsInstance(s.as_dict(), dict)
  232. class test_AttributeDict(Case):
  233. def test_getattr__setattr(self):
  234. x = AttributeDict({'foo': 'bar'})
  235. self.assertEqual(x['foo'], 'bar')
  236. with self.assertRaises(AttributeError):
  237. x.bar
  238. x.bar = 'foo'
  239. self.assertEqual(x['bar'], 'foo')
  240. class test_DependencyGraph(Case):
  241. def graph1(self):
  242. return DependencyGraph([
  243. ('A', []),
  244. ('B', []),
  245. ('C', ['A']),
  246. ('D', ['C', 'B']),
  247. ])
  248. def test_repr(self):
  249. self.assertTrue(repr(self.graph1()))
  250. def test_topsort(self):
  251. order = self.graph1().topsort()
  252. # C must start before D
  253. self.assertLess(order.index('C'), order.index('D'))
  254. # and B must start before D
  255. self.assertLess(order.index('B'), order.index('D'))
  256. # and A must start before C
  257. self.assertLess(order.index('A'), order.index('C'))
  258. def test_edges(self):
  259. self.assertItemsEqual(
  260. list(self.graph1().edges()),
  261. ['C', 'D'],
  262. )
  263. def test_connect(self):
  264. x, y = self.graph1(), self.graph1()
  265. x.connect(y)
  266. def test_valency_of_when_missing(self):
  267. x = self.graph1()
  268. self.assertEqual(x.valency_of('foobarbaz'), 0)
  269. def test_format(self):
  270. x = self.graph1()
  271. x.formatter = Mock()
  272. obj = Mock()
  273. self.assertTrue(x.format(obj))
  274. x.formatter.assert_called_with(obj)
  275. x.formatter = None
  276. self.assertIs(x.format(obj), obj)
  277. def test_items(self):
  278. self.assertDictEqual(
  279. dict(items(self.graph1())),
  280. {'A': [], 'B': [], 'C': ['A'], 'D': ['C', 'B']},
  281. )
  282. def test_repr_node(self):
  283. x = self.graph1()
  284. self.assertTrue(x.repr_node('fasdswewqewq'))
  285. def test_to_dot(self):
  286. s = WhateverIO()
  287. self.graph1().to_dot(s)
  288. self.assertTrue(s.getvalue())