mirror of
				https://github.com/django/django.git
				synced 2025-10-25 22:56:12 +00:00 
			
		
		
		
	[1.7.x] Fixed #23415 -- Added fields for unmanaged and proxy model migrations.
Thanks sky-chen for the report.
Backport of 215aa4f53b from master
			
			
This commit is contained in:
		
				
					committed by
					
						 Tim Graham
						Tim Graham
					
				
			
			
				
	
			
			
			
						parent
						
							b5d6092707
						
					
				
				
					commit
					00c2a721e2
				
			| @@ -3,6 +3,8 @@ from __future__ import unicode_literals | |||||||
| import re | import re | ||||||
| import datetime | import datetime | ||||||
|  |  | ||||||
|  | from itertools import chain | ||||||
|  |  | ||||||
| from django.utils import six | from django.utils import six | ||||||
| from django.db import models | from django.db import models | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
| @@ -172,8 +174,6 @@ class MigrationAutodetector(object): | |||||||
|         self.generate_created_models() |         self.generate_created_models() | ||||||
|         self.generate_deleted_proxies() |         self.generate_deleted_proxies() | ||||||
|         self.generate_created_proxies() |         self.generate_created_proxies() | ||||||
|         self.generate_deleted_unmanaged() |  | ||||||
|         self.generate_created_unmanaged() |  | ||||||
|         self.generate_altered_options() |         self.generate_altered_options() | ||||||
|  |  | ||||||
|         # Generate field operations |         # Generate field operations | ||||||
| @@ -434,20 +434,27 @@ class MigrationAutodetector(object): | |||||||
|  |  | ||||||
|     def generate_created_models(self): |     def generate_created_models(self): | ||||||
|         """ |         """ | ||||||
|         Find all new models and make creation operations for them, |         Find all new models (both managed and unmanaged) and make create | ||||||
|         and separate operations to create any foreign key or M2M relationships |         operations for them as well as separate operations to create any | ||||||
|         (we'll optimise these back in later if we can) |         foreign key or M2M relationships (we'll optimize these back in later | ||||||
|  |         if we can). | ||||||
|  |  | ||||||
|         We also defer any model options that refer to collections of fields |         We also defer any model options that refer to collections of fields | ||||||
|         that might be deferred (e.g. unique_together, index_together) |         that might be deferred (e.g. unique_together, index_together). | ||||||
|         """ |         """ | ||||||
|         added_models = set(self.new_model_keys) - set(self.old_model_keys) |         added_models = set(self.new_model_keys) - set(self.old_model_keys) | ||||||
|         for app_label, model_name in sorted(added_models, key=self.swappable_first_key, reverse=True): |         added_unmanaged_models = set(self.new_unmanaged_keys) - set(self.old_unmanaged_keys) | ||||||
|  |         models = chain( | ||||||
|  |             sorted(added_models, key=self.swappable_first_key, reverse=True), | ||||||
|  |             sorted(added_unmanaged_models, key=self.swappable_first_key, reverse=True) | ||||||
|  |         ) | ||||||
|  |         for app_label, model_name in models: | ||||||
|             model_state = self.to_state.models[app_label, model_name] |             model_state = self.to_state.models[app_label, model_name] | ||||||
|  |             model_opts = self.new_apps.get_model(app_label, model_name)._meta | ||||||
|             # Gather related fields |             # Gather related fields | ||||||
|             related_fields = {} |             related_fields = {} | ||||||
|             primary_key_rel = None |             primary_key_rel = None | ||||||
|             for field in self.new_apps.get_model(app_label, model_name)._meta.local_fields: |             for field in model_opts.local_fields: | ||||||
|                 if field.rel: |                 if field.rel: | ||||||
|                     if field.rel.to: |                     if field.rel.to: | ||||||
|                         if field.primary_key: |                         if field.primary_key: | ||||||
| @@ -458,7 +465,7 @@ class MigrationAutodetector(object): | |||||||
|                     # we can treat lack of through as auto_created=True, though. |                     # we can treat lack of through as auto_created=True, though. | ||||||
|                     if getattr(field.rel, "through", None) and not field.rel.through._meta.auto_created: |                     if getattr(field.rel, "through", None) and not field.rel.through._meta.auto_created: | ||||||
|                         related_fields[field.name] = field |                         related_fields[field.name] = field | ||||||
|             for field in self.new_apps.get_model(app_label, model_name)._meta.local_many_to_many: |             for field in model_opts.local_many_to_many: | ||||||
|                 if field.rel.to: |                 if field.rel.to: | ||||||
|                     related_fields[field.name] = field |                     related_fields[field.name] = field | ||||||
|                 if getattr(field.rel, "through", None) and not field.rel.through._meta.auto_created: |                 if getattr(field.rel, "through", None) and not field.rel.through._meta.auto_created: | ||||||
| @@ -496,6 +503,11 @@ class MigrationAutodetector(object): | |||||||
|                 dependencies=dependencies, |                 dependencies=dependencies, | ||||||
|                 beginning=True, |                 beginning=True, | ||||||
|             ) |             ) | ||||||
|  |  | ||||||
|  |             # Don't add operations which modify the database for unmanaged models | ||||||
|  |             if not model_opts.managed: | ||||||
|  |                 continue | ||||||
|  |  | ||||||
|             # Generate operations for each related field |             # Generate operations for each related field | ||||||
|             for name, field in sorted(related_fields.items()): |             for name, field in sorted(related_fields.items()): | ||||||
|                 # Account for FKs to swappable models |                 # Account for FKs to swappable models | ||||||
| @@ -563,23 +575,17 @@ class MigrationAutodetector(object): | |||||||
|                     ] |                     ] | ||||||
|                 ) |                 ) | ||||||
|  |  | ||||||
|     def generate_created_proxies(self, unmanaged=False): |     def generate_created_proxies(self): | ||||||
|         """ |         """ | ||||||
|         Makes CreateModel statements for proxy models. |         Makes CreateModel statements for proxy models. | ||||||
|         We use the same statements as that way there's less code duplication, |         We use the same statements as that way there's less code duplication, | ||||||
|         but of course for proxy models we can skip all that pointless field |         but of course for proxy models we can skip all that pointless field | ||||||
|         stuff and just chuck out an operation. |         stuff and just chuck out an operation. | ||||||
|         """ |         """ | ||||||
|         if unmanaged: |         added = set(self.new_proxy_keys) - set(self.old_proxy_keys) | ||||||
|             added = set(self.new_unmanaged_keys) - set(self.old_unmanaged_keys) |  | ||||||
|         else: |  | ||||||
|             added = set(self.new_proxy_keys) - set(self.old_proxy_keys) |  | ||||||
|         for app_label, model_name in sorted(added): |         for app_label, model_name in sorted(added): | ||||||
|             model_state = self.to_state.models[app_label, model_name] |             model_state = self.to_state.models[app_label, model_name] | ||||||
|             if unmanaged: |             assert model_state.options.get("proxy", False) | ||||||
|                 assert not model_state.options.get("managed", True) |  | ||||||
|             else: |  | ||||||
|                 assert model_state.options.get("proxy", False) |  | ||||||
|             # Depend on the deletion of any possible non-proxy version of us |             # Depend on the deletion of any possible non-proxy version of us | ||||||
|             dependencies = [ |             dependencies = [ | ||||||
|                 (app_label, model_name, None, False), |                 (app_label, model_name, None, False), | ||||||
| @@ -602,28 +608,32 @@ class MigrationAutodetector(object): | |||||||
|                 dependencies=dependencies, |                 dependencies=dependencies, | ||||||
|             ) |             ) | ||||||
|  |  | ||||||
|     def generate_created_unmanaged(self): |  | ||||||
|         """ |  | ||||||
|         Similar to generate_created_proxies but for unmanaged |  | ||||||
|         (they are similar to us in that we need to supply them, but they don't |  | ||||||
|         affect the DB) |  | ||||||
|         """ |  | ||||||
|         # Just re-use the same code in *_proxies |  | ||||||
|         self.generate_created_proxies(unmanaged=True) |  | ||||||
|  |  | ||||||
|     def generate_deleted_models(self): |     def generate_deleted_models(self): | ||||||
|         """ |         """ | ||||||
|         Find all deleted models and make creation operations for them, |         Find all deleted models (managed and unmanaged) and make delete | ||||||
|         and separate operations to delete any foreign key or M2M relationships |         operations for them as well as separate operations to delete any | ||||||
|         (we'll optimise these back in later if we can) |         foreign key or M2M relationships (we'll optimize these back in later | ||||||
|  |         if we can). | ||||||
|  |  | ||||||
|         We also bring forward removal of any model options that refer to |         We also bring forward removal of any model options that refer to | ||||||
|         collections of fields - the inverse of generate_created_models. |         collections of fields - the inverse of generate_created_models(). | ||||||
|         """ |         """ | ||||||
|         deleted_models = set(self.old_model_keys) - set(self.new_model_keys) |         deleted_models = set(self.old_model_keys) - set(self.new_model_keys) | ||||||
|         for app_label, model_name in sorted(deleted_models): |         deleted_unmanaged_models = set(self.old_unmanaged_keys) - set(self.new_unmanaged_keys) | ||||||
|  |         models = chain(sorted(deleted_models), sorted(deleted_unmanaged_models)) | ||||||
|  |         for app_label, model_name in models: | ||||||
|             model_state = self.from_state.models[app_label, model_name] |             model_state = self.from_state.models[app_label, model_name] | ||||||
|             model = self.old_apps.get_model(app_label, model_name) |             model = self.old_apps.get_model(app_label, model_name) | ||||||
|  |             if not model._meta.managed: | ||||||
|  |                 self.add_operation( | ||||||
|  |                     app_label, | ||||||
|  |                     operations.DeleteModel( | ||||||
|  |                         name=model_state.name, | ||||||
|  |                     ), | ||||||
|  |                 ) | ||||||
|  |                 # Skip here, no need to handle fields for unmanaged models | ||||||
|  |                 continue | ||||||
|  |  | ||||||
|             # Gather related fields |             # Gather related fields | ||||||
|             related_fields = {} |             related_fields = {} | ||||||
|             for field in model._meta.local_fields: |             for field in model._meta.local_fields: | ||||||
| @@ -707,20 +717,14 @@ class MigrationAutodetector(object): | |||||||
|                 dependencies=list(set(dependencies)), |                 dependencies=list(set(dependencies)), | ||||||
|             ) |             ) | ||||||
|  |  | ||||||
|     def generate_deleted_proxies(self, unmanaged=False): |     def generate_deleted_proxies(self): | ||||||
|         """ |         """ | ||||||
|         Makes DeleteModel statements for proxy models. |         Makes DeleteModel statements for proxy models. | ||||||
|         """ |         """ | ||||||
|         if unmanaged: |         deleted = set(self.old_proxy_keys) - set(self.new_proxy_keys) | ||||||
|             deleted = set(self.old_unmanaged_keys) - set(self.new_unmanaged_keys) |  | ||||||
|         else: |  | ||||||
|             deleted = set(self.old_proxy_keys) - set(self.new_proxy_keys) |  | ||||||
|         for app_label, model_name in sorted(deleted): |         for app_label, model_name in sorted(deleted): | ||||||
|             model_state = self.from_state.models[app_label, model_name] |             model_state = self.from_state.models[app_label, model_name] | ||||||
|             if unmanaged: |             assert model_state.options.get("proxy", False) | ||||||
|                 assert not model_state.options.get("managed", True) |  | ||||||
|             else: |  | ||||||
|                 assert model_state.options.get("proxy", False) |  | ||||||
|             self.add_operation( |             self.add_operation( | ||||||
|                 app_label, |                 app_label, | ||||||
|                 operations.DeleteModel( |                 operations.DeleteModel( | ||||||
| @@ -728,12 +732,6 @@ class MigrationAutodetector(object): | |||||||
|                 ), |                 ), | ||||||
|             ) |             ) | ||||||
|  |  | ||||||
|     def generate_deleted_unmanaged(self): |  | ||||||
|         """ |  | ||||||
|         Makes DeleteModel statements for unmanaged models |  | ||||||
|         """ |  | ||||||
|         self.generate_deleted_proxies(unmanaged=True) |  | ||||||
|  |  | ||||||
|     def generate_renamed_fields(self): |     def generate_renamed_fields(self): | ||||||
|         """ |         """ | ||||||
|         Works out renamed fields |         Works out renamed fields | ||||||
|   | |||||||
| @@ -70,3 +70,6 @@ Bugfixes | |||||||
|  |  | ||||||
| * Made the :setting:`SERIALIZE <TEST_SERIALIZE>` entry in the | * Made the :setting:`SERIALIZE <TEST_SERIALIZE>` entry in the | ||||||
|   :setting:`TEST <DATABASE-TEST>` dictionary usable (:ticket:`23421`). |   :setting:`TEST <DATABASE-TEST>` dictionary usable (:ticket:`23421`). | ||||||
|  |  | ||||||
|  | * Fixed bug in migrations that prevented foreign key constraints to unmanaged | ||||||
|  |   models with a custom primary key (:ticket:`23415`). | ||||||
|   | |||||||
| @@ -31,6 +31,7 @@ class AutodetectorTests(TestCase): | |||||||
|     author_name_default = ModelState("testapp", "Author", [("id", models.AutoField(primary_key=True)), ("name", models.CharField(max_length=200, default='Ada Lovelace'))]) |     author_name_default = ModelState("testapp", "Author", [("id", models.AutoField(primary_key=True)), ("name", models.CharField(max_length=200, default='Ada Lovelace'))]) | ||||||
|     author_name_deconstructable_1 = ModelState("testapp", "Author", [("id", models.AutoField(primary_key=True)), ("name", models.CharField(max_length=200, default=DeconstructableObject()))]) |     author_name_deconstructable_1 = ModelState("testapp", "Author", [("id", models.AutoField(primary_key=True)), ("name", models.CharField(max_length=200, default=DeconstructableObject()))]) | ||||||
|     author_name_deconstructable_2 = ModelState("testapp", "Author", [("id", models.AutoField(primary_key=True)), ("name", models.CharField(max_length=200, default=DeconstructableObject()))]) |     author_name_deconstructable_2 = ModelState("testapp", "Author", [("id", models.AutoField(primary_key=True)), ("name", models.CharField(max_length=200, default=DeconstructableObject()))]) | ||||||
|  |     author_custom_pk = ModelState("testapp", "Author", [("pk_field", models.IntegerField(primary_key=True))]) | ||||||
|     author_with_book = ModelState("testapp", "Author", [("id", models.AutoField(primary_key=True)), ("name", models.CharField(max_length=200)), ("book", models.ForeignKey("otherapp.Book"))]) |     author_with_book = ModelState("testapp", "Author", [("id", models.AutoField(primary_key=True)), ("name", models.CharField(max_length=200)), ("book", models.ForeignKey("otherapp.Book"))]) | ||||||
|     author_with_book_order_wrt = ModelState("testapp", "Author", [("id", models.AutoField(primary_key=True)), ("name", models.CharField(max_length=200)), ("book", models.ForeignKey("otherapp.Book"))], options={"order_with_respect_to": "book"}) |     author_with_book_order_wrt = ModelState("testapp", "Author", [("id", models.AutoField(primary_key=True)), ("name", models.CharField(max_length=200)), ("book", models.ForeignKey("otherapp.Book"))], options={"order_with_respect_to": "book"}) | ||||||
|     author_renamed_with_book = ModelState("testapp", "Writer", [("id", models.AutoField(primary_key=True)), ("name", models.CharField(max_length=200)), ("book", models.ForeignKey("otherapp.Book"))]) |     author_renamed_with_book = ModelState("testapp", "Writer", [("id", models.AutoField(primary_key=True)), ("name", models.CharField(max_length=200)), ("book", models.ForeignKey("otherapp.Book"))]) | ||||||
| @@ -44,6 +45,10 @@ class AutodetectorTests(TestCase): | |||||||
|     author_proxy_proxy = ModelState("testapp", "AAuthorProxyProxy", [], {"proxy": True}, ("testapp.authorproxy", )) |     author_proxy_proxy = ModelState("testapp", "AAuthorProxyProxy", [], {"proxy": True}, ("testapp.authorproxy", )) | ||||||
|     author_unmanaged = ModelState("testapp", "AuthorUnmanaged", [], {"managed": False}, ("testapp.author", )) |     author_unmanaged = ModelState("testapp", "AuthorUnmanaged", [], {"managed": False}, ("testapp.author", )) | ||||||
|     author_unmanaged_managed = ModelState("testapp", "AuthorUnmanaged", [], {}, ("testapp.author", )) |     author_unmanaged_managed = ModelState("testapp", "AuthorUnmanaged", [], {}, ("testapp.author", )) | ||||||
|  |     author_unmanaged_default_pk = ModelState("testapp", "Author", [("id", models.AutoField(primary_key=True))]) | ||||||
|  |     author_unmanaged_custom_pk = ModelState("testapp", "Author", [ | ||||||
|  |         ("pk_field", models.IntegerField(primary_key=True)), | ||||||
|  |     ]) | ||||||
|     author_with_m2m = ModelState("testapp", "Author", [ |     author_with_m2m = ModelState("testapp", "Author", [ | ||||||
|         ("id", models.AutoField(primary_key=True)), |         ("id", models.AutoField(primary_key=True)), | ||||||
|         ("publishers", models.ManyToManyField("testapp.Publisher")), |         ("publishers", models.ManyToManyField("testapp.Publisher")), | ||||||
| @@ -644,6 +649,24 @@ class AutodetectorTests(TestCase): | |||||||
|         self.assertOperationAttributes(changes, "testapp", 0, 0, name="AuthorProxy") |         self.assertOperationAttributes(changes, "testapp", 0, 0, name="AuthorProxy") | ||||||
|         self.assertOperationAttributes(changes, "testapp", 0, 1, name="AuthorProxy", options={}) |         self.assertOperationAttributes(changes, "testapp", 0, 1, name="AuthorProxy", options={}) | ||||||
|  |  | ||||||
|  |     def test_proxy_custom_pk(self): | ||||||
|  |         "#23415 - The autodetector must correctly deal with custom FK on proxy models." | ||||||
|  |         # First, we test the default pk field name | ||||||
|  |         before = self.make_project_state([]) | ||||||
|  |         after = self.make_project_state([self.author_empty, self.author_proxy_third, self.book_proxy_fk]) | ||||||
|  |         autodetector = MigrationAutodetector(before, after) | ||||||
|  |         changes = autodetector._detect_changes() | ||||||
|  |         # The field name the FK on the book model points to | ||||||
|  |         self.assertEqual(changes['otherapp'][0].operations[0].fields[2][1].rel.field_name, 'id') | ||||||
|  |  | ||||||
|  |         # Now, we test the custom pk field name | ||||||
|  |         before = self.make_project_state([]) | ||||||
|  |         after = self.make_project_state([self.author_custom_pk, self.author_proxy_third, self.book_proxy_fk]) | ||||||
|  |         autodetector = MigrationAutodetector(before, after) | ||||||
|  |         changes = autodetector._detect_changes() | ||||||
|  |         # The field name the FK on the book model points to | ||||||
|  |         self.assertEqual(changes['otherapp'][0].operations[0].fields[2][1].rel.field_name, 'pk_field') | ||||||
|  |  | ||||||
|     def test_unmanaged(self): |     def test_unmanaged(self): | ||||||
|         "Tests that the autodetector correctly deals with managed models" |         "Tests that the autodetector correctly deals with managed models" | ||||||
|         # First, we test adding an unmanaged model |         # First, we test adding an unmanaged model | ||||||
| @@ -667,6 +690,24 @@ class AutodetectorTests(TestCase): | |||||||
|         self.assertOperationAttributes(changes, 'testapp', 0, 0, name="AuthorUnmanaged") |         self.assertOperationAttributes(changes, 'testapp', 0, 0, name="AuthorUnmanaged") | ||||||
|         self.assertOperationAttributes(changes, 'testapp', 0, 1, name="AuthorUnmanaged") |         self.assertOperationAttributes(changes, 'testapp', 0, 1, name="AuthorUnmanaged") | ||||||
|  |  | ||||||
|  |     def test_unmanaged_custom_pk(self): | ||||||
|  |         "#23415 - The autodetector must correctly deal with custom FK on unmanaged models." | ||||||
|  |         # First, we test the default pk field name | ||||||
|  |         before = self.make_project_state([]) | ||||||
|  |         after = self.make_project_state([self.author_unmanaged_default_pk, self.book]) | ||||||
|  |         autodetector = MigrationAutodetector(before, after) | ||||||
|  |         changes = autodetector._detect_changes() | ||||||
|  |         # The field name the FK on the book model points to | ||||||
|  |         self.assertEqual(changes['otherapp'][0].operations[0].fields[2][1].rel.field_name, 'id') | ||||||
|  |  | ||||||
|  |         # Now, we test the custom pk field name | ||||||
|  |         before = self.make_project_state([]) | ||||||
|  |         after = self.make_project_state([self.author_unmanaged_custom_pk, self.book]) | ||||||
|  |         autodetector = MigrationAutodetector(before, after) | ||||||
|  |         changes = autodetector._detect_changes() | ||||||
|  |         # The field name the FK on the book model points to | ||||||
|  |         self.assertEqual(changes['otherapp'][0].operations[0].fields[2][1].rel.field_name, 'pk_field') | ||||||
|  |  | ||||||
|     @override_settings(AUTH_USER_MODEL="thirdapp.CustomUser") |     @override_settings(AUTH_USER_MODEL="thirdapp.CustomUser") | ||||||
|     def test_swappable(self): |     def test_swappable(self): | ||||||
|         before = self.make_project_state([self.custom_user]) |         before = self.make_project_state([self.custom_user]) | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user