HttpClient로 Restful server의 데이터를 가져올 때, 간혹 서버 문제로 응답이 없을 수 있습니다. 처음부터 서버에 문제가 있다면 HttpHostConnectException 등의 예외가 발생하여 예외처리를 할 수 있지만, 연결이 느려지는 등의 문제가 발생할 때 오랜 시간동안 대기하게 됩니다. 기본적으로 Timeout이 적용되어있지 않기 때문에 오랜 시간 기다리게 됩니다.
다음과 같은 코드로 HttpClient에 Timeout을 설정할 수 있습니다. RequestConfig 생성 및 Timeout 시간을 설정하고 setConfig()
로 request에 적용합니다. 이 Request를 사용할 때 Timeout이 발생하면 ConnectTimeoutException
이 발생하여 예외 처리를 할 수 있습니다.
final String URL = "http://localhost:3000/root";
final HttpGet request = new HttpGet(URL);
// Apply timeout for the request
RequestConfig requestConfig = RequestConfig.custom()
.setSocketTimeout(1 * 1000)
.setConnectTimeout(1 * 1000)
.setConnectionRequestTimeout(1 * 1000)
.build();
request.setConfig(requestConfig);
1. Example : Timeout이 적용되지 않은 코드
다음은 단순히 Restful server에서 데이터를 가져오는 코드입니다. Timeout이 적용되지 않았기 때문에 서버 연결에 문제가 있을 때 오랜 시간 대기할 수 있습니다.
import org.apache.http.HttpEntity;
import org.apache.http.client.config.RequestConfig;
import org.apache.http.client.methods.CloseableHttpResponse;
import org.apache.http.client.methods.HttpGet;
import org.apache.http.impl.client.CloseableHttpClient;
import org.apache.http.impl.client.HttpClients;
import org.apache.http.util.EntityUtils;
import java.io.IOException;
public class RestApiExample {
public static void main(String[] args) {
final String URL = "http://localhost:3000/root";
final HttpGet request = new HttpGet(URL);
final CloseableHttpClient httpClient = HttpClients.createDefault();
try (CloseableHttpResponse response = httpClient.execute(request)) {
HttpEntity entity = response.getEntity();
String result = EntityUtils.toString(entity);
System.out.println(result);
} catch (IOException e) {
e.printStackTrace();
}
}
}
2. Example : Timeout이 적용된 코드
다음은 위의 예제에서 Timeout을 1초로 적용한 코드입니다.
import org.apache.http.HttpEntity;
import org.apache.http.client.config.RequestConfig;
import org.apache.http.client.methods.CloseableHttpResponse;
import org.apache.http.client.methods.HttpGet;
import org.apache.http.impl.client.CloseableHttpClient;
import org.apache.http.impl.client.HttpClients;
import org.apache.http.util.EntityUtils;
import java.io.IOException;
public class RestApiExample {
public static void main(String[] args) {
final String URL = "http://localhost:3000/root";
final HttpGet request = new HttpGet(URL);
final CloseableHttpClient httpClient = HttpClients.createDefault();
// Apply timeout for the request
RequestConfig requestConfig = RequestConfig.custom()
.setSocketTimeout(1 * 1000)
.setConnectTimeout(1 * 1000)
.setConnectionRequestTimeout(1 * 1000)
.build();
request.setConfig(requestConfig);
try (CloseableHttpResponse response = httpClient.execute(request)) {
HttpEntity entity = response.getEntity();
String result = EntityUtils.toString(entity);
System.out.println(result);
} catch (IOException e) {
e.printStackTrace();
}
}
}
Timeout 발생 시, 다음과 같이 ConnectTimeoutException가 발생할 수 있습니다. 이 예외에 대해서 처리해주시면 됩니다.
org.apache.http.conn.ConnectTimeoutException: Connect to localhost:3000 [localhost/127.0.0.1, localhost/0:0:0:0:0:0:0:1] failed: connect timed out
at org.apache.http.impl.conn.DefaultHttpClientConnectionOperator.connect(DefaultHttpClientConnectionOperator.java:151)
at org.apache.http.impl.conn.PoolingHttpClientConnectionManager.connect(PoolingHttpClientConnectionManager.java:374)
at org.apache.http.impl.execchain.MainClientExec.establishRoute(MainClientExec.java:393)
at org.apache.http.impl.execchain.MainClientExec.execute(MainClientExec.java:236)
at org.apache.http.impl.execchain.ProtocolExec.execute(ProtocolExec.java:186)
at org.apache.http.impl.execchain.RetryExec.execute(RetryExec.java:89)
at org.apache.http.impl.execchain.RedirectExec.execute(RedirectExec.java:110)
at org.apache.http.impl.client.InternalHttpClient.doExecute(InternalHttpClient.java:185)
at org.apache.http.impl.client.CloseableHttpClient.execute(CloseableHttpClient.java:83)
at org.apache.http.impl.client.CloseableHttpClient.execute(CloseableHttpClient.java:108)
at RestApiExample.main(RestApiExample.java:30)
Restful server에 문제가 있을 때, 아래와 같이 HttpHostConnectException 예외가 발생할 수 있습니다. 이 예외가 발생하는 경우, 대기하지 않아서 다음 작업을 수행할 수 있지만, 이 예외가 발생하지 않으면 계속 대기할 수 있기 때문에 Timeout을 적용하는 것을 고려해볼만 합니다.
org.apache.http.conn.HttpHostConnectException: Connect to localhost:3000 [localhost/127.0.0.1, localhost/0:0:0:0:0:0:0:1] failed: Connection refused: connect
at org.apache.http.impl.conn.DefaultHttpClientConnectionOperator.connect(DefaultHttpClientConnectionOperator.java:156)
at org.apache.http.impl.conn.PoolingHttpClientConnectionManager.connect(PoolingHttpClientConnectionManager.java:374)
at org.apache.http.impl.execchain.MainClientExec.establishRoute(MainClientExec.java:393)
at org.apache.http.impl.execchain.MainClientExec.execute(MainClientExec.java:236)
at org.apache.http.impl.execchain.ProtocolExec.execute(ProtocolExec.java:186)
Related Posts
- Java - Unsupported class file major version 61 에러
- Java - String.matches()로 문자열 패턴 확인 및 다양한 예제 소개
- Java - 문자열 공백제거 (trim, replace)
- Java - replace()와 replaceAll()의 차이점
- Java - ArrayList 초기화, 4가지 방법
- Java - 배열 정렬(Sorting) (오름차순, 내림차순)
- Java - 문자열(String)을 비교하는 방법 (==, equals, compare)
- Java - StringBuilder 사용 방법, 예제
- Java - 로그 출력, 파일 저장 방법 (Logger 라이브러리)
- Java IllegalArgumentException 의미, 발생 이유
- Java - NullPointerException 원인, 해결 방법
- Seleninum의 ConnectionFailedException: Unable to establish websocket connection 해결
- Java - compareTo(), 객체 크기 비교
- Java - BufferedWriter로 파일 쓰기
- Java - BufferedReader로 파일 읽기
- Java charAt() 함수 알아보기
- Java - BigInteger 범위, 비교, 연산, 형변환
- Java contains()로 문자(대소문자 X) 포함 확인
- Java - Set(HashSet)를 배열로 변환
- Java - 문자열 첫번째 문자, 마지막 문자 확인
- Java - 문자열 한글자씩 자르기
- Java - 문자열 단어 개수 가져오기
- Java - 1초마다 반복 실행
- Java - 배열을 Set(HashSet)로 변환
- Java - 여러 Set(HashSet) 합치기
- Java - 명령행 인자 입력 받기
- Java - 리스트 역순으로 순회, 3가지 방법
- Java - 특정 조건으로 리스트 필터링, 3가지 방법
- Java - HashMap 모든 요소들의 합계, 평균 계산
- Java - 특정 조건으로 HashMap 필터링
- Java - 싱글톤(Singleton) 패턴 구현
- Java - 숫자 왼쪽에 0으로 채우기
- Java - String 배열 초기화 방법
- Java - 정렬된 순서로 Map(HashMap) 순회
- Java - HashMap에서 key, value 가져오기