Bottle框架:如何在JSON响应中返回datetime

时间:2014-01-22 11:46:35

标签: python json datetime bottle bson

当我尝试返回包含datetime值的JSON时,我正在

  File "/usr/lib/python2.7/json/encoder.py", line 178, in default
    raise TypeError(repr(o) + " is not JSON serializable")
TypeError: datetime.datetime(2014, 2, 1, 0, 0) is not JSON serializable

这是正常的。有没有一种简单的方法可以将对象挂钩添加到bottle,如

from bson import json_util
import json
json.dumps(anObject, default=json_util.default)

要转换datetime个值吗?

1 个答案:

答案 0 :(得分:4)

有趣的问题!我可以看到几种方法。一个是编写一个包装JSONPlugin

的自定义插件
from bottle import route, run, install, JSONPlugin
from bson import json_util

class JSONDefaultPlugin(JSONPlugin):
    def __init__(self):
        super(JSONDefaultPlugin, self).__init__()
        self.plain_dump = self.json_dumps
        self.json_dumps = lambda body: self.plain_dump(body, default=json_util.default)

然后可以这样使用:

@route('/hello')
def index(name):
    return {'test': datetime.datetime(2014, 2, 1, 0, 0)}

install(JSONDefaultPlugin())
run(host='localhost', port=8080)

并将提供如下输出:

{"test": {"$date": 1391212800000}}

另一种更短的方法是在实例化JSONPlugin类时简单地指定json_loads参数:

import json
from bson import json_util

install(JSONPlugin(json_dumps=lambda body: json.dumps(body, default=json_util.default)))

这会产生相同的结果。

背景

当您查看source code for bottle时,这一切都会更有意义(为简洁起见,下面删除了一些部分):

class JSONPlugin(object):
    name = 'json'
    api  = 2

    def __init__(self, json_dumps=json_dumps):
        self.json_dumps = json_dumps

    def apply(self, callback, route):
        dumps = self.json_dumps
        if not dumps: return callback
        def wrapper(*a, **ka):
            ... 

            if isinstance(rv, dict):
                ...
            elif isinstance(rv, HTTPResponse) and isinstance(rv.body, dict):
                rv.body = dumps(rv.body)
                rv.content_type = 'application/json'
            return rv

        return wrapper

我们需要做的就是确保拨打dumps的电话会收到您希望提供的default关键字参数。