Django 外键休息框架

时间:2021-02-10 10:40:27

标签: python django django-rest-framework

有什么办法可以在视图的“likedBy”部分显示用户名而不是用户名?使用 Django 休息框架 从我得到的观点来看,忽略评论:

[
    {
        "id": 3,
        "title": "ddsa",
        "content": "dsadsa",
        "created": "2021-02-10T08:07:42.758400Z",
        "updated": "2021-02-10T08:07:42.758400Z",
        "author": 1,
        "category": [
            {
                "pk": 1,
                "name": "Life"
            }
        ],
        "likedBy": [
            1
        ],
        "comments": [
            {
                "id": 2,
                "content": "ghfa",
                "created": "2021-02-10T08:08:02.407950Z",
                "post": 3,
                "author": 1,
                "likedBy": [
                    1
                ]
            }
        ]
    }
]

Views.py:

class PostViewSet(FlexFieldsMixin, generics.ListAPIView):
    queryset = Post.objects.all()
    serializer_class = PostSerializer
    permit_list_expands = ['category', 'comments']

模型.py


class Post(models.Model):
    title = models.CharField(max_length=255)
    content = models.TextField()
    category = models.ManyToManyField(Category, related_name='posts')
    author = models.ForeignKey(User, on_delete=models.CASCADE)
    created = models.DateTimeField(auto_now_add=True)
    updated = models.DateTimeField(auto_now=True)
    likedBy = models.ManyToManyField(User, related_name='posts', blank=True)

    class Meta:
        ordering = ['-created']

Serializers.py:


class PostSerializer(FlexFieldsModelSerializer):
    class Meta:
        model = Post
        fields = '__all__'
        expandable_fields = {
            'category': ('blogApi.CategorySerializer', {'many': True}),
            'comments': ('blogApi.CommentSerializer', {'many': True}),
        }

如何序列化 ManyToMany 字段以显示文本值

3 个答案:

答案 0 :(得分:0)

鉴于您没有序列化关系,而是序列化与用户相关的模型属性,我相信您必须使用 serializer.SerializerMethodField()。这允许您执行以下操作:

class PostSerializer(FlexFieldsModelSerializer):
    liked_by = serializers.SerializerMethodField(method_name="get_user_likes")

    class Meta:
        model = Post
        fields = (
            "title",
            "content",
            "category",
            "author",
            "created",
            "update",
            "liked_by"
        )
        expandable_fields = {
            'category': ('blogApi.CategorySerializer', {'many': True}),
            'comments': ('blogApi.CommentSerializer', {'many': True}),
        }

    @classmethod
    def get_user_likes(obj):
       # do whatever you like here, but I tend to call a 
       # method on my model to keep my serializer file
       # nice and tidy
       # you'll need to define a UserSerializer
       return UserSerializer(obj.get_user_likes(), many=True, read_only=True)

在你的 Post 模型中:

class Post(models.Model):
    title = models.CharField(max_length=255)
    content = models.TextField()
    category = models.ManyToManyField(Category, related_name='posts')
    author = models.ForeignKey(User, on_delete=models.CASCADE)
    created = models.DateTimeField(auto_now_add=True)
    updated = models.DateTimeField(auto_now=True)
    likedBy = models.ManyToManyField(User, related_name='posts', blank=True)

    class Meta:
        ordering = ['-created']

    def get_user_likes(self):
        return self.likedBy.all()

您当然可以在序列化器中定义完整的方法,但正如我所说,我喜欢保持序列化器尽可能干净,并将与模型关联的所有方法放在我的 models.py 中。

    @classmethod
    def get_user_likes(obj):
       # you'll need to define a UserSerializer
       return UserSerializer(obj.likedBy.all(), many=True, read_only=True)

答案 1 :(得分:0)

所以你可以设置

likedBy = serializers.ReadOnlyField(source='get_likedBy')

在你的 PostSerializer 类中 并在您的 Post 模型类中定义函数,如下所示:

@property
def get_likedBy(self):
   liked_by = []
   for user in self.users_liked_post.all():
       liked_by.append(user.name)
   return liked_by

使用正确的related_name 而不是users_liked_post

答案 2 :(得分:0)

如果您将 likeBy 添加到您的可扩展字段,然后将 ?expand=likedBy 添加到您的 url,它应该会提供您在 UserSerializer 中概述的所有信息,或者编写一个名为 {{ 1}}。同样作为一般规则,尽量不要使用 LikedBySerializer,这是泄漏数据的好方法。快乐编码!

'__all__'