python - 返回带有序列化程序列表的响应 Django REST Framework

标签 python django django-rest-framework

我正在使用 Django 和 DjangoRestFramework 为二手销售应用程序编写一些后端软件。现在,我正在尝试发送一个包含产品列表的 Response 对象,但我似乎无法返回实际的产品列表,因为我收到一条错误消息

ListSerializer is not JSON serializable.

我已经尝试过像这样使用序列化器构造函数:

ProductoSerializer(products, many=True)

然后创建 ProductoSerializer.data 列表,然后用它创建 Response 对象。

这是我正在使用的序列化程序:

class UserSerializer(serializers.HyperlinkedModelSerializer):
    ciudad = serializers.SerializerMethodField()
    conectado = serializers.SerializerMethodField()
    class Meta:
        model = Usuario
        fields = ('uid', 'nombre', 'ciudad', 'conectado')
    def get_ciudad(self, obj):
        geolocator = Nominatim(user_agent="bookalo")
        location = geolocator.reverse(str(obj.latitud_registro) + ',' + str(obj.longitud_registro))
        return location.raw['address']['city']

    def get_conectado(self, obj):
        ahora = timezone.now()
        result = relativedelta(ahora, obj.ultima_conexion)
        return result.days == 0 and result.hours == 0 and result.months == 0 and result.years == 0 and result.minutes < 5

class TagSerializer(serializers.HyperlinkedModelSerializer):
    class Meta:
        model = Tag
        fields = ('nombre')

class MultimediaSerializer(serializers.HyperlinkedModelSerializer):
    contenido_url = serializers.SerializerMethodField()
    class Meta:
        model = ContenidoMultimedia
        fields = ('contenido_url', 'orden_en_producto')

    def get_contenido_url(self, obj):
        return obj.contenido.url

class MiniProductoSerializer(serializers.HyperlinkedModelSerializer):
    contenido_multimedia = serializers.SerializerMethodField()
    class Meta:
        model = Producto
        fields = ('nombre', 'precio', 'estado_venta', 'contenido_multimedia')

    def get_contenido_multimedia(self, obj):
        contenido = ContenidoMultimedia.objects.get(producto=obj.pk, orden_en_producto=0)
        return MultimediaSerializer(contenido)

class ProductoSerializer(serializers.HyperlinkedModelSerializer):
    vendido_por = UserSerializer(read_only=True)
    tiene_tags = TagSerializer(many=True, read_only=True)
    contenido_multimedia = serializers.SerializerMethodField()
    valoracion_media_usuario = serializers.SerializerMethodField()
    class Meta:
        model = Producto
        fields = ('nombre', 'precio', 'estado_producto', 'estado_venta', 'latitud', 'longitud', 'tipo_envio', 'descripcion', 'vendido_por', 'tiene_tags', 'num_likes', 'contenido_multimedia')

    def get_contenido_multimedia(self, obj):
        contenido = ContenidoMultimedia.objects.filter(producto=obj.pk).order_by('orden_en_producto')
        return MultimediaSerializer(contenido, many=True)

    def get_valoracion_media_usuario(self, obj):
        return Usuario.objects.get(pk=obj.vendido_por).media_valoraciones

class ValidacionEstrellaSerializer(serializers.HyperlinkedModelSerializer):
    usuario_que_valora = UserSerializer(read_only=True)
    producto_asociado = serializers.SerializerMethodField()
    class Meta:
        model = ValidacionEstrella
        fields = ('estrellas', 'comentario', 'timestamp', 'usuario_que_valora', 'producto_asociado')

    def get_producto_asociado(self, obj):
        producto = Producto.objects.get(pk=obj.producto)
        return MiniProductoSerializer(producto)

class UserProfileSerializer(serializers.HyperlinkedModelSerializer):
    usuario_valorado_estrella = serializers.SerializerMethodField()
    productos_favoritos = serializers.SerializerMethodField()
    class Meta:
        model = Usuario
        fields = ('uid', 'nombre', 'esta_baneado', 'usuario_valorado_estrella', 'producto_del_usuario')

    def get_usuario_valorado_estrella(self, obj):
        validaciones = ValidacionEstrella.objects.filter(usuario_valorado=obj.pk).order_by('-timestamp')
        return ValidacionEstrellaSerializer(validaciones, many=True, read_only=True)

    def get_productos_favoritos(self, obj):
        favoritos = Producto.objects.filter(le_gusta_a__in=[obj.pk])
        return ProductoSerializer(favoritos, many=True, read_only=True)

class ReportSerializer(serializers.HyperlinkedModelSerializer):
    #usuario_reportado = serializers.SerializerMethodField()
    usuario_reportado = UserSerializer(read_only=True)
    class Meta:
        model = Report
        fields = ('usuario_reportado', 'causa')

这是我尝试编写的 views.py 函数:

@api_view(['POST'])
@permission_classes((permissions.AllowAny,))
def SearchProduct(request, format=None):
    if request.method != 'POST':
        return Response(status=status.HTTP_400_BAD_REQUEST)
    preposiciones = ['a','ante','bajo','cabe','con','contra','de','desde','en','entre',
    'hacia','hasta','para','por','segun','sin','so','sobre','tras']
    try:
        search = request.POST.get('busqueda')
    except:
        return Response(status=status.HTTP_404_NOT_FOUND)
    products = Producto.objects.none()
    for word in search.split():
        if word not in preposiciones:
            productos_palabra = Producto.objects.filter(nombre__contains=word)
            products = products | productos_palabra
    products.distinct()
    product_list = []
    for prod in products:
        product_list.append(ProductoSerializer(prod).data)
    return Response({'productos': product_list}, status=status.HTTP_200_OK)

我正在使用请求对象,因为我还必须为具有相同功能的网页提供服务,而不仅仅是移动应用程序(尽管网页部分仍未编码)。

它应该返回包含至少一个用户搜索词的所有产品,并且所有产品都应该基于 ProductoSerializer 对象进行结构化,但由于某种原因,它输出了该错误而且我不太确定如何修复它。

提前致谢,如果您需要任何我遗漏的额外信息,请务必提出要求...今天是漫长的一天,我可能遗漏了什么。

最佳答案

似乎当您使用 SerializerMethodField 时,您返回的是序列化程序实例,但不是 data:

例如:

contenido_multimedia = serializers.SerializerMethodField()


def get_contenido_multimedia(self, obj):
        contenido = ContenidoMultimedia.objects.filter(producto=obj.pk).order_by('orden_en_producto')
        return MultimediaSerializer(contenido, many=True).data  # <-- here try to add .data

所有 SerializerMethodField 方法都应该更改它。

关于python - 返回带有序列化程序列表的响应 Django REST Framework,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/55559023/

相关文章:

python - 为什么覆盖 resetful JSONWebTokenSerializer 只保持返回 token ? Python

python - 如何正确记录查询参数(搜索,过滤)Django Rest Framework?

javascript - 如何更新Backbone JS模型属性?

python - 替代 if-elif 子句

python - pyPdf如何理解文档边界?

django - 如何将登录用户附加到 Django 异常错误消息?

django - 如何在 django 2 中上传任何类型的文件

python - 安装了 Nose 但不能在命令行上使用

python - 追加到列表不会在程序重新启动时保留最后的输入吗?

python - django 的 gentelella 无法识别 custom.css 上的更改