datastructures.py 9.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277
  1. """
  2. Tests for stuff in django.utils.datastructures.
  3. """
  4. import pickle
  5. import unittest
  6. from django.utils.copycompat import copy
  7. from django.utils.datastructures import *
  8. class DatastructuresTestCase(unittest.TestCase):
  9. def assertRaisesErrorWithMessage(self, error, message, callable,
  10. *args, **kwargs):
  11. self.assertRaises(error, callable, *args, **kwargs)
  12. try:
  13. callable(*args, **kwargs)
  14. except error, e:
  15. self.assertEqual(message, str(e))
  16. class SortedDictTests(DatastructuresTestCase):
  17. def setUp(self):
  18. self.d1 = SortedDict()
  19. self.d1[7] = 'seven'
  20. self.d1[1] = 'one'
  21. self.d1[9] = 'nine'
  22. self.d2 = SortedDict()
  23. self.d2[1] = 'one'
  24. self.d2[9] = 'nine'
  25. self.d2[0] = 'nil'
  26. self.d2[7] = 'seven'
  27. def test_basic_methods(self):
  28. self.assertEqual(self.d1.keys(), [7, 1, 9])
  29. self.assertEqual(self.d1.values(), ['seven', 'one', 'nine'])
  30. self.assertEqual(self.d1.items(), [(7, 'seven'), (1, 'one'), (9, 'nine')])
  31. def test_overwrite_ordering(self):
  32. """ Overwriting an item keeps it's place. """
  33. self.d1[1] = 'ONE'
  34. self.assertEqual(self.d1.values(), ['seven', 'ONE', 'nine'])
  35. def test_append_items(self):
  36. """ New items go to the end. """
  37. self.d1[0] = 'nil'
  38. self.assertEqual(self.d1.keys(), [7, 1, 9, 0])
  39. def test_delete_and_insert(self):
  40. """
  41. Deleting an item, then inserting the same key again will place it
  42. at the end.
  43. """
  44. del self.d2[7]
  45. self.assertEqual(self.d2.keys(), [1, 9, 0])
  46. self.d2[7] = 'lucky number 7'
  47. self.assertEqual(self.d2.keys(), [1, 9, 0, 7])
  48. def test_change_keys(self):
  49. """
  50. Changing the keys won't do anything, it's only a copy of the
  51. keys dict.
  52. """
  53. k = self.d2.keys()
  54. k.remove(9)
  55. self.assertEqual(self.d2.keys(), [1, 9, 0, 7])
  56. def test_init_keys(self):
  57. """
  58. Initialising a SortedDict with two keys will just take the first one.
  59. A real dict will actually take the second value so we will too, but
  60. we'll keep the ordering from the first key found.
  61. """
  62. tuples = ((2, 'two'), (1, 'one'), (2, 'second-two'))
  63. d = SortedDict(tuples)
  64. self.assertEqual(d.keys(), [2, 1])
  65. real_dict = dict(tuples)
  66. self.assertEqual(sorted(real_dict.values()), ['one', 'second-two'])
  67. # Here the order of SortedDict values *is* what we are testing
  68. self.assertEqual(d.values(), ['second-two', 'one'])
  69. def test_overwrite(self):
  70. self.d1[1] = 'not one'
  71. self.assertEqual(self.d1[1], 'not one')
  72. self.assertEqual(self.d1.keys(), self.d1.copy().keys())
  73. def test_append(self):
  74. self.d1[13] = 'thirteen'
  75. self.assertEqual(
  76. repr(self.d1),
  77. "{7: 'seven', 1: 'one', 9: 'nine', 13: 'thirteen'}"
  78. )
  79. def test_pop(self):
  80. self.assertEqual(self.d1.pop(1, 'missing'), 'one')
  81. self.assertEqual(self.d1.pop(1, 'missing'), 'missing')
  82. # We don't know which item will be popped in popitem(), so we'll
  83. # just check that the number of keys has decreased.
  84. l = len(self.d1)
  85. self.d1.popitem()
  86. self.assertEqual(l - len(self.d1), 1)
  87. def test_dict_equality(self):
  88. d = SortedDict((i, i) for i in xrange(3))
  89. self.assertEqual(d, {0: 0, 1: 1, 2: 2})
  90. def test_tuple_init(self):
  91. d = SortedDict(((1, "one"), (0, "zero"), (2, "two")))
  92. self.assertEqual(repr(d), "{1: 'one', 0: 'zero', 2: 'two'}")
  93. def test_pickle(self):
  94. self.assertEqual(
  95. pickle.loads(pickle.dumps(self.d1, 2)),
  96. {7: 'seven', 1: 'one', 9: 'nine'}
  97. )
  98. def test_clear(self):
  99. self.d1.clear()
  100. self.assertEqual(self.d1, {})
  101. self.assertEqual(self.d1.keyOrder, [])
  102. class MergeDictTests(DatastructuresTestCase):
  103. def test_simple_mergedict(self):
  104. d1 = {'chris':'cool', 'camri':'cute', 'cotton':'adorable',
  105. 'tulip':'snuggable', 'twoofme':'firstone'}
  106. d2 = {'chris2':'cool2', 'camri2':'cute2', 'cotton2':'adorable2',
  107. 'tulip2':'snuggable2'}
  108. d3 = {'chris3':'cool3', 'camri3':'cute3', 'cotton3':'adorable3',
  109. 'tulip3':'snuggable3'}
  110. d4 = {'twoofme': 'secondone'}
  111. md = MergeDict(d1, d2, d3)
  112. self.assertEqual(md['chris'], 'cool')
  113. self.assertEqual(md['camri'], 'cute')
  114. self.assertEqual(md['twoofme'], 'firstone')
  115. md2 = md.copy()
  116. self.assertEqual(md2['chris'], 'cool')
  117. def test_mergedict_merges_multivaluedict(self):
  118. """ MergeDict can merge MultiValueDicts """
  119. multi1 = MultiValueDict({'key1': ['value1'],
  120. 'key2': ['value2', 'value3']})
  121. multi2 = MultiValueDict({'key2': ['value4'],
  122. 'key4': ['value5', 'value6']})
  123. mm = MergeDict(multi1, multi2)
  124. # Although 'key2' appears in both dictionaries,
  125. # only the first value is used.
  126. self.assertEqual(mm.getlist('key2'), ['value2', 'value3'])
  127. self.assertEqual(mm.getlist('key4'), ['value5', 'value6'])
  128. self.assertEqual(mm.getlist('undefined'), [])
  129. self.assertEqual(sorted(mm.keys()), ['key1', 'key2', 'key4'])
  130. self.assertEqual(len(mm.values()), 3)
  131. self.assertTrue('value1' in mm.values())
  132. self.assertEqual(sorted(mm.items(), key=lambda k: k[0]),
  133. [('key1', 'value1'), ('key2', 'value3'),
  134. ('key4', 'value6')])
  135. self.assertEqual([(k,mm.getlist(k)) for k in sorted(mm)],
  136. [('key1', ['value1']),
  137. ('key2', ['value2', 'value3']),
  138. ('key4', ['value5', 'value6'])])
  139. class MultiValueDictTests(DatastructuresTestCase):
  140. def test_multivaluedict(self):
  141. d = MultiValueDict({'name': ['Adrian', 'Simon'],
  142. 'position': ['Developer']})
  143. self.assertEqual(d['name'], 'Simon')
  144. self.assertEqual(d.get('name'), 'Simon')
  145. self.assertEqual(d.getlist('name'), ['Adrian', 'Simon'])
  146. self.assertEqual(list(d.iteritems()),
  147. [('position', 'Developer'), ('name', 'Simon')])
  148. self.assertEqual(list(d.iterlists()),
  149. [('position', ['Developer']),
  150. ('name', ['Adrian', 'Simon'])])
  151. # MultiValueDictKeyError: "Key 'lastname' not found in
  152. # <MultiValueDict: {'position': ['Developer'],
  153. # 'name': ['Adrian', 'Simon']}>"
  154. self.assertRaisesErrorWithMessage(MultiValueDictKeyError,
  155. '"Key \'lastname\' not found in <MultiValueDict: {\'position\':'\
  156. ' [\'Developer\'], \'name\': [\'Adrian\', \'Simon\']}>"',
  157. d.__getitem__, 'lastname')
  158. self.assertEqual(d.get('lastname'), None)
  159. self.assertEqual(d.get('lastname', 'nonexistent'), 'nonexistent')
  160. self.assertEqual(d.getlist('lastname'), [])
  161. d.setlist('lastname', ['Holovaty', 'Willison'])
  162. self.assertEqual(d.getlist('lastname'), ['Holovaty', 'Willison'])
  163. self.assertEqual(d.values(), ['Developer', 'Simon', 'Willison'])
  164. self.assertEqual(list(d.itervalues()),
  165. ['Developer', 'Simon', 'Willison'])
  166. def test_copy(self):
  167. for copy_func in [copy, lambda d: d.copy()]:
  168. d1 = MultiValueDict({
  169. "developers": ["Carl", "Fred"]
  170. })
  171. self.assertEqual(d1["developers"], "Fred")
  172. d2 = copy_func(d1)
  173. d2.update({"developers": "Groucho"})
  174. self.assertEqual(d2["developers"], "Groucho")
  175. self.assertEqual(d1["developers"], "Fred")
  176. d1 = MultiValueDict({
  177. "key": [[]]
  178. })
  179. self.assertEqual(d1["key"], [])
  180. d2 = copy_func(d1)
  181. d2["key"].append("Penguin")
  182. self.assertEqual(d1["key"], ["Penguin"])
  183. self.assertEqual(d2["key"], ["Penguin"])
  184. class DotExpandedDictTests(DatastructuresTestCase):
  185. def test_dotexpandeddict(self):
  186. d = DotExpandedDict({'person.1.firstname': ['Simon'],
  187. 'person.1.lastname': ['Willison'],
  188. 'person.2.firstname': ['Adrian'],
  189. 'person.2.lastname': ['Holovaty']})
  190. self.assertEqual(d['person']['1']['lastname'], ['Willison'])
  191. self.assertEqual(d['person']['2']['lastname'], ['Holovaty'])
  192. self.assertEqual(d['person']['2']['firstname'], ['Adrian'])
  193. class ImmutableListTests(DatastructuresTestCase):
  194. def test_sort(self):
  195. d = ImmutableList(range(10))
  196. # AttributeError: ImmutableList object is immutable.
  197. self.assertRaisesErrorWithMessage(AttributeError,
  198. 'ImmutableList object is immutable.', d.sort)
  199. self.assertEqual(repr(d), '(0, 1, 2, 3, 4, 5, 6, 7, 8, 9)')
  200. def test_custom_warning(self):
  201. d = ImmutableList(range(10), warning="Object is immutable!")
  202. self.assertEqual(d[1], 1)
  203. # AttributeError: Object is immutable!
  204. self.assertRaisesErrorWithMessage(AttributeError,
  205. 'Object is immutable!', d.__setitem__, 1, 'test')
  206. class DictWrapperTests(DatastructuresTestCase):
  207. def test_dictwrapper(self):
  208. f = lambda x: "*%s" % x
  209. d = DictWrapper({'a': 'a'}, f, 'xx_')
  210. self.assertEqual("Normal: %(a)s. Modified: %(xx_a)s" % d,
  211. 'Normal: a. Modified: *a')