Подтвердить что ты не робот

Порядок размещения вложенных сериализаторов Django Rest-Framework

Есть ли способ заказать вложенный сериализатор _set, например order by pk или time-stamp.

Таким образом, в основном порядок song_set показан в json-данных ниже с самого последнего клика на последний созданный объект, в данном случае by order_by('-timestamp') или order_by('-pk').

Данные Json

{
    "pk": 151,
    "album_name": "Name",
    "song_set": [
         {
           pk: 3,
           timestamp: '5 seconds'
         },
         {
           pk: 2,
           timestamp: '10 seconds'
         },
         {
           pk: 1,
           timestamp: '15 seconds'
         }
    ]
}

Модель

class Album(models.Model):
    album_name     = models.CharField(max_length=100, blank=True)


class Song(models.Model):
    album          = models.ForeignKey('album.Album', default=1)
    timestamp      = models.DateTimeField(auto_now_add=True, auto_now=False)

Searilizer

class SongListSerializer(HyperlinkedModelSerializer):
    class Meta:
        model = Song
        fields = [
            'pk',
            'timestamp'
        ]

class AlbumSerializer(HyperlinkedModelSerializer):
    song_set = SongListSerializer(many=True, read_only=True)
    class Meta:
        model = Album
        fields = [
            'pk',
            'timestamp',
            'song_set'
        ]
4b9b3361

Ответ 1

Вы можете использовать SerializerMethodField и написать собственный метод для этого.

class AlbumSerializer(HyperlinkedModelSerializer):
    song_set = serializers.SerializerMethodField()
    class Meta:
        model = Album
        fields = [
            'pk',
            'timestamp',
            'song_set'
        ]

    def get_song_set(self, instance):
        songs = instance.song_set.all().order_by('-timestamp')
        return SongListSerializer(songs, many=True).data

Ответ 2

Старая ветка, но поскольку она все еще появляется в Google, я хочу поделиться своим ответом. Попробуйте перезаписать метод Serializer.to_representation. Теперь вы можете делать все, что захотите, в том числе настраивать сортировку ответов. В твоем случае:

class AlbumSerializer(HyperlinkedModelSerializer):
    song_set = SongListSerializer(many=True, read_only=True)
    class Meta:
        model = Album
        fields = [
            'pk',
            'timestamp',
            'song_set'
        ]

    def to_representation(self, instance):
        response = super().to_representation(instance)
        response["song_set"] = sorted(response["song_set"], key=lambda x: x["timestamp"])
        return response

Ответ 3

В вашем ViewSet, вы можете указать QuerySet с настраиваемым Prefetch объекта, который можно фильтровать и порядок, как вам нравится. Предварительная выборка вызывает только один дополнительный запрос к базе данных (вместо одного на родительский объект при использовании SerializerMethodField), что значительно повышает производительность.

from rest_framework import viewsets
from django.db.models import Prefetch

class AlbumViewSet(viewsets.ModelViewSet):
    queryset = Album.objects.prefetch_related(Prefetch('song_set',
        queryset=Song.objects.order_by('-timestamp')))