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.