Расширенная маршрутизация WebApp2
-
27-10-2019 - |
Вопрос
В Old School WebApp, маршрутизация на основе App.yaml позволила вам проверить регулярные выражения, и, если она соответствует, обработайте запрос по обработчику, чье имя основано на найденном шаблоне. Например, 1_handler.py будет отправлять на user_handler.py, если соответствующий шаблон был «пользователем».
Есть ли способ сделать то же самое с WebApp2.Route? Может ли ленивый обработчик или параметры method_handler быть основаны на соответствующих шаблонах в шаблоне?
Решение
WebApp2 не позволяет таким маршрутизации. Я думаю, что самое разумное решение - написать пользовательский диспетчер для webapp2.Router
.
Можно установить пользовательский диспетчер, как это:
app = WSGIApplication(...)
app.router.set_dispatcher(custom_dispatcher)
Вот не тестируемый эскиз Для диспетчера код основан на webapp2.Router.default_dispatcher
:
from webapp2 import import_string
def custom_dispatcher(router, request, response):
route, args, kwargs = rv = router.match(request)
request.route, request.route_args, request.route_kwargs = rv
handler = route.handler
if isinstance(handler, basestring):
handler, args, kwargs = _parse_handler_template(handler, args, kwargs)
if handler not in self.handlers:
router.handlers[handler] = handler = import_string(handler)
else:
handler = router.handlers[handler]
return router.adapt(handler)(request, response)
def _parse_handler_template(handler, args, kwargs):
"""replace {key} in `handler` with values from `args` or `kwargs`.
Replaced values are removed from args/kwargs."""
args = list(args)
kwargs = dict(kwargs)
def sub(match):
if kwargs:
return kwargs.pop(match.group().strip('{}'))
else:
return args.pop(int(match.group().strip('{}'))
return re.sub('{.*?}', sub, handler), args, kwargs
Этот код должен разрешить зарегистрировать такие правила:
app = WSGIApplication([
(r'module/<module>/<action>/<argument>', 'modules.{module}.action_{action}'),
])
Этот пример не позволяет использовать переменные из шаблона в имени метода, например: module.Class:action_{method}
. Анкет В Route
класс эта конечная точка разделена с помощью полуколона и значения, хранящиеся в route.method_name
а также route.handler
.