更改时迁移依赖模型 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,它认为现有模型没有改变。
我可能看不到全貌,但我用不同的现有模型和新模型对其进行了测试,它对我有用。如果有人告诉我为什么这个解决方案不好,我将不胜感激。
我正在使用 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,它认为现有模型没有改变。
我可能看不到全貌,但我用不同的现有模型和新模型对其进行了测试,它对我有用。如果有人告诉我为什么这个解决方案不好,我将不胜感激。