Проблемы с колбой при обработке запроса с параметрами URLencoded

У меня странная проблема; Я использую 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)

Я обнаружил, что все хорошо, когда я удаляю закодированный расширенный символ á (% C3% A1) из строки URL.

Я также могу решить эту проблему, добавив этот обходной путь, который я нашел в другом месте моего приложения.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

Код переходит к исключением: и возвращает ошибку 400.

Спасибо за любые указатели, которые вы можете дать мне.

1 ответ

Решение

Нашел решение; И, похоже, я пропустил некоторую важную информацию в своем вопросе, которая была необходима для решения проблемы.

Мы попытались продублировать проблему в очень простом одностраничном приложении Flask, поэтому я могу привести его здесь, в качестве примера, в Stackru. Однако простая тестовая страница 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 за потраченное время и проблемы, и извините за то, что не укомплектованы в моем вопросе.

Другие вопросы по тегам