Таймаут WebException против таймаута HttpWebResponse

Я использую Asp.Net WebClient для публикации http. У меня есть попытка поймать код, чтобы поймать WebException.

        try
        {
            using (MyWebClient wc = new MyWebClient())
            {
                wc.Headers[HttpRequestHeader.ContentType] = _lender.ContentType;
                wc.Timeout = 200;

                return _lender.GetResult(wc.UploadString(_lender.PostUri, _lender.PostValues));
            }
        }
        catch (WebException ex)
        {
            return new ServiceError(ex.Status.ToString());
        }

Основные исключения, которые я ищу, - это таймауты. Я расширил WebClient, чтобы установить время ожидания.

Когда я устанавливаю тайм-аут, равный 100 мс, как и ожидалось, генерируется исключение. Я могу получить статус webexception в соответствии с примером (он возвращает «таймаут»), однако я хочу также вернуть коды статуса.

Если я развернусь, чтобы получить httpwebresponse с помощью ex.Response, я получу возвращаемое значение null, когда я ожидал связанный код состояния. Почему я не получаю HttpStatus.Request.Timeout?


person dotnetnoob    schedule 06.05.2013    source источник


Ответы (1)


У меня та же проблема, и я понимаю несколько вещей, пока ищу решение.

  • WebExceptionStatus enum не эквивалентен коду состояния http, возвращаемому вызываемым API. Вместо этого это список возможных ошибок, которые могут произойти во время HTTP-вызова.
  • Код ошибки WebExceptionStatus, который будет возвращен, когда вы получите сообщение об ошибке (от 400 до 599) от вашего API, - это WebExceptionStatus.ProtocolError, он же номер 7 как int.
  • Когда вам нужно получить тело ответа или реальный код статуса http, возвращенный из api, сначала вам нужно проверить, является ли WebException.Status WebExceptionStatus.ProtocolError. Тогда вы сможете получить реальный ответ от WebExceptionStatus.Response и прочитать его содержание.
  • Иногда тайм-аут обрабатывается вызывающей стороной (также известной как ваш код), поэтому в этом случае у вас нет ответа. Итак, вы можете посмотреть, WebException.Status WebExceptionStatus.Timeout

Это пример:

try
{
    ...
}
catch (WebException webException)
{
    if (webException.Status == WebExceptionStatus.ProtocolError)
    {
        var httpResponse = (HttpWebResponse)webException.Response;
        var responseText = "";
        using (var content = new StreamReader(httpResponse.GetResponseStream()))
        {
            responseText = content.ReadToEnd(); // Get response body as text
        }
        int statusCode = (int)httpResponse.StatusCode; // Get the status code
    }
    else if (webException.Status == WebExceptionStatus.ProtocolError)
    {
       // Timeout handled by your code. You do not have a response here.
    }

    // Handle other webException.Status errors. You do not have a response here.
}
person Lutti Coelho    schedule 29.07.2020