Поддерживает ли Flask регулярные выражения в маршрутизации URL-адресов?

101

Я понимаю, что во Flask есть преобразователи int, float и path, но в разрабатываемом нами приложении есть более сложные шаблоны в URL-адресах.

Есть ли способ использовать регулярные выражения, как в Django?

Алистер
источник

Ответы:

193

Несмотря на то, что Армин победил меня принятым ответом, я подумал, что покажу сокращенный пример того, как я реализовал сопоставление регулярных выражений во Flask, на случай, если кому-то понадобится рабочий пример того, как это можно сделать.

from flask import Flask
from werkzeug.routing import BaseConverter

app = Flask(__name__)

class RegexConverter(BaseConverter):
    def __init__(self, url_map, *items):
        super(RegexConverter, self).__init__(url_map)
        self.regex = items[0]


app.url_map.converters['regex'] = RegexConverter

@app.route('/<regex("[abcABC0-9]{4,6}"):uid>-<slug>/')
def example(uid, slug):
    return "uid: %s, slug: %s" % (uid, slug)


if __name__ == '__main__':
    app.run(debug=True, host='0.0.0.0', port=5000)

этот URL должен возвращать 200: http: // localhost: 5000 / abc0-foo /

этот URL-адрес должен вернуть 404: http: // localhost: 5000 / abcd-foo /

Филип Саутэм
источник
4
Но означает ли это, что регулярные выражения компилируются или вычисляются на лету?
Games Brainiac
1
Похоже, регулярное выражение будет оцениваться непосредственно во время выполнения. Это не должно быть проблемой для небольших приложений (или приложений, которые повторно используют регулярные выражения несколько раз, я бы подумал), поскольку последние несколько шаблонов регулярных выражений хранятся в скомпилированном виде в памяти.
bbenne10
6
Как это работает? Шаблон установлен self.regex, но где происходит совпадение?
Джастин
@Justin Сопоставление происходит во внутреннем устройстве Werkzeug Здесь и где-то в определении правила, которое я не нашел.
AlexLordThorsen
49

Вы можете подключить пользовательские преобразователи, которые соответствуют произвольным выражениям: Пользовательский преобразователь

from random import randrange
from werkzeug.routing import Rule, Map, BaseConverter, ValidationError

class BooleanConverter(BaseConverter):

    def __init__(self, url_map, randomify=False):
        super(BooleanConverter, self).__init__(url_map)
        self.randomify = randomify
        self.regex = '(?:yes|no|maybe)'

    def to_python(self, value):
        if value == 'maybe':
            if self.randomify:
                return not randrange(2)
            raise ValidationError()
        return value == 'yes'

    def to_url(self, value):
        return value and 'yes' or 'no'

url_map = Map([
    Rule('/vote/<bool:werkzeug_rocks>', endpoint='vote'),
    Rule('/vote/<bool(randomify=True):foo>', endpoint='foo')
], converters={'bool': BooleanConverter})
Армин Ронахер
источник
Я не понимаю, что to_pythonзначит
corvid
17

Вы также можете написать перехват любого типа маршрута и выполнить сложную маршрутизацию внутри метода:

from flask import Flask
app = Flask(__name__)

@app.route('/', methods=['GET', 'POST'], defaults={'path': ''})
@app.route('/<path:path>', methods=['GET', 'POST'])
def catch_all(path):
    return 'You want path: %s' % path

if __name__ == '__main__':
    app.run()

Это будет соответствовать любому запросу. Подробнее см. Здесь: Catch-All URL .

Зорайр
источник
У меня ошибка. Не могли бы вы подсказать? Файл «/app/catch_all.py», строка 234, в <module> @ app.route ('/ <path: path>', methods = ['GET']) Файл «/ usr / local / lib / python2. 7 / dist-packages / flask / app.py ", строка 1080, в файле декоратора" /usr/local/lib/python2.7/dist-packages/flask/app.py ", строка 64, в файле wrapper_func" / usr / local / lib / python2.7 / dist-packages / flask / app.py ", строка 1051, в add_url_rule 'существующая функция конечной точки:% s'% конечная точка) AssertionError: отображение функции просмотра перезаписывает существующую функцию конечной точки: test
Spark