Как превратить запрос MongoDB в JSON? - PullRequest
31 голосов
/ 10 декабря 2010
for p in db.collection.find({"test_set":"abc"}):
    posts.append(p)
thejson = json.dumps({'results':posts})
return  HttpResponse(thejson, mimetype="application/javascript")

В моем коде Django / Python я не могу вернуть JSON из запроса монго из-за "ObjectID".Ошибка говорит о том, что «ObjectID» не сериализуем.

Что мне делать?Хакерский способ был бы перебрать:

for p in posts:
    p['_id'] = ""

Ответы [ 4 ]

29 голосов
/ 10 декабря 2010

Модуль json не будет работать из-за таких вещей, как ObjectID.

К счастью PyMongo обеспечивает json_util который ...

... разрешить [s] для специализированного кодирования и декодирование документов BSON в Mongo Расширенный строгий режим JSON. Это позволяет Вы кодируете / декодируете документы BSON в JSON, даже когда они используют специальные BSON типы.

23 голосов
/ 18 июля 2012

Вот простой пример использования pymongo 2.2.1

.
import os
import sys
import json
import pymongo
from bson import BSON
from bson import json_util

if __name__ == '__main__':
  try:
    connection = pymongo.Connection('mongodb://localhost:27017')
    database = connection['mongotest']
  except:
    print('Error: Unable to Connect')
    connection = None

  if connection is not None:
    database["test"].insert({'name': 'foo'})
    doc = database["test"].find_one({'name': 'foo'})
    return json.dumps(doc, sort_keys=True, indent=4, default=json_util.default)
8 голосов
/ 10 декабря 2010

Довольно легко написать собственный сериализатор, который справляется с ObjectIds.Django уже включает в себя тот, который обрабатывает десятичные дроби и даты, поэтому вы можете расширить это:

from django.core.serializers.json import DjangoJSONEncoder
from bson import objectid

class MongoAwareEncoder(DjangoJSONEncoder):
    """JSON encoder class that adds support for Mongo objectids."""
    def default(self, o):
        if isinstance(o, objectid.ObjectId):
            return str(o)
        else:
            return super(MongoAwareEncoder, self).default(o)

Теперь вы можете просто указать json использовать ваш собственный сериализатор:

thejson = json.dumps({'results':posts}, cls=MongoAwareEncoder)
5 голосов
/ 30 декабря 2016

Что-то еще более простое, что работает для меня на Python 3.6 с использованием мотора == 1.1 pymongo == 3.4.0

from bson.json_util import dumps, loads

for mongo_doc in await cursor.to_list(length=10):
    # mongo_doc is a <class 'dict'> returned from the async mongo driver, in this acse motor / pymongo.
    # result of executing a simple find() query.

    json_string = dumps(mongo_doc)
    # serialize the <class 'dict'> into a <class 'str'> 

    back_to_dict = loads(json_string)
    # to unserialize, thus return the string back to a <class 'dict'> with the original 'ObjectID' type.
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...