This document is for Kombu's development version, which can be significantly different from previous releases. Get the stable docs here: 5.0.

Source code for kombu.utils.json

"""JSON Serialization Utilities."""

from __future__ import annotations

import base64
import datetime
import decimal
import json
import uuid

    from django.utils.functional import Promise as DjangoPromise
except ImportError:  # pragma: no cover
[docs] class DjangoPromise: """Dummy object."""
class _DecodeError(Exception): pass _encoder_cls = type(json._default_encoder) _default_encoder = None # ... set to JSONEncoder below.
[docs]class JSONEncoder(_encoder_cls): """Kombu custom json encoder."""
[docs] def default(self, o, dates=(datetime.datetime,, times=(datetime.time,), textual=(decimal.Decimal, DjangoPromise), isinstance=isinstance, datetime=datetime.datetime, text_t=str): reducer = getattr(o, '__json__', None) if reducer is not None: return reducer() else: if isinstance(o, dates): marker = "__date__" if not isinstance(o, datetime): o = datetime(o.year, o.month,, 0, 0, 0, 0) else: marker = "__datetime__" r = o.isoformat() return {"datetime": r, marker: True} elif isinstance(o, times): return o.isoformat() elif isinstance(o, uuid.UUID): return {"uuid": str(o), "__uuid__": True, "version": o.version} elif isinstance(o, textual): return text_t(o) elif isinstance(o, bytes): try: return {"bytes": o.decode("utf-8"), "__bytes__": True} except UnicodeDecodeError: return { "bytes": base64.b64encode(o).decode("utf-8"), "__base64__": True, } return super().default(o)
_default_encoder = JSONEncoder
[docs]def dumps(s, _dumps=json.dumps, cls=None, default_kwargs=None, **kwargs): """Serialize object to json string.""" default_kwargs = default_kwargs or {} return _dumps(s, cls=cls or _default_encoder, **dict(default_kwargs, **kwargs))
[docs]def object_hook(dct): """Hook function to perform custom deserialization.""" if "__date__" in dct: return datetime.datetime.fromisoformat(dct["datetime"]).date() if "__datetime__" in dct: return datetime.datetime.fromisoformat(dct["datetime"]) if "__bytes__" in dct: return dct["bytes"].encode("utf-8") if "__base64__" in dct: return base64.b64decode(dct["bytes"].encode("utf-8")) if "__uuid__" in dct: return uuid.UUID(dct["uuid"], version=dct["version"]) return dct
[docs]def loads(s, _loads=json.loads, decode_bytes=True, object_hook=object_hook): """Deserialize json from string.""" # None of the json implementations supports decoding from # a buffer/memoryview, or even reading from a stream # (load is just loads( # but this is Python, we love copying strings, preferably many times # over. Note that pickle does support buffer/memoryview # </rant> if isinstance(s, memoryview): s = s.tobytes().decode('utf-8') elif isinstance(s, bytearray): s = s.decode('utf-8') elif decode_bytes and isinstance(s, bytes): s = s.decode('utf-8') try: return _loads(s, object_hook=object_hook) except _DecodeError: # catch "Unpaired high surrogate" error return json.loads(s)