123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301 |
- # -*- coding: utf-8 -*-
- from __future__ import unicode_literals
- import json
- from django.core import exceptions, serializers
- from django.forms import Form
- from . import PostgreSQLTestCase
- from .models import HStoreModel
- try:
- from django.contrib.postgres import forms
- from django.contrib.postgres.fields import HStoreField
- from django.contrib.postgres.validators import KeysValidator
- except ImportError:
- pass
- class SimpleTests(PostgreSQLTestCase):
- apps = ['django.contrib.postgres']
- def test_save_load_success(self):
- value = {'a': 'b'}
- instance = HStoreModel(field=value)
- instance.save()
- reloaded = HStoreModel.objects.get()
- self.assertEqual(reloaded.field, value)
- def test_null(self):
- instance = HStoreModel(field=None)
- instance.save()
- reloaded = HStoreModel.objects.get()
- self.assertIsNone(reloaded.field)
- def test_value_null(self):
- value = {'a': None}
- instance = HStoreModel(field=value)
- instance.save()
- reloaded = HStoreModel.objects.get()
- self.assertEqual(reloaded.field, value)
- def test_key_val_cast_to_string(self):
- value = {'a': 1, 'b': 'B', 2: 'c', 'ï': 'ê', b'x': b'test'}
- expected_value = {'a': '1', 'b': 'B', '2': 'c', 'ï': 'ê', 'x': 'test'}
- instance = HStoreModel.objects.create(field=value)
- instance = HStoreModel.objects.get()
- self.assertDictEqual(instance.field, expected_value)
- instance = HStoreModel.objects.get(field__a=1)
- self.assertDictEqual(instance.field, expected_value)
- instance = HStoreModel.objects.get(field__has_keys=[2, 'a', 'ï'])
- self.assertDictEqual(instance.field, expected_value)
- class TestQuerying(PostgreSQLTestCase):
- def setUp(self):
- self.objs = [
- HStoreModel.objects.create(field={'a': 'b'}),
- HStoreModel.objects.create(field={'a': 'b', 'c': 'd'}),
- HStoreModel.objects.create(field={'c': 'd'}),
- HStoreModel.objects.create(field={}),
- HStoreModel.objects.create(field=None),
- ]
- def test_exact(self):
- self.assertSequenceEqual(
- HStoreModel.objects.filter(field__exact={'a': 'b'}),
- self.objs[:1]
- )
- def test_contained_by(self):
- self.assertSequenceEqual(
- HStoreModel.objects.filter(field__contained_by={'a': 'b', 'c': 'd'}),
- self.objs[:4]
- )
- def test_contains(self):
- self.assertSequenceEqual(
- HStoreModel.objects.filter(field__contains={'a': 'b'}),
- self.objs[:2]
- )
- def test_in_generator(self):
- def search():
- yield {'a': 'b'}
- self.assertSequenceEqual(
- HStoreModel.objects.filter(field__in=search()),
- self.objs[:1]
- )
- def test_has_key(self):
- self.assertSequenceEqual(
- HStoreModel.objects.filter(field__has_key='c'),
- self.objs[1:3]
- )
- def test_has_keys(self):
- self.assertSequenceEqual(
- HStoreModel.objects.filter(field__has_keys=['a', 'c']),
- self.objs[1:2]
- )
- def test_has_any_keys(self):
- self.assertSequenceEqual(
- HStoreModel.objects.filter(field__has_any_keys=['a', 'c']),
- self.objs[:3]
- )
- def test_key_transform(self):
- self.assertSequenceEqual(
- HStoreModel.objects.filter(field__a='b'),
- self.objs[:2]
- )
- def test_keys(self):
- self.assertSequenceEqual(
- HStoreModel.objects.filter(field__keys=['a']),
- self.objs[:1]
- )
- def test_values(self):
- self.assertSequenceEqual(
- HStoreModel.objects.filter(field__values=['b']),
- self.objs[:1]
- )
- def test_field_chaining(self):
- self.assertSequenceEqual(
- HStoreModel.objects.filter(field__a__contains='b'),
- self.objs[:2]
- )
- def test_keys_contains(self):
- self.assertSequenceEqual(
- HStoreModel.objects.filter(field__keys__contains=['a']),
- self.objs[:2]
- )
- def test_values_overlap(self):
- self.assertSequenceEqual(
- HStoreModel.objects.filter(field__values__overlap=['b', 'd']),
- self.objs[:3]
- )
- def test_key_isnull(self):
- obj = HStoreModel.objects.create(field={'a': None})
- self.assertSequenceEqual(
- HStoreModel.objects.filter(field__a__isnull=True),
- self.objs[2:5] + [obj]
- )
- self.assertSequenceEqual(
- HStoreModel.objects.filter(field__a__isnull=False),
- self.objs[:2]
- )
- def test_usage_in_subquery(self):
- self.assertSequenceEqual(
- HStoreModel.objects.filter(id__in=HStoreModel.objects.filter(field__a='b')),
- self.objs[:2]
- )
- class TestSerialization(PostgreSQLTestCase):
- test_data = ('[{"fields": {"field": "{\\"a\\": \\"b\\"}"}, '
- '"model": "postgres_tests.hstoremodel", "pk": null}]')
- def test_dumping(self):
- instance = HStoreModel(field={'a': 'b'})
- data = serializers.serialize('json', [instance])
- self.assertEqual(json.loads(data), json.loads(self.test_data))
- def test_loading(self):
- instance = list(serializers.deserialize('json', self.test_data))[0].object
- self.assertEqual(instance.field, {'a': 'b'})
- def test_roundtrip_with_null(self):
- instance = HStoreModel(field={'a': 'b', 'c': None})
- data = serializers.serialize('json', [instance])
- new_instance = list(serializers.deserialize('json', data))[0].object
- self.assertEqual(instance.field, new_instance.field)
- class TestValidation(PostgreSQLTestCase):
- def test_not_a_string(self):
- field = HStoreField()
- with self.assertRaises(exceptions.ValidationError) as cm:
- field.clean({'a': 1}, None)
- self.assertEqual(cm.exception.code, 'not_a_string')
- self.assertEqual(cm.exception.message % cm.exception.params, 'The value of "a" is not a string.')
- class TestFormField(PostgreSQLTestCase):
- def test_valid(self):
- field = forms.HStoreField()
- value = field.clean('{"a": "b"}')
- self.assertEqual(value, {'a': 'b'})
- def test_invalid_json(self):
- field = forms.HStoreField()
- with self.assertRaises(exceptions.ValidationError) as cm:
- field.clean('{"a": "b"')
- self.assertEqual(cm.exception.messages[0], 'Could not load JSON data.')
- self.assertEqual(cm.exception.code, 'invalid_json')
- def test_non_dict_json(self):
- field = forms.HStoreField()
- msg = 'Input must be a JSON dictionary.'
- with self.assertRaisesMessage(exceptions.ValidationError, msg) as cm:
- field.clean('["a", "b", 1]')
- self.assertEqual(cm.exception.code, 'invalid_format')
- def test_not_string_values(self):
- field = forms.HStoreField()
- value = field.clean('{"a": 1}')
- self.assertEqual(value, {'a': '1'})
- def test_empty(self):
- field = forms.HStoreField(required=False)
- value = field.clean('')
- self.assertEqual(value, {})
- def test_model_field_formfield(self):
- model_field = HStoreField()
- form_field = model_field.formfield()
- self.assertIsInstance(form_field, forms.HStoreField)
- def test_field_has_changed(self):
- class HStoreFormTest(Form):
- f1 = forms.HStoreField()
- form_w_hstore = HStoreFormTest()
- self.assertFalse(form_w_hstore.has_changed())
- form_w_hstore = HStoreFormTest({'f1': '{"a": 1}'})
- self.assertTrue(form_w_hstore.has_changed())
- form_w_hstore = HStoreFormTest({'f1': '{"a": 1}'}, initial={'f1': '{"a": 1}'})
- self.assertFalse(form_w_hstore.has_changed())
- form_w_hstore = HStoreFormTest({'f1': '{"a": 2}'}, initial={'f1': '{"a": 1}'})
- self.assertTrue(form_w_hstore.has_changed())
- form_w_hstore = HStoreFormTest({'f1': '{"a": 1}'}, initial={'f1': {"a": 1}})
- self.assertFalse(form_w_hstore.has_changed())
- form_w_hstore = HStoreFormTest({'f1': '{"a": 2}'}, initial={'f1': {"a": 1}})
- self.assertTrue(form_w_hstore.has_changed())
- class TestValidator(PostgreSQLTestCase):
- def test_simple_valid(self):
- validator = KeysValidator(keys=['a', 'b'])
- validator({'a': 'foo', 'b': 'bar', 'c': 'baz'})
- def test_missing_keys(self):
- validator = KeysValidator(keys=['a', 'b'])
- with self.assertRaises(exceptions.ValidationError) as cm:
- validator({'a': 'foo', 'c': 'baz'})
- self.assertEqual(cm.exception.messages[0], 'Some keys were missing: b')
- self.assertEqual(cm.exception.code, 'missing_keys')
- def test_strict_valid(self):
- validator = KeysValidator(keys=['a', 'b'], strict=True)
- validator({'a': 'foo', 'b': 'bar'})
- def test_extra_keys(self):
- validator = KeysValidator(keys=['a', 'b'], strict=True)
- with self.assertRaises(exceptions.ValidationError) as cm:
- validator({'a': 'foo', 'b': 'bar', 'c': 'baz'})
- self.assertEqual(cm.exception.messages[0], 'Some unknown keys were provided: c')
- self.assertEqual(cm.exception.code, 'extra_keys')
- def test_custom_messages(self):
- messages = {
- 'missing_keys': 'Foobar',
- }
- validator = KeysValidator(keys=['a', 'b'], strict=True, messages=messages)
- with self.assertRaises(exceptions.ValidationError) as cm:
- validator({'a': 'foo', 'c': 'baz'})
- self.assertEqual(cm.exception.messages[0], 'Foobar')
- self.assertEqual(cm.exception.code, 'missing_keys')
- with self.assertRaises(exceptions.ValidationError) as cm:
- validator({'a': 'foo', 'b': 'bar', 'c': 'baz'})
- self.assertEqual(cm.exception.messages[0], 'Some unknown keys were provided: c')
- self.assertEqual(cm.exception.code, 'extra_keys')
- def test_deconstruct(self):
- messages = {
- 'missing_keys': 'Foobar',
- }
- validator = KeysValidator(keys=['a', 'b'], strict=True, messages=messages)
- path, args, kwargs = validator.deconstruct()
- self.assertEqual(path, 'django.contrib.postgres.validators.KeysValidator')
- self.assertEqual(args, ())
- self.assertEqual(kwargs, {'keys': ['a', 'b'], 'strict': True, 'messages': messages})
|