恢复已删除的 Django 迁移

Recover deleted Django migrations

在我的生产服务器上,我不小心删除了我的一个应用程序中的迁移目录。多个人在应用程序上工作,所以在制作应用程序时,我们所做的是在不在本地进行任何迁移的情况下推送我们的模型更改,而不是在生产服务器上进行迁移,然后迁移到那里。我们这样做是因为我们过去遇到过迁移合并问题,并且认为这可能是一个很好的解决方案。由于没有迁移文件夹的本地副本,我担心它永远消失了。我的问题是这样的。我可以在 Webfaction 中访问 phpPgAdmin,并且数据库有一个 django_migrations table。我认为一个解决方案是找到最新的迁移,例如 0009_migration.py,而不是简单地重命名一个新的迁移文件,0010_migration.py 用于我的下一次迁移,如果我将来进行任何模型更改的话在那个应用程序中。这样我就可以简单地 运行 迁移,而且它只会被认为是尚未 运行、0010_migration.py 的迁移。但是出于好奇,是否有一些命令可以查看您的 PostgreSQL 数据库,并根据它在 django_migrations table 中存档的迁移记录在您的应用程序迁移目录中创建迁移文件?

我想问这个问题的一种更简单的方法是“有没有办法将 django_migrations table 行逆向工程到 migrations.py 文件中?" 这是我在 PostgreSQL 中的 django_migration table 行的图像,以及我希望文件看起来像什么,只是使用不同的命令。

Django 迁移

class Migration(migrations.Migration):

    dependencies = [
        ('portfolio', '0001_initial'),
    ]

    operations = [
        migrations.AddField(
            model_name='project',
            name='contact_form',
            field=models.BooleanField(default=False),
        ),
    ]

PostgreSQL django_migrations table 行

1) 从数据库的迁移 table 中删除迁移文件已删除的那个应用程序的所有内容。

2) $ python manage.py makemigrations <app>

这会创建 1 个新的迁移文件(初始迁移)。

3) $ python manage.py migrate <app> --fake

将初始迁移写入迁移 table 但不触及任何其他 table(实际上不迁移)。

只要迁移文件被删除的应用程序没有发生任何变化,这就应该有效。

这可能会解决意外删除迁移文件的问题(假设它们是由 manage.py 自动创建且未被操纵)。但我不知道重构迁移是否能够或是否值得推荐 - 它也可能导致适应迁移文件的自动创建,需要在它之后手动操作任何迁移文件是自动创建的,甚至需要覆盖 django 后端。

编辑

评论中补充问题的回答:

  • 初始 $ python manage.py makemigrations <app> 创建一个迁移文件,描述给定应用程序中所有模型的当前架构($ python manage.py makemigrations 对所有应用程序执行此操作)。
  • 初始$ python manage.py migrate <app>将给定应用程序的所有迁移文件应用到数据库并写入您的迁移table哪个 迁移文件被应用。在初始迁移时,将创建具有所需属性和关系的所有需要​​的数据库 table($ python manage.py migrate 对所有应用程序执行此操作)。
  • 以下 $ python manage.py makemigrations <app> 将创建描述 差异lates 迁移文件的迁移文件(因此最新的描述此应用中模型的模式)和此应用中模型的当前模式。
  • 数据库的迁移table仅包含有关哪些迁移文件的信息哪些应用程序应用于数据库.
  • 以下 $ python manage.py migrate <app> 将使用该迁移 table 来检查哪些迁移文件已应用于数据库。如果有未应用的迁移文件,它们将被应用然后写入迁移 table.

因此,如果您删除应用程序的迁移文件,您只需删除该应用程序模型架构的 描述历史记录。迁移中的条目也是如此 table:您只需删除有关 哪个 迁移文件应用于您的数据库的信息。

  • $ manage.py migrate <app> --fake 还会检查哪些迁移文件必须应用于您的数据库,并将所有要应用的迁移文件写入您的迁移 table。但是此命令不会将它们应用到您的数据库。

所以我建议的实际解决方案如下:

  1. 删除有关某个特定应用程序将哪些迁移文件应用于您的数据库的信息
  2. 为那个应用程序创建一个新的迁移文件
  3. 将迁移文件写入迁移table但不应用它。 << 这就是为什么自从删除迁移文件后您的应用模型不能有任何变化。

为什么需要删除迁移中的现有条目table?

正如您在问题中提供的图像中看到的那样,迁移中的条目存储了任何应用的迁移文件的名称。不删除这些条目会导致 manage.py migrate 假设已经应用了具有相同名称(或编号 - 我不是 100% 知道这一点)的迁移文件。所以这些将在迁移时被跳过。