org.ocpsoft.rewrite.servlet.impl.HttpRewriteWrappedResponse Maven / Gradle / Ivy
/*
* Copyright 2013 Lincoln Baxter, III
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.ocpsoft.rewrite.servlet.impl;
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.io.OutputStreamWriter;
import java.io.PrintWriter;
import java.io.UnsupportedEncodingException;
import java.nio.charset.Charset;
import java.util.ArrayList;
import java.util.List;
import javax.servlet.ServletContext;
import javax.servlet.ServletOutputStream;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.ocpsoft.common.util.Streams;
import org.ocpsoft.logging.Logger;
import org.ocpsoft.logging.Logger.Level;
import org.ocpsoft.rewrite.AbstractRewrite;
import org.ocpsoft.rewrite.event.Rewrite;
import org.ocpsoft.rewrite.exception.RewriteException;
import org.ocpsoft.rewrite.servlet.RewriteLifecycleContext;
import org.ocpsoft.rewrite.servlet.RewriteWrappedResponse;
import org.ocpsoft.rewrite.servlet.config.response.ResponseContent;
import org.ocpsoft.rewrite.servlet.config.response.ResponseContentInterceptor;
import org.ocpsoft.rewrite.servlet.config.response.ResponseStreamWrapper;
import org.ocpsoft.rewrite.servlet.event.BaseRewrite.ServletRewriteFlow;
import org.ocpsoft.rewrite.servlet.event.OutboundServletRewrite;
import org.ocpsoft.rewrite.servlet.http.event.HttpServletRewrite;
import org.ocpsoft.rewrite.servlet.spi.OutboundRewriteProducer;
import org.ocpsoft.rewrite.servlet.spi.RewriteLifecycleListener;
import org.ocpsoft.rewrite.spi.RewriteProvider;
import org.ocpsoft.urlbuilder.Address;
import org.ocpsoft.urlbuilder.AddressBuilder;
/**
* @author Lincoln Baxter, III
*/
public class HttpRewriteWrappedResponse extends RewriteWrappedResponse
{
private final HttpServletRequest request;
private final ServletContext servletContext;
private static final Logger log = Logger.getLogger(HttpRewriteWrappedResponse.class);
public HttpRewriteWrappedResponse(final HttpServletRequest request, final HttpServletResponse response,
final ServletContext servletContext)
{
super(request, response);
this.request = request;
this.servletContext = servletContext;
if (getCurrentInstance(request) == null) {
super.setCurrentInstance(this);
}
}
/*
* Buffering Facilities
*/
private final ByteArrayOutputStream bufferedResponseContent = new ByteArrayOutputStream();
private List responseContentInterceptors = new ArrayList();
private List responseStreamWrappers = new ArrayList();
private boolean contentWritten = false;
private PrintWriter printWriter;
private ServletOutputStream outputStream = null;
private ServletOutputStream wrappedOutputStream = null;
@Override
public boolean isResponseContentIntercepted()
{
return !responseContentInterceptors.isEmpty();
}
@Override
public boolean isResponseStreamWrapped()
{
return !responseStreamWrappers.isEmpty();
}
@Override
public void addContentInterceptor(ResponseContentInterceptor stage) throws IllegalStateException
{
if (areStreamsLocked())
{
throw new IllegalStateException(
"Cannot add output buffers to response once request processing has been passed to the application.");
}
this.responseContentInterceptors.add(stage);
}
@Override
public void addStreamWrapper(ResponseStreamWrapper wrapper)
{
if (areStreamsLocked())
{
throw new IllegalStateException(
"Cannot add output stream wrappers to response once request processing has been passed to the application.");
}
this.responseStreamWrappers.add(wrapper);
}
private boolean areStreamsLocked()
{
return contentWritten;
}
private void lockStreams()
{
this.contentWritten = true;
}
/**
* Cause any buffered {@link ServletResponse} content to be processed and flushed to the client.
*/
@Override
public void flushBufferedContent()
{
if (isResponseContentIntercepted())
{
try {
bufferedResponseContent.close();
ResponseContent buffer = new ResponseContentImpl(bufferedResponseContent.toByteArray(),
Charset.forName(getCharacterEncoding()));
new ResponseContentInterceptorChainImpl(responseContentInterceptors).begin(new HttpBufferRewriteImpl(
request, this, servletContext), buffer);
if (!Charset.forName(getCharacterEncoding()).equals(buffer.getCharset()))
setCharacterEncoding(buffer.getCharset().name());
ServletOutputStream outputStream = isResponseStreamWrapped() ? wrappedOutputStream : super
.getOutputStream();
if (outputStream != null)
Streams.copy(new ByteArrayInputStream(buffer.getContents()), outputStream);
if (printWriter != null) {
printWriter.close();
}
}
catch (IOException e) {
throw new RewriteException("Error occurred when flushing response content buffered by "
+ responseContentInterceptors, e);
}
}
}
@Override
public void finishStreamWrappers()
{
if (isResponseStreamWrapped())
{
HttpServletRewrite event = new HttpBufferRewriteImpl(request, this, servletContext);
for (ResponseStreamWrapper wrapper : responseStreamWrappers) {
wrapper.finish(event);
}
}
}
@Override
public String toString()
{
if (isResponseContentIntercepted())
{
try {
return bufferedResponseContent.toString(getCharacterEncoding());
}
catch (UnsupportedEncodingException e) {
throw new RewriteException("Response accepted invalid character encoding " + getCharacterEncoding(), e);
}
}
else
return super.toString();
}
@Override
public PrintWriter getWriter()
{
if (printWriter == null)
{
if (isResponseContentIntercepted())
{
printWriter = new PrintWriter(new OutputStreamWriter(bufferedResponseContent,
Charset.forName(getCharacterEncoding())), true);
}
else if (isResponseStreamWrapped())
{
printWriter = new PrintWriter(new OutputStreamWriter(getOutputStream(),
Charset.forName(getCharacterEncoding())), true);
}
else
{
try {
lockStreams();
return super.getWriter();
}
catch (IOException e) {
throw new RewriteException("Could not get response writer.", e);
}
}
}
return printWriter;
}
@Override
public ServletOutputStream getOutputStream()
{
if (outputStream == null)
{
if (isResponseContentIntercepted())
{
outputStream = new RewriteServletOutputStream(bufferedResponseContent);
}
else
{
try {
lockStreams();
outputStream = super.getOutputStream();
}
catch (IOException e) {
throw new RewriteException("Could not get response output stream.", e);
}
}
if (isResponseStreamWrapped())
{
if (wrappedOutputStream == null)
{
HttpServletRewrite event = new HttpBufferRewriteImpl(request, this, servletContext);
try {
OutputStream wrapped = super.getOutputStream();
for (ResponseStreamWrapper wrapper : responseStreamWrappers) {
wrapped = wrapper.wrap(event, wrapped);
}
wrappedOutputStream = new RewriteServletOutputStream(wrapped);
if (!isResponseContentIntercepted())
{
outputStream = wrappedOutputStream;
}
}
catch (IOException e) {
throw new RewriteException("Could not get response output stream.", e);
}
}
}
}
return outputStream;
}
@Override
public void setContentLength(int contentLength)
{
lockStreams();
/*
* Prevent content-length being set as the page might be modified.
*/
if (!isResponseContentIntercepted())
{
if (isResponseStreamWrapped())
{
setHeader("X-Uncompressed-Content-Length", String.valueOf(contentLength));
}
else
super.setContentLength(contentLength);
}
}
@Override
public void flushBuffer() throws IOException
{
if (isResponseContentIntercepted())
bufferedResponseContent.flush();
else
{
lockStreams();
super.flushBuffer();
}
}
/**
* Buffered {@link ServletOutputStream} implementation.
*
* @author Lincoln Baxter, III
*/
private class RewriteServletOutputStream extends ServletOutputStream
{
private OutputStream stream;
public RewriteServletOutputStream(OutputStream outputStream)
{
this.stream = outputStream;
}
@Override
public void write(int b)
{
try {
stream.write(b);
}
catch (IOException e) {
throw new RewriteException("Error writing int to stream [" + stream + "]", e);
}
}
@Override
public void write(byte[] bytes) throws IOException
{
stream.write(bytes);
}
@Override
public void write(byte[] bytes, int off, int len)
{
try {
stream.write(bytes, off, len);
}
catch (IOException e) {
throw new RewriteException("Error writing bytes to stream [" + stream + "] at offset [" + off
+ "] with length [" + len + "]", e);
}
}
@Override
public String toString()
{
return stream.toString();
}
}
/*
* End buffering facilities
*/
@Override
public String encodeRedirectUrl(final String url)
{
return encodeRedirectURL(url);
}
@Override
public String encodeUrl(final String url)
{
return encodeURL(url);
}
@Override
public String encodeRedirectURL(final String url)
{
Address address = AddressBuilder.create(url);
OutboundServletRewrite event = rewrite(address);
if (event.getFlow().is(ServletRewriteFlow.ABORT_REQUEST))
{
return event.getOutboundAddress().toString();
}
return super.encodeRedirectURL(event.getOutboundAddress().toString());
}
@Override
public String encodeURL(final String url)
{
/*
* In some situations "url" may be not valid according to the rules defined in the RFC.
* In this case AddressBuilder.create() will fail. In these situations we basically
* skip outbound rewriting and just return the result of the super class.
*/
Address address;
try
{
address = AddressBuilder.create(url);
}
catch (IllegalArgumentException e) {
log.warn("Skipping outbound rewriting of invalid URL: " + url);
return super.encodeURL(url);
}
OutboundServletRewrite event = rewrite(address);
if (event.getFlow().is(ServletRewriteFlow.ABORT_REQUEST))
{
return event.getOutboundAddress().toString();
}
return super.encodeURL(event.getOutboundAddress().toString());
}
@SuppressWarnings({ "unchecked", "rawtypes" })
private OutboundServletRewrite rewrite(Address address)
{
OutboundServletRewrite event = null;
try {
RewriteLifecycleContext context = (RewriteLifecycleContext) request
.getAttribute(RewriteLifecycleContext.LIFECYCLE_CONTEXT_KEY);
for (OutboundRewriteProducer producer : context.getOutboundProducers()) {
if (producer.handles(address))
{
event = ((OutboundRewriteProducer) producer)
.createOutboundRewrite(request, getResponse(), servletContext, address);
}
}
if (event == null)
{
log.warn("No instance of [" + OutboundServletRewrite.class
+ "] was produced. Rewriting is disabled on this outbound event.");
}
else
{
for (RewriteLifecycleListener listener : context.getRewriteLifecycleListeners())
{
listener.beforeOutboundRewrite(event);
}
for (RewriteProvider p : context.getRewriteProviders())
{
if (p.handles(event))
{
p.rewrite(event);
if (event.getFlow().is(ServletRewriteFlow.HANDLED))
{
break;
}
}
}
for (RewriteLifecycleListener listener : context.getRewriteLifecycleListeners())
{
listener.afterOutboundRewrite(event);
}
}
AbstractRewrite.logEvaluatedRules(event, Level.DEBUG);
return event;
}
catch (RuntimeException e)
{
if (event != null)
AbstractRewrite.logEvaluatedRules(event, Level.ERROR);
throw e;
}
}
@Override
public void sendError(int sc, String msg) throws IOException
{
lockStreams();
super.sendError(sc, msg);
}
@Override
public void sendError(int sc) throws IOException
{
lockStreams();
super.sendError(sc);
}
@Override
public void sendRedirect(String location) throws IOException
{
lockStreams();
super.sendRedirect(location);
}
@Override
public void reset()
{
bufferedResponseContent.reset();
super.reset();
}
@Override
public void resetBuffer()
{
bufferedResponseContent.reset();
super.resetBuffer();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy