Flask 在处理带有 URL 编码参数的请求时出现问题

Flask problems handling a request with URLencoded parameters

我有一个奇怪的问题;我是 运行 Flask,我有一个 API 取消会员资格的功能。 URL 还包含一个参数来设置取消的原因。这是一个短文本,它可以包含扩展字符,因此该字符串由调用方 URL 编码。

然而,在某些情况下,Flask return在它到达我自己的处理代码之前就出现了 400 错误。例如下面的URL:

curl -X "DELETE" "http://localhost:5000/contracts/C9ABA4AA-834E-4711-91A8-F21057DF693B?date=2015-8-1&canceldate=2015-6-27&booktoday=true&overrideEnddate=true&cancelreason=traslado+a+m%C3%A1s+de+15+km&correctionreason="

给我一个 400 错误:浏览器(或代理)发送了该服务器无法理解的请求。 尽管它似乎是完全有效的 URL . 当我捕获错误时,底层错误数据是:

'ascii' codec can't encode character u'\xe1' in position 12: ordinal not in range(128)

当我从 URL 字符串中删除编码的扩展字符 á (%C3%A1) 时,我发现一切正常。

我也可以通过将我在其他地方找到的解决方法添加到我的应用程序来解决它。init()

import sys
reload(sys)
sys.setdefaultencoding("utf-8")

我明白这个语句强制 Python 解释器默认使用 UTF-8 来解码字节串,而不是 ASCII。
这为我的问题提供了解决方案,但感觉过于复杂和不必要。

所以,真正的问题是,我在这里缺少什么? Flask 中是否有一个我不知道的设置可以解决这个问题,或者 Flask 真的不能处理特定的 URL 编码字符串吗? 我希望 Flask 能够自行处理 URL 编码参数,而无需在非库代码中使用变通方法...

我没有回溯,因为 Flask 会捕获它并将其作为 HTTP 错误和 return 400 HTTP 状态代码进行处理。

编辑: 异常的起源在这部分Flask代码中(site-packages\flask\app.py):

def full_dispatch_request(self):
    """Dispatches the request and on top of that performs request
    pre and postprocessing as well as HTTP exception catching and
    error handling.

    .. versionadded:: 0.7
"""
self.try_trigger_before_first_request_functions()
try:
    request_started.send(self)
    rv = self.preprocess_request()
    if rv is None:
        rv = self.dispatch_request()
except Exception as e:
    rv = self.handle_user_exception(e)
response = self.make_response(rv)
response = self.process_response(response)
request_finished.send(self, response=response)
return response

代码跳到例外:return出现 400 错误。

感谢您的指点。

找到解决办法;看来我在问题中遗漏了一些解决问题所需的基本信息。

我们试图在一个非常简单的单页 Flask 应用程序中重现该问题,因此我可以将其作为示例放在 Whosebug 上。然而,简单的 Flask 测试页面运行良好,所以我们去寻找差异。

我完全忘记了,而我们刚刚注意到的是,我们使用 webargs 库来解析 http 参数。

参数被解析为字符串,并且在webargs中也有一个类型unicode。将类型更改为 unicode 解决了问题。

@api.route('/contracts/<ppl_mshp_id>', methods=['DELETE'])
@use_args({'date': Arg(type_=str, validate=is_date, required=False),
           'canceldate': Arg(type_=str, validate=is_date, required=False),
           'cancelreason': Arg(type_=unicode, required=False, default=u""),
           'correction': Arg(type_=float, required=False),
           'correctionreason': Arg(type_=unicode, required=False, default=u""),
           'restitution': Arg(type_=bool, required=False, default=False),
           'booktoday': Arg(type_=bool, required=False, default=False),
           'overrideEnddate': Arg(type_=bool, required=False, default=False),
           })
def cancel_contract(args, ppl_mshp_id):
...

感谢@ketouem 的时间和麻烦,很抱歉我的问题不完整。