Coverage for /pythoncovmergedfiles/medio/medio/usr/local/lib/python3.11/site-packages/flask/views.py: 52%
Shortcuts on this page
r m x toggle line displays
j k next/prev highlighted chunk
0 (zero) top of page
1 (one) first highlighted chunk
Shortcuts on this page
r m x toggle line displays
j k next/prev highlighted chunk
0 (zero) top of page
1 (one) first highlighted chunk
1from __future__ import annotations
3import typing as t
5from . import typing as ft
6from .globals import current_app
7from .globals import request
9F = t.TypeVar("F", bound=t.Callable[..., t.Any])
11http_method_funcs = frozenset(
12 ["get", "post", "head", "options", "delete", "put", "trace", "patch"]
13)
16class View:
17 """Subclass this class and override :meth:`dispatch_request` to
18 create a generic class-based view. Call :meth:`as_view` to create a
19 view function that creates an instance of the class with the given
20 arguments and calls its ``dispatch_request`` method with any URL
21 variables.
23 See :doc:`views` for a detailed guide.
25 .. code-block:: python
27 class Hello(View):
28 init_every_request = False
30 def dispatch_request(self, name):
31 return f"Hello, {name}!"
33 app.add_url_rule(
34 "/hello/<name>", view_func=Hello.as_view("hello")
35 )
37 Set :attr:`methods` on the class to change what methods the view
38 accepts.
40 Set :attr:`decorators` on the class to apply a list of decorators to
41 the generated view function. Decorators applied to the class itself
42 will not be applied to the generated view function!
44 Set :attr:`init_every_request` to ``False`` for efficiency, unless
45 you need to store request-global data on ``self``.
46 """
48 #: The methods this view is registered for. Uses the same default
49 #: (``["GET", "HEAD", "OPTIONS"]``) as ``route`` and
50 #: ``add_url_rule`` by default.
51 methods: t.ClassVar[t.Collection[str] | None] = None
53 #: Control whether the ``OPTIONS`` method is handled automatically.
54 #: Uses the same default (``True``) as ``route`` and
55 #: ``add_url_rule`` by default.
56 provide_automatic_options: t.ClassVar[bool | None] = None
58 #: A list of decorators to apply, in order, to the generated view
59 #: function. Remember that ``@decorator`` syntax is applied bottom
60 #: to top, so the first decorator in the list would be the bottom
61 #: decorator.
62 #:
63 #: .. versionadded:: 0.8
64 decorators: t.ClassVar[list[t.Callable[..., t.Any]]] = []
66 #: Create a new instance of this view class for every request by
67 #: default. If a view subclass sets this to ``False``, the same
68 #: instance is used for every request.
69 #:
70 #: A single instance is more efficient, especially if complex setup
71 #: is done during init. However, storing data on ``self`` is no
72 #: longer safe across requests, and :data:`~flask.g` should be used
73 #: instead.
74 #:
75 #: .. versionadded:: 2.2
76 init_every_request: t.ClassVar[bool] = True
78 def dispatch_request(self) -> ft.ResponseReturnValue:
79 """The actual view function behavior. Subclasses must override
80 this and return a valid response. Any variables from the URL
81 rule are passed as keyword arguments.
82 """
83 raise NotImplementedError()
85 @classmethod
86 def as_view(
87 cls, name: str, *class_args: t.Any, **class_kwargs: t.Any
88 ) -> ft.RouteCallable:
89 """Convert the class into a view function that can be registered
90 for a route.
92 By default, the generated view will create a new instance of the
93 view class for every request and call its
94 :meth:`dispatch_request` method. If the view class sets
95 :attr:`init_every_request` to ``False``, the same instance will
96 be used for every request.
98 Except for ``name``, all other arguments passed to this method
99 are forwarded to the view class ``__init__`` method.
101 .. versionchanged:: 2.2
102 Added the ``init_every_request`` class attribute.
103 """
104 if cls.init_every_request:
106 def view(**kwargs: t.Any) -> ft.ResponseReturnValue:
107 self = view.view_class( # type: ignore[attr-defined]
108 *class_args, **class_kwargs
109 )
110 return current_app.ensure_sync(self.dispatch_request)(**kwargs) # type: ignore[no-any-return]
112 else:
113 self = cls(*class_args, **class_kwargs) # pyright: ignore
115 def view(**kwargs: t.Any) -> ft.ResponseReturnValue:
116 return current_app.ensure_sync(self.dispatch_request)(**kwargs) # type: ignore[no-any-return]
118 if cls.decorators:
119 view.__name__ = name
120 view.__module__ = cls.__module__
121 for decorator in cls.decorators:
122 view = decorator(view)
124 # We attach the view class to the view function for two reasons:
125 # first of all it allows us to easily figure out what class-based
126 # view this thing came from, secondly it's also used for instantiating
127 # the view class so you can actually replace it with something else
128 # for testing purposes and debugging.
129 view.view_class = cls # type: ignore
130 view.__name__ = name
131 view.__doc__ = cls.__doc__
132 view.__module__ = cls.__module__
133 view.methods = cls.methods # type: ignore
134 view.provide_automatic_options = cls.provide_automatic_options # type: ignore
135 return view
138class MethodView(View):
139 """Dispatches request methods to the corresponding instance methods.
140 For example, if you implement a ``get`` method, it will be used to
141 handle ``GET`` requests.
143 This can be useful for defining a REST API.
145 :attr:`methods` is automatically set based on the methods defined on
146 the class.
148 See :doc:`views` for a detailed guide.
150 .. code-block:: python
152 class CounterAPI(MethodView):
153 def get(self):
154 return str(session.get("counter", 0))
156 def post(self):
157 session["counter"] = session.get("counter", 0) + 1
158 return redirect(url_for("counter"))
160 app.add_url_rule(
161 "/counter", view_func=CounterAPI.as_view("counter")
162 )
163 """
165 def __init_subclass__(cls, **kwargs: t.Any) -> None:
166 super().__init_subclass__(**kwargs)
168 if "methods" not in cls.__dict__:
169 methods = set()
171 for base in cls.__bases__:
172 if getattr(base, "methods", None):
173 methods.update(base.methods) # type: ignore[attr-defined]
175 for key in http_method_funcs:
176 if hasattr(cls, key):
177 methods.add(key.upper())
179 if methods:
180 cls.methods = methods
182 def dispatch_request(self, **kwargs: t.Any) -> ft.ResponseReturnValue:
183 meth = getattr(self, request.method.lower(), None)
185 # If the request method is HEAD and we don't have a handler for it
186 # retry with GET.
187 if meth is None and request.method == "HEAD":
188 meth = getattr(self, "get", None)
190 assert meth is not None, f"Unimplemented method {request.method!r}"
191 return current_app.ensure_sync(meth)(**kwargs) # type: ignore[no-any-return]