ShallowEtagHeaderFilter writes body early on sendError/sendRedirect and interprets setContentLength/setBufferSize as a hint for capacity increase

Issue: SPR-11705
Issue: SPR-11717
This commit is contained in:
Juergen Hoeller 2014-04-22 23:14:48 +02:00
parent 700c3b257f
commit 3f392e32f5
2 changed files with 129 additions and 30 deletions

View File

@ -72,14 +72,15 @@ public class ShallowEtagHeaderFilter extends OncePerRequestFilter {
protected void doFilterInternal(HttpServletRequest request, HttpServletResponse response, FilterChain filterChain) protected void doFilterInternal(HttpServletRequest request, HttpServletResponse response, FilterChain filterChain)
throws ServletException, IOException { throws ServletException, IOException {
HttpServletResponse responseToUse = response;
if (!isAsyncDispatch(request)) { if (!isAsyncDispatch(request)) {
response = new ShallowEtagResponseWrapper(response); responseToUse = new ShallowEtagResponseWrapper(response);
} }
filterChain.doFilter(request, response); filterChain.doFilter(request, responseToUse);
if (!isAsyncStarted(request)) { if (!isAsyncStarted(request)) {
updateResponse(request, response); updateResponse(request, responseToUse);
} }
} }
@ -88,41 +89,44 @@ public class ShallowEtagHeaderFilter extends OncePerRequestFilter {
WebUtils.getNativeResponse(response, ShallowEtagResponseWrapper.class); WebUtils.getNativeResponse(response, ShallowEtagResponseWrapper.class);
Assert.notNull(responseWrapper, "ShallowEtagResponseWrapper not found"); Assert.notNull(responseWrapper, "ShallowEtagResponseWrapper not found");
response = (HttpServletResponse) responseWrapper.getResponse(); HttpServletResponse rawResponse = (HttpServletResponse) responseWrapper.getResponse();
byte[] body = responseWrapper.toByteArray();
int statusCode = responseWrapper.getStatusCode(); int statusCode = responseWrapper.getStatusCode();
byte[] body = responseWrapper.toByteArray();
if (isEligibleForEtag(request, responseWrapper, statusCode, body)) { if (rawResponse.isCommitted()) {
if (body.length > 0) {
StreamUtils.copy(body, rawResponse.getOutputStream());
}
}
else if (isEligibleForEtag(request, responseWrapper, statusCode, body)) {
String responseETag = generateETagHeaderValue(body); String responseETag = generateETagHeaderValue(body);
response.setHeader(HEADER_ETAG, responseETag); rawResponse.setHeader(HEADER_ETAG, responseETag);
String requestETag = request.getHeader(HEADER_IF_NONE_MATCH); String requestETag = request.getHeader(HEADER_IF_NONE_MATCH);
if (responseETag.equals(requestETag)) { if (responseETag.equals(requestETag)) {
if (logger.isTraceEnabled()) { if (logger.isTraceEnabled()) {
logger.trace("ETag [" + responseETag + "] equal to If-None-Match, sending 304"); logger.trace("ETag [" + responseETag + "] equal to If-None-Match, sending 304");
} }
response.setStatus(HttpServletResponse.SC_NOT_MODIFIED); rawResponse.setStatus(HttpServletResponse.SC_NOT_MODIFIED);
} }
else { else {
if (logger.isTraceEnabled()) { if (logger.isTraceEnabled()) {
logger.trace("ETag [" + responseETag + "] not equal to If-None-Match [" + requestETag + logger.trace("ETag [" + responseETag + "] not equal to If-None-Match [" + requestETag +
"], sending normal response"); "], sending normal response");
} }
copyBodyToResponse(body, response); if (body.length > 0) {
rawResponse.setContentLength(body.length);
StreamUtils.copy(body, rawResponse.getOutputStream());
}
} }
} }
else { else {
if (logger.isTraceEnabled()) { if (logger.isTraceEnabled()) {
logger.trace("Response with status code [" + statusCode + "] not eligible for ETag"); logger.trace("Response with status code [" + statusCode + "] not eligible for ETag");
} }
copyBodyToResponse(body, response); if (body.length > 0) {
} rawResponse.setContentLength(body.length);
} StreamUtils.copy(body, rawResponse.getOutputStream());
}
private void copyBodyToResponse(byte[] body, HttpServletResponse response) throws IOException {
if (body.length > 0) {
response.setContentLength(body.length);
StreamUtils.copy(body, response.getOutputStream());
} }
} }
@ -202,19 +206,22 @@ public class ShallowEtagHeaderFilter extends OncePerRequestFilter {
@Override @Override
public void sendError(int sc) throws IOException { public void sendError(int sc) throws IOException {
copyBodyToResponse();
super.sendError(sc); super.sendError(sc);
this.statusCode = sc; this.statusCode = sc;
} }
@Override @Override
public void sendError(int sc, String msg) throws IOException { public void sendError(int sc, String msg) throws IOException {
copyBodyToResponse();
super.sendError(sc, msg); super.sendError(sc, msg);
this.statusCode = sc; this.statusCode = sc;
} }
@Override @Override
public void setContentLength(int len) { public void sendRedirect(String location) throws IOException {
this.content.resize(len); copyBodyToResponse();
super.sendRedirect(location);
} }
@Override @Override
@ -233,9 +240,17 @@ public class ShallowEtagHeaderFilter extends OncePerRequestFilter {
} }
@Override @Override
public void reset() { public void setContentLength(int len) {
super.reset(); if (len > this.content.capacity()) {
resetBuffer(); this.content.resize(len);
}
}
@Override
public void setBufferSize(int size) {
if (size > this.content.capacity()) {
this.content.resize(size);
}
} }
@Override @Override
@ -243,6 +258,12 @@ public class ShallowEtagHeaderFilter extends OncePerRequestFilter {
this.content.reset(); this.content.reset();
} }
@Override
public void reset() {
super.reset();
this.content.reset();
}
public int getStatusCode() { public int getStatusCode() {
return this.statusCode; return this.statusCode;
} }
@ -251,6 +272,14 @@ public class ShallowEtagHeaderFilter extends OncePerRequestFilter {
return this.content.toByteArray(); return this.content.toByteArray();
} }
private void copyBodyToResponse() throws IOException {
if (this.content.size() > 0) {
getResponse().setContentLength(this.content.size());
StreamUtils.copy(this.content.toByteArray(), getResponse().getOutputStream());
this.content.reset();
}
}
private class ResponseServletOutputStream extends ServletOutputStream { private class ResponseServletOutputStream extends ServletOutputStream {

View File

@ -32,6 +32,11 @@ import org.springframework.util.FileCopyUtils;
import static org.junit.Assert.*; import static org.junit.Assert.*;
/**
* @author Arjen Poutsma
* @author Brian Clozel
* @author Juergen Hoeller
*/
public class ShallowEtagHeaderFilterTests { public class ShallowEtagHeaderFilterTests {
private ShallowEtagHeaderFilter filter; private ShallowEtagHeaderFilter filter;
@ -64,7 +69,6 @@ public class ShallowEtagHeaderFilterTests {
final byte[] responseBody = "Hello World".getBytes("UTF-8"); final byte[] responseBody = "Hello World".getBytes("UTF-8");
FilterChain filterChain = new FilterChain() { FilterChain filterChain = new FilterChain() {
@Override @Override
public void doFilter(ServletRequest filterRequest, ServletResponse filterResponse) public void doFilter(ServletRequest filterRequest, ServletResponse filterResponse)
throws IOException, ServletException { throws IOException, ServletException {
@ -73,7 +77,6 @@ public class ShallowEtagHeaderFilterTests {
FileCopyUtils.copy(responseBody, filterResponse.getOutputStream()); FileCopyUtils.copy(responseBody, filterResponse.getOutputStream());
} }
}; };
filter.doFilter(request, response, filterChain); filter.doFilter(request, response, filterChain);
assertEquals("Invalid status", 200, response.getStatus()); assertEquals("Invalid status", 200, response.getStatus());
@ -90,7 +93,6 @@ public class ShallowEtagHeaderFilterTests {
MockHttpServletResponse response = new MockHttpServletResponse(); MockHttpServletResponse response = new MockHttpServletResponse();
FilterChain filterChain = new FilterChain() { FilterChain filterChain = new FilterChain() {
@Override @Override
public void doFilter(ServletRequest filterRequest, ServletResponse filterResponse) public void doFilter(ServletRequest filterRequest, ServletResponse filterResponse)
throws IOException, ServletException { throws IOException, ServletException {
@ -100,7 +102,6 @@ public class ShallowEtagHeaderFilterTests {
filterResponse.setContentLength(responseBody.length); filterResponse.setContentLength(responseBody.length);
} }
}; };
filter.doFilter(request, response, filterChain); filter.doFilter(request, response, filterChain);
assertEquals("Invalid status", 304, response.getStatus()); assertEquals("Invalid status", 304, response.getStatus());
@ -117,7 +118,6 @@ public class ShallowEtagHeaderFilterTests {
MockHttpServletResponse response = new MockHttpServletResponse(); MockHttpServletResponse response = new MockHttpServletResponse();
FilterChain filterChain = new FilterChain() { FilterChain filterChain = new FilterChain() {
@Override @Override
public void doFilter(ServletRequest filterRequest, ServletResponse filterResponse) public void doFilter(ServletRequest filterRequest, ServletResponse filterResponse)
throws IOException, ServletException { throws IOException, ServletException {
@ -127,7 +127,6 @@ public class ShallowEtagHeaderFilterTests {
FileCopyUtils.copy(responseBody, filterResponse.getWriter()); FileCopyUtils.copy(responseBody, filterResponse.getWriter());
} }
}; };
filter.doFilter(request, response, filterChain); filter.doFilter(request, response, filterChain);
assertEquals("Invalid status", 304, response.getStatus()); assertEquals("Invalid status", 304, response.getStatus());
@ -136,4 +135,75 @@ public class ShallowEtagHeaderFilterTests {
assertArrayEquals("Invalid content", new byte[0], response.getContentAsByteArray()); assertArrayEquals("Invalid content", new byte[0], response.getContentAsByteArray());
} }
} @Test
public void filterSendError() throws Exception {
final MockHttpServletRequest request = new MockHttpServletRequest("GET", "/hotels");
MockHttpServletResponse response = new MockHttpServletResponse();
final byte[] responseBody = "Hello World".getBytes("UTF-8");
FilterChain filterChain = new FilterChain() {
@Override
public void doFilter(ServletRequest filterRequest, ServletResponse filterResponse)
throws IOException, ServletException {
assertEquals("Invalid request passed", request, filterRequest);
FileCopyUtils.copy(responseBody, filterResponse.getOutputStream());
((HttpServletResponse) filterResponse).sendError(HttpServletResponse.SC_FORBIDDEN);
}
};
filter.doFilter(request, response, filterChain);
assertEquals("Invalid status", 403, response.getStatus());
assertNull("Invalid ETag header", response.getHeader("ETag"));
assertTrue("Invalid Content-Length header", response.getContentLength() > 0);
assertArrayEquals("Invalid content", responseBody, response.getContentAsByteArray());
}
@Test
public void filterSendErrorMessage() throws Exception {
final MockHttpServletRequest request = new MockHttpServletRequest("GET", "/hotels");
MockHttpServletResponse response = new MockHttpServletResponse();
final byte[] responseBody = "Hello World".getBytes("UTF-8");
FilterChain filterChain = new FilterChain() {
@Override
public void doFilter(ServletRequest filterRequest, ServletResponse filterResponse)
throws IOException, ServletException {
assertEquals("Invalid request passed", request, filterRequest);
FileCopyUtils.copy(responseBody, filterResponse.getOutputStream());
((HttpServletResponse) filterResponse).sendError(HttpServletResponse.SC_FORBIDDEN, "ERROR");
}
};
filter.doFilter(request, response, filterChain);
assertEquals("Invalid status", 403, response.getStatus());
assertNull("Invalid ETag header", response.getHeader("ETag"));
assertTrue("Invalid Content-Length header", response.getContentLength() > 0);
assertArrayEquals("Invalid content", responseBody, response.getContentAsByteArray());
assertEquals("Invalid error message", "ERROR", response.getErrorMessage());
}
@Test
public void filterSendRedirect() throws Exception {
final MockHttpServletRequest request = new MockHttpServletRequest("GET", "/hotels");
MockHttpServletResponse response = new MockHttpServletResponse();
final byte[] responseBody = "Hello World".getBytes("UTF-8");
FilterChain filterChain = new FilterChain() {
@Override
public void doFilter(ServletRequest filterRequest, ServletResponse filterResponse)
throws IOException, ServletException {
assertEquals("Invalid request passed", request, filterRequest);
FileCopyUtils.copy(responseBody, filterResponse.getOutputStream());
((HttpServletResponse) filterResponse).sendRedirect("http://www.google.com");
}
};
filter.doFilter(request, response, filterChain);
assertEquals("Invalid status", 302, response.getStatus());
assertNull("Invalid ETag header", response.getHeader("ETag"));
assertTrue("Invalid Content-Length header", response.getContentLength() > 0);
assertArrayEquals("Invalid content", responseBody, response.getContentAsByteArray());
assertEquals("Invalid redirect URL", "http://www.google.com", response.getRedirectedUrl());
}
}