如何在 Django 迁移中将 ManyToMany 字段设置为现有对象?

时间:2021-06-30 18:25:16

标签: django django-models django-migrations

我正在尝试将一对多字段迁移到现有多对多字段。我来解释一下。

如果我完全理解发生了什么,这个问题会更短。不过,我会尽量提供所有细节,希望有人能发现问题。

为了将 1 对多迁移到新的多对多,我阅读了 this blog post(也喜欢 this question),这很有效。总结一下:

他们从一对多模型开始(披萨只有一个配料):

from django.db import models

class Topping(models.Model):
    name = models.CharField(max_length=20)

class Pizza(models.Model):
    name = models.CharField(max_length=20)
    topping = models.ForeignKey(Topping)

并展示了每个披萨从一个浇头到多个浇头的迁移。

在三个迁移中的第二个中,他们在迁移的 forward 方法中向新的多对多浇头字段添加浇头,如下所示:

def forward(apps, schema_editor):
    Pizza = apps.get_model("pizza", "Pizza")
    for pizza in Pizza.objects.all():
        pizza.toppings.add(pizza.topping)

但是,现在我正在尝试迁移不同的模型以使用现有的浇头add 不对,因为我不想创建新关系,我想使用现有关系。

这是扩展比萨示例,但假设我有一些比萨饼的配料说明(我不知道,卡路里?)。当每个披萨只有一个配料时,这很有效,但是现在有多个配料,我希望该注释移动到比萨饼配料而不是比萨饼。类似的东西:

class Note(models.Model):
  # We're planning to get rid of pizza after copying
  pizza = models.ForeignKey(Pizza)
  # I named the many-to-many field with a through object called PizzaTopping.
  # null=True because we want to migrate existing values into this field
  topping = db.ForeignKey(PizzaTopping, on_delete=db.CASCADE, null=True)

现在我有一个 Note 的迁移:

    def forward(apps, schema_editor):
        Note = apps.get_model('pizza', 'Note')
        for note in Note.objects.all():
            # Find the existing pizza toppings for this Note
            # There's only one, since we just migrated pizza topping to toppings
            pizza_topping = PizzaTopping.objects.get(pizza_id=note.pizza)
            note.topping = pizza_topping

我收到一个错误:

ValueError: Cannot assign "<PizzaTopping: PizzaTopping object (606)>":
  "Note.topping" must be a "PizzaTopping" instance.

这很令人困惑,这些类型看起来都一样!但是,我在 Django 的 related_descriptor.py 对象的 ForwardManyToOneDescriptor 对象、__set__ 方法中找到了一行:

        # An object must be an instance of the related class.
        if value is not None and not isinstance(
            value, self.field.remote_field.model._meta.concrete_model):

问题可能在于 valuePizzaTopping,而 self.field.remote_field.model._meta.concrete_model__fake__.PizzaTopping

这里引入 __fake__ 的复杂性是什么?它是 ForwardManyToOneDescriptor 吗?是我在迁移吗?

如何将现有的 PizzaTopping 复制到此迁移代码中的注释中?

0 个答案:

没有答案