Чи підтримує Flask регулярні вирази в маршрутизації URL-адрес?


100

Я розумію, що Flask має перетворювачі int, float і path, але додаток, який ми розробляємо, має більш складні зразки у своїх URL-адресах.

Чи є спосіб, як ми можемо використовувати регулярні вирази, як у Django?

Відповіді:


192

Навіть незважаючи на те, що Армін побив мене до удару прийнятою відповіддю, я подумав, що я покажу скорочений приклад того, як я реалізував мат-регекс у 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
Але чи означає це, що регулярні вирази складаються, або вони оцінюються на льоту?
Ігри Brainiac

1
Схоже, що регулярне вираження буде оцінено безпосередньо під час виконання. Це не повинно бути проблематично для менших додатків (або програм, які повторно використовують regex кілька разів, я думаю, що), оскільки остання пара зразків регулярних виразів зберігається зібраними в пам'яті.
bbenne10

5
Як це працює? Встановлено шаблон self.regex, але де відбувається збіг?
Джастін

@Justin Збіг відбувається у внутрішніх документах Веркзеуга Тут і десь у визначенні правила, яке я не знайшов.
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робить
сердечний

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()

Це відповідатиме будь-якому запиту. Докладнішу інформацію див. Тут: URL-адреса Catch-All .


У мене помилка, ви б дали мені підказки? Файл "/app/catch_all.py", рядок 234, у <module> @ app.route ('/ <шлях: шлях>', методи = ['GET']) Файл "/ usr / local / lib / python2. 7 / dist-пакети / flask / app.py ", рядок 1080, у файлі декоратора" /usr/local/lib/python2.7/dist-packages/flask/app.py ", рядок 64, у файлі wrapper_func" / usr / local / lib / python2.7 / dist-пакети / flask / app.py ", рядок 1051, у add_url_rule" існуюча функція кінцевої точки:% s '% endpoint) AssertionError: Перегляд функції відображення замінює існуючу функцію кінцевої точки: тест
іскра
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.