Як отримати тіло відповіді за допомогою HttpURLConnection, коли повертається код, відмінний від 2xx?


95

У мене проблеми з отриманням відповіді Json у випадку, коли сервер повертає помилку. Дивіться деталі нижче.

Як я виконую запит

Я використовую java.net.HttpURLConnection. Я налаштовую властивості запиту, то роблю:

conn = (HttpURLConnection) url.openConnection();

Після цього, коли запит буде успішним, я отримую відповідь Json:

br = new BufferedReader(new InputStreamReader((conn.getInputStream())));
sb = new StringBuilder();
String output;
while ((output = br.readLine()) != null) {
  sb.append(output);
}
return sb.toString();

... і проблема полягає в тому, що:

Я не можу отримати Json, отриманий, коли сервер повертає якусь помилку, наприклад 50x або 40x. Наступний рядок кидає IOException:

br = new BufferedReader(new InputStreamReader((conn.getInputStream())));
// throws java.io.IOException: Server returned HTTP response code: 401 for URL: www.example.com

Сервер точно надсилає тіло, я бачу це у зовнішньому інструменті Burp Suite:

HTTP/1.1 401 Unauthorized

{"type":"AuthApiException","message":"AuthApiException","errors":[{"field":"email","message":"Invalid username and/or password."}]}

Я можу отримати повідомлення відповіді (тобто "Внутрішня помилка сервера") та код (тобто "500"), використовуючи такі методи:

conn.getResponseMessage();
conn.getResponseCode();

Але я не можу отримати тіло запиту ... можливо, є якийсь метод, який я не помітив у бібліотеці?

Відповіді:


132

Якщо код відповіді не 200 або 2xx, використовуйте getErrorStream()замістьgetInputStream().


2
у моєму випадку, маючи код відповіді 403, getErrorStream () повертає null
Sip

68

Щоб зробити речі чіткими, ось мій робочий код:

if (200 <= conn.getResponseCode() && conn.getResponseCode() <= 399) {
    br = new BufferedReader(new InputStreamReader(conn.getInputStream()));
} else {
    br = new BufferedReader(new InputStreamReader(conn.getErrorStream()));
}

3
HttpURLConnection.getErrorStream()( sun.net.www.protocol.httpреалізація) не повертається, nullякщо responseCode >= 400ваш чек на 299, ймовірно, невірний.
vladr

5
Якщо ви використовуєте Java 8, ви можете отримати відповідь у вигляді рядка. String responseBody = br.lines().collect(Collectors.joining());
Lanil Marasinghe

1
Правильний діапазон - 200 ... 399, оскільки 3xx - це стан "перенаправлення", а не помилка.
luca.vercelli

1

Це простий спосіб отримати успішну відповідь від сервера, наприклад, PHP відлуння, інакше повідомлення про помилку.

BufferedReader br = null;
if (conn.getResponseCode() == 200) {
    br = new BufferedReader(new InputStreamReader(conn.getInputStream()));
    String strCurrentLine;
        while ((strCurrentLine = br.readLine()) != null) {
               System.out.println(strCurrentLine);
        }
} else {
    br = new BufferedReader(new InputStreamReader(conn.getErrorStream()));
    String strCurrentLine;
        while ((strCurrentLine = br.readLine()) != null) {
               System.out.println(strCurrentLine);
        }
}
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.