mirror of
				https://github.com/django/django.git
				synced 2025-10-25 14:46:09 +00:00 
			
		
		
		
	Fixed #29868 -- Retained database constraints on SQLite table rebuilds.
Refs #11964. Thanks Scott Stevens for testing this upcoming feature and the report.
This commit is contained in:
		
				
					committed by
					
						 Carlton Gibson
						Carlton Gibson
					
				
			
			
				
	
			
			
			
						parent
						
							f77fc56c96
						
					
				
				
					commit
					95bda03f2d
				
			| @@ -126,8 +126,7 @@ class DatabaseSchemaEditor(BaseDatabaseSchemaEditor): | ||||
|         else: | ||||
|             super().alter_field(model, old_field, new_field, strict=strict) | ||||
|  | ||||
|     def _remake_table(self, model, create_field=None, delete_field=None, alter_field=None, | ||||
|                       add_constraint=None, remove_constraint=None): | ||||
|     def _remake_table(self, model, create_field=None, delete_field=None, alter_field=None): | ||||
|         """ | ||||
|         Shortcut to transform a model from old_model into new_model | ||||
|  | ||||
| @@ -224,13 +223,6 @@ class DatabaseSchemaEditor(BaseDatabaseSchemaEditor): | ||||
|             ] | ||||
|  | ||||
|         constraints = list(model._meta.constraints) | ||||
|         if add_constraint: | ||||
|             constraints.append(add_constraint) | ||||
|         if remove_constraint: | ||||
|             constraints = [ | ||||
|                 constraint for constraint in constraints | ||||
|                 if remove_constraint.name != constraint.name | ||||
|             ] | ||||
|  | ||||
|         # Construct a new model for the new state | ||||
|         meta_contents = { | ||||
| @@ -383,7 +375,7 @@ class DatabaseSchemaEditor(BaseDatabaseSchemaEditor): | ||||
|         self.delete_model(old_field.remote_field.through) | ||||
|  | ||||
|     def add_constraint(self, model, constraint): | ||||
|         self._remake_table(model, add_constraint=constraint) | ||||
|         self._remake_table(model) | ||||
|  | ||||
|     def remove_constraint(self, model, constraint): | ||||
|         self._remake_table(model, remove_constraint=constraint) | ||||
|         self._remake_table(model) | ||||
|   | ||||
| @@ -819,6 +819,7 @@ class AddConstraint(IndexOperation): | ||||
|     def state_forwards(self, app_label, state): | ||||
|         model_state = state.models[app_label, self.model_name_lower] | ||||
|         model_state.options[self.option_name] = [*model_state.options[self.option_name], self.constraint] | ||||
|         state.reload_model(app_label, self.model_name_lower, delay=True) | ||||
|  | ||||
|     def database_forwards(self, app_label, schema_editor, from_state, to_state): | ||||
|         model = to_state.apps.get_model(app_label, self.model_name) | ||||
| @@ -851,9 +852,10 @@ class RemoveConstraint(IndexOperation): | ||||
|         model_state = state.models[app_label, self.model_name_lower] | ||||
|         constraints = model_state.options[self.option_name] | ||||
|         model_state.options[self.option_name] = [c for c in constraints if c.name != self.name] | ||||
|         state.reload_model(app_label, self.model_name_lower, delay=True) | ||||
|  | ||||
|     def database_forwards(self, app_label, schema_editor, from_state, to_state): | ||||
|         model = from_state.apps.get_model(app_label, self.model_name) | ||||
|         model = to_state.apps.get_model(app_label, self.model_name) | ||||
|         if self.allow_migrate_model(schema_editor.connection.alias, model): | ||||
|             from_model_state = from_state.models[app_label, self.model_name_lower] | ||||
|             constraint = from_model_state.get_constraint_by_name(self.name) | ||||
|   | ||||
| @@ -54,7 +54,7 @@ class OperationTestBase(MigrationTestBase): | ||||
|     def set_up_test_model( | ||||
|             self, app_label, second_model=False, third_model=False, index=False, multicol_index=False, | ||||
|             related_model=False, mti_model=False, proxy_model=False, manager_model=False, | ||||
|             unique_together=False, options=False, db_table=None, index_together=False, check_constraint=False): | ||||
|             unique_together=False, options=False, db_table=None, index_together=False, constraints=None): | ||||
|         """ | ||||
|         Creates a test model state and database table. | ||||
|         """ | ||||
| @@ -107,11 +107,12 @@ class OperationTestBase(MigrationTestBase): | ||||
|                 "Pony", | ||||
|                 models.Index(fields=["pink", "weight"], name="pony_test_idx") | ||||
|             )) | ||||
|         if check_constraint: | ||||
|             operations.append(migrations.AddConstraint( | ||||
|                 "Pony", | ||||
|                 models.CheckConstraint(check=models.Q(pink__gt=2), name="pony_test_constraint") | ||||
|             )) | ||||
|         if constraints: | ||||
|             for constraint in constraints: | ||||
|                 operations.append(migrations.AddConstraint( | ||||
|                     "Pony", | ||||
|                     constraint, | ||||
|                 )) | ||||
|         if second_model: | ||||
|             operations.append(migrations.CreateModel( | ||||
|                 "Stable", | ||||
| @@ -1788,11 +1789,24 @@ class OperationTests(OperationTestBase): | ||||
|         gt_operation.state_forwards("test_addconstraint", new_state) | ||||
|         self.assertEqual(len(new_state.models["test_addconstraint", "pony"].options["constraints"]), 1) | ||||
|         Pony = new_state.apps.get_model("test_addconstraint", "Pony") | ||||
|         self.assertEqual(len(Pony._meta.constraints), 1) | ||||
|         # Test the database alteration | ||||
|         with connection.schema_editor() as editor: | ||||
|             gt_operation.database_forwards("test_addconstraint", editor, project_state, new_state) | ||||
|         with self.assertRaises(IntegrityError), transaction.atomic(): | ||||
|             Pony.objects.create(pink=1, weight=1.0) | ||||
|         # Add another one. | ||||
|         lt_check = models.Q(pink__lt=100) | ||||
|         lt_constraint = models.CheckConstraint(check=lt_check, name="test_constraint_pony_pink_lt_100") | ||||
|         lt_operation = migrations.AddConstraint("Pony", lt_constraint) | ||||
|         lt_operation.state_forwards("test_addconstraint", new_state) | ||||
|         self.assertEqual(len(new_state.models["test_addconstraint", "pony"].options["constraints"]), 2) | ||||
|         Pony = new_state.apps.get_model("test_addconstraint", "Pony") | ||||
|         self.assertEqual(len(Pony._meta.constraints), 2) | ||||
|         with connection.schema_editor() as editor: | ||||
|             lt_operation.database_forwards("test_addconstraint", editor, project_state, new_state) | ||||
|         with self.assertRaises(IntegrityError), transaction.atomic(): | ||||
|             Pony.objects.create(pink=100, weight=1.0) | ||||
|         # Test reversal | ||||
|         with connection.schema_editor() as editor: | ||||
|             gt_operation.database_backwards("test_addconstraint", editor, new_state, project_state) | ||||
| @@ -1805,28 +1819,43 @@ class OperationTests(OperationTestBase): | ||||
|  | ||||
|     @skipUnlessDBFeature('supports_table_check_constraints') | ||||
|     def test_remove_constraint(self): | ||||
|         project_state = self.set_up_test_model("test_removeconstraint", check_constraint=True) | ||||
|         operation = migrations.RemoveConstraint("Pony", "pony_test_constraint") | ||||
|         self.assertEqual(operation.describe(), "Remove constraint pony_test_constraint from model Pony") | ||||
|         project_state = self.set_up_test_model("test_removeconstraint", constraints=[ | ||||
|             models.CheckConstraint(check=models.Q(pink__gt=2), name="test_constraint_pony_pink_gt_2"), | ||||
|             models.CheckConstraint(check=models.Q(pink__lt=100), name="test_constraint_pony_pink_lt_100"), | ||||
|         ]) | ||||
|         gt_operation = migrations.RemoveConstraint("Pony", "test_constraint_pony_pink_gt_2") | ||||
|         self.assertEqual(gt_operation.describe(), "Remove constraint test_constraint_pony_pink_gt_2 from model Pony") | ||||
|         # Test state alteration | ||||
|         new_state = project_state.clone() | ||||
|         operation.state_forwards("test_removeconstraint", new_state) | ||||
|         self.assertEqual(len(new_state.models["test_removeconstraint", "pony"].options['constraints']), 0) | ||||
|         gt_operation.state_forwards("test_removeconstraint", new_state) | ||||
|         self.assertEqual(len(new_state.models["test_removeconstraint", "pony"].options['constraints']), 1) | ||||
|         Pony = new_state.apps.get_model("test_removeconstraint", "Pony") | ||||
|         self.assertEqual(len(Pony._meta.constraints), 1) | ||||
|         # Test database alteration | ||||
|         with connection.schema_editor() as editor: | ||||
|             operation.database_forwards("test_removeconstraint", editor, project_state, new_state) | ||||
|             gt_operation.database_forwards("test_removeconstraint", editor, project_state, new_state) | ||||
|         Pony.objects.create(pink=1, weight=1.0).delete() | ||||
|         with self.assertRaises(IntegrityError), transaction.atomic(): | ||||
|             Pony.objects.create(pink=100, weight=1.0) | ||||
|         # Remove the other one. | ||||
|         lt_operation = migrations.RemoveConstraint("Pony", "test_constraint_pony_pink_lt_100") | ||||
|         lt_operation.state_forwards("test_removeconstraint", new_state) | ||||
|         self.assertEqual(len(new_state.models["test_removeconstraint", "pony"].options['constraints']), 0) | ||||
|         Pony = new_state.apps.get_model("test_removeconstraint", "Pony") | ||||
|         self.assertEqual(len(Pony._meta.constraints), 0) | ||||
|         with connection.schema_editor() as editor: | ||||
|             lt_operation.database_forwards("test_removeconstraint", editor, project_state, new_state) | ||||
|         Pony.objects.create(pink=100, weight=1.0).delete() | ||||
|         # Test reversal | ||||
|         with connection.schema_editor() as editor: | ||||
|             operation.database_backwards("test_removeconstraint", editor, new_state, project_state) | ||||
|             gt_operation.database_backwards("test_removeconstraint", editor, new_state, project_state) | ||||
|         with self.assertRaises(IntegrityError), transaction.atomic(): | ||||
|             Pony.objects.create(pink=1, weight=1.0) | ||||
|         # Test deconstruction | ||||
|         definition = operation.deconstruct() | ||||
|         definition = gt_operation.deconstruct() | ||||
|         self.assertEqual(definition[0], "RemoveConstraint") | ||||
|         self.assertEqual(definition[1], []) | ||||
|         self.assertEqual(definition[2], {'model_name': "Pony", 'name': "pony_test_constraint"}) | ||||
|         self.assertEqual(definition[2], {'model_name': "Pony", 'name': "test_constraint_pony_pink_gt_2"}) | ||||
|  | ||||
|     def test_alter_model_options(self): | ||||
|         """ | ||||
|   | ||||
		Reference in New Issue
	
	Block a user