Source code for flask_unchained.bundles.controller.route

import inspect
from types import FunctionType
from typing import *

from flask import Blueprint
from flask_unchained.flask_unchained import FlaskUnchained
from py_meta_utils import _missing
from werkzeug.utils import cached_property

from .utils import (join, method_name_to_url, rename_parent_resource_param_name,
                    controller_name, get_param_tuples)


[docs]class Route: """ This is a semi-private class that you most likely shouldn't use directly. Instead, you should use the public functions in :ref:`bundles/controller:Declarative Routing`, and the :func:`~flask_unchained.route` and :func:`~flask_unchained.no_route` decorators. This class is used to store an *intermediate* representation of route details as an attribute on view functions and class view methods. Most notably, this class's :attr:`rule` and :attr:`full_rule` attributes may not represent the final url rule that gets registered with :class:`~flask.Flask`. Further gotchas with :class:`~flask_unchained.Controller` and :class:`~flask_unchained.Resource` routes include that their view_func must be finalized from the outside using ``TheControllerClass.method_as_view``. """ def __init__(self, rule: Union[str, None], view_func: Union[str, FunctionType], blueprint: Optional[Blueprint] = None, defaults: Optional[Dict[str, Any]] = None, endpoint: Optional[str] = None, is_member: bool = False, methods: Optional[Union[List[str], Tuple[str, ...]]] = None, only_if: Optional[Union[bool, FunctionType]] = _missing, **rule_options, ) -> None: self._blueprint: Optional[Blueprint] = blueprint self._defaults: Dict[str, Any] = defaults or {} self._endpoint: str = endpoint self._methods: Optional[Union[List[str], Tuple[str, ...]]] = methods self.only_if: Optional[Union[bool, FunctionType]] = only_if self._rule: str = rule self.rule_options: Dict[str, Any] = rule_options self.view_func: Union[str, FunctionType] = view_func # private self._controller_cls = None self._member_param: Optional[str] = None self._unique_member_param: Optional[str] = None self._parent_resource_cls = None self._parent_member_param: Optional[str] = None self._is_member: bool = is_member """ Whether or not this route should be a member method of the parent resource. """ self._is_member_method: bool = False """ Whether or not this route is a member method of this route's resource class. """
[docs] def should_register(self, app: FlaskUnchained) -> bool: """ Determines whether or not this route should be registered with the app, based on :attr:`only_if`. """ if self.only_if in {None, _missing}: return True elif callable(self.only_if): return self.only_if(app) return bool(self.only_if)
@property def blueprint(self) -> Union[Blueprint, None]: if self._blueprint is _missing: return None return self._blueprint @blueprint.setter def blueprint(self, blueprint: Blueprint): self._blueprint = blueprint @property def bp_prefix(self) -> Union[str, None]: if not self.blueprint: return None return self.blueprint.url_prefix @property def bp_name(self) -> Union[str, None]: if not self.blueprint: return None return self.blueprint.name @property def defaults(self) -> Dict[str, Any]: """ The URL defaults for this route. """ if self._defaults is _missing: return {} return self._defaults @defaults.setter def defaults(self, defaults: Dict[str, Any]): self._defaults = defaults or {} @property def endpoint(self) -> str: """ The endpoint for this route. """ if self._endpoint: return self._endpoint elif self._controller_cls: endpoint = f'{self._controller_cls.Meta.endpoint_prefix}.{self.method_name}' return endpoint if not self.bp_name else f'{self.bp_name}.{endpoint}' elif self.bp_name: return f'{self.bp_name}.{self.method_name}' return self.method_name @endpoint.setter def endpoint(self, endpoint: str): self._endpoint = endpoint @property def is_member(self) -> bool: """ Whether or not this route is for a resource member route. """ if self._is_member is _missing: return False return self._is_member @is_member.setter def is_member(self, is_member: bool): self._is_member = is_member @property def method_name(self) -> str: """ The string name of this route's view function. """ if isinstance(self.view_func, str): return self.view_func return self.view_func.__name__ @property def methods(self) -> Union[List[str], Tuple[str, ...]]: """ The HTTP methods supported by this route. """ return getattr(self.view_func, 'methods', self._methods) or ['GET'] @methods.setter def methods(self, methods: Union[List[str], Tuple[str, ...]]): self._methods = methods
[docs] @cached_property def module_name(self) -> Union[str, None]: """ The module where this route's view function was defined. """ if not self.view_func: return None elif self._controller_cls: return inspect.getmodule(self._controller_cls).__name__ return inspect.getmodule(self.view_func).__name__
@property def rule(self) -> str: """ The (partial) url rule for this route. """ if self._rule: return self._rule return self._make_rule(member_param=self._member_param, unique_member_param=self._unique_member_param) @rule.setter def rule(self, rule: str): if rule is not None and not rule.startswith('/'): rule = '/' + rule self._rule = rule @property def full_rule(self) -> str: """ The full url rule for this route, including any blueprint prefix. """ rule = self.rule return join(self.bp_prefix, rule, trailing_slash=rule.endswith('/')) def _make_rule(self, url_prefix: Optional[str] = None, member_param: Optional[str] = None, unique_member_param: Optional[str] = None, ) -> str: if member_param is not None: self._member_param = member_param if unique_member_param is not None: self._unique_member_param = unique_member_param if self._rule: return join(url_prefix, self._rule, trailing_slash=( self._rule != '/' and self._rule.endswith('/') )) elif self._controller_cls: rule = method_name_to_url(self.method_name) if (self._is_member or self._is_member_method) and not member_param: raise Exception('member_param argument is required') if self._is_member_method: rule = member_param elif self._is_member: rule = rename_parent_resource_param_name(self, join(member_param, rule)) return join(url_prefix, rule) return method_name_to_url(self.method_name) @property def unique_member_param(self) -> Union[str, None]: if not (self.is_member or self._is_member_method): return None elif self._unique_member_param: return self._unique_member_param # FIXME should probably use the snake_case singular of the resource's model name ctrl_name = controller_name(self._controller_cls) type_, name = get_param_tuples(self._member_param)[0] return f'<{type_}:{ctrl_name}_{name}>' def copy(self): new = object.__new__(Route) new.__dict__ = self.__dict__.copy() return new @property def full_name(self) -> Union[str, None]: """ The full name of this route's view function, including the module path and controller name, if any. """ if not self.view_func: return None prefix = self.view_func.__module__ if self._controller_cls: module_name = self._controller_cls.__module__ class_name = self._controller_cls.__name__ prefix = f'{module_name}.{class_name}' return f'{prefix}.{self.method_name}' def __repr__(self): props = [prop for prop in ['full_name', 'endpoint', 'methods', 'defaults'] if getattr(self, prop)] try: self.rule and props.insert(0, 'rule') except: pass return f"Route({', '.join(f'{k}={repr(getattr(self, k))}' for k in props)})"
__all__ = [ 'Route', ]