The spring integration flow I wrote has to get files (some of them are as big as 4G) from a rest service and transfer them to a remote shared drive. For downloading them from the rest service I configured this simple component:
@Bean
public HttpRequestExecutingMessagehandler httpDownloader (RestTemplate template){
Expression expr = (new SpelExpressionParser()).parseExpression("payload.url");
HttpRequestExecutingMessagehandler handler = new HttpRequestExecutingMessagehandler (expr, template);
handler.setExpectedResponseType(byte[].class);
handler.setHttpMethod(GET);
return handler;
}
Unfortunately this won't scale meaning for larger files it will eventually throw java.lang.OutOfMemoryError: Java heap space
, even if i add more memory with -Xmx
or -XXMaxPermSize
So my question is, what to do in order to avoid these problems no matter how big the files will be?
I think I have answered you in some other similar your question that Spring
RestTemplate
is not designed for streaming response body. It is explained in this SO thread: Getting InputStream with RestTemplate.One of the solution which may work for your is to write a custom
HttpMessageConverter
which would return aFile
object containing data from HTTP response. This article explains how to do that with theResponseExtractor
, but something likeFileHttpMessageConverter
is not so hard to implement based on experience from that article. SeeStreamUtils.copy(InputStream in, OutputStream out)
Then you inject this
FileHttpMessageConverter
into yourHttpRequestExecutingMessagehandler
-setMessageConverters(List<HttpMessageConverter<?>> messageConverters)
.Your service for remote shared drive should already deal with this local temporary file to get that large content without consuming memory.
See also this one about possible approach via
WebFlux
: https://www.amitph.com/spring-webclient-large-file-download/