Spaces:
Running
Running
from __future__ import annotations | |
import dataclasses | |
import decimal | |
import json | |
import typing as t | |
import uuid | |
import weakref | |
from datetime import date | |
from werkzeug.http import http_date | |
if t.TYPE_CHECKING: # pragma: no cover | |
from werkzeug.sansio.response import Response | |
from ..sansio.app import App | |
class JSONProvider: | |
"""A standard set of JSON operations for an application. Subclasses | |
of this can be used to customize JSON behavior or use different | |
JSON libraries. | |
To implement a provider for a specific library, subclass this base | |
class and implement at least :meth:`dumps` and :meth:`loads`. All | |
other methods have default implementations. | |
To use a different provider, either subclass ``Flask`` and set | |
:attr:`~flask.Flask.json_provider_class` to a provider class, or set | |
:attr:`app.json <flask.Flask.json>` to an instance of the class. | |
:param app: An application instance. This will be stored as a | |
:class:`weakref.proxy` on the :attr:`_app` attribute. | |
.. versionadded:: 2.2 | |
""" | |
def __init__(self, app: App) -> None: | |
self._app: App = weakref.proxy(app) | |
def dumps(self, obj: t.Any, **kwargs: t.Any) -> str: | |
"""Serialize data as JSON. | |
:param obj: The data to serialize. | |
:param kwargs: May be passed to the underlying JSON library. | |
""" | |
raise NotImplementedError | |
def dump(self, obj: t.Any, fp: t.IO[str], **kwargs: t.Any) -> None: | |
"""Serialize data as JSON and write to a file. | |
:param obj: The data to serialize. | |
:param fp: A file opened for writing text. Should use the UTF-8 | |
encoding to be valid JSON. | |
:param kwargs: May be passed to the underlying JSON library. | |
""" | |
fp.write(self.dumps(obj, **kwargs)) | |
def loads(self, s: str | bytes, **kwargs: t.Any) -> t.Any: | |
"""Deserialize data as JSON. | |
:param s: Text or UTF-8 bytes. | |
:param kwargs: May be passed to the underlying JSON library. | |
""" | |
raise NotImplementedError | |
def load(self, fp: t.IO[t.AnyStr], **kwargs: t.Any) -> t.Any: | |
"""Deserialize data as JSON read from a file. | |
:param fp: A file opened for reading text or UTF-8 bytes. | |
:param kwargs: May be passed to the underlying JSON library. | |
""" | |
return self.loads(fp.read(), **kwargs) | |
def _prepare_response_obj( | |
self, args: tuple[t.Any, ...], kwargs: dict[str, t.Any] | |
) -> t.Any: | |
if args and kwargs: | |
raise TypeError("app.json.response() takes either args or kwargs, not both") | |
if not args and not kwargs: | |
return None | |
if len(args) == 1: | |
return args[0] | |
return args or kwargs | |
def response(self, *args: t.Any, **kwargs: t.Any) -> Response: | |
"""Serialize the given arguments as JSON, and return a | |
:class:`~flask.Response` object with the ``application/json`` | |
mimetype. | |
The :func:`~flask.json.jsonify` function calls this method for | |
the current application. | |
Either positional or keyword arguments can be given, not both. | |
If no arguments are given, ``None`` is serialized. | |
:param args: A single value to serialize, or multiple values to | |
treat as a list to serialize. | |
:param kwargs: Treat as a dict to serialize. | |
""" | |
obj = self._prepare_response_obj(args, kwargs) | |
return self._app.response_class(self.dumps(obj), mimetype="application/json") | |
def _default(o: t.Any) -> t.Any: | |
if isinstance(o, date): | |
return http_date(o) | |
if isinstance(o, (decimal.Decimal, uuid.UUID)): | |
return str(o) | |
if dataclasses and dataclasses.is_dataclass(o): | |
return dataclasses.asdict(o) # type: ignore[arg-type] | |
if hasattr(o, "__html__"): | |
return str(o.__html__()) | |
raise TypeError(f"Object of type {type(o).__name__} is not JSON serializable") | |
class DefaultJSONProvider(JSONProvider): | |
"""Provide JSON operations using Python's built-in :mod:`json` | |
library. Serializes the following additional data types: | |
- :class:`datetime.datetime` and :class:`datetime.date` are | |
serialized to :rfc:`822` strings. This is the same as the HTTP | |
date format. | |
- :class:`uuid.UUID` is serialized to a string. | |
- :class:`dataclasses.dataclass` is passed to | |
:func:`dataclasses.asdict`. | |
- :class:`~markupsafe.Markup` (or any object with a ``__html__`` | |
method) will call the ``__html__`` method to get a string. | |
""" | |
default: t.Callable[[t.Any], t.Any] = staticmethod(_default) # type: ignore[assignment] | |
"""Apply this function to any object that :meth:`json.dumps` does | |
not know how to serialize. It should return a valid JSON type or | |
raise a ``TypeError``. | |
""" | |
ensure_ascii = True | |
"""Replace non-ASCII characters with escape sequences. This may be | |
more compatible with some clients, but can be disabled for better | |
performance and size. | |
""" | |
sort_keys = True | |
"""Sort the keys in any serialized dicts. This may be useful for | |
some caching situations, but can be disabled for better performance. | |
When enabled, keys must all be strings, they are not converted | |
before sorting. | |
""" | |
compact: bool | None = None | |
"""If ``True``, or ``None`` out of debug mode, the :meth:`response` | |
output will not add indentation, newlines, or spaces. If ``False``, | |
or ``None`` in debug mode, it will use a non-compact representation. | |
""" | |
mimetype = "application/json" | |
"""The mimetype set in :meth:`response`.""" | |
def dumps(self, obj: t.Any, **kwargs: t.Any) -> str: | |
"""Serialize data as JSON to a string. | |
Keyword arguments are passed to :func:`json.dumps`. Sets some | |
parameter defaults from the :attr:`default`, | |
:attr:`ensure_ascii`, and :attr:`sort_keys` attributes. | |
:param obj: The data to serialize. | |
:param kwargs: Passed to :func:`json.dumps`. | |
""" | |
kwargs.setdefault("default", self.default) | |
kwargs.setdefault("ensure_ascii", self.ensure_ascii) | |
kwargs.setdefault("sort_keys", self.sort_keys) | |
return json.dumps(obj, **kwargs) | |
def loads(self, s: str | bytes, **kwargs: t.Any) -> t.Any: | |
"""Deserialize data as JSON from a string or bytes. | |
:param s: Text or UTF-8 bytes. | |
:param kwargs: Passed to :func:`json.loads`. | |
""" | |
return json.loads(s, **kwargs) | |
def response(self, *args: t.Any, **kwargs: t.Any) -> Response: | |
"""Serialize the given arguments as JSON, and return a | |
:class:`~flask.Response` object with it. The response mimetype | |
will be "application/json" and can be changed with | |
:attr:`mimetype`. | |
If :attr:`compact` is ``False`` or debug mode is enabled, the | |
output will be formatted to be easier to read. | |
Either positional or keyword arguments can be given, not both. | |
If no arguments are given, ``None`` is serialized. | |
:param args: A single value to serialize, or multiple values to | |
treat as a list to serialize. | |
:param kwargs: Treat as a dict to serialize. | |
""" | |
obj = self._prepare_response_obj(args, kwargs) | |
dump_args: dict[str, t.Any] = {} | |
if (self.compact is None and self._app.debug) or self.compact is False: | |
dump_args.setdefault("indent", 2) | |
else: | |
dump_args.setdefault("separators", (",", ":")) | |
return self._app.response_class( | |
f"{self.dumps(obj, **dump_args)}\n", mimetype=self.mimetype | |
) | |