Como atualizar muitos para muitos Django resto?

votos
2

Como eu iria sobre a atualização de muitos para muitos relação no framework Django resto?

Aqui é o meu modelo.

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')

Aqui é a minha serializer:

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

Como faço para conseguir a actualização subjects? Atualmente, eu só pode adicionar e não excluir temas já existentes.

Publicado 19/12/2018 em 14:09
fonte usuário
Em outras línguas...                            


2 respostas

votos
0

Aqui está uma solução alternativa. Eu desejo no entanto alguém postar uma resposta melhor. A idéia aqui é para apagar todos os assuntos primeiro, em seguida, adicioná-los. No entanto, desde é claro que não podemos fazer teacher.subjects.all().delete(), nós podemos fazer

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

    for subject in subjects:
        teacher.subjects.add(subject)
Respondeu 19/12/2018 em 16:30
fonte usuário

votos
0

Como estamos a falar de muitos-para-muitos, ambos os objetos precisam ser persistentes antes que você pode atribuí-los um ao outro.

Uma vez que você tem isso, você pode simplesmente adicionar um para o respeito da coleta do outro, como este (fonte: 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)

Ou, em uma única etapa:

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

Eu não descobri como expressar "contém" ainda, mas o próximo passo é remover os temas que não estão na nova lista e adicione os que estão no novo, mas não a lista de idade. Provavelmente mais eficiente com listas ordenadas ou melhores hashes. Mas o meu python não é bom o suficiente para isso ainda.

Respondeu 19/12/2018 em 14:55
fonte usuário

Cookies help us deliver our services. By using our services, you agree to our use of cookies. Learn more