Fixed #23275: Unmanaged models kept by autodetector, ignored by ops
This commit is contained in:
parent
6745b6fd7a
commit
8f9862cd4d
|
@ -108,11 +108,15 @@ class MigrationAutodetector(object):
|
||||||
self.new_apps = self.to_state.render()
|
self.new_apps = self.to_state.render()
|
||||||
self.old_model_keys = []
|
self.old_model_keys = []
|
||||||
self.old_proxy_keys = []
|
self.old_proxy_keys = []
|
||||||
|
self.old_unmanaged_keys = []
|
||||||
self.new_model_keys = []
|
self.new_model_keys = []
|
||||||
self.new_proxy_keys = []
|
self.new_proxy_keys = []
|
||||||
|
self.new_unmanaged_keys = []
|
||||||
for al, mn in sorted(self.from_state.models.keys()):
|
for al, mn in sorted(self.from_state.models.keys()):
|
||||||
model = self.old_apps.get_model(al, mn)
|
model = self.old_apps.get_model(al, mn)
|
||||||
if model._meta.managed and al not in self.from_state.real_apps:
|
if not model._meta.managed:
|
||||||
|
self.old_unmanaged_keys.append((al, mn))
|
||||||
|
elif al not in self.from_state.real_apps:
|
||||||
if model._meta.proxy:
|
if model._meta.proxy:
|
||||||
self.old_proxy_keys.append((al, mn))
|
self.old_proxy_keys.append((al, mn))
|
||||||
else:
|
else:
|
||||||
|
@ -120,7 +124,9 @@ class MigrationAutodetector(object):
|
||||||
|
|
||||||
for al, mn in sorted(self.to_state.models.keys()):
|
for al, mn in sorted(self.to_state.models.keys()):
|
||||||
model = self.new_apps.get_model(al, mn)
|
model = self.new_apps.get_model(al, mn)
|
||||||
if model._meta.managed and (
|
if not model._meta.managed:
|
||||||
|
self.new_unmanaged_keys.append((al, mn))
|
||||||
|
elif (
|
||||||
al not in self.from_state.real_apps or
|
al not in self.from_state.real_apps or
|
||||||
(convert_apps and al in convert_apps)
|
(convert_apps and al in convert_apps)
|
||||||
):
|
):
|
||||||
|
@ -136,6 +142,8 @@ class MigrationAutodetector(object):
|
||||||
# through models in the old state so we can make dependencies
|
# through models in the old state so we can make dependencies
|
||||||
# from the through model deletion to the field that uses it.
|
# from the through model deletion to the field that uses it.
|
||||||
self.kept_model_keys = set(self.old_model_keys).intersection(self.new_model_keys)
|
self.kept_model_keys = set(self.old_model_keys).intersection(self.new_model_keys)
|
||||||
|
self.kept_proxy_keys = set(self.old_proxy_keys).intersection(self.new_proxy_keys)
|
||||||
|
self.kept_unmanaged_keys = set(self.old_unmanaged_keys).intersection(self.new_unmanaged_keys)
|
||||||
self.through_users = {}
|
self.through_users = {}
|
||||||
self.old_field_keys = set()
|
self.old_field_keys = set()
|
||||||
self.new_field_keys = set()
|
self.new_field_keys = set()
|
||||||
|
@ -164,6 +172,8 @@ 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
|
||||||
|
@ -539,16 +549,22 @@ class MigrationAutodetector(object):
|
||||||
]
|
]
|
||||||
)
|
)
|
||||||
|
|
||||||
def generate_created_proxies(self):
|
def generate_created_proxies(self, unmanaged=False):
|
||||||
"""
|
"""
|
||||||
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.
|
||||||
"""
|
"""
|
||||||
added_proxies = set(self.new_proxy_keys) - set(self.old_proxy_keys)
|
if unmanaged:
|
||||||
for app_label, model_name in sorted(added_proxies):
|
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):
|
||||||
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 = [
|
||||||
|
@ -572,6 +588,15 @@ 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 and make creation operations for them,
|
||||||
|
@ -668,13 +693,19 @@ class MigrationAutodetector(object):
|
||||||
dependencies=list(set(dependencies)),
|
dependencies=list(set(dependencies)),
|
||||||
)
|
)
|
||||||
|
|
||||||
def generate_deleted_proxies(self):
|
def generate_deleted_proxies(self, unmanaged=False):
|
||||||
"""
|
"""
|
||||||
Makes DeleteModel statements for proxy models.
|
Makes DeleteModel statements for proxy models.
|
||||||
"""
|
"""
|
||||||
deleted_proxies = set(self.old_proxy_keys) - set(self.new_proxy_keys)
|
if unmanaged:
|
||||||
for app_label, model_name in sorted(deleted_proxies):
|
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):
|
||||||
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,
|
||||||
|
@ -683,6 +714,12 @@ 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
|
||||||
|
@ -852,7 +889,7 @@ class MigrationAutodetector(object):
|
||||||
makes an operation to represent them in state changes (in case Python
|
makes an operation to represent them in state changes (in case Python
|
||||||
code in migrations needs them)
|
code in migrations needs them)
|
||||||
"""
|
"""
|
||||||
models_to_check = self.kept_model_keys.union(set(self.new_proxy_keys).intersection(self.old_proxy_keys))
|
models_to_check = self.kept_model_keys.union(self.kept_proxy_keys).union(self.kept_unmanaged_keys)
|
||||||
for app_label, model_name in sorted(models_to_check):
|
for app_label, model_name in sorted(models_to_check):
|
||||||
old_model_name = self.renamed_models.get((app_label, model_name), model_name)
|
old_model_name = self.renamed_models.get((app_label, model_name), model_name)
|
||||||
old_model_state = self.from_state.models[app_label, old_model_name]
|
old_model_state = self.from_state.models[app_label, old_model_name]
|
||||||
|
|
|
@ -101,12 +101,13 @@ class Operation(object):
|
||||||
def allowed_to_migrate(self, connection_alias, model):
|
def allowed_to_migrate(self, connection_alias, model):
|
||||||
"""
|
"""
|
||||||
Returns if we're allowed to migrate the model. Checks the router,
|
Returns if we're allowed to migrate the model. Checks the router,
|
||||||
if it's a proxy, and if it's swapped out.
|
if it's a proxy, if it's managed, and if it's swapped out.
|
||||||
"""
|
"""
|
||||||
return (
|
return (
|
||||||
router.allow_migrate(connection_alias, model) and
|
router.allow_migrate(connection_alias, model) and
|
||||||
not model._meta.proxy and
|
not model._meta.proxy and
|
||||||
not model._meta.swapped
|
not model._meta.swapped and
|
||||||
|
model._meta.managed
|
||||||
)
|
)
|
||||||
|
|
||||||
def __repr__(self):
|
def __repr__(self):
|
||||||
|
|
|
@ -639,30 +639,28 @@ 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_unmanaged_ignorance(self):
|
def test_unmanaged(self):
|
||||||
"Tests that the autodetector correctly ignores 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
|
||||||
before = self.make_project_state([self.author_empty])
|
before = self.make_project_state([self.author_empty])
|
||||||
after = self.make_project_state([self.author_empty, self.author_unmanaged])
|
after = self.make_project_state([self.author_empty, self.author_unmanaged])
|
||||||
autodetector = MigrationAutodetector(before, after)
|
autodetector = MigrationAutodetector(before, after)
|
||||||
changes = autodetector._detect_changes()
|
changes = autodetector._detect_changes()
|
||||||
# Right number of migrations?
|
# Right number of migrations?
|
||||||
self.assertEqual(len(changes), 0)
|
self.assertNumberMigrations(changes, 'testapp', 1)
|
||||||
|
self.assertOperationTypes(changes, 'testapp', 0, ["CreateModel"])
|
||||||
|
self.assertOperationAttributes(changes, 'testapp', 0, 0, name="AuthorUnmanaged")
|
||||||
|
self.assertEqual(changes['testapp'][0].operations[0].options['managed'], False)
|
||||||
# Now, we test turning an unmanaged model into a managed model
|
# Now, we test turning an unmanaged model into a managed model
|
||||||
before = self.make_project_state([self.author_empty, self.author_unmanaged])
|
before = self.make_project_state([self.author_empty, self.author_unmanaged])
|
||||||
after = self.make_project_state([self.author_empty, self.author_unmanaged_managed])
|
after = self.make_project_state([self.author_empty, self.author_unmanaged_managed])
|
||||||
autodetector = MigrationAutodetector(before, after)
|
autodetector = MigrationAutodetector(before, after)
|
||||||
changes = autodetector._detect_changes()
|
changes = autodetector._detect_changes()
|
||||||
# Right number of migrations?
|
# Right number of migrations?
|
||||||
self.assertEqual(len(changes['testapp']), 1)
|
self.assertNumberMigrations(changes, 'testapp', 1)
|
||||||
# Right number of actions?
|
self.assertOperationTypes(changes, 'testapp', 0, ["DeleteModel", "CreateModel"])
|
||||||
migration = changes['testapp'][0]
|
self.assertOperationAttributes(changes, 'testapp', 0, 0, name="AuthorUnmanaged")
|
||||||
self.assertEqual(len(migration.operations), 1)
|
self.assertOperationAttributes(changes, 'testapp', 0, 1, name="AuthorUnmanaged")
|
||||||
# Right action?
|
|
||||||
action = migration.operations[0]
|
|
||||||
self.assertEqual(action.__class__.__name__, "CreateModel")
|
|
||||||
self.assertEqual(action.name, "AuthorUnmanaged")
|
|
||||||
|
|
||||||
@override_settings(AUTH_USER_MODEL="thirdapp.CustomUser")
|
@override_settings(AUTH_USER_MODEL="thirdapp.CustomUser")
|
||||||
def test_swappable(self):
|
def test_swappable(self):
|
||||||
|
|
|
@ -326,6 +326,35 @@ class OperationTests(OperationTestBase):
|
||||||
self.assertTableNotExists("test_crprmo_proxypony")
|
self.assertTableNotExists("test_crprmo_proxypony")
|
||||||
self.assertTableExists("test_crprmo_pony")
|
self.assertTableExists("test_crprmo_pony")
|
||||||
|
|
||||||
|
def test_create_unmanaged_model(self):
|
||||||
|
"""
|
||||||
|
Tests that CreateModel ignores unmanaged models.
|
||||||
|
"""
|
||||||
|
project_state = self.set_up_test_model("test_crummo")
|
||||||
|
# Test the state alteration
|
||||||
|
operation = migrations.CreateModel(
|
||||||
|
"UnmanagedPony",
|
||||||
|
[],
|
||||||
|
options={"proxy": True},
|
||||||
|
bases=("test_crummo.Pony", ),
|
||||||
|
)
|
||||||
|
self.assertEqual(operation.describe(), "Create proxy model UnmanagedPony")
|
||||||
|
new_state = project_state.clone()
|
||||||
|
operation.state_forwards("test_crummo", new_state)
|
||||||
|
self.assertIn(("test_crummo", "unmanagedpony"), new_state.models)
|
||||||
|
# Test the database alteration
|
||||||
|
self.assertTableNotExists("test_crummo_unmanagedpony")
|
||||||
|
self.assertTableExists("test_crummo_pony")
|
||||||
|
with connection.schema_editor() as editor:
|
||||||
|
operation.database_forwards("test_crummo", editor, project_state, new_state)
|
||||||
|
self.assertTableNotExists("test_crummo_unmanagedpony")
|
||||||
|
self.assertTableExists("test_crummo_pony")
|
||||||
|
# And test reversal
|
||||||
|
with connection.schema_editor() as editor:
|
||||||
|
operation.database_backwards("test_crummo", editor, new_state, project_state)
|
||||||
|
self.assertTableNotExists("test_crummo_unmanagedpony")
|
||||||
|
self.assertTableExists("test_crummo_pony")
|
||||||
|
|
||||||
def test_delete_model(self):
|
def test_delete_model(self):
|
||||||
"""
|
"""
|
||||||
Tests the DeleteModel operation.
|
Tests the DeleteModel operation.
|
||||||
|
|
Loading…
Reference in New Issue