使用 Webargs Flaskparser 验证 URL 路径和查询参数中的变量参数

stw*_*ite 2 python flask python-3.x webargs

我正在尝试验证发送到我的 api 的所有数据。我的 url 结构中包含一个变量 ,/api/v2/users/<string:username>/collections/以及实际的查询字符串参数,所有这些都需要通过验证。

完整的网址如下所示: https://127.0.0.1:5000/api/v2/users/<string:username>/collections/?page=5&per_page=10

需要验证的两个变量是:username page、 和per_page。简单的解决方案是更改我的 url 结构,但我想知道是否可以完成我需要的内容并保持我当前的简单性,而无需在资源类中添加额外的验证。如果可能的话,你会怎么做?

class UserCollections(Resource):
    @use_args({
        'username': fields.Str(
            required=True,
            validate=username_length,
            error_messages=dict(
                required='Username is required.',
                validator_failed='Username can be between 3 and 25 characters.',
            )
        ),
        'page': fields.Int(
            #required=True,
            missing=1,
            validate=feed_minmax_pages,
            error_messages=dict(
                validator_failed='Maximum number of pages reached.',
            )
        ),
        'per_page': fields.Int(
            #required=True,
            missing=5,
            validate=validate.Range(min=5,max=25),
            error_messages=dict(
                validator_failed='Test number of pages reached.',
            )
        ),
    }, locations=('query',))

    def get(self, args, username):
        print(args)
        print(username)

        return default_schema(
            data={},
            http_status=200
        )
Run Code Online (Sandbox Code Playgroud)

当我运行代码时,我收到验证错误,username因为它不存在于参数中。

stw*_*ite 5

经过一段时间的摸索后,我找到了解决方案,我现在在webargs Flaskparser 文档中看到了该解决方案

use_args location除了参数中的其他参数之外,还可以轻松地使用它。看来请求方法 function getpost等仍然需要您传入该 url 变量。就我而言,是的<username>

class UserCollections(Resource):
    @use_args({
        'username': fields.Str(
            location='view_args',
            required=True,
            validate=username_length,
            error_messages=dict(
                required='Username is required.',
                validator_failed='Username can be between 3 and 25 characters.',
            )
        ),
        'page': fields.Int(
            location='query',
            missing=1,
            validate=feed_minmax_pages,
            error_messages=dict(
                validator_failed='Maximum number of pages reached.',
            )
        ),
        'per_page': fields.Int(
            location='query',
            missing=5,
            validate=validate.Range(min=5,max=25),
            error_messages=dict(
                validator_failed='Test number of pages reached.',
            )
        ),
    })

    def get(self, args, username):
        print(args) # access with args['username']
        print(username) # would be nice to not have a second of the same var

        return default_schema(
            data={},
            http_status=200
        )
Run Code Online (Sandbox Code Playgroud)