test_hstore.py 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301
  1. # -*- coding: utf-8 -*-
  2. from __future__ import unicode_literals
  3. import json
  4. from django.core import exceptions, serializers
  5. from django.forms import Form
  6. from . import PostgreSQLTestCase
  7. from .models import HStoreModel
  8. try:
  9. from django.contrib.postgres import forms
  10. from django.contrib.postgres.fields import HStoreField
  11. from django.contrib.postgres.validators import KeysValidator
  12. except ImportError:
  13. pass
  14. class SimpleTests(PostgreSQLTestCase):
  15. apps = ['django.contrib.postgres']
  16. def test_save_load_success(self):
  17. value = {'a': 'b'}
  18. instance = HStoreModel(field=value)
  19. instance.save()
  20. reloaded = HStoreModel.objects.get()
  21. self.assertEqual(reloaded.field, value)
  22. def test_null(self):
  23. instance = HStoreModel(field=None)
  24. instance.save()
  25. reloaded = HStoreModel.objects.get()
  26. self.assertIsNone(reloaded.field)
  27. def test_value_null(self):
  28. value = {'a': None}
  29. instance = HStoreModel(field=value)
  30. instance.save()
  31. reloaded = HStoreModel.objects.get()
  32. self.assertEqual(reloaded.field, value)
  33. def test_key_val_cast_to_string(self):
  34. value = {'a': 1, 'b': 'B', 2: 'c', 'ï': 'ê', b'x': b'test'}
  35. expected_value = {'a': '1', 'b': 'B', '2': 'c', 'ï': 'ê', 'x': 'test'}
  36. instance = HStoreModel.objects.create(field=value)
  37. instance = HStoreModel.objects.get()
  38. self.assertDictEqual(instance.field, expected_value)
  39. instance = HStoreModel.objects.get(field__a=1)
  40. self.assertDictEqual(instance.field, expected_value)
  41. instance = HStoreModel.objects.get(field__has_keys=[2, 'a', 'ï'])
  42. self.assertDictEqual(instance.field, expected_value)
  43. class TestQuerying(PostgreSQLTestCase):
  44. def setUp(self):
  45. self.objs = [
  46. HStoreModel.objects.create(field={'a': 'b'}),
  47. HStoreModel.objects.create(field={'a': 'b', 'c': 'd'}),
  48. HStoreModel.objects.create(field={'c': 'd'}),
  49. HStoreModel.objects.create(field={}),
  50. HStoreModel.objects.create(field=None),
  51. ]
  52. def test_exact(self):
  53. self.assertSequenceEqual(
  54. HStoreModel.objects.filter(field__exact={'a': 'b'}),
  55. self.objs[:1]
  56. )
  57. def test_contained_by(self):
  58. self.assertSequenceEqual(
  59. HStoreModel.objects.filter(field__contained_by={'a': 'b', 'c': 'd'}),
  60. self.objs[:4]
  61. )
  62. def test_contains(self):
  63. self.assertSequenceEqual(
  64. HStoreModel.objects.filter(field__contains={'a': 'b'}),
  65. self.objs[:2]
  66. )
  67. def test_in_generator(self):
  68. def search():
  69. yield {'a': 'b'}
  70. self.assertSequenceEqual(
  71. HStoreModel.objects.filter(field__in=search()),
  72. self.objs[:1]
  73. )
  74. def test_has_key(self):
  75. self.assertSequenceEqual(
  76. HStoreModel.objects.filter(field__has_key='c'),
  77. self.objs[1:3]
  78. )
  79. def test_has_keys(self):
  80. self.assertSequenceEqual(
  81. HStoreModel.objects.filter(field__has_keys=['a', 'c']),
  82. self.objs[1:2]
  83. )
  84. def test_has_any_keys(self):
  85. self.assertSequenceEqual(
  86. HStoreModel.objects.filter(field__has_any_keys=['a', 'c']),
  87. self.objs[:3]
  88. )
  89. def test_key_transform(self):
  90. self.assertSequenceEqual(
  91. HStoreModel.objects.filter(field__a='b'),
  92. self.objs[:2]
  93. )
  94. def test_keys(self):
  95. self.assertSequenceEqual(
  96. HStoreModel.objects.filter(field__keys=['a']),
  97. self.objs[:1]
  98. )
  99. def test_values(self):
  100. self.assertSequenceEqual(
  101. HStoreModel.objects.filter(field__values=['b']),
  102. self.objs[:1]
  103. )
  104. def test_field_chaining(self):
  105. self.assertSequenceEqual(
  106. HStoreModel.objects.filter(field__a__contains='b'),
  107. self.objs[:2]
  108. )
  109. def test_keys_contains(self):
  110. self.assertSequenceEqual(
  111. HStoreModel.objects.filter(field__keys__contains=['a']),
  112. self.objs[:2]
  113. )
  114. def test_values_overlap(self):
  115. self.assertSequenceEqual(
  116. HStoreModel.objects.filter(field__values__overlap=['b', 'd']),
  117. self.objs[:3]
  118. )
  119. def test_key_isnull(self):
  120. obj = HStoreModel.objects.create(field={'a': None})
  121. self.assertSequenceEqual(
  122. HStoreModel.objects.filter(field__a__isnull=True),
  123. self.objs[2:5] + [obj]
  124. )
  125. self.assertSequenceEqual(
  126. HStoreModel.objects.filter(field__a__isnull=False),
  127. self.objs[:2]
  128. )
  129. def test_usage_in_subquery(self):
  130. self.assertSequenceEqual(
  131. HStoreModel.objects.filter(id__in=HStoreModel.objects.filter(field__a='b')),
  132. self.objs[:2]
  133. )
  134. class TestSerialization(PostgreSQLTestCase):
  135. test_data = ('[{"fields": {"field": "{\\"a\\": \\"b\\"}"}, '
  136. '"model": "postgres_tests.hstoremodel", "pk": null}]')
  137. def test_dumping(self):
  138. instance = HStoreModel(field={'a': 'b'})
  139. data = serializers.serialize('json', [instance])
  140. self.assertEqual(json.loads(data), json.loads(self.test_data))
  141. def test_loading(self):
  142. instance = list(serializers.deserialize('json', self.test_data))[0].object
  143. self.assertEqual(instance.field, {'a': 'b'})
  144. def test_roundtrip_with_null(self):
  145. instance = HStoreModel(field={'a': 'b', 'c': None})
  146. data = serializers.serialize('json', [instance])
  147. new_instance = list(serializers.deserialize('json', data))[0].object
  148. self.assertEqual(instance.field, new_instance.field)
  149. class TestValidation(PostgreSQLTestCase):
  150. def test_not_a_string(self):
  151. field = HStoreField()
  152. with self.assertRaises(exceptions.ValidationError) as cm:
  153. field.clean({'a': 1}, None)
  154. self.assertEqual(cm.exception.code, 'not_a_string')
  155. self.assertEqual(cm.exception.message % cm.exception.params, 'The value of "a" is not a string.')
  156. class TestFormField(PostgreSQLTestCase):
  157. def test_valid(self):
  158. field = forms.HStoreField()
  159. value = field.clean('{"a": "b"}')
  160. self.assertEqual(value, {'a': 'b'})
  161. def test_invalid_json(self):
  162. field = forms.HStoreField()
  163. with self.assertRaises(exceptions.ValidationError) as cm:
  164. field.clean('{"a": "b"')
  165. self.assertEqual(cm.exception.messages[0], 'Could not load JSON data.')
  166. self.assertEqual(cm.exception.code, 'invalid_json')
  167. def test_non_dict_json(self):
  168. field = forms.HStoreField()
  169. msg = 'Input must be a JSON dictionary.'
  170. with self.assertRaisesMessage(exceptions.ValidationError, msg) as cm:
  171. field.clean('["a", "b", 1]')
  172. self.assertEqual(cm.exception.code, 'invalid_format')
  173. def test_not_string_values(self):
  174. field = forms.HStoreField()
  175. value = field.clean('{"a": 1}')
  176. self.assertEqual(value, {'a': '1'})
  177. def test_empty(self):
  178. field = forms.HStoreField(required=False)
  179. value = field.clean('')
  180. self.assertEqual(value, {})
  181. def test_model_field_formfield(self):
  182. model_field = HStoreField()
  183. form_field = model_field.formfield()
  184. self.assertIsInstance(form_field, forms.HStoreField)
  185. def test_field_has_changed(self):
  186. class HStoreFormTest(Form):
  187. f1 = forms.HStoreField()
  188. form_w_hstore = HStoreFormTest()
  189. self.assertFalse(form_w_hstore.has_changed())
  190. form_w_hstore = HStoreFormTest({'f1': '{"a": 1}'})
  191. self.assertTrue(form_w_hstore.has_changed())
  192. form_w_hstore = HStoreFormTest({'f1': '{"a": 1}'}, initial={'f1': '{"a": 1}'})
  193. self.assertFalse(form_w_hstore.has_changed())
  194. form_w_hstore = HStoreFormTest({'f1': '{"a": 2}'}, initial={'f1': '{"a": 1}'})
  195. self.assertTrue(form_w_hstore.has_changed())
  196. form_w_hstore = HStoreFormTest({'f1': '{"a": 1}'}, initial={'f1': {"a": 1}})
  197. self.assertFalse(form_w_hstore.has_changed())
  198. form_w_hstore = HStoreFormTest({'f1': '{"a": 2}'}, initial={'f1': {"a": 1}})
  199. self.assertTrue(form_w_hstore.has_changed())
  200. class TestValidator(PostgreSQLTestCase):
  201. def test_simple_valid(self):
  202. validator = KeysValidator(keys=['a', 'b'])
  203. validator({'a': 'foo', 'b': 'bar', 'c': 'baz'})
  204. def test_missing_keys(self):
  205. validator = KeysValidator(keys=['a', 'b'])
  206. with self.assertRaises(exceptions.ValidationError) as cm:
  207. validator({'a': 'foo', 'c': 'baz'})
  208. self.assertEqual(cm.exception.messages[0], 'Some keys were missing: b')
  209. self.assertEqual(cm.exception.code, 'missing_keys')
  210. def test_strict_valid(self):
  211. validator = KeysValidator(keys=['a', 'b'], strict=True)
  212. validator({'a': 'foo', 'b': 'bar'})
  213. def test_extra_keys(self):
  214. validator = KeysValidator(keys=['a', 'b'], strict=True)
  215. with self.assertRaises(exceptions.ValidationError) as cm:
  216. validator({'a': 'foo', 'b': 'bar', 'c': 'baz'})
  217. self.assertEqual(cm.exception.messages[0], 'Some unknown keys were provided: c')
  218. self.assertEqual(cm.exception.code, 'extra_keys')
  219. def test_custom_messages(self):
  220. messages = {
  221. 'missing_keys': 'Foobar',
  222. }
  223. validator = KeysValidator(keys=['a', 'b'], strict=True, messages=messages)
  224. with self.assertRaises(exceptions.ValidationError) as cm:
  225. validator({'a': 'foo', 'c': 'baz'})
  226. self.assertEqual(cm.exception.messages[0], 'Foobar')
  227. self.assertEqual(cm.exception.code, 'missing_keys')
  228. with self.assertRaises(exceptions.ValidationError) as cm:
  229. validator({'a': 'foo', 'b': 'bar', 'c': 'baz'})
  230. self.assertEqual(cm.exception.messages[0], 'Some unknown keys were provided: c')
  231. self.assertEqual(cm.exception.code, 'extra_keys')
  232. def test_deconstruct(self):
  233. messages = {
  234. 'missing_keys': 'Foobar',
  235. }
  236. validator = KeysValidator(keys=['a', 'b'], strict=True, messages=messages)
  237. path, args, kwargs = validator.deconstruct()
  238. self.assertEqual(path, 'django.contrib.postgres.validators.KeysValidator')
  239. self.assertEqual(args, ())
  240. self.assertEqual(kwargs, {'keys': ['a', 'b'], 'strict': True, 'messages': messages})