/ / Publikowanie relacji jeden do wielu - django, jeden do wielu, django-rest-framework

Publikowanie relacji jeden-do-wielu - django, jeden-do-wielu, framework django-rest

Próbuję udostępnić API do mojego modelu Django za pośrednictwem frameworka Django REST.

Mam obiekt Observation. Obserwacja może zawierać wiele rzeczy, które zostały zaobserwowane. Więc przedstawiłem to w ten sposób:

class Observation(models.Model):

photo_file = models.ImageField( upload_to=img_dir,   blank=True, null=True )
titestamp = models.DateTimeField(blank=True, null=True)
latitude = models.FloatField()
longitude = models.FloatField()


class ObservedThing(models.Model):
thing = models.ForeignKey(Thing) # the thing being observed
observation = models.ForeignKey(Observation, related_name="observed_thing")
value = models.FloatField()

Jak rozumiem, jest to relacja jeden do wielu.

Mam teraz widok API:

class ObsvList(generics.ListCreateAPIView):
"""
API endpoint that represents a list of observations.
"""
model = Observation
serializer_class = ObsvSerializer

i odpowiedni serializator:

class ObsvSerializer(serializers.ModelSerializer):

observed_thing = serializers.PrimaryKeyRelatedField(many=True)

class Meta:
model = Observation

Co muszę zrobić, aby móc OPUBLIKOWAĆ obserwację, w której wykryto kilka rzeczy? Nie mogę tego rozgryźć. Wielkie dzięki.

Odpowiedzi:

8 dla odpowiedzi № 1

(odpowiedź mniej więcej skopiowana z innego podobne, ale mniej jasne pytanie)

Aby utworzyć wiele powiązanych obiektów w jednym POST, wymaga zapisywalne zagnieżdżone serializatory które nie są jeszcze dostępne.

Pełne wsparcie to Praca w toku, ale w międzyczasie jednym (hackerskim) rozwiązaniem jest zastąpienie create metoda na widoku w każdym przypadku:

class FooListCreateView(ListCreateAPIView):
model = Foo
serializer_class = FooSerializer

def create(self, request, *args, **kwargs):
data=request.DATA

f = Foo.objects.create()

# ... create nested objects from request data ...

# ...
return Response(serializer.data,
status=status.HTTP_201_CREATED,
headers=headers)

Prawdopodobnie nie jest idealny, ale działa dla mnie, dopóki nie nadejdzie właściwa droga.

Inną opcją jest utworzenie powiązanego pliku Observation obiekty indywidualnie z oddzielnymi POST i użycie PrimaryKeyRelatedField lub HyperlinkedRelatedField zrobić skojarzenia w finale ObservedThing POCZTA.


4 dla odpowiedzi nr 2

Wiem, że ten wątek ma już odpowiedź, ale jaZacząłem pracować nad rozwiązaniem tego problemu, a ponieważ ten wpis był jedną z moich inspiracji, chciałbym podzielić się moim ostatecznym rozwiązaniem. To może być przydatne dla kogoś. Mam modele, więc klasa nadrzędna:

#parent model class
class Parent(models.Model):

id = models.AutoField(primary_key=True)
field = models.CharField(max_length=45)

class Meta:
managed = False
db_table = "parent"

następnie klasa podrzędna:

#child model class
class Child(models.Model):

id = models.AutoField(primary_key=True)
field = models.CharField(max_length=45)
parent = models.ForeignKey(Parent, related_name="children")

class Meta:
managed = False
db_table = "child"

Musiałem zdefiniować serializatory, ponieważ tego nie zrobiłemchcę utworzyć adres URL dostępny dla routera, aby bezpośrednio zarządzać obiektami Children, ale chciałem je utworzyć za pomocą ModelViewSet nadrzędnego ModelViewSet, potrzebowałem tego:

class ChildSerializer(serializers.ModelSerializer):
class Meta:
model = Child
read_only_fields = ("id",)

class ParentSerializer(serializers.ModelSerializer):
class Meta:
model = Banner
read_only_fields = ("id",)

class ParentSerializerNested(ParentSerializer):
children = ChildSerializer(many=True)

Byłem wtedy gotowy, aby utworzyć ModelViewSet, nadpisując / rozszerzając mieszanki tworzenia / aktualizowania i uczynić go ogólnym w celu ponownego użycia w innych przypadkach:

class ParentChildViewSet(viewsets.ModelViewSet):

def create(self, request, *args, **kwargs):
serializer = self.serializer_parent(data=request.DATA,
files=request.FILES)

try:
if serializer.is_valid():
with transaction.commit_on_success():
self.pre_save(serializer.object)
parent = serializer.save(force_insert=True)
self.post_save(parent, created=True)

# need to insert children records
for child in request.DATA[self.child_field]:
child[self.parent_field] = parent.id
child_record = self.serializer_child(data=child)
if child_record.is_valid():
child_record.save(force_insert=True)
else:
raise ValidationError("Child validation failed")

headers = self.get_success_headers(serializer.data)

serializer.data[self.child_field] = self.serializer_child(
self.model_child.objects.filter(
**{self.parent_field: parent.id}).all(),
many=True).data
return Response(serializer.data,
status=status.HTTP_201_CREATED,
headers=headers)
except ValidationError:
pass
return Response(serializer.errors, status=status.HTTP_400_BAD_REQUEST)

Mogę więc użyć go ponownie dla każdego zagnieżdżonego przypadku relacji, który mam w mojej aplikacji w następujący sposób:

class ParentViewSet(ParentChildViewSet):
child_field = "children"
parent_field = "parent"
model = Parent
model_child = Child
serializer_class = ParentSerializerNested
serializer_parent = ParentSerializer
serializer_child = ChildSerializer

I na koniec routing:

router = routers.DefaultRouter()
router.register(r"parents", ParentViewSet)

To działa jak urok!


1 dla odpowiedzi nr 3
thing = models.ManyToManyField("Thing")

musisz użyć relacji wiele do wielu, aby utworzyć tymczasową tabelę, która będzie przechowywać klucze i automatycznie kojarzyć dane.