Home TOC |
![]() ![]() ![]() |
Filtering Requests and Responses
A filter is an object that can transform the header and/or content of a request or response. Filters differ from Web components in that they usually do not themselves create a response. Instead, a filter provides functionality that can be "attached" to any kind of Web resource. As a consequence, a filter should not have any dependencies on a Web resource for which it is acting as a filter so that it can be composable with more than one type of Web resource. The main tasks that a filter can perform are:
- Query the request and act accordingly
- Block the request and response pair from passing any further.
- Modify the request headers and data. You do this by providing a customized version of the request.
- Modify the response headers and data. You do this by providing a customized version of the response.
- Interact with external resources
Applications of filters include authentication, logging, image conversion, data compression, encryption, tokenizing streams, XML transformations, and so on.
You can configure a Web resource to be filtered by a chain of zero, one, or more filters in a specific order. This chain is specified when the Web application containing the component is deployed and instantiated when a Web container loads the component.
In summary, the tasks involved in using filters include:
- Programming the filter
- Programming customized requests and responses
- Specifying the filter chain for each Web resource
Programming Filters
The filtering API is defined by the
Filter
,FilterChain
, andFilterConfig
interfaces in thejavax.servlet
package. You define a filter by implementing theFilter
interface. The most important method in this interface is thedoFilter
method, which is passed request, response, and filter chain objects. This method can perform the following actions:
- Examine the request headers
- Customize the request object if it wishes to modify request headers or data
- Customize the response object if it wishes to modify response headers or data
- Invoke the next entity in the filter chain. If the current filter is the last filter in the chain that ends with the target Web component or static resource, the next entity is the resource at the end of the chain; otherwise, it is the next filter that was configured in the WAR. It invokes the next entity by calling the
doFilter
method on the chain object (passing in the request and response it was called with, or the wrapped versions it may have created). Alternatively, it can choose to block the request by not making the call to invoke the next entity. In the latter case, the filter is responsible for filling out the response.- Examine response headers after it has invoked the next filter in the chain
- Throw an exception to indicate an error in processing
In addition to
doFilter
, you must implement theinit
anddestroy
methods. The init
method is called by the container when the filter is instantiated. If you wish to pass initialization parameters to the filter you retrieve them from theFilterConfig
object passed toinit
.The Duke's Bookstore application uses the filters
HitCounterFilter
andOrderFilter
to increment and log the value of a counter when the entry and receipt servlets are accessed.In the
doFilter
method, both filters retrieve the servlet context from the filter configuration object so that they can access the counters stored as context attributes. After the filters have completed application-specific processing, they invokedoFilter
on the filter chain object passed into the originaldoFilter
method. The elided code is discussed in the next section.public final class HitCounterFilter implements Filter { private FilterConfig filterConfig = null; public void init(FilterConfig filterConfig) throws ServletException { this.filterConfig = filterConfig; } public void destroy() { this.filterConfig = null; } public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException { if (filterConfig == null) return; StringWriter sw = new StringWriter(); PrintWriter writer = new PrintWriter(sw); Counter counter = (Counter)filterConfig. getServletContext(). getAttribute("hitCounter"); writer.println(); writer.println("==============="); writer.println("The number of hits is: " + counter.incCounter()); writer.println("==============="); // Log the resulting string writer.flush(); filterConfig.getServletContext(). log(sw.getBuffer().toString()); ... chain.doFilter(request, wrapper); ... } }Programming Customized Requests and Responses
There are many ways for a filter to modify a request or response. For example, a filter could add an attribute to the request or insert data in the response. In the Duke's Bookstore example,
HitCounterFilter
inserts the value of the counter into the response.A filter that modifies a response must usually capture the response before it is returned to the client. The way to do this is to pass the servlet that generates the response a stand-in stream. The stand-in stream prevents the servlet from closing the original response stream when it completes and allows the filter to modify the servlet's response.
In order to pass this stand-in stream to the servlet, the filter creates a response "wrapper" that overrides the
getWriter
orgetOutputStream
method to return this stand-in stream. The wrapper is passed to thedoFilter
method of the filter chain. Wrapper methods default to calling through to the wrapped request or response object. This approach follows the well-known Wrapper or Decorator pattern described in Design Patterns, Elements of Reusable Object-Oriented Software. The following sections describe how the hit counter filter described earlier and other types of filters use wrappers.To override request methods, you wrap the request in an object that extends
ServletRequestWrapper
orHttpServletRequestWrapper
. To override response methods, you wrap the response in an object that extendsServletResponseWrapper
orHttpServletResponseWrapper
.
HitCounterFilter
wraps the response in aCharResponseWrapper
. The wrapped response is passed to the next object in the filter chain, which isBookStoreServlet
.BookStoreServlet
writes its response into the stream created byCharResponseWrapper
. Whenchain.doFilter
returns,HitCounterFilter
retrieves the servlet's response fromPrintWriter
and writes it to a buffer. The filter inserts the value of the counter into the buffer, resets the content length header of the response, and finally writes the contents of the buffer to the response stream.PrintWriter out = response.getWriter(); CharResponseWrapper wrapper = new CharResponseWrapper( (HttpServletResponse)response); chain.doFilter(request, wrapper);CharArrayWriter caw = new CharArrayWriter(); caw.write(wrapper.toString().substring(0, wrapper.toString().indexOf("</body>")-1)); caw.write("<p>\n<center><center>" + messages.getString("Visitor") + "<font color='red'>" + counter.getCounter() + "</font><center>"); caw.write("\n</body></html>"); response.setContentLength(caw.toString().length()); out.write(caw.toString());
out.close(); public class CharResponseWrapper extends HttpServletResponseWrapper { private CharArrayWriter output; public String toString() { return output.toString(); } public CharResponseWrapper(HttpServletResponse response){ super(response); output = new CharArrayWriter(); } public PrintWriter getWriter(){ return new PrintWriter(output); } }Figure 3 shows the entry page for Duke's Bookstore with the hit counter.
Specifying Filter Mappings
A Web container uses filter mappings to decide how to apply filters to Web resources. A filter mapping matches a filter to a Web component by name or to Web resources by URL pattern. The filters are invoked in the order that filter mappings appear in the filter mapping list of a WAR.
To map a filter to a Web resources you:
- Declare the filter using the
<filter>
element in the Web application deployment descriptor. This element creates a name for the filter and declares the filter's implementation class and initialization parameters.- Map the filter to a Web resource by defining a
<filter-mapping>
element in the deployment descriptor. This element maps a filter name to a Web resource by name or by URL pattern.The following elements show how to specify the hit counter and order filters. To define a filter you provide a name for the filter, the class that implements the filter, and optionally some initialization parameters.
<filter> <filter-name>OrderFilter</filter-name> <filter-class>filters.OrderFilter</filter-class> </filter> <filter> <filter-name>HitCounterFilter</filter-name> <filter-class>filters.HitCounterFilter</filter-class> </filter>The
filter-mapping
element maps the order filter to the/receipt
URL. The mapping could also have specified the servletReceiptServlet
. Note that thefilter
,filter-mapping
,servlet
, andservlet-mapping
elements must appear in the Web application deployment descriptor in that order.<filter-mapping> <filter-name>OrderFilter</filter-name> <url-pattern>/receipt</url-pattern> </filter-mapping> <filter-mapping> <filter-name>HitCounterFilter</filter-name> <url-pattern>/enter</url-pattern> </filter-mapping>
If you want to log every request to a Web application, you would map the hit counter filter to the URL pattern
/*
. Table 6 summarizes the filter mapping list for the Duke's Bookstore application. The filters are matched by URL pattern and each filter chain contains only one filter.
Table 6 Duke's Bookstore Filter Mapping List URL
Filter
/enter
HitCounterFilter
/receipt
OrderFilter
You can map a filter to one or more Web resource and you can map more than one filter to a Web resource. This is illustrated in Figure 4, where filter F1 is mapped to servlets S1, S2, and S3, filter F2 is mapped to S2, and filter F3 is mapped to S1 and S2.
Figure 4 Filter to Servlet Mapping
Recall that a filter chain is one of the objects passed to the
doFilter
method of a filter. This chain is formed indirectly via filter mappings. The order of the filters in the chain is the same as the order that filter mappings appear in the Web application deployment descriptor.When a filter is mapped to servlet S1, the Web container invokes the
doFilter
method of F1. ThedoFilter
method of each filter in S1's filter chain is invoked by the preceding filter in the chain via thechain.doFilter
method. Since S1's filter chain contains filters F1 and F3, F1's call tochain.doFilter
invokes thedoFilter
method of filter F3. When F3'sdoFilter
method completes, control returns to F1'sdoFilter
method.
Home TOC |
![]() ![]() ![]() |