mirror of
				https://github.com/django/django.git
				synced 2025-10-25 06:36:07 +00:00 
			
		
		
		
	Fixed #23745 -- Reused states as much as possible in migrations
Thanks Tim Graham and Markus Holtermann for the reviews.
This commit is contained in:
		| @@ -98,13 +98,13 @@ class MigrationExecutor(object): | |||||||
|             self.progress_callback("apply_start", migration, fake) |             self.progress_callback("apply_start", migration, fake) | ||||||
|         if not fake: |         if not fake: | ||||||
|             # Test to see if this is an already-applied initial migration |             # Test to see if this is an already-applied initial migration | ||||||
|             if self.detect_soft_applied(state, migration): |             applied, state = self.detect_soft_applied(state, migration) | ||||||
|  |             if applied: | ||||||
|                 fake = True |                 fake = True | ||||||
|             else: |             else: | ||||||
|                 # Alright, do it normally |                 # Alright, do it normally | ||||||
|                 with self.connection.schema_editor() as schema_editor: |                 with self.connection.schema_editor() as schema_editor: | ||||||
|                     project_state = self.loader.project_state((migration.app_label, migration.name), at_end=False) |                     state = migration.apply(state, schema_editor) | ||||||
|                     migration.apply(project_state, schema_editor) |  | ||||||
|         # For replacement migrations, record individual statuses |         # For replacement migrations, record individual statuses | ||||||
|         if migration.replaces: |         if migration.replaces: | ||||||
|             for app_label, name in migration.replaces: |             for app_label, name in migration.replaces: | ||||||
| @@ -124,8 +124,7 @@ class MigrationExecutor(object): | |||||||
|             self.progress_callback("unapply_start", migration, fake) |             self.progress_callback("unapply_start", migration, fake) | ||||||
|         if not fake: |         if not fake: | ||||||
|             with self.connection.schema_editor() as schema_editor: |             with self.connection.schema_editor() as schema_editor: | ||||||
|                 project_state = self.loader.project_state((migration.app_label, migration.name), at_end=False) |                 state = migration.unapply(state, schema_editor) | ||||||
|                 migration.unapply(project_state, schema_editor) |  | ||||||
|         # For replacement migrations, record individual statuses |         # For replacement migrations, record individual statuses | ||||||
|         if migration.replaces: |         if migration.replaces: | ||||||
|             for app_label, name in migration.replaces: |             for app_label, name in migration.replaces: | ||||||
| @@ -143,12 +142,15 @@ class MigrationExecutor(object): | |||||||
|         tables it would create exist. This is intended only for use |         tables it would create exist. This is intended only for use | ||||||
|         on initial migrations (as it only looks for CreateModel). |         on initial migrations (as it only looks for CreateModel). | ||||||
|         """ |         """ | ||||||
|         project_state = self.loader.project_state((migration.app_label, migration.name), at_end=True) |  | ||||||
|         apps = project_state.apps |  | ||||||
|         found_create_migration = False |  | ||||||
|         # Bail if the migration isn't the first one in its app |         # Bail if the migration isn't the first one in its app | ||||||
|         if [name for app, name in migration.dependencies if app == migration.app_label]: |         if [name for app, name in migration.dependencies if app == migration.app_label]: | ||||||
|             return False |             return False, project_state | ||||||
|  |         if project_state is None: | ||||||
|  |             after_state = self.loader.project_state((migration.app_label, migration.name), at_end=True) | ||||||
|  |         else: | ||||||
|  |             after_state = migration.mutate_state(project_state) | ||||||
|  |         apps = after_state.apps | ||||||
|  |         found_create_migration = False | ||||||
|         # Make sure all create model are done |         # Make sure all create model are done | ||||||
|         for operation in migration.operations: |         for operation in migration.operations: | ||||||
|             if isinstance(operation, migrations.CreateModel): |             if isinstance(operation, migrations.CreateModel): | ||||||
| @@ -158,8 +160,8 @@ class MigrationExecutor(object): | |||||||
|                     # main app cache, as it's not a direct dependency. |                     # main app cache, as it's not a direct dependency. | ||||||
|                     model = global_apps.get_model(model._meta.swapped) |                     model = global_apps.get_model(model._meta.swapped) | ||||||
|                 if model._meta.db_table not in self.connection.introspection.table_names(self.connection.cursor()): |                 if model._meta.db_table not in self.connection.introspection.table_names(self.connection.cursor()): | ||||||
|                     return False |                     return False, project_state | ||||||
|                 found_create_migration = True |                 found_create_migration = True | ||||||
|         # If we get this far and we found at least one CreateModel migration, |         # If we get this far and we found at least one CreateModel migration, | ||||||
|         # the migration is considered implicitly applied. |         # the migration is considered implicitly applied. | ||||||
|         return found_create_migration |         return found_create_migration, after_state | ||||||
|   | |||||||
| @@ -97,19 +97,17 @@ class Migration(object): | |||||||
|                 schema_editor.collected_sql.append("-- %s" % operation.describe()) |                 schema_editor.collected_sql.append("-- %s" % operation.describe()) | ||||||
|                 schema_editor.collected_sql.append("--") |                 schema_editor.collected_sql.append("--") | ||||||
|                 continue |                 continue | ||||||
|             # Get the state after the operation has run |             # Save the state before the operation has run | ||||||
|             new_state = project_state.clone() |             old_state = project_state.clone() | ||||||
|             operation.state_forwards(self.app_label, new_state) |             operation.state_forwards(self.app_label, project_state) | ||||||
|             # Run the operation |             # Run the operation | ||||||
|             if not schema_editor.connection.features.can_rollback_ddl and operation.atomic: |             if not schema_editor.connection.features.can_rollback_ddl and operation.atomic: | ||||||
|                 # We're forcing a transaction on a non-transactional-DDL backend |                 # We're forcing a transaction on a non-transactional-DDL backend | ||||||
|                 with atomic(schema_editor.connection.alias): |                 with atomic(schema_editor.connection.alias): | ||||||
|                     operation.database_forwards(self.app_label, schema_editor, project_state, new_state) |                     operation.database_forwards(self.app_label, schema_editor, old_state, project_state) | ||||||
|             else: |             else: | ||||||
|                 # Normal behaviour |                 # Normal behaviour | ||||||
|                 operation.database_forwards(self.app_label, schema_editor, project_state, new_state) |                 operation.database_forwards(self.app_label, schema_editor, old_state, project_state) | ||||||
|             # Switch states |  | ||||||
|             project_state = new_state |  | ||||||
|         return project_state |         return project_state | ||||||
|  |  | ||||||
|     def unapply(self, project_state, schema_editor, collect_sql=False): |     def unapply(self, project_state, schema_editor, collect_sql=False): | ||||||
| @@ -133,10 +131,9 @@ class Migration(object): | |||||||
|             # If it's irreversible, error out |             # If it's irreversible, error out | ||||||
|             if not operation.reversible: |             if not operation.reversible: | ||||||
|                 raise Migration.IrreversibleError("Operation %s in %s is not reversible" % (operation, self)) |                 raise Migration.IrreversibleError("Operation %s in %s is not reversible" % (operation, self)) | ||||||
|             new_state = project_state.clone() |             old_state = project_state.clone() | ||||||
|             operation.state_forwards(self.app_label, new_state) |             operation.state_forwards(self.app_label, project_state) | ||||||
|             to_run.append((operation, project_state, new_state)) |             to_run.append((operation, old_state, project_state)) | ||||||
|             project_state = new_state |  | ||||||
|         # Now run them in reverse |         # Now run them in reverse | ||||||
|         to_run.reverse() |         to_run.reverse() | ||||||
|         for operation, to_state, from_state in to_run: |         for operation, to_state, from_state in to_run: | ||||||
|   | |||||||
| @@ -38,6 +38,7 @@ class AddField(Operation): | |||||||
|         else: |         else: | ||||||
|             field = self.field |             field = self.field | ||||||
|         state.models[app_label, self.model_name.lower()].fields.append((self.name, field)) |         state.models[app_label, self.model_name.lower()].fields.append((self.name, field)) | ||||||
|  |         state.reload_model(app_label, self.model_name) | ||||||
|  |  | ||||||
|     def database_forwards(self, app_label, schema_editor, from_state, to_state): |     def database_forwards(self, app_label, schema_editor, from_state, to_state): | ||||||
|         to_model = to_state.apps.get_model(app_label, self.model_name) |         to_model = to_state.apps.get_model(app_label, self.model_name) | ||||||
| @@ -94,6 +95,7 @@ class RemoveField(Operation): | |||||||
|             if name != self.name: |             if name != self.name: | ||||||
|                 new_fields.append((name, instance)) |                 new_fields.append((name, instance)) | ||||||
|         state.models[app_label, self.model_name.lower()].fields = new_fields |         state.models[app_label, self.model_name.lower()].fields = new_fields | ||||||
|  |         state.reload_model(app_label, self.model_name) | ||||||
|  |  | ||||||
|     def database_forwards(self, app_label, schema_editor, from_state, to_state): |     def database_forwards(self, app_label, schema_editor, from_state, to_state): | ||||||
|         from_model = from_state.apps.get_model(app_label, self.model_name) |         from_model = from_state.apps.get_model(app_label, self.model_name) | ||||||
| @@ -150,6 +152,7 @@ class AlterField(Operation): | |||||||
|         state.models[app_label, self.model_name.lower()].fields = [ |         state.models[app_label, self.model_name.lower()].fields = [ | ||||||
|             (n, field if n == self.name else f) for n, f in state.models[app_label, self.model_name.lower()].fields |             (n, field if n == self.name else f) for n, f in state.models[app_label, self.model_name.lower()].fields | ||||||
|         ] |         ] | ||||||
|  |         state.reload_model(app_label, self.model_name) | ||||||
|  |  | ||||||
|     def database_forwards(self, app_label, schema_editor, from_state, to_state): |     def database_forwards(self, app_label, schema_editor, from_state, to_state): | ||||||
|         to_model = to_state.apps.get_model(app_label, self.model_name) |         to_model = to_state.apps.get_model(app_label, self.model_name) | ||||||
| @@ -220,6 +223,7 @@ class RenameField(Operation): | |||||||
|                     [self.new_name if n == self.old_name else n for n in together] |                     [self.new_name if n == self.old_name else n for n in together] | ||||||
|                     for together in options[option] |                     for together in options[option] | ||||||
|                 ] |                 ] | ||||||
|  |         state.reload_model(app_label, self.model_name) | ||||||
|  |  | ||||||
|     def database_forwards(self, app_label, schema_editor, from_state, to_state): |     def database_forwards(self, app_label, schema_editor, from_state, to_state): | ||||||
|         to_model = to_state.apps.get_model(app_label, self.model_name) |         to_model = to_state.apps.get_model(app_label, self.model_name) | ||||||
|   | |||||||
| @@ -39,14 +39,14 @@ class CreateModel(Operation): | |||||||
|         ) |         ) | ||||||
|  |  | ||||||
|     def state_forwards(self, app_label, state): |     def state_forwards(self, app_label, state): | ||||||
|         state.models[app_label, self.name.lower()] = ModelState( |         state.add_model(ModelState( | ||||||
|             app_label, |             app_label, | ||||||
|             self.name, |             self.name, | ||||||
|             list(self.fields), |             list(self.fields), | ||||||
|             dict(self.options), |             dict(self.options), | ||||||
|             tuple(self.bases), |             tuple(self.bases), | ||||||
|             list(self.managers), |             list(self.managers), | ||||||
|         ) |         )) | ||||||
|  |  | ||||||
|     def database_forwards(self, app_label, schema_editor, from_state, to_state): |     def database_forwards(self, app_label, schema_editor, from_state, to_state): | ||||||
|         model = to_state.apps.get_model(app_label, self.name) |         model = to_state.apps.get_model(app_label, self.name) | ||||||
| @@ -98,7 +98,7 @@ class DeleteModel(Operation): | |||||||
|         ) |         ) | ||||||
|  |  | ||||||
|     def state_forwards(self, app_label, state): |     def state_forwards(self, app_label, state): | ||||||
|         del state.models[app_label, self.name.lower()] |         state.remove_model(app_label, self.name) | ||||||
|  |  | ||||||
|     def database_forwards(self, app_label, schema_editor, from_state, to_state): |     def database_forwards(self, app_label, schema_editor, from_state, to_state): | ||||||
|         model = from_state.apps.get_model(app_label, self.name) |         model = from_state.apps.get_model(app_label, self.name) | ||||||
| @@ -141,12 +141,13 @@ class RenameModel(Operation): | |||||||
|         # Get all of the related objects we need to repoint |         # Get all of the related objects we need to repoint | ||||||
|         apps = state.apps |         apps = state.apps | ||||||
|         model = apps.get_model(app_label, self.old_name) |         model = apps.get_model(app_label, self.old_name) | ||||||
|  |         model._meta.apps = apps | ||||||
|         related_objects = model._meta.get_all_related_objects() |         related_objects = model._meta.get_all_related_objects() | ||||||
|         related_m2m_objects = model._meta.get_all_related_many_to_many_objects() |         related_m2m_objects = model._meta.get_all_related_many_to_many_objects() | ||||||
|         # Rename the model |         # Rename the model | ||||||
|         state.models[app_label, self.new_name.lower()] = state.models[app_label, self.old_name.lower()] |         state.models[app_label, self.new_name.lower()] = state.models[app_label, self.old_name.lower()] | ||||||
|         state.models[app_label, self.new_name.lower()].name = self.new_name |         state.models[app_label, self.new_name.lower()].name = self.new_name | ||||||
|         del state.models[app_label, self.old_name.lower()] |         state.remove_model(app_label, self.old_name) | ||||||
|         # Repoint the FKs and M2Ms pointing to us |         # Repoint the FKs and M2Ms pointing to us | ||||||
|         for related_object in (related_objects + related_m2m_objects): |         for related_object in (related_objects + related_m2m_objects): | ||||||
|             # Use the new related key for self referential related objects. |             # Use the new related key for self referential related objects. | ||||||
| @@ -164,7 +165,8 @@ class RenameModel(Operation): | |||||||
|                     field.rel.to = "%s.%s" % (app_label, self.new_name) |                     field.rel.to = "%s.%s" % (app_label, self.new_name) | ||||||
|                 new_fields.append((name, field)) |                 new_fields.append((name, field)) | ||||||
|             state.models[related_key].fields = new_fields |             state.models[related_key].fields = new_fields | ||||||
|         del state.apps  # FIXME: this should be replaced by a logic in state (update_model?) |             state.reload_model(*related_key) | ||||||
|  |         state.reload_model(app_label, self.new_name) | ||||||
|  |  | ||||||
|     def database_forwards(self, app_label, schema_editor, from_state, to_state): |     def database_forwards(self, app_label, schema_editor, from_state, to_state): | ||||||
|         new_model = to_state.apps.get_model(app_label, self.new_name) |         new_model = to_state.apps.get_model(app_label, self.new_name) | ||||||
| @@ -235,6 +237,7 @@ class AlterModelTable(Operation): | |||||||
|  |  | ||||||
|     def state_forwards(self, app_label, state): |     def state_forwards(self, app_label, state): | ||||||
|         state.models[app_label, self.name.lower()].options["db_table"] = self.table |         state.models[app_label, self.name.lower()].options["db_table"] = self.table | ||||||
|  |         state.reload_model(app_label, self.name) | ||||||
|  |  | ||||||
|     def database_forwards(self, app_label, schema_editor, from_state, to_state): |     def database_forwards(self, app_label, schema_editor, from_state, to_state): | ||||||
|         new_model = to_state.apps.get_model(app_label, self.name) |         new_model = to_state.apps.get_model(app_label, self.name) | ||||||
| @@ -290,6 +293,7 @@ class AlterUniqueTogether(Operation): | |||||||
|     def state_forwards(self, app_label, state): |     def state_forwards(self, app_label, state): | ||||||
|         model_state = state.models[app_label, self.name.lower()] |         model_state = state.models[app_label, self.name.lower()] | ||||||
|         model_state.options[self.option_name] = self.unique_together |         model_state.options[self.option_name] = self.unique_together | ||||||
|  |         state.reload_model(app_label, self.name) | ||||||
|  |  | ||||||
|     def database_forwards(self, app_label, schema_editor, from_state, to_state): |     def database_forwards(self, app_label, schema_editor, from_state, to_state): | ||||||
|         new_model = to_state.apps.get_model(app_label, self.name) |         new_model = to_state.apps.get_model(app_label, self.name) | ||||||
| @@ -337,6 +341,7 @@ class AlterIndexTogether(Operation): | |||||||
|     def state_forwards(self, app_label, state): |     def state_forwards(self, app_label, state): | ||||||
|         model_state = state.models[app_label, self.name.lower()] |         model_state = state.models[app_label, self.name.lower()] | ||||||
|         model_state.options[self.option_name] = self.index_together |         model_state.options[self.option_name] = self.index_together | ||||||
|  |         state.reload_model(app_label, self.name) | ||||||
|  |  | ||||||
|     def database_forwards(self, app_label, schema_editor, from_state, to_state): |     def database_forwards(self, app_label, schema_editor, from_state, to_state): | ||||||
|         new_model = to_state.apps.get_model(app_label, self.name) |         new_model = to_state.apps.get_model(app_label, self.name) | ||||||
| @@ -381,6 +386,7 @@ class AlterOrderWithRespectTo(Operation): | |||||||
|     def state_forwards(self, app_label, state): |     def state_forwards(self, app_label, state): | ||||||
|         model_state = state.models[app_label, self.name.lower()] |         model_state = state.models[app_label, self.name.lower()] | ||||||
|         model_state.options['order_with_respect_to'] = self.order_with_respect_to |         model_state.options['order_with_respect_to'] = self.order_with_respect_to | ||||||
|  |         state.reload_model(app_label, self.name) | ||||||
|  |  | ||||||
|     def database_forwards(self, app_label, schema_editor, from_state, to_state): |     def database_forwards(self, app_label, schema_editor, from_state, to_state): | ||||||
|         to_model = to_state.apps.get_model(app_label, self.name) |         to_model = to_state.apps.get_model(app_label, self.name) | ||||||
| @@ -451,6 +457,7 @@ class AlterModelOptions(Operation): | |||||||
|         for key in self.ALTER_OPTION_KEYS: |         for key in self.ALTER_OPTION_KEYS: | ||||||
|             if key not in self.options and key in model_state.options: |             if key not in self.options and key in model_state.options: | ||||||
|                 del model_state.options[key] |                 del model_state.options[key] | ||||||
|  |         state.reload_model(app_label, self.name) | ||||||
|  |  | ||||||
|     def database_forwards(self, app_label, schema_editor, from_state, to_state): |     def database_forwards(self, app_label, schema_editor, from_state, to_state): | ||||||
|         pass |         pass | ||||||
|   | |||||||
| @@ -1,4 +1,6 @@ | |||||||
| from __future__ import unicode_literals | from __future__ import unicode_literals | ||||||
|  | from collections import OrderedDict | ||||||
|  | import copy | ||||||
|  |  | ||||||
| from django.apps import AppConfig | from django.apps import AppConfig | ||||||
| from django.apps.registry import Apps, apps as global_apps | from django.apps.registry import Apps, apps as global_apps | ||||||
| @@ -30,15 +32,52 @@ class ProjectState(object): | |||||||
|         # Apps to include from main registry, usually unmigrated ones |         # Apps to include from main registry, usually unmigrated ones | ||||||
|         self.real_apps = real_apps or [] |         self.real_apps = real_apps or [] | ||||||
|  |  | ||||||
|     def add_model_state(self, model_state): |     def add_model(self, model_state): | ||||||
|         self.models[(model_state.app_label, model_state.name.lower())] = model_state |         app_label, model_name = model_state.app_label, model_state.name.lower() | ||||||
|  |         self.models[(app_label, model_name)] = model_state | ||||||
|  |         if 'apps' in self.__dict__:  # hasattr would cache the property | ||||||
|  |             self.reload_model(app_label, model_name) | ||||||
|  |  | ||||||
|  |     def remove_model(self, app_label, model_name): | ||||||
|  |         model_name = model_name.lower() | ||||||
|  |         del self.models[app_label, model_name] | ||||||
|  |         if 'apps' in self.__dict__:  # hasattr would cache the property | ||||||
|  |             self.apps.unregister_model(app_label, model_name) | ||||||
|  |  | ||||||
|  |     def reload_model(self, app_label, model_name): | ||||||
|  |         if 'apps' in self.__dict__:  # hasattr would cache the property | ||||||
|  |             # Get relations before reloading the models, as _meta.apps may change | ||||||
|  |             model_name = model_name.lower() | ||||||
|  |             try: | ||||||
|  |                 related_old = { | ||||||
|  |                     f.model for f in | ||||||
|  |                     self.apps.get_model(app_label, model_name)._meta.get_all_related_objects() | ||||||
|  |                 } | ||||||
|  |             except LookupError: | ||||||
|  |                 related_old = set() | ||||||
|  |             self._reload_one_model(app_label, model_name) | ||||||
|  |             # Reload models if there are relations | ||||||
|  |             model = self.apps.get_model(app_label, model_name) | ||||||
|  |             related_m2m = {f.rel.to for f, _ in model._meta.get_m2m_with_model()} | ||||||
|  |             for rel_model in related_old.union(related_m2m): | ||||||
|  |                 self._reload_one_model(rel_model._meta.app_label, rel_model._meta.model_name) | ||||||
|  |             if related_m2m: | ||||||
|  |                 # Re-render this model after related models have been reloaded | ||||||
|  |                 self._reload_one_model(app_label, model_name) | ||||||
|  |  | ||||||
|  |     def _reload_one_model(self, app_label, model_name): | ||||||
|  |         self.apps.unregister_model(app_label, model_name) | ||||||
|  |         self.models[app_label, model_name].render(self.apps) | ||||||
|  |  | ||||||
|     def clone(self): |     def clone(self): | ||||||
|         "Returns an exact copy of this ProjectState" |         "Returns an exact copy of this ProjectState" | ||||||
|         return ProjectState( |         new_state = ProjectState( | ||||||
|             models={k: v.clone() for k, v in self.models.items()}, |             models={k: v.clone() for k, v in self.models.items()}, | ||||||
|             real_apps=self.real_apps, |             real_apps=self.real_apps, | ||||||
|         ) |         ) | ||||||
|  |         if 'apps' in self.__dict__: | ||||||
|  |             new_state.apps = self.apps.clone() | ||||||
|  |         return new_state | ||||||
|  |  | ||||||
|     @cached_property |     @cached_property | ||||||
|     def apps(self): |     def apps(self): | ||||||
| @@ -147,6 +186,31 @@ class StateApps(Apps): | |||||||
|             else: |             else: | ||||||
|                 do_pending_lookups(model) |                 do_pending_lookups(model) | ||||||
|  |  | ||||||
|  |     def clone(self): | ||||||
|  |         """ | ||||||
|  |         Return a clone of this registry, mainly used by the migration framework. | ||||||
|  |         """ | ||||||
|  |         clone = StateApps([], {}) | ||||||
|  |         clone.all_models = copy.deepcopy(self.all_models) | ||||||
|  |         clone.app_configs = copy.deepcopy(self.app_configs) | ||||||
|  |         return clone | ||||||
|  |  | ||||||
|  |     def register_model(self, app_label, model): | ||||||
|  |         self.all_models[app_label][model._meta.model_name] = model | ||||||
|  |         if app_label not in self.app_configs: | ||||||
|  |             self.app_configs[app_label] = AppConfigStub(app_label) | ||||||
|  |             self.app_configs[app_label].models = OrderedDict() | ||||||
|  |         self.app_configs[app_label].models[model._meta.model_name] = model | ||||||
|  |         self.clear_cache() | ||||||
|  |  | ||||||
|  |     def unregister_model(self, app_label, model_name): | ||||||
|  |         try: | ||||||
|  |             del self.all_models[app_label][model_name] | ||||||
|  |             del self.app_configs[app_label].models[model_name] | ||||||
|  |         except KeyError: | ||||||
|  |             pass | ||||||
|  |         self.clear_cache() | ||||||
|  |  | ||||||
|  |  | ||||||
| class ModelState(object): | class ModelState(object): | ||||||
|     """ |     """ | ||||||
| @@ -368,7 +432,7 @@ class ModelState(object): | |||||||
|         for mgr_name, manager in self.managers: |         for mgr_name, manager in self.managers: | ||||||
|             body[mgr_name] = manager |             body[mgr_name] = manager | ||||||
|  |  | ||||||
|         # Then, make a Model object |         # Then, make a Model object (apps.register_model is called in __new__) | ||||||
|         return type( |         return type( | ||||||
|             str(self.name), |             str(self.name), | ||||||
|             bases, |             bases, | ||||||
|   | |||||||
| @@ -410,7 +410,7 @@ class AutodetectorTests(TestCase): | |||||||
|         "Shortcut to make ProjectStates from lists of predefined models" |         "Shortcut to make ProjectStates from lists of predefined models" | ||||||
|         project_state = ProjectState() |         project_state = ProjectState() | ||||||
|         for model_state in model_states: |         for model_state in model_states: | ||||||
|             project_state.add_model_state(model_state.clone()) |             project_state.add_model(model_state.clone()) | ||||||
|         return project_state |         return project_state | ||||||
|  |  | ||||||
|     def test_arrange_for_graph(self): |     def test_arrange_for_graph(self): | ||||||
|   | |||||||
| @@ -223,7 +223,7 @@ class ExecutorTests(MigrationTestBase): | |||||||
|         global_apps.get_app_config("migrations").models["author"] = migrations_apps.get_model("migrations", "author") |         global_apps.get_app_config("migrations").models["author"] = migrations_apps.get_model("migrations", "author") | ||||||
|         try: |         try: | ||||||
|             migration = executor.loader.get_migration("auth", "0001_initial") |             migration = executor.loader.get_migration("auth", "0001_initial") | ||||||
|             self.assertEqual(executor.detect_soft_applied(None, migration), True) |             self.assertEqual(executor.detect_soft_applied(None, migration)[0], True) | ||||||
|         finally: |         finally: | ||||||
|             connection.introspection.table_names = old_table_names |             connection.introspection.table_names = old_table_names | ||||||
|             del global_apps.get_app_config("migrations").models["author"] |             del global_apps.get_app_config("migrations").models["author"] | ||||||
|   | |||||||
| @@ -828,12 +828,13 @@ class OperationTests(OperationTestBase): | |||||||
|         ]) |         ]) | ||||||
|         self.assertTableExists("test_rmflmm_pony_stables") |         self.assertTableExists("test_rmflmm_pony_stables") | ||||||
|  |  | ||||||
|  |         with_field_state = project_state.clone() | ||||||
|         operations = [migrations.RemoveField("Pony", "stables")] |         operations = [migrations.RemoveField("Pony", "stables")] | ||||||
|         self.apply_operations("test_rmflmm", project_state, operations=operations) |         project_state = self.apply_operations("test_rmflmm", project_state, operations=operations) | ||||||
|         self.assertTableNotExists("test_rmflmm_pony_stables") |         self.assertTableNotExists("test_rmflmm_pony_stables") | ||||||
|  |  | ||||||
|         # And test reversal |         # And test reversal | ||||||
|         self.unapply_operations("test_rmflmm", project_state, operations=operations) |         self.unapply_operations("test_rmflmm", with_field_state, operations=operations) | ||||||
|         self.assertTableExists("test_rmflmm_pony_stables") |         self.assertTableExists("test_rmflmm_pony_stables") | ||||||
|  |  | ||||||
|     def test_remove_field_m2m_with_through(self): |     def test_remove_field_m2m_with_through(self): | ||||||
|   | |||||||
| @@ -159,7 +159,7 @@ class StateTests(TestCase): | |||||||
|         Tests rendering a ProjectState into an Apps. |         Tests rendering a ProjectState into an Apps. | ||||||
|         """ |         """ | ||||||
|         project_state = ProjectState() |         project_state = ProjectState() | ||||||
|         project_state.add_model_state(ModelState( |         project_state.add_model(ModelState( | ||||||
|             app_label="migrations", |             app_label="migrations", | ||||||
|             name="Tag", |             name="Tag", | ||||||
|             fields=[ |             fields=[ | ||||||
| @@ -168,7 +168,7 @@ class StateTests(TestCase): | |||||||
|                 ("hidden", models.BooleanField()), |                 ("hidden", models.BooleanField()), | ||||||
|             ], |             ], | ||||||
|         )) |         )) | ||||||
|         project_state.add_model_state(ModelState( |         project_state.add_model(ModelState( | ||||||
|             app_label="migrations", |             app_label="migrations", | ||||||
|             name="SubTag", |             name="SubTag", | ||||||
|             fields=[ |             fields=[ | ||||||
| @@ -187,7 +187,7 @@ class StateTests(TestCase): | |||||||
|         base_mgr = models.Manager() |         base_mgr = models.Manager() | ||||||
|         mgr1 = FoodManager('a', 'b') |         mgr1 = FoodManager('a', 'b') | ||||||
|         mgr2 = FoodManager('x', 'y', c=3, d=4) |         mgr2 = FoodManager('x', 'y', c=3, d=4) | ||||||
|         project_state.add_model_state(ModelState( |         project_state.add_model(ModelState( | ||||||
|             app_label="migrations", |             app_label="migrations", | ||||||
|             name="Food", |             name="Food", | ||||||
|             fields=[ |             fields=[ | ||||||
| @@ -324,21 +324,21 @@ class StateTests(TestCase): | |||||||
|  |  | ||||||
|         # Make a ProjectState and render it |         # Make a ProjectState and render it | ||||||
|         project_state = ProjectState() |         project_state = ProjectState() | ||||||
|         project_state.add_model_state(ModelState.from_model(A)) |         project_state.add_model(ModelState.from_model(A)) | ||||||
|         project_state.add_model_state(ModelState.from_model(B)) |         project_state.add_model(ModelState.from_model(B)) | ||||||
|         project_state.add_model_state(ModelState.from_model(C)) |         project_state.add_model(ModelState.from_model(C)) | ||||||
|         project_state.add_model_state(ModelState.from_model(D)) |         project_state.add_model(ModelState.from_model(D)) | ||||||
|         project_state.add_model_state(ModelState.from_model(E)) |         project_state.add_model(ModelState.from_model(E)) | ||||||
|         project_state.add_model_state(ModelState.from_model(F)) |         project_state.add_model(ModelState.from_model(F)) | ||||||
|         final_apps = project_state.apps |         final_apps = project_state.apps | ||||||
|         self.assertEqual(len(final_apps.get_models()), 6) |         self.assertEqual(len(final_apps.get_models()), 6) | ||||||
|  |  | ||||||
|         # Now make an invalid ProjectState and make sure it fails |         # Now make an invalid ProjectState and make sure it fails | ||||||
|         project_state = ProjectState() |         project_state = ProjectState() | ||||||
|         project_state.add_model_state(ModelState.from_model(A)) |         project_state.add_model(ModelState.from_model(A)) | ||||||
|         project_state.add_model_state(ModelState.from_model(B)) |         project_state.add_model(ModelState.from_model(B)) | ||||||
|         project_state.add_model_state(ModelState.from_model(C)) |         project_state.add_model(ModelState.from_model(C)) | ||||||
|         project_state.add_model_state(ModelState.from_model(F)) |         project_state.add_model(ModelState.from_model(F)) | ||||||
|         with self.assertRaises(InvalidBasesError): |         with self.assertRaises(InvalidBasesError): | ||||||
|             project_state.apps |             project_state.apps | ||||||
|  |  | ||||||
| @@ -358,8 +358,8 @@ class StateTests(TestCase): | |||||||
|  |  | ||||||
|         # Make a ProjectState and render it |         # Make a ProjectState and render it | ||||||
|         project_state = ProjectState() |         project_state = ProjectState() | ||||||
|         project_state.add_model_state(ModelState.from_model(A)) |         project_state.add_model(ModelState.from_model(A)) | ||||||
|         project_state.add_model_state(ModelState.from_model(B)) |         project_state.add_model(ModelState.from_model(B)) | ||||||
|         self.assertEqual(len(project_state.apps.get_models()), 2) |         self.assertEqual(len(project_state.apps.get_models()), 2) | ||||||
|  |  | ||||||
|     def test_equality(self): |     def test_equality(self): | ||||||
| @@ -369,7 +369,7 @@ class StateTests(TestCase): | |||||||
|  |  | ||||||
|         # Test two things that should be equal |         # Test two things that should be equal | ||||||
|         project_state = ProjectState() |         project_state = ProjectState() | ||||||
|         project_state.add_model_state(ModelState( |         project_state.add_model(ModelState( | ||||||
|             "migrations", |             "migrations", | ||||||
|             "Tag", |             "Tag", | ||||||
|             [ |             [ | ||||||
| @@ -388,7 +388,7 @@ class StateTests(TestCase): | |||||||
|  |  | ||||||
|         # Make a very small change (max_len 99) and see if that affects it |         # Make a very small change (max_len 99) and see if that affects it | ||||||
|         project_state = ProjectState() |         project_state = ProjectState() | ||||||
|         project_state.add_model_state(ModelState( |         project_state.add_model(ModelState( | ||||||
|             "migrations", |             "migrations", | ||||||
|             "Tag", |             "Tag", | ||||||
|             [ |             [ | ||||||
| @@ -428,20 +428,20 @@ class StateTests(TestCase): | |||||||
|  |  | ||||||
|         # Make a valid ProjectState and render it |         # Make a valid ProjectState and render it | ||||||
|         project_state = ProjectState() |         project_state = ProjectState() | ||||||
|         project_state.add_model_state(ModelState.from_model(Author)) |         project_state.add_model(ModelState.from_model(Author)) | ||||||
|         project_state.add_model_state(ModelState.from_model(Book)) |         project_state.add_model(ModelState.from_model(Book)) | ||||||
|         project_state.add_model_state(ModelState.from_model(Magazine)) |         project_state.add_model(ModelState.from_model(Magazine)) | ||||||
|         self.assertEqual(len(project_state.apps.get_models()), 3) |         self.assertEqual(len(project_state.apps.get_models()), 3) | ||||||
|  |  | ||||||
|         # now make an invalid one with a ForeignKey |         # now make an invalid one with a ForeignKey | ||||||
|         project_state = ProjectState() |         project_state = ProjectState() | ||||||
|         project_state.add_model_state(ModelState.from_model(Book)) |         project_state.add_model(ModelState.from_model(Book)) | ||||||
|         with self.assertRaises(ValueError): |         with self.assertRaises(ValueError): | ||||||
|             project_state.apps |             project_state.apps | ||||||
|  |  | ||||||
|         # and another with ManyToManyField |         # and another with ManyToManyField | ||||||
|         project_state = ProjectState() |         project_state = ProjectState() | ||||||
|         project_state.add_model_state(ModelState.from_model(Magazine)) |         project_state.add_model(ModelState.from_model(Magazine)) | ||||||
|         with self.assertRaises(ValueError): |         with self.assertRaises(ValueError): | ||||||
|             project_state.apps |             project_state.apps | ||||||
|  |  | ||||||
| @@ -461,13 +461,13 @@ class StateTests(TestCase): | |||||||
|  |  | ||||||
|         # If we just stick it into an empty state it should fail |         # If we just stick it into an empty state it should fail | ||||||
|         project_state = ProjectState() |         project_state = ProjectState() | ||||||
|         project_state.add_model_state(ModelState.from_model(TestModel)) |         project_state.add_model(ModelState.from_model(TestModel)) | ||||||
|         with self.assertRaises(ValueError): |         with self.assertRaises(ValueError): | ||||||
|             project_state.apps |             project_state.apps | ||||||
|  |  | ||||||
|         # If we include the real app it should succeed |         # If we include the real app it should succeed | ||||||
|         project_state = ProjectState(real_apps=["contenttypes"]) |         project_state = ProjectState(real_apps=["contenttypes"]) | ||||||
|         project_state.add_model_state(ModelState.from_model(TestModel)) |         project_state.add_model(ModelState.from_model(TestModel)) | ||||||
|         rendered_state = project_state.apps |         rendered_state = project_state.apps | ||||||
|         self.assertEqual( |         self.assertEqual( | ||||||
|             len([x for x in rendered_state.get_models() if x._meta.app_label == "migrations"]), |             len([x for x in rendered_state.get_models() if x._meta.app_label == "migrations"]), | ||||||
| @@ -498,8 +498,8 @@ class StateTests(TestCase): | |||||||
|  |  | ||||||
|         # Make a valid ProjectState and render it |         # Make a valid ProjectState and render it | ||||||
|         project_state = ProjectState() |         project_state = ProjectState() | ||||||
|         project_state.add_model_state(ModelState.from_model(Author)) |         project_state.add_model(ModelState.from_model(Author)) | ||||||
|         project_state.add_model_state(ModelState.from_model(Book)) |         project_state.add_model(ModelState.from_model(Book)) | ||||||
|         self.assertEqual( |         self.assertEqual( | ||||||
|             [name for name, field in project_state.models["migrations", "book"].fields], |             [name for name, field in project_state.models["migrations", "book"].fields], | ||||||
|             ["id", "author"], |             ["id", "author"], | ||||||
| @@ -534,6 +534,6 @@ class ModelStateTests(TestCase): | |||||||
|         self.assertEqual(repr(state), "<ModelState: 'app.Model'>") |         self.assertEqual(repr(state), "<ModelState: 'app.Model'>") | ||||||
|  |  | ||||||
|         project_state = ProjectState() |         project_state = ProjectState() | ||||||
|         project_state.add_model_state(state) |         project_state.add_model(state) | ||||||
|         with self.assertRaisesMessage(InvalidBasesError, "Cannot resolve bases for [<ModelState: 'app.Model'>]"): |         with self.assertRaisesMessage(InvalidBasesError, "Cannot resolve bases for [<ModelState: 'app.Model'>]"): | ||||||
|             project_state.apps |             project_state.apps | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user