Source code for flask_unchained.bundles.controller.hooks.register_routes_hook

import inspect
import itertools

from flask_unchained import AppFactoryHook, Bundle, FlaskUnchained, unchained
from flask_unchained._compat import is_local_proxy
from typing import *

from ..attr_constants import CONTROLLER_ROUTES_ATTR, FN_ROUTES_ATTR
from ..controller import Controller
from ..resource import Resource
from ..route import Route
from ..routes import _reduce_routes, controller, resource, include


[docs]class RegisterRoutesHook(AppFactoryHook): """ Registers routes. """ name = 'routes' """ The name of this hook. """ bundle_module_name = 'routes' """ The default module this hook loads from. Override by setting the ``routes_module_name`` attribute on your bundle class. """ require_exactly_one_bundle_module = True run_before = ['blueprints', 'bundle_blueprints']
[docs] def run_hook(self, app: FlaskUnchained, bundles: List[Bundle], unchained_config: Optional[Dict[str, Any]] = None, ) -> None: """ Discover and register routes. """ app_bundle = bundles[-1] try: self.import_bundle_modules(app_bundle)[0] except IndexError: routes = self.collect_from_bundle(app_bundle) else: try: routes = self.get_explicit_routes(app_bundle) except AttributeError as e: if not app_bundle.is_single_module: raise e routes = self.collect_from_bundle(app_bundle) self.process_objects(app, routes)
# skipcq: PYL-W0221 (parameters mismatch in overridden method)
[docs] def process_objects(self, app: FlaskUnchained, routes: Iterable[Route]): """ Organize routes by where they came from, and then register them with the app. """ for route in _reduce_routes(routes): if route.should_register(app): existing_routes = (self.bundle.endpoints.get(route.endpoint, None) if route.module_name else None) if existing_routes: import warnings warnings.warn(f'Skipping duplicate latter route: ' f'{existing_routes[0]} precedes {route}') continue self.bundle.endpoints[route.endpoint].append(route) if route._controller_cls: key = f'{route._controller_cls.__name__}.{route.method_name}' self.bundle.controller_endpoints[key].append(route) # build up a list of bundles with views: bundle_module_names = [] # [tuple(top_bundle_module_name, hierarchy_module_names)] for bundle in app.unchained.bundles.values(): hierarchy = [bundle_super.module_name for bundle_super in bundle._iter_class_hierarchy() if bundle_super._has_views] if hierarchy: bundle_module_names.append((bundle.module_name, hierarchy)) # for each route, figure out which bundle hierarchy it's from, and assign the # route to the top bundle for that hierarchy bundle_route_endpoints = set() for endpoint, endpoint_routes in self.bundle.endpoints.items(): for route in endpoint_routes: if not route.module_name: continue # FIXME would be nice for routes to know which bundle they're from... for top_level_bundle_module_name, hierarchy in bundle_module_names: for bundle_module_name in hierarchy: if route.module_name.startswith(bundle_module_name): self.bundle.bundle_routes[ top_level_bundle_module_name ].append(route) bundle_route_endpoints.add(endpoint) break # get all the remaining routes not belonging to a bundle self.bundle.other_routes = itertools.chain.from_iterable([ routes for endpoint, routes in self.bundle.endpoints.items() if endpoint not in bundle_route_endpoints ]) # we register non-bundle routes with the app here, and # the RegisterBundleBlueprintsHook registers the bundle routes for route in self.bundle.other_routes: app.add_url_rule(route.full_rule, defaults=route.defaults, endpoint=route.endpoint, methods=route.methods, view_func=route.view_func, **route.rule_options)
[docs] def get_explicit_routes(self, bundle: Bundle): """ Collect routes from a bundle using declarative routing. """ routes_module = self.import_bundle_modules(bundle)[0] try: return getattr(routes_module, 'routes')() except AttributeError: module_name = self.get_module_names(bundle)[0] raise AttributeError(f'Could not find a variable named `routes` ' f'in the {module_name} module!')
[docs] def collect_from_bundle(self, bundle: Bundle): """ Collect routes from a bundle when not using declarative routing. """ if not bundle._has_views: return () from flask_unchained.hooks.views_hook import ViewsHook for views_module in ViewsHook.import_bundle_modules(bundle): for _, obj in inspect.getmembers(views_module, self.type_check): if hasattr(obj, FN_ROUTES_ATTR): yield getattr(obj, FN_ROUTES_ATTR) elif issubclass(obj, Resource): yield from resource(obj) elif issubclass(obj, Controller): yield from controller(obj) else: raise NotImplementedError try: yield from include(views_module.__name__) except AttributeError: return ()
[docs] def type_check(self, obj): """ Returns True if ``obj`` was decorated with :func:`~flask_unchained.route` or if ``obj`` is a controller or resource with views. """ if obj is unchained or is_local_proxy(obj): return False is_controller = isinstance(getattr(obj, CONTROLLER_ROUTES_ATTR, None), dict) is_view_fn = isinstance(getattr(obj, FN_ROUTES_ATTR, None), list) return is_controller or is_view_fn