Spaces:
Runtime error
Runtime error
| 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 | |
| ) | |