Cómo actualizar muchos a muchos Django RESTO?

votos
2

¿Cómo hago para actualizar una relación de muchos a muchos en el marco de Django RESTO?

Aquí está mi 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')

Aquí está mi serializador:

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

¿Cómo logro actualización subjects? Actualmente, sólo puedo añadir y eliminar los sujetos no existentes.

Publicado el 19/12/2018 a las 14:09
fuente por usuario
En otros idiomas...                            


2 respuestas

votos
0

Aquí es una solución. Deseo sin embargo alguien podría publicar una mejor respuesta. La idea aquí es eliminar todos los sujetos primero y luego añadirlos. Sin embargo, dado que, por supuesto, no podemos hacerlo teacher.subjects.all().delete(), podemos hacerlo

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

    for subject in subjects:
        teacher.subjects.add(subject)
Respondida el 19/12/2018 a las 16:30
fuente por usuario

votos
0

Como estamos hablando de muchos-a-muchos, ambos objetos tienen que ser mantenido para poder asignar el uno al otro.

Una vez conseguido eso, sólo tiene que añadir una a la colección con respecto de la otra, como esto (fuente: 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)

O bien, en un solo paso:

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

No he encontrado la manera de expresar "contiene", sin embargo, pero el siguiente paso es eliminar los temas que no están en la nueva lista y añadir los que se encuentran en la nueva lista, pero no la edad. Probablemente más eficiente con listas ordenadas o mejores valores hash. Pero mi pitón no es lo suficientemente bueno para eso.

Respondida el 19/12/2018 a las 14:55
fuente por usuario

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