Czy Flask obsługuje wyrażenia regularne w routingu adresów URL?

101

Rozumiem, że Flask ma konwertery int, float i path, ale tworzona przez nas aplikacja ma bardziej złożone wzorce w swoich adresach URL.

Czy jest sposób, w jaki możemy używać wyrażeń regularnych, jak w Django?

Alistair
źródło

Odpowiedzi:

193

Mimo że Armin pokonał mnie po raz pierwszy z akceptowaną odpowiedzią, pomyślałem, że pokażę skrócony przykład tego, jak zaimplementowałem dopasowywanie wyrażeń regularnych w Flasku, na wypadek gdyby ktoś chciał mieć działający przykład, jak można to zrobić.

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)

ten adres URL powinien zwrócić 200: http: // localhost: 5000 / abc0-foo /

ten adres URL powinien zwrócić 404: http: // localhost: 5000 / abcd-foo /

Philip Southam
źródło
4
Ale czy to oznacza, że ​​wyrażenia regularne są kompilowane, czy też są oceniane w locie?
Games Brainiac
1
Wygląda na to, że wyrażenie regularne zostanie ocenione bezpośrednio w czasie wykonywania. Nie powinno to stanowić problemu dla mniejszych aplikacji (lub aplikacji, które wielokrotnie używają wyrażeń regularnych, jak sądzę), ponieważ ostatnie kilka wzorców wyrażeń regularnych jest przechowywanych w pamięci.
bbenne10
6
Jak to działa? Wzorzec jest ustawiony na self.regex, ale gdzie ma miejsce dopasowanie?
Justin
@Justin Dopasowanie zachodzi we wnętrzu Werkzeuga Tutaj i gdzieś w definicji reguły, której nie znalazłem.
AlexLordThorsen
49

Możesz podłączyć niestandardowe konwertery pasujące do dowolnych wyrażeń: Custom Converter

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})
Armin Ronacher
źródło
Nie rozumiem, co to_pythonrobi
corvid
17

Możesz również napisać przechwytywanie wszystkich typów tras i wykonywać złożone trasy w ramach metody:

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

To pasuje do każdego żądania. Zobacz więcej szczegółów tutaj: Catch-All URL .

Zorayr
źródło
Mam błąd, czy możesz dać mi jakieś wskazówki? Plik "/app/catch_all.py", wiersz 234, w <module> @ app.route ('/ <path: path>', methods = ['GET']) File "/ usr / local / lib / python2. 7 / dist-packages / flask / app.py ", wiersz 1080, w pliku dekoratora" /usr/local/lib/python2.7/dist-packages/flask/app.py ", wiersz 64, w pliku wrapper_func" / usr / local / lib / python2.7 / dist-packages / flask / app.py ", wiersz 1051, w istniejącej funkcji punktu końcowego add_url_rule:% s '% endpoint) AssertionError: Mapowanie funkcji widoku zastępuje istniejącą funkcję punktu końcowego: test
iskra