JSON Сериализация моделей Django с simplejson

Я хотел бы использовать simplejson для сериализации модели Django. Django serializer не поддерживает словари... и simplejson не поддерживает Django Querysets. Это довольно загадка.

В модели есть спонсоры, у которых есть внешний ключ для уровня спонсоров, я пытаюсь объединить всех спонсоров, принадлежащих определенному уровню спонсоров. Здесь код, который генерирует список:

from django.shortcuts import get_list_or_404
from special_event.models import Sponsor, SponsorLevel

sponsor_dict = {}

roadie_sponsors = get_list_or_404(Sponsor, level__category = SponsorLevel.ROADIE_CHOICE)

for item in roadie_sponsors:
    try:
        sponsor_dict[item.level.name].append(item)
    except KeyError:
        sponsor_dict[item.level.name] = [item]

Здесь sponsor_dict выглядит, как только он "сделан"

{
    'Fan': [<Sponsor: Fan Sponsor>], 
    'VIP': [<Sponsor: VIP Sponsor>],
    'Groupie': [<Sponsor: Groupie Sponsor>],
    'Silver': [<Sponsor: Silver Sponsor>],
    'Bronze': [<Sponsor: Another Bronze Sponsor>, <Sponsor: Bronze Sponsor>]
}

Я добавил только одного спонсора на каждом уровне, кроме бронзы, чтобы показать, как он работает. Все, что я хочу сделать, это получить "все" в JSON, поэтому jQuery может легко интерпретировать его. Может ли Django использовать другие сериализаторы (например, XML или YAML)? Могу ли я "расширить" Django JSON Serializer для обработки словарей или "расширить" simplejson для обработки объектов Django QuerySet?

Ответ 1

Я бы пошел с расширением simplejson. В принципе, вы хотите подключить сериализацию django, когда кодер JSON встречает QuerySet. Вы можете использовать что-то вроде:

from json import dumps, loads, JSONEncoder

from django.core.serializers import serialize
from django.db.models.query import QuerySet
from django.utils.functional import curry

class DjangoJSONEncoder(JSONEncoder):
    def default(self, obj):
        if isinstance(obj, QuerySet):
            # `default` must return a python serializable
            # structure, the easiest way is to load the JSON
            # string produced by `serialize` and return it
            return loads(serialize('json', obj))
        return JSONEncoder.default(self,obj)

# partial function, we can now use dumps(my_dict) instead
# of dumps(my_dict, cls=DjangoJSONEncoder)
dumps = curry(dumps, cls=DjangoJSONEncoder)

Для получения дополнительной информации о методе default см. simplejson documentation. Поместите это в модуль python, затем импортируйте dumps, и вам хорошо идти. Но учтите, что эта функция поможет вам только сериализовать экземпляры QuerySet, а не Model.

Ответ 2

Действительно гибким способом сериализации большинства структур в django является использование класса serializer, найденного здесь

Ответ 3

на основе ответа Клемента, я сделал это, чтобы получить модели в JSON.

def toJSON(obj):
   if isinstance(obj, QuerySet):
       return simplejson.dumps(obj, cls=DjangoJSONEncoder)
   if isinstance(obj, models.Model):
       #do the same as above by making it a queryset first
       set_obj = [obj]
       set_str = simplejson.dumps(simplejson.loads(serialize('json', set_obj)))
       #eliminate brackets in the beginning and the end 
       str_obj = set_str[1:len(set_str)-2]
   return str_obj