Fixed #25817 -- Made RenameField repoint to_field/to_fields references.
Also updated the autodetector to assume the RenameField operation will perform the required repointing.
This commit is contained in:
parent
2faeb21d2f
commit
dcdd219ee1
|
@ -895,6 +895,24 @@ class MigrationAutodetector:
|
||||||
)
|
)
|
||||||
if rename_key in self.renamed_models:
|
if rename_key in self.renamed_models:
|
||||||
new_field.remote_field.model = old_field.remote_field.model
|
new_field.remote_field.model = old_field.remote_field.model
|
||||||
|
# Handle ForeignKey which can only have a single to_field.
|
||||||
|
remote_field_name = getattr(new_field.remote_field, 'field_name', None)
|
||||||
|
if remote_field_name:
|
||||||
|
to_field_rename_key = rename_key + (remote_field_name,)
|
||||||
|
if to_field_rename_key in self.renamed_fields:
|
||||||
|
new_field.remote_field.field_name = old_field.remote_field.field_name
|
||||||
|
# Handle ForeignObjects which can have multiple from_fields/to_fields.
|
||||||
|
from_fields = getattr(new_field, 'from_fields', None)
|
||||||
|
if from_fields:
|
||||||
|
from_rename_key = (app_label, model_name)
|
||||||
|
new_field.from_fields = tuple([
|
||||||
|
self.renamed_fields.get(from_rename_key + (from_field,), from_field)
|
||||||
|
for from_field in from_fields
|
||||||
|
])
|
||||||
|
new_field.to_fields = tuple([
|
||||||
|
self.renamed_fields.get(rename_key + (to_field,), to_field)
|
||||||
|
for to_field in new_field.to_fields
|
||||||
|
])
|
||||||
if hasattr(new_field, "remote_field") and getattr(new_field.remote_field, "through", None):
|
if hasattr(new_field, "remote_field") and getattr(new_field.remote_field, "through", None):
|
||||||
rename_key = (
|
rename_key = (
|
||||||
new_field.remote_field.through._meta.app_label,
|
new_field.remote_field.through._meta.app_label,
|
||||||
|
|
|
@ -270,8 +270,9 @@ class RenameField(FieldOperation):
|
||||||
model_state = state.models[app_label, self.model_name_lower]
|
model_state = state.models[app_label, self.model_name_lower]
|
||||||
# Rename the field
|
# Rename the field
|
||||||
fields = model_state.fields
|
fields = model_state.fields
|
||||||
|
found = False
|
||||||
for index, (name, field) in enumerate(fields):
|
for index, (name, field) in enumerate(fields):
|
||||||
if name == self.old_name:
|
if not found and name == self.old_name:
|
||||||
fields[index] = (self.new_name, field)
|
fields[index] = (self.new_name, field)
|
||||||
# Delay rendering of relationships if it's not a relational
|
# Delay rendering of relationships if it's not a relational
|
||||||
# field and not referenced by a foreign key.
|
# field and not referenced by a foreign key.
|
||||||
|
@ -279,8 +280,15 @@ class RenameField(FieldOperation):
|
||||||
not field.is_relation and
|
not field.is_relation and
|
||||||
not is_referenced_by_foreign_key(state, self.model_name_lower, field, self.name)
|
not is_referenced_by_foreign_key(state, self.model_name_lower, field, self.name)
|
||||||
)
|
)
|
||||||
break
|
found = True
|
||||||
else:
|
# Fix from_fields to refer to the new field.
|
||||||
|
from_fields = getattr(field, 'from_fields', None)
|
||||||
|
if from_fields:
|
||||||
|
field.from_fields = tuple([
|
||||||
|
self.new_name if from_field_name == self.old_name else from_field_name
|
||||||
|
for from_field_name in from_fields
|
||||||
|
])
|
||||||
|
if not found:
|
||||||
raise FieldDoesNotExist(
|
raise FieldDoesNotExist(
|
||||||
"%s.%s has no field named '%s'" % (app_label, self.model_name, self.old_name)
|
"%s.%s has no field named '%s'" % (app_label, self.model_name, self.old_name)
|
||||||
)
|
)
|
||||||
|
@ -292,6 +300,24 @@ class RenameField(FieldOperation):
|
||||||
[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]
|
||||||
]
|
]
|
||||||
|
# Fix to_fields to refer to the new field.
|
||||||
|
model_tuple = app_label, self.model_name_lower
|
||||||
|
for (model_app_label, model_name), model_state in state.models.items():
|
||||||
|
for index, (name, field) in enumerate(model_state.fields):
|
||||||
|
remote_field = field.remote_field
|
||||||
|
if remote_field:
|
||||||
|
remote_model_tuple = self._get_model_tuple(
|
||||||
|
remote_field.model, model_app_label, model_name
|
||||||
|
)
|
||||||
|
if remote_model_tuple == model_tuple:
|
||||||
|
if getattr(remote_field, 'field_name', None) == self.old_name:
|
||||||
|
remote_field.field_name = self.new_name
|
||||||
|
to_fields = getattr(field, 'to_fields', None)
|
||||||
|
if to_fields:
|
||||||
|
field.to_fields = tuple([
|
||||||
|
self.new_name if to_field_name == self.old_name else to_field_name
|
||||||
|
for to_field_name in to_fields
|
||||||
|
])
|
||||||
state.reload_model(app_label, self.model_name_lower, delay=delay)
|
state.reload_model(app_label, self.model_name_lower, delay=delay)
|
||||||
|
|
||||||
def database_forwards(self, app_label, schema_editor, from_state, to_state):
|
def database_forwards(self, app_label, schema_editor, from_state, to_state):
|
||||||
|
|
|
@ -821,6 +821,102 @@ class AutodetectorTests(TestCase):
|
||||||
self.assertOperationTypes(changes, 'testapp', 0, ["RenameField"])
|
self.assertOperationTypes(changes, 'testapp', 0, ["RenameField"])
|
||||||
self.assertOperationAttributes(changes, 'testapp', 0, 0, old_name="name", new_name="names")
|
self.assertOperationAttributes(changes, 'testapp', 0, 0, old_name="name", new_name="names")
|
||||||
|
|
||||||
|
def test_rename_field_foreign_key_to_field(self):
|
||||||
|
before = [
|
||||||
|
ModelState('app', 'Foo', [
|
||||||
|
('id', models.AutoField(primary_key=True)),
|
||||||
|
('field', models.IntegerField(unique=True)),
|
||||||
|
]),
|
||||||
|
ModelState('app', 'Bar', [
|
||||||
|
('id', models.AutoField(primary_key=True)),
|
||||||
|
('foo', models.ForeignKey('app.Foo', models.CASCADE, to_field='field')),
|
||||||
|
]),
|
||||||
|
]
|
||||||
|
after = [
|
||||||
|
ModelState('app', 'Foo', [
|
||||||
|
('id', models.AutoField(primary_key=True)),
|
||||||
|
('renamed_field', models.IntegerField(unique=True)),
|
||||||
|
]),
|
||||||
|
ModelState('app', 'Bar', [
|
||||||
|
('id', models.AutoField(primary_key=True)),
|
||||||
|
('foo', models.ForeignKey('app.Foo', models.CASCADE, to_field='renamed_field')),
|
||||||
|
]),
|
||||||
|
]
|
||||||
|
changes = self.get_changes(before, after, MigrationQuestioner({'ask_rename': True}))
|
||||||
|
# Right number/type of migrations?
|
||||||
|
self.assertNumberMigrations(changes, 'app', 1)
|
||||||
|
self.assertOperationTypes(changes, 'app', 0, ['RenameField'])
|
||||||
|
self.assertOperationAttributes(changes, 'app', 0, 0, old_name='field', new_name='renamed_field')
|
||||||
|
|
||||||
|
def test_rename_foreign_object_fields(self):
|
||||||
|
fields = ('first', 'second')
|
||||||
|
renamed_fields = ('first_renamed', 'second_renamed')
|
||||||
|
before = [
|
||||||
|
ModelState('app', 'Foo', [
|
||||||
|
('id', models.AutoField(primary_key=True)),
|
||||||
|
('first', models.IntegerField()),
|
||||||
|
('second', models.IntegerField()),
|
||||||
|
], options={'unique_together': {fields}}),
|
||||||
|
ModelState('app', 'Bar', [
|
||||||
|
('id', models.AutoField(primary_key=True)),
|
||||||
|
('first', models.IntegerField()),
|
||||||
|
('second', models.IntegerField()),
|
||||||
|
('foo', models.ForeignObject(
|
||||||
|
'app.Foo', models.CASCADE, from_fields=fields, to_fields=fields,
|
||||||
|
)),
|
||||||
|
]),
|
||||||
|
]
|
||||||
|
# Case 1: to_fields renames.
|
||||||
|
after = [
|
||||||
|
ModelState('app', 'Foo', [
|
||||||
|
('id', models.AutoField(primary_key=True)),
|
||||||
|
('first_renamed', models.IntegerField()),
|
||||||
|
('second_renamed', models.IntegerField()),
|
||||||
|
], options={'unique_together': {renamed_fields}}),
|
||||||
|
ModelState('app', 'Bar', [
|
||||||
|
('id', models.AutoField(primary_key=True)),
|
||||||
|
('first', models.IntegerField()),
|
||||||
|
('second', models.IntegerField()),
|
||||||
|
('foo', models.ForeignObject(
|
||||||
|
'app.Foo', models.CASCADE, from_fields=fields, to_fields=renamed_fields,
|
||||||
|
)),
|
||||||
|
]),
|
||||||
|
]
|
||||||
|
changes = self.get_changes(before, after, MigrationQuestioner({'ask_rename': True}))
|
||||||
|
self.assertNumberMigrations(changes, 'app', 1)
|
||||||
|
self.assertOperationTypes(changes, 'app', 0, ['RenameField', 'RenameField', 'AlterUniqueTogether'])
|
||||||
|
self.assertOperationAttributes(
|
||||||
|
changes, 'app', 0, 0, model_name='foo', old_name='first', new_name='first_renamed',
|
||||||
|
)
|
||||||
|
self.assertOperationAttributes(
|
||||||
|
changes, 'app', 0, 1, model_name='foo', old_name='second', new_name='second_renamed',
|
||||||
|
)
|
||||||
|
# Case 2: from_fields renames.
|
||||||
|
after = [
|
||||||
|
ModelState('app', 'Foo', [
|
||||||
|
('id', models.AutoField(primary_key=True)),
|
||||||
|
('first', models.IntegerField()),
|
||||||
|
('second', models.IntegerField()),
|
||||||
|
], options={'unique_together': {fields}}),
|
||||||
|
ModelState('app', 'Bar', [
|
||||||
|
('id', models.AutoField(primary_key=True)),
|
||||||
|
('first_renamed', models.IntegerField()),
|
||||||
|
('second_renamed', models.IntegerField()),
|
||||||
|
('foo', models.ForeignObject(
|
||||||
|
'app.Foo', models.CASCADE, from_fields=renamed_fields, to_fields=fields,
|
||||||
|
)),
|
||||||
|
]),
|
||||||
|
]
|
||||||
|
changes = self.get_changes(before, after, MigrationQuestioner({'ask_rename': True}))
|
||||||
|
self.assertNumberMigrations(changes, 'app', 1)
|
||||||
|
self.assertOperationTypes(changes, 'app', 0, ['RenameField', 'RenameField'])
|
||||||
|
self.assertOperationAttributes(
|
||||||
|
changes, 'app', 0, 0, model_name='bar', old_name='first', new_name='first_renamed',
|
||||||
|
)
|
||||||
|
self.assertOperationAttributes(
|
||||||
|
changes, 'app', 0, 1, model_name='bar', old_name='second', new_name='second_renamed',
|
||||||
|
)
|
||||||
|
|
||||||
def test_rename_model(self):
|
def test_rename_model(self):
|
||||||
"""Tests autodetection of renamed models."""
|
"""Tests autodetection of renamed models."""
|
||||||
changes = self.get_changes(
|
changes = self.get_changes(
|
||||||
|
|
|
@ -1455,6 +1455,34 @@ class OperationTests(OperationTestBase):
|
||||||
with self.assertRaisesMessage(FieldDoesNotExist, "app.model has no field named 'field'"):
|
with self.assertRaisesMessage(FieldDoesNotExist, "app.model has no field named 'field'"):
|
||||||
migrations.RenameField('model', 'field', 'new_field').state_forwards('app', state)
|
migrations.RenameField('model', 'field', 'new_field').state_forwards('app', state)
|
||||||
|
|
||||||
|
def test_rename_referenced_field_state_forward(self):
|
||||||
|
state = ProjectState()
|
||||||
|
state.add_model(ModelState('app', 'Model', [
|
||||||
|
('id', models.AutoField(primary_key=True)),
|
||||||
|
('field', models.IntegerField(unique=True)),
|
||||||
|
]))
|
||||||
|
state.add_model(ModelState('app', 'OtherModel', [
|
||||||
|
('id', models.AutoField(primary_key=True)),
|
||||||
|
('fk', models.ForeignKey('Model', models.CASCADE, to_field='field')),
|
||||||
|
('fo', models.ForeignObject('Model', models.CASCADE, from_fields=('fk',), to_fields=('field',))),
|
||||||
|
]))
|
||||||
|
operation = migrations.RenameField('Model', 'field', 'renamed')
|
||||||
|
new_state = state.clone()
|
||||||
|
operation.state_forwards('app', new_state)
|
||||||
|
self.assertEqual(new_state.models['app', 'othermodel'].fields[1][1].remote_field.field_name, 'renamed')
|
||||||
|
self.assertEqual(new_state.models['app', 'othermodel'].fields[1][1].from_fields, ['self'])
|
||||||
|
self.assertEqual(new_state.models['app', 'othermodel'].fields[1][1].to_fields, ('renamed',))
|
||||||
|
self.assertEqual(new_state.models['app', 'othermodel'].fields[2][1].from_fields, ('fk',))
|
||||||
|
self.assertEqual(new_state.models['app', 'othermodel'].fields[2][1].to_fields, ('renamed',))
|
||||||
|
operation = migrations.RenameField('OtherModel', 'fk', 'renamed_fk')
|
||||||
|
new_state = state.clone()
|
||||||
|
operation.state_forwards('app', new_state)
|
||||||
|
self.assertEqual(new_state.models['app', 'othermodel'].fields[1][1].remote_field.field_name, 'renamed')
|
||||||
|
self.assertEqual(new_state.models['app', 'othermodel'].fields[1][1].from_fields, ('self',))
|
||||||
|
self.assertEqual(new_state.models['app', 'othermodel'].fields[1][1].to_fields, ('renamed',))
|
||||||
|
self.assertEqual(new_state.models['app', 'othermodel'].fields[2][1].from_fields, ('renamed_fk',))
|
||||||
|
self.assertEqual(new_state.models['app', 'othermodel'].fields[2][1].to_fields, ('renamed',))
|
||||||
|
|
||||||
def test_alter_unique_together(self):
|
def test_alter_unique_together(self):
|
||||||
"""
|
"""
|
||||||
Tests the AlterUniqueTogether operation.
|
Tests the AlterUniqueTogether operation.
|
||||||
|
|
Loading…
Reference in New Issue