[2.2.x] Refs #30172 -- Prevented removing a field's check or unique constraint from removing Meta constraints.
Backport of 4bb859e246
from master.
This commit is contained in:
parent
40b0a58f5f
commit
3dd5e71752
|
@ -281,6 +281,10 @@ class BaseDatabaseFeatures:
|
||||||
supports_partial_indexes = True
|
supports_partial_indexes = True
|
||||||
supports_functions_in_partial_indexes = True
|
supports_functions_in_partial_indexes = True
|
||||||
|
|
||||||
|
# Does the database allow more than one constraint or index on the same
|
||||||
|
# field(s)?
|
||||||
|
allows_multiple_constraints_on_same_fields = True
|
||||||
|
|
||||||
def __init__(self, connection):
|
def __init__(self, connection):
|
||||||
self.connection = connection
|
self.connection = connection
|
||||||
|
|
||||||
|
|
|
@ -548,7 +548,11 @@ class BaseDatabaseSchemaEditor:
|
||||||
# Has unique been removed?
|
# Has unique been removed?
|
||||||
if old_field.unique and (not new_field.unique or self._field_became_primary_key(old_field, new_field)):
|
if old_field.unique and (not new_field.unique or self._field_became_primary_key(old_field, new_field)):
|
||||||
# Find the unique constraint for this field
|
# Find the unique constraint for this field
|
||||||
constraint_names = self._constraint_names(model, [old_field.column], unique=True, primary_key=False)
|
meta_constraint_names = {constraint.name for constraint in model._meta.constraints}
|
||||||
|
constraint_names = self._constraint_names(
|
||||||
|
model, [old_field.column], unique=True, primary_key=False,
|
||||||
|
exclude=meta_constraint_names,
|
||||||
|
)
|
||||||
if strict and len(constraint_names) != 1:
|
if strict and len(constraint_names) != 1:
|
||||||
raise ValueError("Found wrong number (%s) of unique constraints for %s.%s" % (
|
raise ValueError("Found wrong number (%s) of unique constraints for %s.%s" % (
|
||||||
len(constraint_names),
|
len(constraint_names),
|
||||||
|
@ -598,7 +602,11 @@ class BaseDatabaseSchemaEditor:
|
||||||
self.execute(self._delete_index_sql(model, index_name))
|
self.execute(self._delete_index_sql(model, index_name))
|
||||||
# Change check constraints?
|
# Change check constraints?
|
||||||
if old_db_params['check'] != new_db_params['check'] and old_db_params['check']:
|
if old_db_params['check'] != new_db_params['check'] and old_db_params['check']:
|
||||||
constraint_names = self._constraint_names(model, [old_field.column], check=True)
|
meta_constraint_names = {constraint.name for constraint in model._meta.constraints}
|
||||||
|
constraint_names = self._constraint_names(
|
||||||
|
model, [old_field.column], check=True,
|
||||||
|
exclude=meta_constraint_names,
|
||||||
|
)
|
||||||
if strict and len(constraint_names) != 1:
|
if strict and len(constraint_names) != 1:
|
||||||
raise ValueError("Found wrong number (%s) of check constraints for %s.%s" % (
|
raise ValueError("Found wrong number (%s) of check constraints for %s.%s" % (
|
||||||
len(constraint_names),
|
len(constraint_names),
|
||||||
|
@ -1089,7 +1097,7 @@ class BaseDatabaseSchemaEditor:
|
||||||
|
|
||||||
def _constraint_names(self, model, column_names=None, unique=None,
|
def _constraint_names(self, model, column_names=None, unique=None,
|
||||||
primary_key=None, index=None, foreign_key=None,
|
primary_key=None, index=None, foreign_key=None,
|
||||||
check=None, type_=None):
|
check=None, type_=None, exclude=None):
|
||||||
"""Return all constraint names matching the columns and conditions."""
|
"""Return all constraint names matching the columns and conditions."""
|
||||||
if column_names is not None:
|
if column_names is not None:
|
||||||
column_names = [
|
column_names = [
|
||||||
|
@ -1113,6 +1121,7 @@ class BaseDatabaseSchemaEditor:
|
||||||
continue
|
continue
|
||||||
if type_ is not None and infodict['type'] != type_:
|
if type_ is not None and infodict['type'] != type_:
|
||||||
continue
|
continue
|
||||||
|
if not exclude or name not in exclude:
|
||||||
result.append(name)
|
result.append(name)
|
||||||
return result
|
return result
|
||||||
|
|
||||||
|
|
|
@ -56,6 +56,7 @@ class DatabaseFeatures(BaseDatabaseFeatures):
|
||||||
supports_ignore_conflicts = False
|
supports_ignore_conflicts = False
|
||||||
max_query_params = 2**16 - 1
|
max_query_params = 2**16 - 1
|
||||||
supports_partial_indexes = False
|
supports_partial_indexes = False
|
||||||
|
allows_multiple_constraints_on_same_fields = False
|
||||||
|
|
||||||
@cached_property
|
@cached_property
|
||||||
def has_fetch_offset_support(self):
|
def has_fetch_offset_support(self):
|
||||||
|
|
|
@ -55,6 +55,13 @@ class AuthorWithIndexedName(models.Model):
|
||||||
apps = new_apps
|
apps = new_apps
|
||||||
|
|
||||||
|
|
||||||
|
class AuthorWithUniqueName(models.Model):
|
||||||
|
name = models.CharField(max_length=255, unique=True)
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
apps = new_apps
|
||||||
|
|
||||||
|
|
||||||
class Book(models.Model):
|
class Book(models.Model):
|
||||||
author = models.ForeignKey(Author, models.CASCADE)
|
author = models.ForeignKey(Author, models.CASCADE)
|
||||||
title = models.CharField(max_length=100, db_index=True)
|
title = models.CharField(max_length=100, db_index=True)
|
||||||
|
|
|
@ -7,7 +7,8 @@ from unittest import mock
|
||||||
from django.db import (
|
from django.db import (
|
||||||
DatabaseError, IntegrityError, OperationalError, connection,
|
DatabaseError, IntegrityError, OperationalError, connection,
|
||||||
)
|
)
|
||||||
from django.db.models import Model
|
from django.db.models import Model, Q
|
||||||
|
from django.db.models.constraints import CheckConstraint, UniqueConstraint
|
||||||
from django.db.models.deletion import CASCADE, PROTECT
|
from django.db.models.deletion import CASCADE, PROTECT
|
||||||
from django.db.models.fields import (
|
from django.db.models.fields import (
|
||||||
AutoField, BigAutoField, BigIntegerField, BinaryField, BooleanField,
|
AutoField, BigAutoField, BigIntegerField, BinaryField, BooleanField,
|
||||||
|
@ -31,9 +32,10 @@ from .fields import (
|
||||||
from .models import (
|
from .models import (
|
||||||
Author, AuthorCharFieldWithIndex, AuthorTextFieldWithIndex,
|
Author, AuthorCharFieldWithIndex, AuthorTextFieldWithIndex,
|
||||||
AuthorWithDefaultHeight, AuthorWithEvenLongerName, AuthorWithIndexedName,
|
AuthorWithDefaultHeight, AuthorWithEvenLongerName, AuthorWithIndexedName,
|
||||||
Book, BookForeignObj, BookWeak, BookWithLongName, BookWithO2O,
|
AuthorWithUniqueName, Book, BookForeignObj, BookWeak, BookWithLongName,
|
||||||
BookWithoutAuthor, BookWithSlug, IntegerPK, Node, Note, NoteRename, Tag,
|
BookWithO2O, BookWithoutAuthor, BookWithSlug, IntegerPK, Node, Note,
|
||||||
TagIndexed, TagM2MTest, TagUniqueRename, Thing, UniqueTest, new_apps,
|
NoteRename, Tag, TagIndexed, TagM2MTest, TagUniqueRename, Thing,
|
||||||
|
UniqueTest, new_apps,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@ -1524,6 +1526,53 @@ class SchemaTests(TransactionTestCase):
|
||||||
if not any(details['columns'] == ['height'] and details['check'] for details in constraints.values()):
|
if not any(details['columns'] == ['height'] and details['check'] for details in constraints.values()):
|
||||||
self.fail("No check constraint for height found")
|
self.fail("No check constraint for height found")
|
||||||
|
|
||||||
|
@skipUnlessDBFeature('supports_column_check_constraints')
|
||||||
|
def test_remove_field_check_does_not_remove_meta_constraints(self):
|
||||||
|
with connection.schema_editor() as editor:
|
||||||
|
editor.create_model(Author)
|
||||||
|
# Add the custom check constraint
|
||||||
|
constraint = CheckConstraint(check=Q(height__gte=0), name='author_height_gte_0_check')
|
||||||
|
custom_constraint_name = constraint.name
|
||||||
|
Author._meta.constraints = [constraint]
|
||||||
|
with connection.schema_editor() as editor:
|
||||||
|
editor.add_constraint(Author, constraint)
|
||||||
|
# Ensure the constraints exist
|
||||||
|
constraints = self.get_constraints(Author._meta.db_table)
|
||||||
|
self.assertIn(custom_constraint_name, constraints)
|
||||||
|
other_constraints = [
|
||||||
|
name for name, details in constraints.items()
|
||||||
|
if details['columns'] == ['height'] and details['check'] and name != custom_constraint_name
|
||||||
|
]
|
||||||
|
self.assertEqual(len(other_constraints), 1)
|
||||||
|
# Alter the column to remove field check
|
||||||
|
old_field = Author._meta.get_field('height')
|
||||||
|
new_field = IntegerField(null=True, blank=True)
|
||||||
|
new_field.set_attributes_from_name('height')
|
||||||
|
with connection.schema_editor() as editor:
|
||||||
|
editor.alter_field(Author, old_field, new_field, strict=True)
|
||||||
|
constraints = self.get_constraints(Author._meta.db_table)
|
||||||
|
self.assertIn(custom_constraint_name, constraints)
|
||||||
|
other_constraints = [
|
||||||
|
name for name, details in constraints.items()
|
||||||
|
if details['columns'] == ['height'] and details['check'] and name != custom_constraint_name
|
||||||
|
]
|
||||||
|
self.assertEqual(len(other_constraints), 0)
|
||||||
|
# Alter the column to re-add field check
|
||||||
|
new_field2 = Author._meta.get_field('height')
|
||||||
|
with connection.schema_editor() as editor:
|
||||||
|
editor.alter_field(Author, new_field, new_field2, strict=True)
|
||||||
|
constraints = self.get_constraints(Author._meta.db_table)
|
||||||
|
self.assertIn(custom_constraint_name, constraints)
|
||||||
|
other_constraints = [
|
||||||
|
name for name, details in constraints.items()
|
||||||
|
if details['columns'] == ['height'] and details['check'] and name != custom_constraint_name
|
||||||
|
]
|
||||||
|
self.assertEqual(len(other_constraints), 1)
|
||||||
|
# Drop the check constraint
|
||||||
|
with connection.schema_editor() as editor:
|
||||||
|
Author._meta.constraints = []
|
||||||
|
editor.remove_constraint(Author, constraint)
|
||||||
|
|
||||||
def test_unique(self):
|
def test_unique(self):
|
||||||
"""
|
"""
|
||||||
Tests removing and adding unique constraints to a single column.
|
Tests removing and adding unique constraints to a single column.
|
||||||
|
@ -1650,6 +1699,53 @@ class SchemaTests(TransactionTestCase):
|
||||||
with self.assertRaises(IntegrityError):
|
with self.assertRaises(IntegrityError):
|
||||||
Tag.objects.create(title='bar', slug='foo')
|
Tag.objects.create(title='bar', slug='foo')
|
||||||
|
|
||||||
|
@skipUnlessDBFeature('allows_multiple_constraints_on_same_fields')
|
||||||
|
def test_remove_field_unique_does_not_remove_meta_constraints(self):
|
||||||
|
with connection.schema_editor() as editor:
|
||||||
|
editor.create_model(AuthorWithUniqueName)
|
||||||
|
# Add the custom unique constraint
|
||||||
|
constraint = UniqueConstraint(fields=['name'], name='author_name_uniq')
|
||||||
|
custom_constraint_name = constraint.name
|
||||||
|
AuthorWithUniqueName._meta.constraints = [constraint]
|
||||||
|
with connection.schema_editor() as editor:
|
||||||
|
editor.add_constraint(AuthorWithUniqueName, constraint)
|
||||||
|
# Ensure the constraints exist
|
||||||
|
constraints = self.get_constraints(AuthorWithUniqueName._meta.db_table)
|
||||||
|
self.assertIn(custom_constraint_name, constraints)
|
||||||
|
other_constraints = [
|
||||||
|
name for name, details in constraints.items()
|
||||||
|
if details['columns'] == ['name'] and details['unique'] and name != custom_constraint_name
|
||||||
|
]
|
||||||
|
self.assertEqual(len(other_constraints), 1)
|
||||||
|
# Alter the column to remove field uniqueness
|
||||||
|
old_field = AuthorWithUniqueName._meta.get_field('name')
|
||||||
|
new_field = CharField(max_length=255)
|
||||||
|
new_field.set_attributes_from_name('name')
|
||||||
|
with connection.schema_editor() as editor:
|
||||||
|
editor.alter_field(AuthorWithUniqueName, old_field, new_field, strict=True)
|
||||||
|
constraints = self.get_constraints(AuthorWithUniqueName._meta.db_table)
|
||||||
|
self.assertIn(custom_constraint_name, constraints)
|
||||||
|
other_constraints = [
|
||||||
|
name for name, details in constraints.items()
|
||||||
|
if details['columns'] == ['name'] and details['unique'] and name != custom_constraint_name
|
||||||
|
]
|
||||||
|
self.assertEqual(len(other_constraints), 0)
|
||||||
|
# Alter the column to re-add field uniqueness
|
||||||
|
new_field2 = AuthorWithUniqueName._meta.get_field('name')
|
||||||
|
with connection.schema_editor() as editor:
|
||||||
|
editor.alter_field(AuthorWithUniqueName, new_field, new_field2, strict=True)
|
||||||
|
constraints = self.get_constraints(AuthorWithUniqueName._meta.db_table)
|
||||||
|
self.assertIn(custom_constraint_name, constraints)
|
||||||
|
other_constraints = [
|
||||||
|
name for name, details in constraints.items()
|
||||||
|
if details['columns'] == ['name'] and details['unique'] and name != custom_constraint_name
|
||||||
|
]
|
||||||
|
self.assertEqual(len(other_constraints), 1)
|
||||||
|
# Drop the unique constraint
|
||||||
|
with connection.schema_editor() as editor:
|
||||||
|
AuthorWithUniqueName._meta.constraints = []
|
||||||
|
editor.remove_constraint(AuthorWithUniqueName, constraint)
|
||||||
|
|
||||||
def test_unique_together(self):
|
def test_unique_together(self):
|
||||||
"""
|
"""
|
||||||
Tests removing and adding unique_together constraints on a model.
|
Tests removing and adding unique_together constraints on a model.
|
||||||
|
|
Loading…
Reference in New Issue