Как сериализовать JSON __dict__ модели Django?

Я хочу сериализовать значения одной модели в Django. Поскольку я хочу использовать get(), values() недоступен. Однако я прочитал в группах Google, что вы можете получить доступ к значениям с __dict__.

from django.http import HttpResponse, Http404
import json
from customer.models import Customer

def single(request, id):
    try:
        model = Customer.objects.get(id=id, user=1)
    except Customer.DoesNotExist:
        raise Http404
    values = model.__dict__
    print(values)
    string = json.dumps(values)
    return HttpResponse(string, content_type='application/json')

Оператор печати выводит это.

{'_state': <django.db.models.base.ModelState object at 0x0000000005556EF0>, 'web
site': 'http://example.com/', 'name': 'Company Name', 'id': 1, 'logo': '', 'use
r_id': 1, 'address3': 'City', 'notes': '', 'address2': 'Street 123', 'address1': 'Company Name', 'ustid': 'AB123456789', 'fullname': 'Full Name Of Company Inc.', 'mail': '[email protected]'}

Из-за ключа _state, который содержит несериализуемое значение, следующая строка завершается ошибкой.

<django.db.models.base.ModelState object at 0x0000000005556EF0> is not JSON serializable

Как я могу сериализовать словарь, возвращенный из __dict__, без включения _state?


person danijar    schedule 19.04.2014    source источник
comment
возможный дубликат django webapi: дамп модели django в JSON   -  person FallenAngel    schedule 19.04.2014
comment
@FallenAngel Сериализация того, что возвращает all(), тривиальна, но мой вопрос касается get(). Это связано, но не дубликат.   -  person danijar    schedule 19.04.2014


Ответы (3)


model_to_dict() — это то, что вам нужно:

from django.forms.models import model_to_dict

data = model_to_dict(model)
data['logo'] = data['logo'].url
return HttpResponse(json.dumps(data), content_type='application/json')

Указав аргументы ключевого слова fields и exclude, вы можете контролировать, какие поля сериализовать.

Кроме того, вы можете упростить блок try/except, используя ярлык get_object_or_404():

model = get_object_or_404(Customer, id=id, user=1)
person alecxe    schedule 19.04.2014
comment
Спасибо! Я добавил необходимый импорт к вашему вопросу. Однако я все еще не могу сериализовать словарь, так как он содержит изображение Django. Я думаю, что словарь должен содержать URL-адрес изображения, а не весь объект Django. Есть идеи? - person danijar; 19.04.2014
comment
model_to_dict() возвращает словарь, подобный {'name': 'Foo', 'user': 1, 'logo': <ImageFieldFile: None>}, который не удается сериализовать. Сериализаторы Django оборачивают словарь полей в другой, который мне не нужен. - person danijar; 19.04.2014
comment
@danijar хорошо, я обновил ответ. Идея состоит в том, чтобы использовать model_to_dict() для получения словаря, а затем перезаписать ключ logo. Попробуйте. Есть и другие варианты, но давайте сначала попробуем этот. Спасибо. - person alecxe; 19.04.2014
comment
Мне пришлось проверить наличие пустых файлов, поэтому values['logo'] = values['logo'].url if values['logo'] else '' сделал свое дело. Спасибо за вашу помощь. - person danijar; 19.04.2014

проверьте исходный код django/core/serializers/__init__.py, комментарий:

Interfaces for serializing Django objects.

Usage::

    from django.core import serializers
    json = serializers.serialize("json", some_queryset)
    objects = list(serializers.deserialize("json", json))

To add your own serializers, use the SERIALIZATION_MODULES setting::

    SERIALIZATION_MODULES = {
        "csv" : "path.to.csv.serializer",
        "txt" : "path.to.txt.serializer",
    }

за один объект

json = serializers.serialize("json", some_queryset[0:1])
person vadimchin    schedule 19.04.2014

Я узнал, что на самом деле можно использовать values() вместе с get(). Вам просто нужно получить значения из отфильтрованного набора.

def single(request, id):
    user = 1
    try:
        models = Customer.objects.filter(id=id, user=user)
        values = models.values().get()
    except Customer.DoesNotExist:
        raise Http404
    string = json.dumps(values)
    return HttpResponse(string, content_type='application/json')
person danijar    schedule 19.04.2014