Flask的网址路由中是否支持正则表达式?


100

我知道Flask具有int,float和path转换器,但是我们正在开发的应用程序的URL中具有更复杂的模式。

有没有办法像Django一样使用正则表达式?

Answers:


192

即使Armin用可接受的答案击败了我,我仍然认为我会显示一个简短的示例,说明如何在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 /

该网址应返回404:http:// localhost:5000 / abcd-foo /


4
但这是否意味着正则表达式已编译,或者它们是即时计算的?
Games Brainiac

1
看起来正则表达式将在运行时直接评估。对于较小的应用程序(或者我认为是多次重复使用regex的应用程序),这应该没有问题,因为最后两个regex模式已编译存储在内存中。
bbenne10

5
这是如何运作的?模式设置为self.regex,但是匹配在哪里发生?
贾斯汀

@Justin匹配发生在Werkzeug的内部,这里和我未找到的规则定义中的某个地方。
AlexLordThorsen '16

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

这将匹配任何请求。在此处查看更多详细信息:捕获所有URL


我有一个错误,您能给我一些提示吗?在<module> @ app.route('/ <path:path>',methods = ['GET'])中的文件“ /app/catch_all.py”,第234行,文件“ / 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
触发
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.