diff options
| author | Simon Charette <charette.s@gmail.com> | 2023-08-02 20:47:49 -0400 |
|---|---|---|
| committer | Mariusz Felisiak <felisiak.mariusz@gmail.com> | 2023-08-04 10:14:03 +0200 |
| commit | 3434dbd39d373df7193ad006b970c09c1a909ea3 (patch) | |
| tree | 1c6efadb6c261581a6df74898b563742083e4e8f /tests/constraints | |
| parent | 2b582387d51c44fa928351ca55f05fc8b8d2986e (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.py | 7 | ||||
| -rw-r--r-- | tests/constraints/tests.py | 20 |
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 |
