summaryrefslogtreecommitdiff
path: root/tests/postgres_tests
diff options
context:
space:
mode:
authorMarc Tamlyn <marc.tamlyn@gmail.com>2015-05-30 22:13:58 +0100
committerMarc Tamlyn <marc.tamlyn@gmail.com>2015-05-30 23:10:30 +0100
commit33ea472f6924480eb8708bc6d0ac90b9f742ba68 (patch)
tree2ca1f92c94638db440e3e23a30bd9a67ca34f8f3 /tests/postgres_tests
parent74fe4428e51030cbed768083489f0497f4453c17 (diff)
Fixed #24604 -- Added JSONField to contrib.postgres.
Diffstat (limited to 'tests/postgres_tests')
-rw-r--r--tests/postgres_tests/fields.py3
-rw-r--r--tests/postgres_tests/migrations/0002_create_test_models.py15
-rw-r--r--tests/postgres_tests/models.py14
-rw-r--r--tests/postgres_tests/test_json.py258
4 files changed, 287 insertions, 3 deletions
diff --git a/tests/postgres_tests/fields.py b/tests/postgres_tests/fields.py
index e21ca1bfcb..0926175e1b 100644
--- a/tests/postgres_tests/fields.py
+++ b/tests/postgres_tests/fields.py
@@ -7,7 +7,7 @@ from django.db import models
try:
from django.contrib.postgres.fields import (
ArrayField, BigIntegerRangeField, DateRangeField, DateTimeRangeField,
- FloatRangeField, HStoreField, IntegerRangeField,
+ FloatRangeField, HStoreField, IntegerRangeField, JSONField,
)
except ImportError:
class DummyArrayField(models.Field):
@@ -29,3 +29,4 @@ except ImportError:
FloatRangeField = models.Field
HStoreField = models.Field
IntegerRangeField = models.Field
+ JSONField = models.Field
diff --git a/tests/postgres_tests/migrations/0002_create_test_models.py b/tests/postgres_tests/migrations/0002_create_test_models.py
index 106818a7c6..4eb2154f02 100644
--- a/tests/postgres_tests/migrations/0002_create_test_models.py
+++ b/tests/postgres_tests/migrations/0002_create_test_models.py
@@ -150,6 +150,19 @@ class Migration(migrations.Migration):
),
]
+ pg_94_operations = [
+ migrations.CreateModel(
+ name='JSONModel',
+ fields=[
+ ('id', models.AutoField(verbose_name='ID', serialize=False, auto_created=True, primary_key=True)),
+ ('field', JSONField(null=True, blank=True)),
+ ],
+ options={
+ },
+ bases=(models.Model,),
+ ),
+ ]
+
def apply(self, project_state, schema_editor, collect_sql=False):
try:
PG_VERSION = schema_editor.connection.pg_version
@@ -158,4 +171,6 @@ class Migration(migrations.Migration):
else:
if PG_VERSION >= 90200:
self.operations = self.operations + self.pg_92_operations
+ if PG_VERSION >= 90400:
+ self.operations = self.operations + self.pg_94_operations
return super(Migration, self).apply(project_state, schema_editor, collect_sql)
diff --git a/tests/postgres_tests/models.py b/tests/postgres_tests/models.py
index aafd529443..329a91c951 100644
--- a/tests/postgres_tests/models.py
+++ b/tests/postgres_tests/models.py
@@ -2,7 +2,7 @@ from django.db import connection, models
from .fields import (
ArrayField, BigIntegerRangeField, DateRangeField, DateTimeRangeField,
- FloatRangeField, HStoreField, IntegerRangeField,
+ FloatRangeField, HStoreField, IntegerRangeField, JSONField,
)
@@ -52,7 +52,7 @@ class TextFieldModel(models.Model):
field = models.TextField()
-# Only create this model for databases which support it
+# Only create this model for postgres >= 9.2
if connection.vendor == 'postgresql' and connection.pg_version >= 90200:
class RangesModel(PostgreSQLModel):
ints = IntegerRangeField(blank=True, null=True)
@@ -66,6 +66,16 @@ else:
pass
+# Only create this model for postgres >= 9.4
+if connection.vendor == 'postgresql' and connection.pg_version >= 90400:
+ class JSONModel(models.Model):
+ field = JSONField(blank=True, null=True)
+else:
+ # create an object with this name so we don't have failing imports
+ class JSONModel(object):
+ pass
+
+
class ArrayFieldSubclass(ArrayField):
def __init__(self, *args, **kwargs):
super(ArrayFieldSubclass, self).__init__(models.IntegerField())
diff --git a/tests/postgres_tests/test_json.py b/tests/postgres_tests/test_json.py
new file mode 100644
index 0000000000..b59abe7898
--- /dev/null
+++ b/tests/postgres_tests/test_json.py
@@ -0,0 +1,258 @@
+import datetime
+import unittest
+
+from django.core import exceptions, serializers
+from django.db import connection
+from django.test import TestCase
+
+from . import PostgresSQLTestCase
+from .models import JSONModel
+
+try:
+ from django.contrib.postgres import forms
+ from django.contrib.postgres.fields import JSONField
+except ImportError:
+ pass
+
+
+def skipUnlessPG94(test):
+ try:
+ PG_VERSION = connection.pg_version
+ except AttributeError:
+ PG_VERSION = 0
+ if PG_VERSION < 90400:
+ return unittest.skip('PostgreSQL >= 9.4 required')(test)
+ return test
+
+
+@skipUnlessPG94
+class TestSaveLoad(TestCase):
+ def test_null(self):
+ instance = JSONModel()
+ instance.save()
+ loaded = JSONModel.objects.get()
+ self.assertEqual(loaded.field, None)
+
+ def test_empty_object(self):
+ instance = JSONModel(field={})
+ instance.save()
+ loaded = JSONModel.objects.get()
+ self.assertEqual(loaded.field, {})
+
+ def test_empty_list(self):
+ instance = JSONModel(field=[])
+ instance.save()
+ loaded = JSONModel.objects.get()
+ self.assertEqual(loaded.field, [])
+
+ def test_boolean(self):
+ instance = JSONModel(field=True)
+ instance.save()
+ loaded = JSONModel.objects.get()
+ self.assertEqual(loaded.field, True)
+
+ def test_string(self):
+ instance = JSONModel(field='why?')
+ instance.save()
+ loaded = JSONModel.objects.get()
+ self.assertEqual(loaded.field, 'why?')
+
+ def test_number(self):
+ instance = JSONModel(field=1)
+ instance.save()
+ loaded = JSONModel.objects.get()
+ self.assertEqual(loaded.field, 1)
+
+ def test_realistic_object(self):
+ obj = {
+ 'a': 'b',
+ 'c': 1,
+ 'd': ['e', {'f': 'g'}],
+ 'h': True,
+ 'i': False,
+ 'j': None,
+ }
+ instance = JSONModel(field=obj)
+ instance.save()
+ loaded = JSONModel.objects.get()
+ self.assertEqual(loaded.field, obj)
+
+
+@skipUnlessPG94
+class TestQuerying(TestCase):
+ @classmethod
+ def setUpTestData(cls):
+ cls.objs = [
+ JSONModel.objects.create(field=None),
+ JSONModel.objects.create(field=True),
+ JSONModel.objects.create(field=False),
+ JSONModel.objects.create(field='yes'),
+ JSONModel.objects.create(field=7),
+ JSONModel.objects.create(field=[]),
+ JSONModel.objects.create(field={}),
+ JSONModel.objects.create(field={
+ 'a': 'b',
+ 'c': 1,
+ }),
+ JSONModel.objects.create(field={
+ 'a': 'b',
+ 'c': 1,
+ 'd': ['e', {'f': 'g'}],
+ 'h': True,
+ 'i': False,
+ 'j': None,
+ 'k': {'l': 'm'},
+ }),
+ JSONModel.objects.create(field=[1, [2]]),
+ JSONModel.objects.create(field={
+ 'k': True,
+ 'l': False,
+ }),
+ ]
+
+ def test_exact(self):
+ self.assertSequenceEqual(
+ JSONModel.objects.filter(field__exact={}),
+ [self.objs[6]]
+ )
+
+ def test_exact_complex(self):
+ self.assertSequenceEqual(
+ JSONModel.objects.filter(field__exact={'a': 'b', 'c': 1}),
+ [self.objs[7]]
+ )
+
+ def test_isnull(self):
+ self.assertSequenceEqual(
+ JSONModel.objects.filter(field__isnull=True),
+ [self.objs[0]]
+ )
+
+ def test_contains(self):
+ self.assertSequenceEqual(
+ JSONModel.objects.filter(field__contains={'a': 'b'}),
+ [self.objs[7], self.objs[8]]
+ )
+
+ def test_contained_by(self):
+ self.assertSequenceEqual(
+ JSONModel.objects.filter(field__contained_by={'a': 'b', 'c': 1, 'h': True}),
+ [self.objs[6], self.objs[7]]
+ )
+
+ def test_has_key(self):
+ self.assertSequenceEqual(
+ JSONModel.objects.filter(field__has_key='a'),
+ [self.objs[7], self.objs[8]]
+ )
+
+ def test_has_keys(self):
+ self.assertSequenceEqual(
+ JSONModel.objects.filter(field__has_keys=['a', 'c', 'h']),
+ [self.objs[8]]
+ )
+
+ def test_has_any_keys(self):
+ self.assertSequenceEqual(
+ JSONModel.objects.filter(field__has_any_keys=['c', 'l']),
+ [self.objs[7], self.objs[8], self.objs[10]]
+ )
+
+ def test_shallow_list_lookup(self):
+ self.assertSequenceEqual(
+ JSONModel.objects.filter(field__0=1),
+ [self.objs[9]]
+ )
+
+ def test_shallow_obj_lookup(self):
+ self.assertSequenceEqual(
+ JSONModel.objects.filter(field__a='b'),
+ [self.objs[7], self.objs[8]]
+ )
+
+ def test_deep_lookup_objs(self):
+ self.assertSequenceEqual(
+ JSONModel.objects.filter(field__k__l='m'),
+ [self.objs[8]]
+ )
+
+ def test_shallow_lookup_obj_target(self):
+ self.assertSequenceEqual(
+ JSONModel.objects.filter(field__k={'l': 'm'}),
+ [self.objs[8]]
+ )
+
+ def test_deep_lookup_array(self):
+ self.assertSequenceEqual(
+ JSONModel.objects.filter(field__1__0=2),
+ [self.objs[9]]
+ )
+
+ def test_deep_lookup_mixed(self):
+ self.assertSequenceEqual(
+ JSONModel.objects.filter(field__d__1__f='g'),
+ [self.objs[8]]
+ )
+
+ def test_deep_lookup_transform(self):
+ self.assertSequenceEqual(
+ JSONModel.objects.filter(field__c__gt=1),
+ []
+ )
+ self.assertSequenceEqual(
+ JSONModel.objects.filter(field__c__lt=5),
+ [self.objs[7], self.objs[8]]
+ )
+
+
+@skipUnlessPG94
+class TestSerialization(TestCase):
+ test_data = '[{"fields": {"field": {"a": "b"}}, "model": "postgres_tests.jsonmodel", "pk": null}]'
+
+ def test_dumping(self):
+ instance = JSONModel(field={'a': 'b'})
+ data = serializers.serialize('json', [instance])
+ self.assertJSONEqual(data, self.test_data)
+
+ def test_loading(self):
+ instance = list(serializers.deserialize('json', self.test_data))[0].object
+ self.assertEqual(instance.field, {'a': 'b'})
+
+
+class TestValidation(PostgresSQLTestCase):
+
+ def test_not_serializable(self):
+ field = JSONField()
+ with self.assertRaises(exceptions.ValidationError) as cm:
+ field.clean(datetime.timedelta(days=1), None)
+ self.assertEqual(cm.exception.code, 'invalid')
+ self.assertEqual(cm.exception.message % cm.exception.params, "Value must be valid JSON.")
+
+
+class TestFormField(PostgresSQLTestCase):
+
+ def test_valid(self):
+ field = forms.JSONField()
+ value = field.clean('{"a": "b"}')
+ self.assertEqual(value, {'a': 'b'})
+
+ def test_valid_empty(self):
+ field = forms.JSONField(required=False)
+ value = field.clean('')
+ self.assertEqual(value, None)
+
+ def test_invalid(self):
+ field = forms.JSONField()
+ with self.assertRaises(exceptions.ValidationError) as cm:
+ field.clean('{some badly formed: json}')
+ self.assertEqual(cm.exception.messages[0], "'{some badly formed: json}' value must be valid JSON.")
+
+ def test_formfield(self):
+ model_field = JSONField()
+ form_field = model_field.formfield()
+ self.assertIsInstance(form_field, forms.JSONField)
+
+ def test_prepare_value(self):
+ field = forms.JSONField()
+ self.assertEqual(field.prepare_value({'a': 'b'}), '{"a": "b"}')
+ self.assertEqual(field.prepare_value(None), 'null')