All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.jruby.rack.servlet.ResponseCapture Maven / Gradle / Ivy

/*
 * Copyright (c) 2010-2012 Engine Yard, Inc.
 * Copyright (c) 2007-2009 Sun Microsystems, Inc.
 * This source code is available under the MIT license.
 * See the file LICENSE.txt for details.
 */

package org.jruby.rack.servlet;

import java.io.IOException;
import java.io.OutputStreamWriter;
import java.io.PrintWriter;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.Collection;
import java.util.Collections;

import javax.servlet.ServletOutputStream;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpServletResponseWrapper;

/**
 * Response wrapper passed to filter chain.
 */
public class ResponseCapture extends HttpServletResponseWrapper {

    private static final String STREAM = "stream";
    private static final String WRITER = "writer";

    private static final Collection defaultNotHandledStatuses = Collections.singleton(404);

    private Integer status;
    private Object output;

    private boolean handledByDefault;
    private Collection notHandledStatuses = defaultNotHandledStatuses;

    /**
     * Wrap a response
     * @param response
     */
    public ResponseCapture(HttpServletResponse response) {
        super(response);
    }

    /**
     * @return the status set using one of the set status methods
     * @see #handleStatus(int, boolean)
     */
    public int getStatus() {
        return status != null ? status : 200;
    }

    public boolean isStatusSet() {
        return status != null;
    }

    /**
     * Status code handler customizable by sub-classes.
     *
     * Besides serving as a setter, should return whether the status has been
     * "accepted" (super methods will be called when this is invoked from response
     * API methods such as {@link #setStatus(int)}).
     *
     * @param status the new HTTP status
     * @param error whether the status comes from a {@code sendError}
     * @see #isHandled(HttpServletRequest)
     */
    protected boolean handleStatus(int status, boolean error) {
        this.status = status;
        return isHandled(null);
    }

    @Override
    public void setStatus(int status) {
        // no longer check if there ain't an error before calling super ...
        // if an error has been set previously the caller should deal with it
        if ( handleStatus(status, false) ) {
            super.setStatus(status);
        }
    }

    @Override
    public void setStatus(int status, String message) {
        if ( handleStatus(status, false) ) {
            super.setStatus(status, message);
        }
    }

    @Override
    public void sendError(int status) throws IOException {
        if ( handleStatus(status, true) ) {
            super.sendError(status);
        }
        // after using this method, the response should be considered to be
        // committed and should not be written to ... ever again !
    }

    @Override
    public void sendError(int status, String message) throws IOException {
        if ( handleStatus(status, true) ) {
            super.sendError(status, message);
        }
    }

    @Override
    public void sendRedirect(String path) throws IOException {
        if ( handleStatus(302, false) ) {
            super.sendRedirect(path);
        }
    }

    private boolean headerSet;

    public boolean isHeaderSet() {
        return headerSet;
    }

    @Override
    public void addDateHeader(String name, long date) {
        super.addDateHeader(name, date);
        headerSet = true;
    }

    @Override
    public void addHeader(String name, String value) {
        super.addHeader(name, value);
        headerSet = true;
    }

    @Override
    public void addIntHeader(String name, int value) {
        super.addIntHeader(name, value);
        headerSet = true;
    }

    @Override
    public void setHeader(String name, String value) {
        super.setHeader(name, value);
        headerSet = true;
    }

    @Override
    public void setDateHeader(String name, long date) {
        super.setDateHeader(name, date);
        headerSet = true;
    }

    @Override
    public void setIntHeader(String name, int value) {
        super.setIntHeader(name, value);
        headerSet = true;
    }

    @Override
    public ServletOutputStream getOutputStream() throws IOException {
        if ( output == null ) output = STREAM;

        if ( isHandled(null) ) return super.getOutputStream();
        else { // TODO get rid of this in 1.2.0
            // backwards compatibility with isError() :
            return new ServletOutputStream() {
                @Override
                public void write(int b) throws IOException {
                    // swallow output, because we're going to discard it
                }
            };
        }
    }

    @Override
    public PrintWriter getWriter() throws IOException {
        if ( output == null ) output = WRITER;

        // we protect against API limitations as we depend on #getWriter
        // being functional even if getOutputStream has been called ...
        if ( output != WRITER ) {
            String enc = getCharacterEncoding();
            if ( enc == null ) enc = "UTF-8";
            return new PrintWriter(new OutputStreamWriter(getOutputStream(), enc));
        }
        else {
            return super.getWriter();
        }
    }

    @Override
    public void flushBuffer() throws IOException {
        if ( isHandled(null) ) super.flushBuffer();
    }

    public boolean isError() {
        return getStatus() >= 400;
    }

    /**
     * @deprecated no longer to be used from outside
     * @see #isHandled(HttpServletRequest)
     */
    public boolean isHandled() {
        return isHandled(null);
    }

    private boolean handled; // once handled - stays handled

    /**
     * Response is considered to be handled if a status has been set
     * and it is (by default) not a HTTP NOT FOUND (404) status.
     *
     * @return true if this response should be considered as handled
     * @see #handleStatus(int, boolean)
     */
    public boolean isHandled(final HttpServletRequest request) {
        if ( handled ) return true; // ... once handled always handled !

        // setting a header should consider the response to be handled
        if ( ! isStatusSet() ) {
            if ( ! isHeaderSet() ) {
                // true by default seems very weird but this is best to cover
                // "more" containers right out of the box ...
                // e.g. Jetty https://github.com/jruby/jruby-rack/issues/175
                return handled = isHandledByDefault();
            }

            // consider HTTP OPTIONS with "Allow" header unhandled :
            if ( request != null && "OPTIONS".equals( request.getMethod() ) ) {
                final Collection headerNames = getHeaderNamesOrNull();
                if ( headerNames == null || headerNames.isEmpty() ) {
                    // not to happen but there's all kind of beasts out there
                    return false;
                }
                for ( final String headerName : headerNames ) {
                    if ( ! "Allow".equals( headerName ) ) {
                        return handled = true; // not just Allow header - consider handled
                    }
                }
                return false; // OPTIONS with only Allow header set - unhandled
            }
            return handled = true;
        }

        if ( notHandledStatuses.contains( getStatus() ) ) return false;
        return handled = true;
    }

    public Collection getNotHandledStatuses() {
        return this.notHandledStatuses;
    }

    @SuppressWarnings("unchecked")
    public void setNotHandledStatuses(final Collection notHandledStatuses) {
        this.notHandledStatuses =
            notHandledStatuses == null ? Collections.EMPTY_SET : notHandledStatuses;
    }

    boolean isHandledByDefault() {
        return handledByDefault;
    }

    public void setHandledByDefault(boolean handledByDefault) {
        this.handledByDefault = handledByDefault;
    }

    /**
     * @return true if {@link #getOutputStream()} (or {@link #getWriter()}) has
     * been accessed
     */
    public boolean isOutputAccessed() {
        return output != null;
    }

    @SuppressWarnings("unchecked")
    private Collection getHeaderNamesOrNull() {
        // NOTE: getHeaderNames since Servlet API 3.0 JRuby-Rack 1.1 still supports 2.5
        try {
            final Method getHeaderNames = getResponse().getClass().getMethod("getHeaderNames");
            return (Collection) getHeaderNames.invoke( getResponse() );
        }
        catch (NoSuchMethodException e) { return null; }
        catch (IllegalAccessException e) { return null; }
        catch (InvocationTargetException e) { return null; }
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy