Custom datatypes (like datetime) serialization to/from JSON.
Project description
Serialization of Python types to JSON that “just works”.
Forget errors like:
TypeError: datetime.datetime(...) is not JSON serializable
In addition to (de-)serialization of basic types (provided by simplejson), jsonplus provides support for exact (de-)serialization of other commonly used types, like: tuple/namedtuple, set/frozenset, complex/decimal.Decimal/fractions.Fraction, and datetime/date/time/timedelta.
If the exact representation of types is not your cup of tea, and all you wish for is json.dumps to work on your data structure with non-basic types, accepting the loss of “type-precision” along the way, than you should use the compatibility mode (jsonplus.prefer_compat()).
Installation
jsonplus is available as Python package. To install, simply type:
$ pip install jsonplus
Usage
You can treat jsonplus as a friendly drop-in replacement for json/simplejson.
>>> import jsonplus as json
>>> x = json.loads('{"a":1,"b":2}')
>>> y = json.dumps(x, indent=4)
>>> z = json.pretty(x)
Examples
Let’s start with that beloved datetime.
>>> import jsonplus as json
>>> from datetime import datetime
>>> json.dumps({
... "x": [4,3],
... "t": datetime.now()
... })
'{"x":[4,3],"t":{"__class__":"datetime","__value__":"2013-09-06T23:38:55.819791"}}'
>>> json.loads(_)
{u'x': [4, 3], u't': datetime.datetime(2013, 9, 6, 23, 38, 55, 819791)}
Similarly for other datetime.* types, like timedelta, date, and time:
>>> from datetime import timedelta, date, time
>>> print(json.pretty({"dt": timedelta(0, 1234567, 123), "d": date.today(), "t": datetime.now().time()}))
{
"d": {
"__class__": "date",
"__value__": "2013-09-22"
},
"dt": {
"__class__": "timedelta",
"__value__": {
"days": 14,
"microseconds": 123,
"seconds": 24967
}
},
"t": {
"__class__": "time",
"__value__": "23:33:16.335360"
}
}
Also, set and complex:
>>> json.dumps([set(range(3)), 1+2j])
'[{"__class__":"set","__value__":[0,1,2]},{"__class__":"complex","__value__":{"real":1.0,"imag":2.0}}]'
>>> json.loads(_)
[set([0, 1, 2]), (1+2j)]
tuple and namedtuple are also preserved:
>>> from collections import namedtuple
>>> Point = namedtuple('Point', ['x', 'y'])
>>> data = json.pretty({"vect": (1, 2, 3), "dot": Point(3, 4)})
>>> print(data)
{
"dot": {
"__class__": "namedtuple",
"__value__": {
"fields": [
"x",
"y"
],
"name": "Point",
"values": [
3,
4
]
}
},
"vect": {
"__class__": "tuple",
"__value__": [
1,
2,
3
]
}
}
>>> json.loads(data)
{'vect': (1, 2, 3), 'dot': Point(x=3, y=4)}
Compatibility mode
All types supported in the exact mode are also supported in the compatibility mode. JSON representation differs, however.
In the exact mode, type and value are encoded with JSON Object’s __class__ and __value__ keys, while in the compatibility mode, values are “rounded” to the closest JSON type.
For example, tuple``s and ``set``s are represented with ``JSON Array``s, and ``namedtuple``s are coded as plain ``JSON Object``s. ``Decimal``s are represented as ``JSON Number``s with arbitrary precision (which is lost if decoded as ``float).
To switch between the exact and compatibility modes, use the (thread- local) functions prefer_exact() and prefer_compat():
>>> import jsonplus as json
>>> json.prefer_compat()
# or:
>>> json.prefer(json.COMPAT)
# to go back to (default) exact coding:
>>> json.prefer_exact()
The above tuple/namedtuple example run in compatibility coding mode results with:
>>> json.prefer_compat()
>>> print(json.pretty({"vect": (1, 2, 3), "dot": Point(3, 4)}))
{
"point": {
"x": 3,
"y": 4
},
"vector": [
1,
2,
3
]
}
Dates and times are represented with ISO8601 strings:
>>> json.prefer_compat()
>>> json.dumps({"now": datetime.now()})
'{"now":"2017-01-26T00:37:40.293963"}'
So, to be able to decode it as date/time, some additional context has to be provided to the decoder.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Hashes for jsonplus-0.6.0-py2.py3-none-any.whl
Algorithm | Hash digest | |
---|---|---|
SHA256 | 21adb2e717e4c2303e4c825cb399d4898325e57e60df5b8231555dcba69e3658 |
|
MD5 | e801a1b70515c45db44980a71811d4a5 |
|
BLAKE2b-256 | 5408ec44eeca46c052b4d28f416c901a9c347c264f5e8dab49b5851f7dbf62c0 |