Commit e0cd07ec authored by Tim Graham's avatar Tim Graham
Browse files

Fixed #22903 -- Fixed migration generation if index_together or...

Fixed #22903 -- Fixed migration generation if index_together or unique_together is removed from a model.
parent e769846e
Loading
Loading
Loading
Loading
+9 −16
Original line number Diff line number Diff line
@@ -761,33 +761,26 @@ class MigrationAutodetector(object):
                    )
                )

    def generate_altered_unique_together(self):
    def _generate_altered_foo_together(self, operation):
        option_name = operation.option_name
        for app_label, model_name in sorted(self.kept_model_keys):
            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]
            new_model_state = self.to_state.models[app_label, model_name]
            if old_model_state.options.get("unique_together", None) != new_model_state.options.get("unique_together", None):
            if old_model_state.options.get(option_name) != new_model_state.options.get(option_name):
                self.add_operation(
                    app_label,
                    operations.AlterUniqueTogether(
                    operation(
                        name=model_name,
                        unique_together=new_model_state.options['unique_together'],
                        **{option_name: new_model_state.options.get(option_name)}
                    )
                )

    def generate_altered_unique_together(self):
        self._generate_altered_foo_together(operations.AlterUniqueTogether)

    def generate_altered_index_together(self):
        for app_label, model_name in sorted(self.kept_model_keys):
            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]
            new_model_state = self.to_state.models[app_label, model_name]
            if old_model_state.options.get("index_together", None) != new_model_state.options.get("index_together", None):
                self.add_operation(
                    app_label,
                    operations.AlterIndexTogether(
                        name=model_name,
                        index_together=new_model_state.options['index_together'],
                    )
                )
        self._generate_altered_foo_together(operations.AlterIndexTogether)

    def generate_altered_options(self):
        """
+17 −11
Original line number Diff line number Diff line
@@ -213,18 +213,21 @@ class AlterModelTable(Operation):

class AlterUniqueTogether(Operation):
    """
    Changes the value of index_together to the target one.
    Changes the value of unique_together to the target one.
    Input value of unique_together must be a set of tuples.
    """
    option_name = "unique_together"

    def __init__(self, name, unique_together):
        self.name = name
        unique_together = normalize_together(unique_together)
        self.unique_together = set(tuple(cons) for cons in unique_together)
        # need None rather than an empty set to prevent infinite migrations
        # after removing unique_together from a model
        self.unique_together = set(tuple(cons) for cons in unique_together) or None

    def state_forwards(self, app_label, state):
        model_state = state.models[app_label, self.name.lower()]
        model_state.options["unique_together"] = self.unique_together
        model_state.options[self.option_name] = self.unique_together

    def database_forwards(self, app_label, schema_editor, from_state, to_state):
        old_apps = from_state.render()
@@ -234,8 +237,8 @@ class AlterUniqueTogether(Operation):
        if self.allowed_to_migrate(schema_editor.connection.alias, new_model):
            schema_editor.alter_unique_together(
                new_model,
                getattr(old_model._meta, "unique_together", set()),
                getattr(new_model._meta, "unique_together", set()),
                getattr(old_model._meta, self.option_name, set()),
                getattr(new_model._meta, self.option_name, set()),
            )

    def database_backwards(self, app_label, schema_editor, from_state, to_state):
@@ -245,7 +248,7 @@ class AlterUniqueTogether(Operation):
        return name.lower() == self.name.lower()

    def describe(self):
        return "Alter unique_together for %s (%s constraints)" % (self.name, len(self.unique_together))
        return "Alter %s for %s (%s constraints)" % (self.option_name, self.name, len(self.unique_together))


class AlterIndexTogether(Operation):
@@ -253,15 +256,18 @@ class AlterIndexTogether(Operation):
    Changes the value of index_together to the target one.
    Input value of index_together must be a set of tuples.
    """
    option_name = "index_together"

    def __init__(self, name, index_together):
        self.name = name
        index_together = normalize_together(index_together)
        self.index_together = set(tuple(cons) for cons in index_together)
        # need None rather than an empty set to prevent infinite migrations
        # after removing unique_together from a model
        self.index_together = set(tuple(cons) for cons in index_together) or None

    def state_forwards(self, app_label, state):
        model_state = state.models[app_label, self.name.lower()]
        model_state.options["index_together"] = self.index_together
        model_state.options[self.option_name] = self.index_together

    def database_forwards(self, app_label, schema_editor, from_state, to_state):
        old_apps = from_state.render()
@@ -271,8 +277,8 @@ class AlterIndexTogether(Operation):
        if self.allowed_to_migrate(schema_editor.connection.alias, new_model):
            schema_editor.alter_index_together(
                new_model,
                getattr(old_model._meta, "index_together", set()),
                getattr(new_model._meta, "index_together", set()),
                getattr(old_model._meta, self.option_name, set()),
                getattr(new_model._meta, self.option_name, set()),
            )

    def database_backwards(self, app_label, schema_editor, from_state, to_state):
@@ -282,7 +288,7 @@ class AlterIndexTogether(Operation):
        return name.lower() == self.name.lower()

    def describe(self):
        return "Alter index_together for %s (%s constraints)" % (self.name, len(self.index_together))
        return "Alter %s for %s (%s constraints)" % (self.self.option_name, self.name, len(self.index_together))


class AlterOrderWithRespectTo(Operation):
+38 −0
Original line number Diff line number Diff line
@@ -567,6 +567,44 @@ class AutodetectorTests(TestCase):
        self.assertEqual(action2.__class__.__name__, "AlterUniqueTogether")
        self.assertEqual(action2.unique_together, set([("title", "newfield")]))

    def test_remove_index_together(self):
        author_index_together = ModelState("testapp", "Author", [
            ("id", models.AutoField(primary_key=True)), ("name", models.CharField(max_length=200))
        ], {"index_together": [("id", "name")]})

        before = self.make_project_state([author_index_together])
        after = self.make_project_state([self.author_name])
        autodetector = MigrationAutodetector(before, after)
        changes = autodetector._detect_changes()
        # Right number of migrations?
        self.assertEqual(len(changes['testapp']), 1)
        migration = changes['testapp'][0]
        # Right number of actions?
        self.assertEqual(len(migration.operations), 1)
        # Right actions?
        action = migration.operations[0]
        self.assertEqual(action.__class__.__name__, "AlterIndexTogether")
        self.assertEqual(action.index_together, None)

    def test_remove_unique_together(self):
        author_unique_together = ModelState("testapp", "Author", [
            ("id", models.AutoField(primary_key=True)), ("name", models.CharField(max_length=200))
        ], {"unique_together": [("id", "name")]})

        before = self.make_project_state([author_unique_together])
        after = self.make_project_state([self.author_name])
        autodetector = MigrationAutodetector(before, after)
        changes = autodetector._detect_changes()
        # Right number of migrations?
        self.assertEqual(len(changes['testapp']), 1)
        migration = changes['testapp'][0]
        # Right number of actions?
        self.assertEqual(len(migration.operations), 1)
        # Right actions?
        action = migration.operations[0]
        self.assertEqual(action.__class__.__name__, "AlterUniqueTogether")
        self.assertEqual(action.unique_together, None)

    def test_proxy(self):
        "Tests that the autodetector correctly deals with proxy models"
        # First, we test adding a proxy model