我正在尝试将一对多字段迁移到现有多对多字段。我来解释一下。
如果我完全理解发生了什么,这个问题会更短。不过,我会尽量提供所有细节,希望有人能发现问题。
为了将 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):
问题可能在于 value
是 PizzaTopping
,而 self.field.remote_field.model._meta.concrete_model
是 __fake__.PizzaTopping
。
这里引入 __fake__
的复杂性是什么?它是 ForwardManyToOneDescriptor 吗?是我在迁移吗?
如何将现有的 PizzaTopping 复制到此迁移代码中的注释中?