From 3765efad2b7033bc78f8ee05ec730a5d5fae6b79 Mon Sep 17 00:00:00 2001 From: Greg Wilkins Date: Mon, 18 Jul 2016 11:38:10 +1000 Subject: [PATCH 1/2] Extensible ErrorHandler for different mimetpyes #727 --- .../jetty/server/handler/ErrorHandler.java | 86 +++++++++++++++---- 1 file changed, 68 insertions(+), 18 deletions(-) diff --git a/jetty-server/src/main/java/org/eclipse/jetty/server/handler/ErrorHandler.java b/jetty-server/src/main/java/org/eclipse/jetty/server/handler/ErrorHandler.java index 73cd498ac05..cad18be57e0 100644 --- a/jetty-server/src/main/java/org/eclipse/jetty/server/handler/ErrorHandler.java +++ b/jetty-server/src/main/java/org/eclipse/jetty/server/handler/ErrorHandler.java @@ -23,13 +23,14 @@ import java.io.PrintWriter; import java.io.StringWriter; import java.io.Writer; import java.nio.ByteBuffer; +import java.util.Collections; +import java.util.List; import javax.servlet.RequestDispatcher; import javax.servlet.ServletException; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; -import org.eclipse.jetty.http.HttpField; import org.eclipse.jetty.http.HttpFields; import org.eclipse.jetty.http.HttpHeader; import org.eclipse.jetty.http.HttpMethod; @@ -110,25 +111,74 @@ public class ErrorHandler extends AbstractHandler } } - baseRequest.setHandled(true); - - // Issue #124 - Don't produce text/html if the request doesn't accept it - HttpField accept = baseRequest.getHttpFields().getField(HttpHeader.ACCEPT); - if (accept == null || accept.contains("text/html") || accept.contains("*/*")) - { - response.setContentType(MimeTypes.Type.TEXT_HTML_8859_1.asString()); - if (_cacheControl != null) - response.setHeader(HttpHeader.CACHE_CONTROL.asString(), _cacheControl); - ByteArrayISO8859Writer writer = new ByteArrayISO8859Writer(4096); - String reason = (response instanceof Response) ? ((Response) response).getReason() : null; - handleErrorPage(request, writer, response.getStatus(), reason); - writer.flush(); - response.setContentLength(writer.size()); - writer.writeTo(response.getOutputStream()); - writer.destroy(); - } + generateAcceptableResponse(baseRequest,request,response); } + /* ------------------------------------------------------------ */ + protected void generateAcceptableResponse(Request baseRequest, HttpServletRequest request, HttpServletResponse response) + throws IOException + { + List acceptable=baseRequest.getHttpFields().getQualityCSV(HttpHeader.ACCEPT); + if (acceptable.isEmpty()) + acceptable=Collections.singletonList("*/*"); + for (String mimeType:acceptable) + { + generateAcceptableResponse(baseRequest,request,response,mimeType); + if (baseRequest.isHandled()) + return; + } + + baseRequest.setHandled(true); + } + + /* ------------------------------------------------------------ */ + protected void generateAcceptableResponse(Request baseRequest, HttpServletRequest request, HttpServletResponse response, String mimeType) + throws IOException + { + switch(mimeType) + { + case "text/html": + case "text/html;charset=iso-8859-1": + case "*/*": + { + baseRequest.setHandled(true); + response.setContentType(MimeTypes.Type.TEXT_HTML_8859_1.asString()); + if (_cacheControl != null) + response.setHeader(HttpHeader.CACHE_CONTROL.asString(), _cacheControl); + ByteArrayISO8859Writer writer = new ByteArrayISO8859Writer(4096); + String reason = (response instanceof Response) ? ((Response) response).getReason() : null; + handleErrorPage(request, writer, response.getStatus(), reason); + writer.flush(); + response.setContentLength(writer.size()); + writer.writeTo(response.getOutputStream()); + writer.destroy(); + return; + } + + case "text/html;charset=utf-8": + { + baseRequest.setHandled(true); + response.setContentType(MimeTypes.Type.TEXT_HTML_UTF_8.asString()); + if (_cacheControl != null) + response.setHeader(HttpHeader.CACHE_CONTROL.asString(), _cacheControl); + String reason = (response instanceof Response) ? ((Response) response).getReason() : null; + handleErrorPage(request, response.getWriter(), response.getStatus(), reason); + return; + } + } + + if (MimeTypes.Type.TEXT_HTML.is(MimeTypes.getContentTypeWithoutCharset(mimeType))) + { + baseRequest.setHandled(true); + response.setContentType(mimeType); + if (_cacheControl != null) + response.setHeader(HttpHeader.CACHE_CONTROL.asString(), _cacheControl); + String reason = (response instanceof Response) ? ((Response) response).getReason() : null; + handleErrorPage(request, response.getWriter(), response.getStatus(), reason); + return; + } + } + /* ------------------------------------------------------------ */ protected void handleErrorPage(HttpServletRequest request, Writer writer, int code, String message) throws IOException From 35bb6620a11171d74fc371769bdd5dc07569754a Mon Sep 17 00:00:00 2001 From: Greg Wilkins Date: Mon, 18 Jul 2016 21:12:48 +1000 Subject: [PATCH 2/2] Extensible ErrorHandler for different mimetypes #230 --- .../jetty/server/handler/ErrorHandler.java | 145 ++++++---- .../jetty/server/ErrorHandlerTest.java | 260 ++++++++++++++++++ 2 files changed, 356 insertions(+), 49 deletions(-) create mode 100644 jetty-server/src/test/java/org/eclipse/jetty/server/ErrorHandlerTest.java diff --git a/jetty-server/src/main/java/org/eclipse/jetty/server/handler/ErrorHandler.java b/jetty-server/src/main/java/org/eclipse/jetty/server/handler/ErrorHandler.java index cad18be57e0..357b49ed957 100644 --- a/jetty-server/src/main/java/org/eclipse/jetty/server/handler/ErrorHandler.java +++ b/jetty-server/src/main/java/org/eclipse/jetty/server/handler/ErrorHandler.java @@ -23,7 +23,8 @@ import java.io.PrintWriter; import java.io.StringWriter; import java.io.Writer; import java.nio.ByteBuffer; -import java.util.Collections; +import java.nio.charset.Charset; +import java.nio.charset.StandardCharsets; import java.util.List; import javax.servlet.RequestDispatcher; @@ -41,7 +42,6 @@ import org.eclipse.jetty.server.Request; import org.eclipse.jetty.server.Response; import org.eclipse.jetty.server.Server; import org.eclipse.jetty.util.BufferUtil; -import org.eclipse.jetty.util.ByteArrayISO8859Writer; import org.eclipse.jetty.util.StringUtil; import org.eclipse.jetty.util.log.Log; import org.eclipse.jetty.util.log.Logger; @@ -103,80 +103,127 @@ public class ErrorHandler extends AbstractHandler return; } } - } else { + } + else + { if (LOG.isDebugEnabled()) { LOG.debug("No Error Page mapping for request({} {}) (using default)",request.getMethod(),request.getRequestURI()); } } } - - generateAcceptableResponse(baseRequest,request,response); + + if (_cacheControl != null) + response.setHeader(HttpHeader.CACHE_CONTROL.asString(), _cacheControl); + generateAcceptableResponse(baseRequest,request,response,response.getStatus(),baseRequest.getResponse().getReason()); } /* ------------------------------------------------------------ */ - protected void generateAcceptableResponse(Request baseRequest, HttpServletRequest request, HttpServletResponse response) + /** Generate an acceptable error response. + *

This method is called to generate an Error page of a mime type that is + * acceptable to the user-agent. The Accept header is evaluated in + * quality order and the method + * {@link #generateAcceptableResponse(Request, HttpServletRequest, HttpServletResponse, String)} + * is called for each mimetype until {@link Request#isHandled()} is true.

+ * @param baseRequest The base request + * @param request The servlet request (may be wrapped) + * @param response The response (may be wrapped) + * @throws IOException + */ + protected void generateAcceptableResponse(Request baseRequest, HttpServletRequest request, HttpServletResponse response, int code, String message) throws IOException { List acceptable=baseRequest.getHttpFields().getQualityCSV(HttpHeader.ACCEPT); - if (acceptable.isEmpty()) - acceptable=Collections.singletonList("*/*"); - for (String mimeType:acceptable) + + if (acceptable.isEmpty() && !baseRequest.getHttpFields().contains(HttpHeader.ACCEPT)) + generateAcceptableResponse(baseRequest,request,response,code,message,MimeTypes.Type.TEXT_HTML.asString()); + else { - generateAcceptableResponse(baseRequest,request,response,mimeType); - if (baseRequest.isHandled()) - return; + for (String mimeType:acceptable) + { + generateAcceptableResponse(baseRequest,request,response,code,message,mimeType); + if (baseRequest.isHandled()) + return; + } } - baseRequest.setHandled(true); } /* ------------------------------------------------------------ */ - protected void generateAcceptableResponse(Request baseRequest, HttpServletRequest request, HttpServletResponse response, String mimeType) + /** get an acceptable writer for an error page. + *

Uses the user-agent's Accept-Charset to get response + * {@link Writer}. The acceptable charsets are tested in quality order + * if they are known to the JVM and the first known is set on + * {@link HttpServletResponse#setCharacterEncoding(String)} and the + * {@link HttpServletResponse#getWriter()} method used to return a writer. + * If there is no Accept-Charset header then + * ISO-8859-1 is used. If '*' is the highest quality known + * charset, then utf-8 is used. + *

+ * @param baseRequest The base request + * @param request The servlet request (may be wrapped) + * @param response The response (may be wrapped) + * @return A {@link Writer} if there is a known acceptable charset or null + * @throws IOException + */ + protected Writer getAcceptableWriter(Request baseRequest, HttpServletRequest request, HttpServletResponse response) + throws IOException + { + List acceptable=baseRequest.getHttpFields().getQualityCSV(HttpHeader.ACCEPT_CHARSET); + if (acceptable.isEmpty()) + { + response.setCharacterEncoding(StandardCharsets.ISO_8859_1.name()); + return response.getWriter(); + } + + for (String charset:acceptable) + { + try + { + if ("*".equals(charset)) + response.setCharacterEncoding(StandardCharsets.UTF_8.name()); + else + response.setCharacterEncoding(Charset.forName(charset).name()); + return response.getWriter(); + } + catch(Exception e) + { + LOG.ignore(e); + } + } + return null; + } + + /* ------------------------------------------------------------ */ + /** Generate an acceptable error response for a mime type. + *

This method is called for each mime type in the users agent's + * Accept header, until {@link Request#isHandled()} is true and a + * response of the appropriate type is generated. + * + * @param baseRequest The base request + * @param request The servlet request (may be wrapped) + * @param response The response (may be wrapped) + * @param mimeType The mimetype to generate (may be */*or other wildcard) + * @throws IOException + */ + protected void generateAcceptableResponse(Request baseRequest, HttpServletRequest request, HttpServletResponse response, int code, String message, String mimeType) throws IOException { switch(mimeType) { case "text/html": - case "text/html;charset=iso-8859-1": + case "text/*": case "*/*": { baseRequest.setHandled(true); - response.setContentType(MimeTypes.Type.TEXT_HTML_8859_1.asString()); - if (_cacheControl != null) - response.setHeader(HttpHeader.CACHE_CONTROL.asString(), _cacheControl); - ByteArrayISO8859Writer writer = new ByteArrayISO8859Writer(4096); - String reason = (response instanceof Response) ? ((Response) response).getReason() : null; - handleErrorPage(request, writer, response.getStatus(), reason); - writer.flush(); - response.setContentLength(writer.size()); - writer.writeTo(response.getOutputStream()); - writer.destroy(); - return; + Writer writer = getAcceptableWriter(baseRequest,request,response); + if (writer!=null) + { + response.setContentType(MimeTypes.Type.TEXT_HTML.asString()); + handleErrorPage(request, writer, code, message); + } } - - case "text/html;charset=utf-8": - { - baseRequest.setHandled(true); - response.setContentType(MimeTypes.Type.TEXT_HTML_UTF_8.asString()); - if (_cacheControl != null) - response.setHeader(HttpHeader.CACHE_CONTROL.asString(), _cacheControl); - String reason = (response instanceof Response) ? ((Response) response).getReason() : null; - handleErrorPage(request, response.getWriter(), response.getStatus(), reason); - return; - } - } - - if (MimeTypes.Type.TEXT_HTML.is(MimeTypes.getContentTypeWithoutCharset(mimeType))) - { - baseRequest.setHandled(true); - response.setContentType(mimeType); - if (_cacheControl != null) - response.setHeader(HttpHeader.CACHE_CONTROL.asString(), _cacheControl); - String reason = (response instanceof Response) ? ((Response) response).getReason() : null; - handleErrorPage(request, response.getWriter(), response.getStatus(), reason); - return; - } + } } /* ------------------------------------------------------------ */ diff --git a/jetty-server/src/test/java/org/eclipse/jetty/server/ErrorHandlerTest.java b/jetty-server/src/test/java/org/eclipse/jetty/server/ErrorHandlerTest.java new file mode 100644 index 00000000000..5040abbdc5c --- /dev/null +++ b/jetty-server/src/test/java/org/eclipse/jetty/server/ErrorHandlerTest.java @@ -0,0 +1,260 @@ +// +// ======================================================================== +// Copyright (c) 1995-2016 Mort Bay Consulting Pty. Ltd. +// ------------------------------------------------------------------------ +// All rights reserved. This program and the accompanying materials +// are made available under the terms of the Eclipse Public License v1.0 +// and Apache License v2.0 which accompanies this distribution. +// +// The Eclipse Public License is available at +// http://www.eclipse.org/legal/epl-v10.html +// +// The Apache License v2.0 is available at +// http://www.opensource.org/licenses/apache2.0.php +// +// You may elect to redistribute this code under either of these licenses. +// ======================================================================== +// + +package org.eclipse.jetty.server; + +import static org.hamcrest.Matchers.containsString; +import static org.hamcrest.Matchers.not; +import static org.hamcrest.Matchers.startsWith; +import static org.junit.Assert.*; + +import java.io.IOException; + +import javax.servlet.http.HttpServletRequest; +import javax.servlet.http.HttpServletResponse; + +import org.eclipse.jetty.server.handler.ErrorHandler; +import org.junit.After; +import org.junit.Before; +import org.junit.Test; + +public class ErrorHandlerTest +{ + Server server; + LocalConnector connector; + + @Before + public void before() throws Exception + { + server = new Server(); + connector = new LocalConnector(server); + server.addConnector(connector); + server.addBean(new ErrorHandler() + { + @Override + protected void generateAcceptableResponse( + Request baseRequest, + HttpServletRequest request, + HttpServletResponse response, + int code, + String message, + String mimeType) throws IOException + { + switch(mimeType) + { + case "text/json": + case "application/json": + { + baseRequest.setHandled(true); + response.setContentType(mimeType); + response.getWriter() + .append("{") + .append("code: \"").append(Integer.toString(code)).append("\",") + .append("message: \"").append(message).append('"') + .append("}"); + break; + } + default: + super.generateAcceptableResponse(baseRequest,request,response,code,message,mimeType); + } + } + + }); + server.start(); + } + + @After + public void after() throws Exception + { + server.stop(); + } + + @Test + public void test404NoAccept() throws Exception + { + String response = connector.getResponse( + "GET / HTTP/1.1\r\n"+ + "Host: Localhost\r\n"+ + "\r\n"); + + assertThat(response,startsWith("HTTP/1.1 404 ")); + assertThat(response,not(containsString("Content-Length: 0"))); + assertThat(response,containsString("Content-Type: text/html;charset=iso-8859-1")); + } + + @Test + public void test404EmptyAccept() throws Exception + { + String response = connector.getResponse( + "GET / HTTP/1.1\r\n"+ + "Accept: \r\n"+ + "Host: Localhost\r\n"+ + "\r\n"); + assertThat(response,startsWith("HTTP/1.1 404 ")); + assertThat(response,containsString("Content-Length: 0")); + assertThat(response,not(containsString("Content-Type"))); + } + + @Test + public void test404UnAccept() throws Exception + { + String response = connector.getResponse( + "GET / HTTP/1.1\r\n"+ + "Accept: text/*;q=0\r\n"+ + "Host: Localhost\r\n"+ + "\r\n"); + + assertThat(response,startsWith("HTTP/1.1 404 ")); + assertThat(response,containsString("Content-Length: 0")); + assertThat(response,not(containsString("Content-Type"))); + } + + @Test + public void test404AllAccept() throws Exception + { + String response = connector.getResponse( + "GET / HTTP/1.1\r\n"+ + "Host: Localhost\r\n"+ + "Accept: */*\r\n"+ + "\r\n"); + + assertThat(response,startsWith("HTTP/1.1 404 ")); + assertThat(response,not(containsString("Content-Length: 0"))); + assertThat(response,containsString("Content-Type: text/html;charset=iso-8859-1")); + } + + @Test + public void test404HtmlAccept() throws Exception + { + String response = connector.getResponse( + "GET / HTTP/1.1\r\n"+ + "Host: Localhost\r\n"+ + "Accept: text/html\r\n"+ + "\r\n"); + + assertThat(response,startsWith("HTTP/1.1 404 ")); + assertThat(response,not(containsString("Content-Length: 0"))); + assertThat(response,containsString("Content-Type: text/html;charset=iso-8859-1")); + } + + @Test + public void test404HtmlAcceptAnyCharset() throws Exception + { + String response = connector.getResponse( + "GET / HTTP/1.1\r\n"+ + "Host: Localhost\r\n"+ + "Accept: text/html\r\n"+ + "Accept-Charset: *\r\n"+ + "\r\n"); + + assertThat(response,startsWith("HTTP/1.1 404 ")); + assertThat(response,not(containsString("Content-Length: 0"))); + assertThat(response,containsString("Content-Type: text/html;charset=utf-8")); + } + + @Test + public void test404HtmlAcceptUtf8Charset() throws Exception + { + String response = connector.getResponse( + "GET / HTTP/1.1\r\n"+ + "Host: Localhost\r\n"+ + "Accept: text/html\r\n"+ + "Accept-Charset: utf-8\r\n"+ + "\r\n"); + + assertThat(response,startsWith("HTTP/1.1 404 ")); + assertThat(response,not(containsString("Content-Length: 0"))); + assertThat(response,containsString("Content-Type: text/html;charset=utf-8")); + } + + @Test + public void test404HtmlAcceptNotUtf8Charset() throws Exception + { + String response = connector.getResponse( + "GET / HTTP/1.1\r\n"+ + "Host: Localhost\r\n"+ + "Accept: text/html\r\n"+ + "Accept-Charset: utf-8;q=0\r\n"+ + "\r\n"); + + assertThat(response,startsWith("HTTP/1.1 404 ")); + assertThat(response,not(containsString("Content-Length: 0"))); + assertThat(response,containsString("Content-Type: text/html;charset=iso-8859-1")); + } + + @Test + public void test404HtmlAcceptNotUtf8UnknownCharset() throws Exception + { + String response = connector.getResponse( + "GET / HTTP/1.1\r\n"+ + "Host: Localhost\r\n"+ + "Accept: text/html\r\n"+ + "Accept-Charset: utf-8;q=0,unknown\r\n"+ + "\r\n"); + + assertThat(response,startsWith("HTTP/1.1 404 ")); + assertThat(response,containsString("Content-Length: 0")); + assertThat(response,not(containsString("Content-Type"))); + } + + @Test + public void test404HtmlAcceptUnknownUtf8Charset() throws Exception + { + String response = connector.getResponse( + "GET / HTTP/1.1\r\n"+ + "Host: Localhost\r\n"+ + "Accept: text/html\r\n"+ + "Accept-Charset: utf-8;q=0.1,unknown\r\n"+ + "\r\n"); + + assertThat(response,startsWith("HTTP/1.1 404 ")); + assertThat(response,not(containsString("Content-Length: 0"))); + assertThat(response,containsString("Content-Type: text/html;charset=utf-8")); + } + + @Test + public void test404PreferHtml() throws Exception + { + String response = connector.getResponse( + "GET / HTTP/1.1\r\n"+ + "Host: Localhost\r\n"+ + "Accept: text/html;q=1.0,text/json;q=0.5,*/*\r\n"+ + "Accept-Charset: *\r\n"+ + "\r\n"); + + assertThat(response,startsWith("HTTP/1.1 404 ")); + assertThat(response,not(containsString("Content-Length: 0"))); + assertThat(response,containsString("Content-Type: text/html;charset=utf-8")); + } + + @Test + public void test404PreferJson() throws Exception + { + String response = connector.getResponse( + "GET / HTTP/1.1\r\n"+ + "Host: Localhost\r\n"+ + "Accept: text/html;q=0.5,text/json;q=1.0,*/*\r\n"+ + "Accept-Charset: *\r\n"+ + "\r\n"); + + assertThat(response,startsWith("HTTP/1.1 404 ")); + assertThat(response,not(containsString("Content-Length: 0"))); + assertThat(response,containsString("Content-Type: text/json")); + } + +}