如何解决如何在 Django 迁移中将 ManyToMany 字段设置为现有对象?
我正在尝试将一对多字段迁移到现有多对多字段。我来解释一下。
如果我完全理解发生了什么,这个问题会更短。不过,我会尽量提供所有细节,希望有人能发现问题。
为了将 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)
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 复制到此迁移代码中的注释中?
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。