Как декодировать сжатое тело запроса gzip в Spring MVC
у меня есть клиент, который отправляет данные с
CONTENT-ENCODING deflate
у меня есть такой код
@RequestMapping(value = "/connect", method = RequestMethod.POST)
@ResponseBody
public Map onConnect(@RequestBody String body){}
в настоящее время "тело" печатает искаженные, сжатые данные. Есть ли способ заставить Spring MVC автоматически распаковывать его?
3 ответов
вы не справляетесь с этим весной. Вместо этого вы используете фильтр, чтобы данные поступали весной уже сдутыми.
Надеюсь, эти две ссылки помогут вам начать.
вам нужно будет написать собственный фильтр для распаковки тела gzipped-запросов. Синус вы будете читать весь входной поток из запроса, вам нужно переопределить метод parcing параметров тоже. Это фильтр, который я использую в своем коде. Поддерживает только gzipped POST запросы, но вы можете обновить его, чтобы использовать другие типы запросов, если это необходимо. Также остерегайтесь разбирать параметры, которые я использую библиотеку guava, вы можете захватить здесь: http://central.maven.org/maven2/com/google/guava/guava/
public class GzipBodyDecompressFilter extends Filter {
@Override
public void init(FilterConfig filterConfig) throws ServletException {
}
/**
* Analyzes servlet request for possible gzipped body.
* When Content-Encoding header has "gzip" value and request method is POST we read all the
* gzipped stream and is it haz any data unzip it. In case when gzip Content-Encoding header
* specified but body is not actually in gzip format we will throw ZipException.
*
* @param servletRequest servlet request
* @param servletResponse servlet response
* @param chain filter chain
* @throws IOException throws when fails
* @throws ServletException thrown when fails
*/
@Override
public final void doFilter(final ServletRequest servletRequest,
final ServletResponse servletResponse,
final FilterChain chain) throws IOException, ServletException {
HttpServletRequest request = (HttpServletRequest) servletRequest;
HttpServletResponse response = (HttpServletResponse) servletResponse;
boolean isGzipped = request.getHeader(HttpHeaders.CONTENT_ENCODING) != null
&& request.getHeader(HttpHeaders.CONTENT_ENCODING).contains("gzip");
boolean requestTypeSupported = HttpMethods.POST.equals(request.getMethod());
if (isGzipped && !requestTypeSupported) {
throw new IllegalStateException(request.getMethod()
+ " is not supports gzipped body of parameters."
+ " Only POST requests are currently supported.");
}
if (isGzipped && requestTypeSupported) {
request = new GzippedInputStreamWrapper((HttpServletRequest) servletRequest);
}
chain.doFilter(request, response);
}
/**
* @inheritDoc
*/
@Override
public final void destroy() {
}
/**
* Wrapper class that detects if the request is gzipped and ungzipps it.
*/
final class GzippedInputStreamWrapper extends HttpServletRequestWrapper {
/**
* Default encoding that is used when post parameters are parsed.
*/
public static final String DEFAULT_ENCODING = "ISO-8859-1";
/**
* Serialized bytes array that is a result of unzipping gzipped body.
*/
private byte[] bytes;
/**
* Constructs a request object wrapping the given request.
* In case if Content-Encoding contains "gzip" we wrap the input stream into byte array
* to original input stream has nothing in it but hew wrapped input stream always returns
* reproducible ungzipped input stream.
*
* @param request request which input stream will be wrapped.
* @throws java.io.IOException when input stream reqtieval failed.
*/
public GzippedInputStreamWrapper(final HttpServletRequest request) throws IOException {
super(request);
try {
final InputStream in = new GZIPInputStream(request.getInputStream());
bytes = ByteStreams.toByteArray(in);
} catch (EOFException e) {
bytes = new byte[0];
}
}
/**
* @return reproduceable input stream that is either equal to initial servlet input
* stream(if it was not zipped) or returns unzipped input stream.
* @throws IOException if fails.
*/
@Override
public ServletInputStream getInputStream() throws IOException {
final ByteArrayInputStream sourceStream = new ByteArrayInputStream(bytes);
return new ServletInputStream() {
public int read() throws IOException {
return sourceStream.read();
}
public void close() throws IOException {
super.close();
sourceStream.close();
}
};
}
/**
* Need to override getParametersMap because we initially read the whole input stream and
* servlet container won't have access to the input stream data.
*
* @return parsed parameters list. Parameters get parsed only when Content-Type
* "application/x-www-form-urlencoded" is set.
*/
@Override
public Map getParameterMap() {
String contentEncodingHeader = getHeader(HttpHeaders.CONTENT_TYPE);
if (!Strings.isNullOrEmpty(contentEncodingHeader)
&& contentEncodingHeader.contains("application/x-www-form-urlencoded")) {
Map params = new HashMap(super.getParameterMap());
try {
params.putAll(parseParams(new String(bytes)));
} catch (UnsupportedEncodingException e) {
e.printStackTrace();
}
return params;
} else {
return super.getParameterMap();
}
}
/**
* parses params from the byte input stream.
*
* @param body request body serialized to string.
* @return parsed parameters map.
* @throws UnsupportedEncodingException if encoding provided is not supported.
*/
private Map<String, String[]> parseParams(final String body)
throws UnsupportedEncodingException {
String characterEncoding = getCharacterEncoding();
if (null == characterEncoding) {
characterEncoding = DEFAULT_ENCODING;
}
final Multimap<String, String> parameters = ArrayListMultimap.create();
for (String pair : body.split("&")) {
if (Strings.isNullOrEmpty(pair)) {
continue;
}
int idx = pair.indexOf("=");
String key = null;
if (idx > 0) {
key = URLDecoder.decode(pair.substring(0, idx), characterEncoding);
} else {
key = pair;
}
String value = null;
if (idx > 0 && pair.length() > idx + 1) {
value = URLDecoder.decode(pair.substring(idx + 1), characterEncoding);
} else {
value = null;
}
parameters.put(key, value);
}
return Maps.transformValues(parameters.asMap(),
new Function<Collection<String>, String[]>() {
@Nullable
@Override
public String[] apply(final Collection<String> input) {
return Iterables.toArray(input, String.class);
}
});
}
}
}
Это должно обрабатываться сервером, а не приложением.
насколько я знаю, Tomcat не поддерживает его, Хотя вы, вероятно, могли бы написать фильтр.
обычный способ справиться с этим-поместить Tomcat ( или любой Java-контейнер, который вы используете) за сервер Apache, настроенный для обработки сжатых тел запросов.