<L'objet Django> n'est pas sérialisable JSON

102

J'ai le code suivant pour sérialiser le jeu de requêtes;

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

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

Et le suivant est mon 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',.........]}]

Ce dont j'ai besoin pour sérialiser. Mais il dit impossible de sérialiser le fichier <Product: hederello ()>. Parce que liste composée à la fois d'objets django et de dictées. Des idées ?

thon
la source
Dupliqué: stackoverflow.com/a/29088221/2172260
Julio Marins

Réponses:

115

simplejsonet jsonne fonctionne pas bien avec les objets django.

Les sérialiseurs intégrés de Django ne peuvent sérialiser que des ensembles de requêtes remplis d'objets django:

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

Dans votre cas, self.get_queryset()contient un mélange d'objets django et de dictées à l'intérieur.

Une option consiste à supprimer les instances de modèle dans le self.get_queryset()et à les remplacer par des dictionnaires en utilisant 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")

J'espère que cela pourra aider.

alecxe
la source
Obtention d'une erreur -> 'NoneType' object has no attribute 'concrete_model'... Et utilisation de Django 1.4+
thon
3
Lorsque le modèle a un champ datetime, cela ne fonctionne pas.
ax003d
cette solution déclenchera de nombreuses requêtes
Julio Marins
pour l'utiliser directement dans JS, utilisez simplement le safetage. stackoverflow.com/a/57939897/4157431
Rami Alloush
63

Le moyen le plus simple est d'utiliser un JsonResponse .

Pour un ensemble de requêtes, vous devez transmettre une liste des valuespour cet ensemble de requêtes, comme ceci:

from django.http import JsonResponse

queryset = YourModel.objects.filter(some__filter="some value").values()
return JsonResponse({"models_to_return": list(queryset)})
YPCrumble
la source
2
merci pour .values ​​(), Dans mon cas, j'ai juste besoin d'ajouter .values ​​() après le filtre
Jze
18

J'ai trouvé que cela peut être fait assez simple en utilisant la méthode ".values", qui donne également des champs nommés:

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

"list" doit être utilisé pour obtenir des données comme itérables, puisque le type "value queryset" n'est qu'un dict s'il est choisi comme iterable.

Documentation: https://docs.djangoproject.com/en/1.7/ref/models/querysets/#values

Danny Agrafe
la source
Cela a bien fonctionné pour moi. Même si le message d'erreur suggère que tout est dans une grande liste, le list()est toujours apparemment nécessaire.
trpt4him
1
Solution la plus simple et la meilleure
Timur
11

À partir de la version 1.9 Un moyen plus simple et officiel d'obtenir json

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


return JsonResponse(  model_to_dict(modelinstance) )
Yash
la source
8

Notre programmeur js m'a demandé de lui renvoyer les données exactes au format JSON au lieu d'une chaîne encodée en json.

Vous trouverez ci-dessous la solution (cela renverra un objet qui peut être utilisé / affiché directement dans le navigateur)

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
la source
Sera mieux justeHttpResponse(tmpObj)
Pablo Díaz
6

J'ai d'abord ajouté une méthode to_dict à mon modèle;

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

Alors j'ai ceci;

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)

et enfin utiliser cette classe pour sérialiser mon jeu de requêtes.

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

Cela fonctionne assez bien

thon
la source