[Demo] Filters and Wrappers

You may use request and response wrappers to extend the functionality of request or response objects. For instance, we can provide an extra layer of buffer using a wrapper with its own internal buffer independent of the output stream supplied by the container. We can pass this to other components and once we finally verify the content, then we can write it into containers output stream and thus actually committing the response. 

Using filters you can pass a wrapper to a servlet without modifying the servlet. We can also add and remove the filter without touching the target servlet at all. You can also pass in our custom wrapper to a RequestDispatcher forward and include, as you seen in a previous demo. However then you will have to modify the target servlet also as you have seen. Wrappers were used along with RequestDispatcher before the addition of filters to the Java EE specification. After the addition of filters to the servlet specification, wrappers should be now used with filters than directly with servlets using RequestDispatcher mechanism. 

 

Setting up the basic servlet

We will reuse the same FirstServlet and ForwardedServlet from our RequestDispatcher basic demo. We will rename our FirstServlet to FirstServletV2. We will also need to change the url-pattern, the java file name, constructor name.

In FirstServletV2, we will remove the initial getWriter part as we will be using an outputstream later and we cannot call both getWriter and getOutputStream.

In FirstServletV2, you may remove or comment out the RD for include, and uncomment both the forwards. We will also add a request attribute to keep a count of the number of forwards we do.

Similarly we will ForwardedServlet to ForwardedServletV2 to get and print the value of the new request attribute.

 

The doGet of FirstServletV2 will now look like:

System.out.println("FirstServletV2.doGet");

 

RequestDispatcher rd2 = getServletContext().getRequestDispatcher(

    "/ForwardedServletV2");

request.setAttribute("CallNumber", "1");

rd2.forward(request, response);

request.setAttribute("CallNumber", "2");

rd2.forward(request, response);

The doGet of ForwardedServletV2 will now look like:

System.out.println("ForwardedServletV2.doGet");

 

PrintWriter out = response.getWriter();

out.println("<br>ForwardedServlet.doGet"

    + request.getAttribute("CallNumber"));

 

Now try deploying and executing the class. We are trying to forward twice and we will get the exception in the console log:

java.lang.IllegalStateException: Cannot forward after response has been committed

 

Creating our wrapper class

We will now create a wrapper class to override the default buffer with one of our own. We will override the below methods in our custom wrapper:

  • void flushBuffer() – We will not provide any implementation and hence the next component will not be able to write any data to the client.

  • PrintWriter getWriter() – We will create and return our own writer object and hence  the next component will not be able to write directly to the response stream.

  • OutputStream getOutputStream() – We will create and return our own output stream.

We will also extend ServletOutputStream to create our own ServletOutputStream so that we can instantiate it within the getOutputStream above to provide our own output stream. We will override only its abstract write method to write to our own output stream.

ServletOutputStream has added two new methods isReady() and setWriteListener(WriteListener) servlet spec 3.1 (Java EE 7). We will simply implement them and provide no code. However if you are using a lower version of servlet spec, then you don’t need to implement these.

Code for our complete wrapper class is given below:

package com.javajee.rdmechanisms;

 

import java.io.ByteArrayOutputStream;

import java.io.PrintWriter;

 

import javax.servlet.ServletOutputStream;

import javax.servlet.WriteListener;

import javax.servlet.http.HttpServletResponse;

import javax.servlet.http.HttpServletResponseWrapper;

 

public class CustCommitHttpRespWrapper extends HttpServletResponseWrapper {

 

  private ByteArrayOutputStream out;

  private ServletOutputStream outBytes;

  private PrintWriter outWriter;

 

  public CustCommitHttpRespWrapper(HttpServletResponse wrapper) {

    super(wrapper);

  }

 

  private class MyServletOutputStream extends ServletOutputStream {

    public void write(int b) {

      out.write(b);

    }

 

    // Below two methods isReady and setWriteListener

    // were added in servlet spec 3.1 (Java EE 7).

    @Override

    public boolean isReady() {

      return false;

    }

 

    @Override

    public void setWriteListener(WriteListener arg0) {

    }

 

  }

 

  @Override

  public void flushBuffer() {

    // Do nothing

  }

 

  @Override

  public PrintWriter getWriter() {

    if (outWriter != null) {

      return outWriter;

    }

 

    if (outBytes != null) {

      throw new IllegalStateException("getOut already invoked");

    }

    out = new ByteArrayOutputStream();

    outWriter = new PrintWriter(out);

    return outWriter;

  }

 

  @Override

  public ServletOutputStream getOutputStream() {

    if (outBytes != null) {

      return outBytes;

    }

 

    if (outWriter != null) {

      throw new IllegalStateException("getWriter was invoked");

    }

 

    out = new ByteArrayOutputStream();

    outBytes = new MyServletOutputStream();

    return outBytes;

  }

 

  public ByteArrayOutputStream getOut() {

    return out;

  }

}

This is not a perfect implementation is just provided to show the possibilities of things we can do with a wrapper.

Instead of going back and modifying FirstServletV2 as we seen in a previous demo using RD mechanism, we will now create a new filter CustCommitHttpRespWrapper:

package com.javajee.filters;

import java.io.IOException;

import javax.servlet.Filter;
import javax.servlet.FilterChain;
import javax.servlet.FilterConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import javax.servlet.annotation.WebFilter;
import javax.servlet.http.HttpServletResponse;

import com.javajee.rdmechanisms.CustCommitHttpRespWrapper;

@WebFilter("/FirstServletV2")
public class CustCommitFilter implements Filter {

  public CustCommitFilter() {
  }

  @Override
  public void doFilter(ServletRequest request, ServletResponse response,
      FilterChain chain) throws IOException, ServletException {

    HttpServletResponse r = (HttpServletResponse) response;
    CustCommitHttpRespWrapper wraper = new CustCommitHttpRespWrapper(r);

    chain.doFilter(request, wraper);

    if (wraper.isCommitted()) {
      return;
    }

    if (wraper.getOut() != null) {
      wraper.getOut().writeTo(r.getOutputStream());
    }

  }

  @Override
  public void init(FilterConfig conf) throws ServletException {
  }

  @Override
  public void destroy() {
  }
}

The code is very similar to the demo using RD that we seen before. We will create an instance of our wrapper class and pass it instead of original response. Since a filter's doFilter() method signature has ServletResponse, you will first need to cast it to an HttpServletResponse before you create the wrapper instance. You will also need this casted reference later to get the output stream. After calling chain.doFilter, we are adding code to actually write it to the client along with some error handling. 

Now deploy and execute our servlet and you will get the output in the browser as:

ForwardedServlet.doGet1

And in the cosole you will still get the output as:

FirstServletV2.doGet
ForwardedServletV2.doGet1
ForwardedServletV2.doGet2

Previously this would have been thrown an error in the console. We can also now decide to send an error to client even after data is written to output. Previously exception would have been thrown only in the console after sending some data as after the data was written to the original output stream, there was no way to tell the client about the error.

Complete Code

FirstServletV2.java

package com.javajee.rdmechanisms;

import java.io.IOException;
import javax.servlet.RequestDispatcher;
import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

@WebServlet("/FirstServletV2")
public class FirstServletV2 extends HttpServlet {
  private static final long serialVersionUID = 1L;

  public FirstServletV2() {
    super();
  }

  protected void doGet(HttpServletRequest request, HttpServletResponse response)
      throws ServletException, IOException {
    System.out.println("FirstServletV2.doGet");

    RequestDispatcher rd2 = getServletContext().getRequestDispatcher(
        "/ForwardedServletV2");
    request.setAttribute("CallNumber", "1");
    rd2.forward(request, response);
    request.setAttribute("CallNumber", "2");
    rd2.forward(request, response);
  }
}

 

ForwardedServletV2.java

package com.javajee.rdmechanisms;

import java.io.IOException;
import java.io.PrintWriter;
import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

 

@WebServlet("/ForwardedServletV2")
public class ForwardedServletV2 extends HttpServlet {
  private static final long serialVersionUID = 1L;

  public ForwardedServletV2() {
    super();
  }

  protected void doGet(HttpServletRequest request, HttpServletResponse response)
      throws ServletException, IOException {
    System.out.println("ForwardedServletV2.doGet"
        + request.getAttribute("CallNumber"));

    PrintWriter out = response.getWriter();
    out.println("<br>ForwardedServlet.doGet"
        + request.getAttribute("CallNumber"));

  }

}

 
The filter and wrapper complete code is already given above.
 

Development and Execution Environment

Eclipse Luna Java EE IDE for Web Developers and Apache Tomcat 8.0.18, using Servlet spec 3.1.

Quick Notes Finder Tags

Activities (1) advanced java (1) agile (3) App Servers (6) archived notes (2) Arrays (1) Best Practices (12) Best Practices (Design) (3) Best Practices (Java) (7) Best Practices (Java EE) (1) BigData (3) Chars & Encodings (6) coding problems (2) Collections (15) contests (3) Core Java (All) (55) course plan (2) Database (12) Design patterns (8) dev tools (3) downloads (2) eclipse (9) Essentials (1) examples (14) Exception (1) Exceptions (4) Exercise (1) exercises (6) Getting Started (18) Groovy (2) hadoop (4) hibernate (77) hibernate interview questions (6) History (1) Hot book (5) http monitoring (2) Inheritance (4) intellij (1) java 8 notes (4) Java 9 (1) Java Concepts (7) Java Core (9) java ee exercises (1) java ee interview questions (2) Java Elements (16) Java Environment (1) Java Features (4) java interview points (4) java interview questions (4) javajee initiatives (1) javajee thoughts (3) Java Performance (6) Java Programmer 1 (11) Java Programmer 2 (7) Javascript Frameworks (1) Java SE Professional (1) JPA 1 - Module (6) JPA 1 - Modules (1) JSP (1) Legacy Java (1) linked list (3) maven (1) Multithreading (16) NFR (1) No SQL (1) Object Oriented (9) OCPJP (4) OCPWCD (1) OOAD (3) Operators (4) Overloading (2) Overriding (2) Overviews (1) policies (1) programming (1) Quartz Scheduler (1) Quizzes (17) RabbitMQ (1) references (2) restful web service (3) Searching (1) security (10) Servlets (8) Servlets and JSP (31) Site Usage Guidelines (1) Sorting (1) source code management (1) spring (4) spring boot (3) Spring Examples (1) Spring Features (1) spring jpa (1) Stack (1) Streams & IO (3) Strings (11) SW Developer Tools (2) testing (1) troubleshooting (1) user interface (1) vxml (8) web services (1) Web Technologies (1) Web Technology Books (1) youtube (1)