<Objeto Django> no es serializable JSON

102

Tengo el siguiente código para serializar el conjunto de consultas;

def render_to_response(self, context, **response_kwargs):

    return HttpResponse(json.simplejson.dumps(list(self.get_queryset())),
                        mimetype="application/json")

Y lo siguiente es mi get_querset()

[{'product': <Product: hederello ()>, u'_id': u'9802', u'_source': {u'code': u'23981', u'facilities': [{u'facility': {u'name': {u'fr': u'G\xe9n\xe9ral', u'en': u'General'}, u'value': {u'fr': [u'bar', u'r\xe9ception ouverte 24h/24', u'chambres non-fumeurs', u'chambres familiales',.........]}]

Que necesito serializar. Pero dice que no se puede serializar <Product: hederello ()>. Porque la lista está compuesta por objetos y dictados de django. Algunas ideas ?

atún
fuente
Duplicado: stackoverflow.com/a/29088221/2172260
Julio Marins

Respuestas:

114

simplejsony jsonno funciona bien con objetos django.

Los serializadores integrados de Django solo pueden serializar conjuntos de consultas llenos de objetos de django:

data = serializers.serialize('json', self.get_queryset())
return HttpResponse(data, content_type="application/json")

En su caso, self.get_queryset()contiene una mezcla de objetos django y dictados en su interior.

Una opción es deshacerse de las instancias del modelo en el self.get_queryset()y reemplazarlas con dictados usando model_to_dict:

from django.forms.models import model_to_dict

data = self.get_queryset()

for item in data:
   item['product'] = model_to_dict(item['product'])

return HttpResponse(json.simplejson.dumps(data), mimetype="application/json")

Espero que ayude.

Alecxe
fuente
Ahora aparece el error -> 'NoneType' object has no attribute 'concrete_model'... Y usando Django 1.4+
tuna
3
Cuando el modelo tiene un campo de fecha y hora, no funciona.
ax003d
esa solución generará muchas consultas
Julio Marins
para usar eso directamente en JS, solo use el safetage. stackoverflow.com/a/57939897/4157431
Rami Alloush
62

La forma más sencilla es utilizar un JsonResponse .

Para un conjunto de consultas, debe pasar una lista de los valuespara ese conjunto de consultas, así:

from django.http import JsonResponse

queryset = YourModel.objects.filter(some__filter="some value").values()
return JsonResponse({"models_to_return": list(queryset)})
YPCrumble
fuente
2
gracias por .values ​​(), en mi caso, solo necesito agregar .values ​​() después del filtro
Jze
18

Descubrí que esto se puede hacer de manera bastante simple usando el método ".values", que también proporciona campos con nombre:

result_list = list(my_queryset.values('first_named_field', 'second_named_field'))
return HttpResponse(json.dumps(result_list))

"list" debe usarse para obtener datos como iterables, ya que el tipo "value queryset" es solo un dict si se selecciona como iterable.

Documentación: https://docs.djangoproject.com/en/1.7/ref/models/querysets/#values

Danny Staple
fuente
Esto funcionó bien para mí. Aunque el mensaje de error sugiere que todo está en una gran lista, list()aparentemente todavía se necesita.
trpt4him
1
La mejor y más simple solución
Timur
11

Desde la versión 1.9 Una forma oficial y más fácil de obtener json

from django.http import JsonResponse
from django.forms.models import model_to_dict


return JsonResponse(  model_to_dict(modelinstance) )
Yash
fuente
8

Nuestro programador js me pidió que le devolviera los datos en formato JSON exactos en lugar de una cadena codificada en json.

A continuación se muestra la solución. (Esto devolverá un objeto que se puede usar / ver directamente en el navegador)

import json
from xxx.models import alert
from django.core import serializers

def test(request):
    alert_list = alert.objects.all()

    tmpJson = serializers.serialize("json",alert_list)
    tmpObj = json.loads(tmpJson)

    return HttpResponse(json.dumps(tmpObj))
Woody Johnson
fuente
Será mejor soloHttpResponse(tmpObj)
Pablo Díaz
6

Primero agregué un método to_dict a mi modelo;

def to_dict(self):
    return {"name": self.woo, "title": self.foo}

Entonces tengo esto;

class DjangoJSONEncoder(JSONEncoder):

    def default(self, obj):
        if isinstance(obj, models.Model):
            return obj.to_dict()
        return JSONEncoder.default(self, obj)


dumps = curry(dumps, cls=DjangoJSONEncoder)

y por fin usar esta clase para serializar mi conjunto de consultas.

def render_to_response(self, context, **response_kwargs):
    return HttpResponse(dumps(self.get_queryset()))

Esto funciona bastante bien

atún
fuente