在Django Admin中保存M2M "Through" 内联

5 投票
1 回答
3462 浏览
提问于 2025-04-16 16:55

看起来Django的ModelAdmin和ModelForm不允许你在有中间表的ManyToManyField中使用save_m2m()。

这是models.py的内容:

from django.db import models


def make_uuid():
    import uuid
    return uuid.uuid4().hex


class MyModel(models.Model):
    id = models.CharField(default=make_uuid, max_length=32, primary_key=True)
    title = models.CharField(max_length=32)
    many = models.ManyToManyField("RelatedModel", through="RelatedToMyModel")

    def save(self, *args, **kwargs):
      if not self.id:
        self.id = make_uuid()
      super(GuidPk, self).save(*args, **kwargs)


class RelatedModel(models.Model):
    field = models.CharField(max_length=32)


class RelatedToMyModel(models.Model):
    my_model = models.ForeignKey(MyModel)
    related_model = models.ForeignKey(RelatedModel)
    additional_field = models.CharField(max_length=32)

这是admin.py的内容:

from django import forms
from django.contrib import admin

from .models import MyModel


class RelatedToMyModelInline(admin.TabularInline):
    model = MyModel.many.through


class MyModelAdminForm(forms.ModelForm):
    class Meta:
        model = MyModel


class MyModelAdmin(admin.ModelAdmin):
    form = MyModelAdminForm
    inlines = (RelatedToMyModelInline, )


admin.site.register(MyModel, MyModelAdmin)

如果我先保存MyModel,然后通过内联添加一个新的相关中间模型,这样是没问题的。但是如果我在添加新MyModel的数据时同时设置内联,就会出现Django Admin的错误提示“请修正下面的错误。”但下面没有任何内容被标记出来。

我该如何先保存MyModel,然后再保存内联的中间模型呢?显然Django在保存MyModel后可以保存中间模型,所以我只是想找到一个可以实现这个的办法。我尝试重写表单的save()方法,在调用instance.save()后调用save_m2m(),但显然这对有中间表的ManyToMany关系不起作用。

我使用的是Django 1.2,但在1.3中这个问题依然存在。

更新:我做了一个像上面那样的测试应用来隔离问题,结果发现它按预期工作,正确地在保存MyModel对象后保存M2M中间对象……只要我让Django在运行python manage.py syncdb时自动创建MyModel.id字段——一旦我添加了GUID id字段,它就不再工作了。

这越来越像是Django的一个bug。

1 个回答

0

在你的 MyModelAdmin 里,你可以试着重写 save_formset 这个方法。这样一来,你就可以选择保存的顺序了。

撰写回答