<Objeto Django> não é serializável JSON

102

Eu tenho o seguinte código para serializar o queryset;

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

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

E o seguinte é o meu 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 preciso serializar. Mas diz que não foi possível serializar o <Product: hederello ()>. Porque lista composta de objetos e dictos django. Alguma ideia ?

atum
fonte
Duplicado: stackoverflow.com/a/29088221/2172260
Julio Marins

Respostas:

114

simplejsone jsonnão funcionam bem com objetos django.

Os serializadores embutidos do Django só podem serializar querysets preenchidos com objetos django:

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

No seu caso, self.get_queryset()contém uma mistura de objetos django e dicts dentro.

Uma opção é livrar-se das instâncias de modelo no self.get_queryset()e substituí-las por dictos 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 ajude.

Alecxe
fonte
Agora obtendo erro -> 'NoneType' object has no attribute 'concrete_model'... E usando Django 1.4+
atum
3
Quando o modelo tem um campo de data e hora, ele não funciona.
ax003d
essa solução vai gerar muitas consultas
Julio Marins
para usar isso diretamente no JS, basta usar o safetage. stackoverflow.com/a/57939897/4157431
Rami Alloush
62

A maneira mais fácil é usar um JsonResponse .

Para um queryset, você deve passar uma lista de valuespara esse queryset, assim:

from django.http import JsonResponse

queryset = YourModel.objects.filter(some__filter="some value").values()
return JsonResponse({"models_to_return": list(queryset)})
YPCrumble
fonte
2
obrigado por .values ​​(), No meu caso, só preciso adicionar .values ​​() após o filtro
Jze
18

Descobri que isso pode ser feito de forma bastante simples usando o método ".values", que também fornece campos nomeados:

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

"list" deve ser usado para obter dados como iteráveis, uma vez que o tipo "value queryset" é apenas um dict se escolhido como iterável.

Documentação: https://docs.djangoproject.com/en/1.7/ref/models/querysets/#values

Danny Staple
fonte
Isto funcionou bem para mim. Mesmo que a mensagem de erro sugira que está tudo em uma grande lista, o list()ainda é aparentemente necessário.
trpt4him
1
Solução mais simples e melhor
Timur,
11

A partir da versão 1.9. Forma mais fácil e oficial de obter json

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


return JsonResponse(  model_to_dict(modelinstance) )
Yash
fonte
8

Nosso programador js me pediu para retornar os dados de formato JSON exatos em vez de uma string codificada em json para ela.

Abaixo está a solução. (Isso retornará um objeto que pode ser usado / visualizado diretamente no 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
fonte
Será melhor apenasHttpResponse(tmpObj)
Pablo Díaz
6

Primeiro, adicionei um método to_dict ao meu modelo;

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

Então eu tenho isso;

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)

e, finalmente, use esta classe para serializar meu queryset.

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

Isso funciona muito bem

atum
fonte