[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:
Markus Holtermann 2014-09-08 02:26:12 +02:00 committed by Tim Graham
parent b5d6092707
commit 00c2a721e2
3 changed files with 89 additions and 47 deletions

View File

@ -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,22 +575,16 @@ 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_unmanaged_keys) - set(self.old_unmanaged_keys)
else:
added = set(self.new_proxy_keys) - set(self.old_proxy_keys) 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 not model_state.options.get("managed", True)
else:
assert model_state.options.get("proxy", False) 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 = [
@ -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,19 +717,13 @@ 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_unmanaged_keys) - set(self.new_unmanaged_keys)
else:
deleted = set(self.old_proxy_keys) - set(self.new_proxy_keys) 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 not model_state.options.get("managed", True)
else:
assert model_state.options.get("proxy", False) assert model_state.options.get("proxy", False)
self.add_operation( self.add_operation(
app_label, app_label,
@ -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

View File

@ -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`).

View File

@ -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])