由于字段默认值中的错误,无法进行 django 迁移?

django migration impossible because of bug in field default value?

我的一个 Django 模型遇到了一个奇怪的情况。 我正在使用 Django 1.10.3 和 python 3.5.2.

模型如下所示(为清楚起见进行了简化):

class Report(models.Model):
    date = models.FieldDate()
    def fieldA_default(self):
        return MyObject.objects.filter(date=self.date).count()
    fieldA = models.IntegerField(default=fieldA_default)

我有创建模型并添加字段的初始迁移,由 django 使用 ./manage.py makemigrations.

自动生成

我将此代码提交到我的 git 存储库,并将其部署到我的生产服务器,但实际上并未使用该模型(我的数据库中没有 Report 对象)。 我刚刚发现此代码不正确 (django set default value of a model field to a self attribute) 并决定改写 save()

但是当我将默认值从 fieldA_default 更改为 0 时,运行ning ./manage.py makemigrations 失败,因为它尝试 运行 旧的默认值函数fieldA_default。在尝试了几个选项之后,我最终决定完全删除该模型。但这也不起作用,因为 makemigrations 仍在尝试 运行 相同的功能。

这是 makemigrations 当我简单地删除模型时的回溯:

Traceback (most recent call last):
  File "./manage.py", line 22, in <module>
    execute_from_command_line(sys.argv)
  File "venv/lib/python3.5/site-packages/django/core/management/__init__.py", line 367, in execute_from_command_line
    utility.execute()
  File "venv/lib/python3.5/site-packages/django/core/management/__init__.py", line 359, in execute
    self.fetch_command(subcommand).run_from_argv(self.argv)
  File "venv/lib/python3.5/site-packages/django/core/management/base.py", line 294, in run_from_argv
    self.execute(*args, **cmd_options)
  File "venv/lib/python3.5/site-packages/django/core/management/base.py", line 345, in execute
    output = self.handle(*args, **options)
  File "venv/lib/python3.5/site-packages/django/core/management/commands/makemigrations.py", line 95, in handle
    loader = MigrationLoader(None, ignore_no_migrations=True)
  File "venv/lib/python3.5/site-packages/django/db/migrations/loader.py", line 52, in __init__
    self.build_graph()
  File "venv/lib/python3.5/site-packages/django/db/migrations/loader.py", line 197, in build_graph
    self.load_disk()
  File "venv/lib/python3.5/site-packages/django/db/migrations/loader.py", line 108, in load_disk
    migration_module = import_module("%s.%s" % (module_name, migration_name))
  File "venv/lib/python3.5/importlib/__init__.py", line 126, in import_module
    return _bootstrap._gcd_import(name[level:], package, level)
  File "<frozen importlib._bootstrap>", line 986, in _gcd_import
  File "<frozen importlib._bootstrap>", line 969, in _find_and_load
  File "<frozen importlib._bootstrap>", line 958, in _find_and_load_unlocked
  File "<frozen importlib._bootstrap>", line 673, in _load_unlocked
  File "<frozen importlib._bootstrap_external>", line 665, in exec_module
  File "<frozen importlib._bootstrap>", line 222, in _call_with_frames_removed
  File "xxx/reporting/migrations/0001_initial.py", line 9, in <module>
    class Migration(migrations.Migration):
  File "xxx/reporting/migrations/0001_initial.py", line 22, in Migration
    ('fieldA', models.IntegerField(default=reporting.models.Report.fieldA_default)),
AttributeError: module 'reporting.models' has no attribute 'Report'

我有几个问题:

why is Django running this "old" code even if I'm deleting the model?

因为迁移文件中仍然引用了模型及其方法0001_initial.py

But when I change the default from fieldA_default to 0, running ./manage.py makemigrations fails because it tries to run the old default value function fieldA_default.

我假设,在重置字段的默认值后,您删除了 - 现在已过时 - 方法 fieldA_default。如上所述,这个方法在初始迁移中被引用,现在肯定会中断。​​

how did I manage to get this invalid code into a migration without Django screaming at me?

创建迁移时,代码并非无效。模型上的某些更改不能通过简单的前向迁移来处理。在你的情况下:

  • 在迁移文件中引用和导入模型时删除模型(它本身只是另一个 python 模块,不能只导入 non-existing 类 )

  • 与删除默认方法相同。

当您的模型代码混乱或与您的 migrations/db 不同步并且 makemigrations 在当前状态下不起作用时,您可以做的一件事如下:

  1. python manage.py migrate app_name zero # undo all existing migrations of app
  2. 从应用程序中删除所有迁移文件。或者,如果数据库中已有有价值的数据,您可以撤消它们 one-by-one 并查看步骤 3 是否已经起作用

  3. python manage.py makemigrations app_name # new start from clean sheet

这在开发过程中很容易,可以被认为是迁移压缩的替代方法,但如果数据库中已经有生产数据,这显然是最后的选择。但在那种情况下,无论如何都应该小心谨慎地应用模型更改 :)

  1. 从您的迁移中手动删除对 Report 的所有引用
  2. 从数据库
  3. 中删除报告table
  4. 运行 python manage.py 迁移。它将为 Report 创建新的迁移。
  5. 推送、部署。