更改时迁移依赖模型 DEFAULT_AUTO_FIELD

Migrating models of dependencies when changing DEFAULT_AUTO_FIELD

我正在使用 Django 3.2。我已将此行添加到 settings.py:

DEFAULT_AUTO_FIELD = 'django.db.models.BigAutoField'

然后我 运行 这些命令:

$ python manage.py makemigrations
$ python manage.py migrate

makemigrations 命令为我的应用程序创建新的迁移文件,不仅是我创建的应用程序,还包括我的依赖项。例如,我正在使用 django-allauth,这个文件是在我的虚拟环境 (virtualenv) 中创建的:

 .venv/lib/python3.8/site-packages/allauth/account/migrations/0003_auto_20210408_1526.py

此文件未随 django-allauth 一起提供。当我从 git 部署此应用程序时,不包括此文件。

我应该怎么做?如何切换 DEFAULT_AUTO_FIELD 而无需为 django-allauth 之类的依赖项创建新的迁移文件?

理想情况下,您的第三方依赖项应在 apps.py:

中的配置中包含此行
from django.apps import AppConfig

class ExampleConfig(AppConfig):
    default_auto_field = 'django.db.models.AutoField'

在等待上游依赖项更新其 apps.py 或迁移文件时,您可以自己覆盖应用程序配置。如果它不存在,请在您的主应用程序目录中创建一个 apps.py 文件(例如:project/apps.py),并覆盖依赖项的配置。在这个例子中,我覆盖了 django-allauth:

的配置
from allauth.account.apps import AccountConfig
from allauth.socialaccount.apps import SocialAccountConfig

class ModifiedAccountConfig(AccountConfig):
    default_auto_field = 'django.db.models.AutoField'

class ModifiedSocialAccountConfig(SocialAccountConfig):
    default_auto_field = 'django.db.models.AutoField'

然后将 settings.py 中的 INSTALLED_APPS 修改为如下所示,在此示例中替换 django-allauth 的旧条目:

INSTALLED_APPS = [
    # ....
    # replace: "allauth.account", with
    "projectname.apps.ModifiedAccountConfig",
    # replace: "allauth.socialaccount", with
    "projectname.apps.ModifiedSocialAccountConfig",
]

如果依赖项没有要覆盖的 apps.py 文件,您仍然可以在 project/apps.py 中创建一个 AppConfig 子 class,如下所示:

from django.apps import AppConfig

class ModifiedExampleDependencyConfig(AppConfig):
    name = 'exampledependency' # the python module
    default_auto_field = 'django.db.models.AutoField'

现在,当您 运行 python manage.py makemigrations 时,不应为依赖项创建迁移文件。

我在做一个大项目,我们从 2.2 升级了 Django。到 3.2,然后需要使用大整数 (Int8) (PostgreSQL) 字段而不是默认整数 (Int4) 创建所有新模型。

当我在settings.py中定义它时:

DEFAULT_AUTO_FIELD = 'django.db.models.BigAutoField'

我遇到了同样的问题,但有自己的应用程序 - Django 试图让我迁移我拥有的 135 个模型,但我不想这样做。我只想用 BigInt 创建新模型并手动操作旧模型。

我找到了下一个解决方案。我将字段更改为自定义:

DEFAULT_AUTO_FIELD = 'project.db.models.CustomBigAutoField'

然后覆盖它的解构:

from django.db import models

class CustomBigAutoField(models.BigAutoField):
    """Int8 field that is applied only for new models."""

    def deconstruct(self):
        name, path, args, kwargs = super().deconstruct()
        if getattr(self, 'model', None):
            path = 'django.db.models.AutoField'
        return name, path, args, kwargs

据我发现,新模型的字段没有对其模型的反向引用,因此 path 不会被它们覆盖。 我们覆盖 path 是因为 Django 检查模型是否被键更改,其中包括该字段的路径。所以我们欺骗 Django,它认为现有模型没有改变。

我可能看不到全貌,但我用不同的现有模型和新模型对其进行了测试,它对我有用。如果有人告诉我为什么这个解决方案不好,我将不胜感激。