216 lines
		
	
	
		
			7.5 KiB
		
	
	
	
		
			Python
		
	
	
	
			
		
		
	
	
			216 lines
		
	
	
		
			7.5 KiB
		
	
	
	
		
			Python
		
	
	
	
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
 | 
						|
        )
 |