mirror of
				https://github.com/django/django.git
				synced 2025-10-25 14:46:09 +00:00 
			
		
		
		
	Fixed #23474 -- Prevented migrating backwards from unapplying the wrong migrations.
This commit is contained in:
		| @@ -36,11 +36,15 @@ class MigrationExecutor(object): | |||||||
|             # If the migration is already applied, do backwards mode, |             # If the migration is already applied, do backwards mode, | ||||||
|             # otherwise do forwards mode. |             # otherwise do forwards mode. | ||||||
|             elif target in applied: |             elif target in applied: | ||||||
|                 backwards_plan = self.loader.graph.backwards_plan(target)[:-1] |                 app_label = target[0] | ||||||
|                 # We only do this if the migration is not the most recent one |                 next_migration_prefix = str(int(target[1][:4]) + 1) | ||||||
|                 # in its app - that is, another migration with the same app |                 try: | ||||||
|                 # label is in the backwards plan |                     next_migration = self.loader.get_migration_by_prefix(app_label, next_migration_prefix) | ||||||
|                 if any(node[0] == target[0] for node in backwards_plan): |                 except KeyError: | ||||||
|  |                     # If `target` is the most recent one in its app, there is nothing to do. | ||||||
|  |                     pass | ||||||
|  |                 else: | ||||||
|  |                     backwards_plan = self.loader.graph.backwards_plan(next_migration) | ||||||
|                     for migration in backwards_plan: |                     for migration in backwards_plan: | ||||||
|                         if migration in applied: |                         if migration in applied: | ||||||
|                             plan.append((self.loader.graph.nodes[migration], True)) |                             plan.append((self.loader.graph.nodes[migration], True)) | ||||||
|   | |||||||
| @@ -46,3 +46,6 @@ Bugfixes | |||||||
|  |  | ||||||
| * Allowed migrations to work with ``app_label``\s that have the same last | * Allowed migrations to work with ``app_label``\s that have the same last | ||||||
|   part (e.g. ``django.contrib.auth`` and ``vendor.auth``) (:ticket:`23483`). |   part (e.g. ``django.contrib.auth`` and ``vendor.auth``) (:ticket:`23483`). | ||||||
|  |  | ||||||
|  | * Fixed bug in migrations that could cause unexpected data loss when executing | ||||||
|  |   a backwards or no-op migration (:ticket:`23474`). | ||||||
|   | |||||||
| @@ -231,3 +231,41 @@ class ExecutorTests(MigrationTestBase): | |||||||
|         executor.migrate([("migrations", None)]) |         executor.migrate([("migrations", None)]) | ||||||
|         self.assertTableNotExists("migrations_author") |         self.assertTableNotExists("migrations_author") | ||||||
|         self.assertTableNotExists("migrations_tribble") |         self.assertTableNotExists("migrations_tribble") | ||||||
|  |  | ||||||
|  |     @override_settings( | ||||||
|  |         MIGRATION_MODULES={ | ||||||
|  |           "migrations": "migrations.test_migrations_backwards_deps_1", | ||||||
|  |           "migrations2": "migrations2.test_migrations_backwards_deps_2", | ||||||
|  |         }, | ||||||
|  |     ) | ||||||
|  |     def test_backwards_deps(self): | ||||||
|  |         """ | ||||||
|  |         #23474 - Migrating backwards shouldn't cause the wrong migrations to be | ||||||
|  |         unapplied. | ||||||
|  |  | ||||||
|  |         Migration dependencies (x -> y === y depends on x): | ||||||
|  |         m.0001 -+-> m.0002 | ||||||
|  |                 +-> m2.0001 | ||||||
|  |  | ||||||
|  |         1) Migrate m2 to 0001, causing { m.0001, m2.0002 } to be applied. | ||||||
|  |         2) Migrate m to 0001. m.0001 has already been applied, so this should | ||||||
|  |            be a noop. | ||||||
|  |         """ | ||||||
|  |         executor = MigrationExecutor(connection) | ||||||
|  |         executor.migrate([("migrations2", "0001_initial")]) | ||||||
|  |         try: | ||||||
|  |             self.assertTableExists("migrations2_example") | ||||||
|  |             # Rebuild the graph to reflect the new DB state | ||||||
|  |             executor.loader.build_graph() | ||||||
|  |             self.assertEqual( | ||||||
|  |                 executor.migration_plan([("migrations", "0001_initial")]), | ||||||
|  |                 [], | ||||||
|  |             ) | ||||||
|  |             executor.migrate([("migrations", "0001_initial")]) | ||||||
|  |             self.assertTableExists("migrations2_example") | ||||||
|  |         finally: | ||||||
|  |             # And migrate back to clean up the database | ||||||
|  |             executor.loader.build_graph() | ||||||
|  |             executor.migrate([("migrations", None)]) | ||||||
|  |             self.assertTableNotExists("migrations_author") | ||||||
|  |             self.assertTableNotExists("migrations_tribble") | ||||||
|   | |||||||
| @@ -0,0 +1,8 @@ | |||||||
|  | # -*- coding: utf-8 -*- | ||||||
|  | from __future__ import unicode_literals | ||||||
|  |  | ||||||
|  | from django.db import migrations, models | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Migration(migrations.Migration): | ||||||
|  |     operations = [] | ||||||
| @@ -0,0 +1,9 @@ | |||||||
|  | # -*- coding: utf-8 -*- | ||||||
|  | from __future__ import unicode_literals | ||||||
|  |  | ||||||
|  | from django.db import migrations, models | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Migration(migrations.Migration): | ||||||
|  |     dependencies = [('migrations', '0001_initial')] | ||||||
|  |     operations = [] | ||||||
| @@ -0,0 +1,16 @@ | |||||||
|  | # -*- coding: utf-8 -*- | ||||||
|  | from __future__ import unicode_literals | ||||||
|  |  | ||||||
|  | from django.db import migrations, models | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Migration(migrations.Migration): | ||||||
|  |     dependencies = [('migrations', '0001_initial')] | ||||||
|  |     operations = [ | ||||||
|  |         migrations.CreateModel( | ||||||
|  |             "Example", | ||||||
|  |             [ | ||||||
|  |                 ("id", models.AutoField(primary_key=True)), | ||||||
|  |             ], | ||||||
|  |         ), | ||||||
|  |     ] | ||||||
		Reference in New Issue
	
	Block a user