Flask 在 URL 路由中支持正则表达式吗?

问题描述 投票:0回答:3

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

有没有一种方法可以像在 Django 中一样使用正则表达式?

python regex flask
3个回答
210
投票

尽管 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/

此 URL 应返回 404:http://localhost:5000/abcd-foo/


50
投票

您可以挂钩与任意表达式匹配的自定义转换器:自定义转换器

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

21
投票

您还可以编写一个捕获所有类型的路由并在方法中执行复杂的路由:

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

© www.soinside.com 2019 - 2024. All rights reserved.