2014-09-26 9 views
17

Django 1.7 wprowadził JsonResponse objects, którego próbuję użyć, aby zwrócić listę wartości do mojego żądania ajax.Jak przekazać pola modelu do obiektu JsonResponse

Chcę przekazać

>>> Genre.objects.values('name', 'color') 
[{'color': '8a3700', 'name': 'rock'}, {'color': 'ffff00', 'name': 'pop'}, {'color': '8f8f00', 'name': 'electronic'}, {'color': '9e009e', 'name': 'chillout'}, {'color': 'ff8838', 'name': 'indie'}, {'color': '0aff0a', 'name': 'techno'}, {'color': 'c20000', 'name': "drum'n'bass"}, {'color': '0000d6', 'name': 'worldmusic'}, {'color': 'a800a8', 'name': 'classic'}, {'color': 'dbdb00', 'name': 'hiphop'}] 

do obiektu JsonResponse.

Jednak moje próby się nie udają.

>>> JsonResponse({'foo': 'bar', 'blib': 'blab'}) # works 
<django.http.response.JsonResponse object at 0x7f53d28bbb00> 

>>> JsonResponse(Genre.objects.values('name', 'color')) # doesn't work 
Traceback (most recent call last): 
    File "<console>", line 1, in <module> 
    File "/home/marcel/Dokumente/django/FlushFM/env/lib/python3.4/site-packages/django/http/response.py", line 476, in __init__ 
    raise TypeError('In order to allow non-dict objects to be ' 
TypeError: In order to allow non-dict objects to be serialized set the safe parameter to False 

Jest to prawdopodobnie ze względu na inną strukturę danych Genre.objects.values().

Jak to zrobić dobrze?

[edytuj]

Z safe=False uzyskać

>>> JsonResponse(Genre.objects.values('name', 'color'), safe=False) 
Traceback (most recent call last): 
    File "<console>", line 1, in <module> 
    File "/home/marcel/Dokumente/django/FlushFM/env/lib/python3.4/site-packages/django/http/response.py", line 479, in __init__ 
    data = json.dumps(data, cls=encoder) 
    File "/usr/lib/python3.4/json/__init__.py", line 237, in dumps 
    **kw).encode(obj) 
    File "/usr/lib/python3.4/json/encoder.py", line 192, in encode 
    chunks = self.iterencode(o, _one_shot=True) 
    File "/usr/lib/python3.4/json/encoder.py", line 250, in iterencode 
    return _iterencode(o, 0) 
    File "/home/marcel/Dokumente/django/FlushFM/env/lib/python3.4/site-packages/django/core/serializers/json.py", line 109, in default 
    return super(DjangoJSONEncoder, self).default(o) 
    File "/usr/lib/python3.4/json/encoder.py", line 173, in default 
    raise TypeError(repr(o) + " is not JSON serializable") 
TypeError: [{'color': '8a3700', 'name': 'rock'}, {'color': 'ffff00', 'name': 'pop'}, {'color': '8f8f00', 'name': 'electronic'}, {'color': '9e009e', 'name': 'chillout'}, {'color': 'ff8838', 'name': 'indie'}, {'color': '0aff0a', 'name': 'techno'}, {'color': 'c20000', 'name': "drum'n'bass"}, {'color': '0000d6', 'name': 'worldmusic'}, {'color': 'a800a8', 'name': 'classic'}, {'color': 'dbdb00', 'name': 'hiphop'}] is not JSON serializable 

Co działa to

>>> JsonResponse(list(Genre.objects.values('name', 'color')), safe=False) 
<django.http.response.JsonResponse object at 0x7f53d28bb9e8> 

Ale tam nie ma lepszego sposobu, aby wygenerować dict Spośród obiektu modelu?

+1

Czy próbowałeś wykonać komunikat o błędzie? –

+0

@DanielRoseman również tak, ale ustawienie "safe = False" spowodowałoby wyświetlenie komunikatu o błędzie. Potrzebujesz wyjścia? – speendo

+1

@speendo Czy to samo dotyczy również 'safe = False'? Być może wypróbuj po przejściu 'ValuesQuerySet' do' list() ':' JsonResponse (list (Genre.objects.values ​​('name', 'color'))) ' –

Odpowiedz

18

Na przyszłość, .values() zwraca ValuesQuerySet że zachowuje się jak iterable pełne słowników, więc używając list() utworzy nową instancję list ze wszystkich słowników w nim. Dzięki temu możesz stworzyć nowy dykt i serializować to.

response = JsonResponse(dict(genres=list(Genre.objects.values('name', 'color')))) 

IIRC, to nie jest bezpieczne, aby obiekt JSON, który ma listę jako root i to pewnie dlatego Django narzeka. Nie mogłem znaleźć żadnego odniesienia na ten temat, aby podać źródło, przepraszam.

+1

Odpowiedź na http://stackoverflow.com/a/26833156/179583 używa 'serializers' z django. core, który może być przydatny, aby wydobyć cały model, ale wydaje się dziwny/grubszy, że muszę przekazać JSON-a już zserializowanego JsonResponse. Technika tej odpowiedzi jest dla mnie lepsza, szczególnie gdy potrzebuję tylko kilku pól z każdego modelu. – natevw

+0

tak długo szukałem takiej odpowiedzi! – Vizjerei