Java/Spring
Springboot OncePerRequestFilter를 이용한 HttpServeletRequest의 body 값 무제한으로 읽어보기, request body 한번만 읽어올수 있는 문제 해결
닥치고개발
2023. 7. 10. 18:05
728x90
우선 다양한 방법으로 request body 한번만 읽어올수 있는 문제를 해결 할 수 있지만 내가 해결 한방 법중에 가장 심플한 방법을 소개한다.
- 필터를 만든다.
- 끝!!!
너무 간단해서 할말을 잃었다.
- 그럼 코드를 보면서 한번 이해해 보자
커스텀 필터를 아래와 같이 만든다!!!
Order는 1번으로 최상단!!
/** * JSON Body를 여러번 읽을 수 있게 커스텀 필터를 추가 */ @Component @Order(1) public class RequestBodyFilter extends OncePerRequestFilter { @Override protected void doFilterInternal(HttpServletRequest request, HttpServletResponse response, FilterChain chain) throws ServletException, IOException { chain.doFilter(new MultiReadHttpServletRequest(request), response); } private class MultiReadHttpServletRequest extends HttpServletRequestWrapper { private ByteArrayOutputStream cachedBytes; public MultiReadHttpServletRequest(HttpServletRequest request) { super(request); } @Override public ServletInputStream getInputStream() throws IOException { if (cachedBytes == null) { cacheInputStream(); } return new CachedServletInputStream(cachedBytes.toByteArray()); } @Override public BufferedReader getReader() throws IOException { return new BufferedReader(new InputStreamReader(getInputStream())); } private void cacheInputStream() throws IOException { /* Cache the inputstream in order to read it multiple times. For * convenience, I use apache.commons IOUtils */ cachedBytes = new ByteArrayOutputStream(); IOUtils.copy(super.getInputStream(), cachedBytes); }
/* An input stream which reads the cached request body */
private static class CachedServletInputStream extends ServletInputStream {
private final ByteArrayInputStream buffer;
public CachedServletInputStream(byte[] contents) {
this.buffer = new ByteArrayInputStream(contents);
}
@Override
public int read() {
return buffer.read();
}
@Override
public boolean isFinished() {
return buffer.available() == 0;
}
@Override
public boolean isReady() {
return true;
}
@Override
public void setReadListener(ReadListener listener) {
throw new RuntimeException("Not implemented");
}
}
}
}
## 이후 나는 이 값을 interceptor에서 처리하기 위해서 Requestbody값을 읽을 필요가 있었다.
- 그래서 인터셉터를 아래와 같이 만들어서 사용중이다
- 인증 관련 처리를 하고 있다.
```java
@Slf4j
public class AuthorizationInterceptor implements HandlerInterceptor {
public static final String X_PARTNER_NAME = "X-Partner-Name";
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
final String requestJsonBody = getRequestJsonBody(request);
log.info("requestJsonBody : " + requestJsonBody);
log.info("start check");
checkIfClientValid(request);
log.info("end check");
// BniTokenService.instance.get()
return HandlerInterceptor.super.preHandle(request, response, handler);
}
/**
* 유효한 Client인지 체크
*
* @param request
*/
private void checkIfClientValid(HttpServletRequest request) {
Client.findByClient(request.getHeader(X_PARTNER_NAME));
}
/**
* JSON Body를 읽어온다
*
* @param request
* @return
*/
private String getRequestJsonBody(HttpServletRequest request) {
try {
// JSON 인 경우만 Body를 읽도록 필터링
if ("application/json".equalsIgnoreCase(request.getHeader("Content-Type"))) {
return IOUtils.readLines(request.getInputStream()).stream().map(f -> f.trim()).collect(Collectors.joining());
}
} catch (Exception e) {
log.warn(e.getMessage(), e);
}
return null;
}
}
여기서 유효한 Client가 아니면 throw를 던진다.
- 끝!!! 쉽지 않은가?
LIST