使用URLencoded参数处理请求的Flask问题

时间:2015-06-29 08:47:43

标签: python flask webargs

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

但在某些情况下,Flask会在它到达我自己的处理代码之前返回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错误处理并返回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

代码跳到except:并返回400错误。

感谢您给我的任何指示。

1 个答案:

答案 0 :(得分:1)

找到解决方案;似乎我在解决问题所需的问题中遗漏了一些基本信息。

我们尝试在一个非常简单的单页Flask应用程序中复制问题,因此我可以将它放在StackOverflow上作为示例。然而,简单的Flask testpage运行良好,所以我们去寻找差异。

我完全忘记了,当时我们注意到,我们使用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的时间和麻烦,对不起我的问题不完整。