summaryrefslogtreecommitdiff
path: root/tests/constraints
diff options
context:
space:
mode:
authorSimon Charette <charette.s@gmail.com>2023-08-02 20:47:49 -0400
committerMariusz Felisiak <felisiak.mariusz@gmail.com>2023-08-04 10:14:03 +0200
commit3434dbd39d373df7193ad006b970c09c1a909ea3 (patch)
tree1c6efadb6c261581a6df74898b563742083e4e8f /tests/constraints
parent2b582387d51c44fa928351ca55f05fc8b8d2986e (diff)
Fixed #34754 -- Fixed JSONField check constraints validation on NULL values.
The __isnull lookup of JSONField must special case Value(None, JSONField()) left-hand-side in order to be coherent with its convoluted null handling. Since psycopg>=3 offers no way to pass a NULL::jsonb the issue is resolved by optimizing IsNull(Value(None), True | False) to True | False. Regression in 5c23d9f0c32f166c81ecb6f3f01d5077a6084318. Thanks Alexandre Collet for the report.
Diffstat (limited to 'tests/constraints')
-rw-r--r--tests/constraints/models.py7
-rw-r--r--tests/constraints/tests.py20
2 files changed, 27 insertions, 0 deletions
diff --git a/tests/constraints/models.py b/tests/constraints/models.py
index ab3d4dc1e0..3ea5cf2323 100644
--- a/tests/constraints/models.py
+++ b/tests/constraints/models.py
@@ -121,3 +121,10 @@ class AbstractModel(models.Model):
class ChildModel(AbstractModel):
pass
+
+
+class JSONFieldModel(models.Model):
+ data = models.JSONField(null=True)
+
+ class Meta:
+ required_db_features = {"supports_json_field"}
diff --git a/tests/constraints/tests.py b/tests/constraints/tests.py
index 7e3d20e40c..f6571084b0 100644
--- a/tests/constraints/tests.py
+++ b/tests/constraints/tests.py
@@ -13,6 +13,7 @@ from django.utils.deprecation import RemovedInDjango60Warning
from .models import (
ChildModel,
ChildUniqueConstraintProduct,
+ JSONFieldModel,
Product,
UniqueConstraintConditionProduct,
UniqueConstraintDeferrable,
@@ -332,6 +333,25 @@ class CheckConstraintTests(TestCase):
)
constraint.validate(Product, Product())
+ @skipUnlessDBFeature("supports_json_field")
+ def test_validate_nullable_jsonfield(self):
+ is_null_constraint = models.CheckConstraint(
+ check=models.Q(data__isnull=True),
+ name="nullable_data",
+ )
+ is_not_null_constraint = models.CheckConstraint(
+ check=models.Q(data__isnull=False),
+ name="nullable_data",
+ )
+ is_null_constraint.validate(JSONFieldModel, JSONFieldModel(data=None))
+ msg = f"Constraint “{is_null_constraint.name}” is violated."
+ with self.assertRaisesMessage(ValidationError, msg):
+ is_null_constraint.validate(JSONFieldModel, JSONFieldModel(data={}))
+ msg = f"Constraint “{is_not_null_constraint.name}” is violated."
+ with self.assertRaisesMessage(ValidationError, msg):
+ is_not_null_constraint.validate(JSONFieldModel, JSONFieldModel(data=None))
+ is_not_null_constraint.validate(JSONFieldModel, JSONFieldModel(data={}))
+
class UniqueConstraintTests(TestCase):
@classmethod