Как обновить многие для многих Django REST? - PullRequest
0 голосов
/ 19 декабря 2018

Как мне обновлять отношения многие ко многим в DEST-среде Django?

Вот моя модель.

class SchoolTeacher(AbstractBase):
    school = models.ForeignKey(School, on_delete=models.CASCADE, 
         related_name='teachers')
    user = models.ForeignKey(User, on_delete=models.CASCADE, 
         related_name='teacher_at',)
    subjects = models.ManyToManyField(SchoolSubject, 
        related_name='teachers')

Вот мой сериализатор:

class SchoolTeacherSerializer(serializers.ModelSerializer):
   ....
   def create(self, validated_data):
    school_class = validated_data.get('school_class', None)
    stream = validated_data.get('stream', None)

    school_teacher_model_fields = [
        f.name for f in SchoolTeacher._meta.get_fields()]
    valid_teacher_data = {
        key: validated_data[key]
        for key in school_teacher_model_fields
        if key in validated_data.keys()
    }
    subjects = valid_teacher_data.pop('subjects')
    teacher = SchoolTeacher.objects.create(**valid_teacher_data)

    for subject in subjects:
        teacher.subjects.add(subject)

    self.add_class_teacher(stream, school_class, teacher)

    return teacher

def update(self, instance, validated_data):
    subjects = validated_data.pop('subjects')
    school_class = validated_data.get('school_class', None)
    stream = validated_data.get('stream', None)
    teacher = instance

    for (key, value) in validated_data.items():
        setattr(teacher, key, value)

    for subject in subjects:
        teacher.subjects.add(subject)

    teacher.save()

    return teacher

Как мне добиться обновления subjects?В настоящее время я могу только добавлять, а не удалять существующие темы.

Ответы [ 2 ]

0 голосов
/ 19 декабря 2018

Вот обходной путь.Однако я хотел бы, чтобы кто-нибудь опубликовал лучший ответ.Идея здесь состоит в том, чтобы сначала удалить все предметы, а затем добавить их.Однако, поскольку мы, конечно, не можем сделать teacher.subjects.all().delete(), мы можем сделать

    for existing_subject in teacher.subjects.all():
        teacher.subjects.remove(existing_subject)

    for subject in subjects:
        teacher.subjects.add(subject)
0 голосов
/ 19 декабря 2018

Поскольку мы говорим о многих ко многим, оба объекта должны быть сохранены, прежде чем вы сможете назначить их друг другу.

Получив это, вы можете просто добавить один в соответствующую коллекциюдругой, как этот (источник: https://docs.djangoproject.com/en/2.1/topics/db/examples/many_to_many/):

a1 = Article(headline='Django lets you build Web apps easily')
a1.save()
p1 = Publication(title='The Python Journal')
p1.save()
a1.publications.add(p1)

Или, в один шаг:

new_publication = a1.publications.create(title='Highlights for Children')

Я еще не нашел, как выразить «содержит», ноСледующий шаг - удалить темы, которых нет в новом списке, и добавить те, которые есть в новом, но не в старом списке. Вероятно, более эффективный с отсортированными списками или лучшими хэшами. Но мой питон еще недостаточно хорош для этого.

...