Source code for flask_shortcut.shortcut

from logging import getLogger
from typing import Union, Tuple, Dict, Any, Callable, Optional
from types import MethodType, FunctionType
from functools import wraps
import inspect
import json

from click import secho
from flask import Flask

from flask_shortcut.util import diff, get_request_data

logger = getLogger(__name__)

#: Shorthand for the return type of something pretending to be a route function.
RESPONSE_ARGS = Tuple[Any, int]


[docs]class Shortcut: """Object that handles the shortcut rerouting. Calling an instance of this class on a view function gives the view an option to behave differently in non-production environments. The way in which the behavior may differ is constrained in three possible ways. In the first one, only the arguments for a response are passed to the shortcut definition, meaning that the original view is effectively disabled and the route will instead just return the shortcut's arguments as its only response. In the second one, any number of shortcut response arguments are mapped to condition-keys. The condition is a json-like string that is used to assert a substructure in request bodies that reach that route, and will only apply its respective shortcut-response iff that substructure can be matched. In the third one, a function represents the shortcut and can run arbitrary code to ensure whatever the user deems necessary on the request body, header, etc. Such a shortcut function may not accept arguments and needs to either return None, to signal that the shortcut condition failed and the original logic should be run, or valid response arguments in the form of a tuple. If none of the condition can be satisfied, the route will run its original view. There are two different ways to register shortcuts, one using decorators on the target functions before the are decorated as routes, and the other by running the a single wire call after all routes were added. Example: - Basic app setup: >>> app = Flask(__name__) >>> short = Shortcut(app) - With shortcut decorator: >>> @app.route('/my_route', methods=['GET']) ... @short.cut(('short_ok', 200)) ... def my_func(): ... return 'ok', 200 - Equivalent post route-definition wiring: >>> @app.route('/my_route', methods=['GET']) ... def my_func(): ... return 'ok', 200 >>> short.wire({'/my_route': ('short_ok', 200)}) """ def __init__(self, app: Flask): self.app = app self.exclude = ["production"] if app.config.get("SHORTCUT_EXCLUSIONS"): exclusions = [x.strip() for x in app.config["SHORTCUT_EXCLUSIONS"].split(",") if x] self.exclude.extend(exclusions) if app.env not in self.exclude: logger.info(f"Setting up flask shortcuts in environment '{app.env}'.") secho(" Route shortcuts enabled. Do not do this in any kind of production environment.", fg="red") else: # make .cut(...) return a wrapper that does nothing self.cut = MethodType(lambda _self, mapping: lambda f: f, self) # type: ignore
[docs] def cut(self, mapping: Union[RESPONSE_ARGS, Dict[str, RESPONSE_ARGS], Callable[[], Optional[RESPONSE_ARGS]]]): """Returns view function wrappers. Depending on the input argument, a different wrapper will be returned. This function can only run in applications that are not listed in the `_EXCLUDE` list. Args: mapping: The mapping that decides which types of shortcuts can be offered under which type of condition. """ # wrapper in case only the response is given, assumes that the 'condition' is always True def simple_map(f): f_name = f"{f.__module__}.{f.__name__}" logger.info(f"Adding simple_map shortcut for routing function '{f_name}'.") assert isinstance(mapping, tuple), "Messed up shortcut wiring, abort." # nosec @wraps(f) def decorated(*_, **__): logger.debug(f"Running shortcut for '{f_name}'.") response, status = mapping return self.app.make_response(response), status return decorated # wrapper for dict-based mappings def dict_map(f): f_name = f"{f.__module__}.{f.__name__}" logger.info(f"Adding dict_map shortcut for routing function '{f_name}'.") assert isinstance(mapping, dict), "Messed up shortcut wiring, abort." # nosec for s in mapping: try: json.loads(s) except Exception as e: raise TypeError(f"'{s}' can't be deserialized into valid json, raises '{str(e)}'.") @wraps(f) def decorated(*args, **kwargs): logger.debug(f"Running shortcut for '{f_name}'.") for condition, (response, status) in mapping.items(): request_data = get_request_data() try: sub_resolves = diff(request_data, json.loads(condition)) except TypeError as e_: logger.debug( f"Couldn't walk '{condition}' in the target request, got error message '{str(e_)}'. " f"This could mean that the shortcut for this function is not well-defined." ) continue if not sub_resolves: continue return self.app.make_response(response), status else: logger.debug(f"Shortcut conditions couldn't be satisfied, defaulting to actual implementation.") return f(*args, **kwargs) return decorated # wrapper for function mappings def func_map(f): f_name = f"{f.__module__}.{f.__name__}" logger.info(f"Adding func_map shortcut for routing function '{f_name}'.") assert isinstance(mapping, (FunctionType, list)), "Messed up shortcut wiring, abort." # nosec func_list = mapping if isinstance(mapping, list) else [mapping] for func in func_list: assert isinstance(func, FunctionType), "All mappings in a shortcut lists need to be functions." # nosec assert not inspect.signature(func).parameters, "Mapping functions can't take arguments." # nosec @wraps(f) def decorated(*args, **kwargs): logger.debug(f"Running shortcut for '{f_name}'.") for function in func_list: response = function() if response is not None: return response else: logger.debug(f"Shortcut conditions couldn't be satisfied, defaulting to actual implementation.") return f(*args, **kwargs) return decorated # assigning the right decorator given the mapping types if isinstance(mapping, tuple): return simple_map elif isinstance(mapping, dict): return dict_map elif isinstance(mapping, (FunctionType, list)): return func_map else: raise TypeError(f"'{type(mapping)}' is not a supported mapping type for shortcuts yet.")
[docs] def wire( self, shortcuts: Dict[str, Union[RESPONSE_ARGS, Dict[str, RESPONSE_ARGS], Callable[[], Optional[RESPONSE_ARGS]]]], ): """Manual wiring function. If you don't want to have the shortcut definitions in your routing file for some reason (e.g. there are lots of shortcuts and it would make the whole thing hard to read), you can use this function at some point after all view functions were registered and before the server is started. Args: shortcuts: A dictionary that maps routes to the mappings that would have been used as arguments in the shortcut decorator. """ route_map = {str(r): r.endpoint for r in self.app.url_map.iter_rules()} for route, mapping in shortcuts.items(): if route in route_map: wrap = self.cut(mapping)(self.app.view_functions[route_map[route]]) self.app.view_functions[route_map[route]] = wrap else: logger.warning(f"Can't resolve route '{route}' in the registered routes '{[*route_map]}'")