EsiRenderer.java
/*
* 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.esigate.extension.parallelesi;
import java.io.IOException;
import java.io.Writer;
import java.util.Map;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.Executor;
import java.util.concurrent.Future;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.TimeoutException;
import java.util.regex.Pattern;
import org.apache.http.HttpStatus;
import org.esigate.HttpErrorPage;
import org.esigate.Renderer;
import org.esigate.impl.DriverRequest;
import org.esigate.parser.future.FutureAppendable;
import org.esigate.parser.future.FutureAppendableAdapter;
import org.esigate.parser.future.FutureParser;
import org.esigate.parser.future.StringBuilderFutureAppendable;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* Retrieves a resource from the provider application and parses it to find ESI tags to be replaced by contents from
* other applications.
*
* For more information about ESI language specification, see <a href="http://www.w3.org/TR/esi-lang">Edge Side
* Include</a>
*
* <p>
* This class is based on EsiRenderer
*
* @see org.esigate.esi.EsiRenderer
*
* @author Nicolas Richeton
*/
public class EsiRenderer implements Renderer, FutureAppendable {
private static final Logger LOG = LoggerFactory.getLogger(EsiRenderer.class);
/**
* Key for the executor for future tasks. This is used with parser#setData().
*/
public static final String DATA_EXECUTOR = "executor";
private static final Pattern PATTERN = Pattern
.compile("(<esi:\\w+((\\s+\\w+(\\s*=\\s*(?:\".*?\"|'.*?'|[^'\">\\s]+))?)+\\s*|\\s*)/?>)|(</esi:[^>]*>)");
private static final Pattern PATTERN_COMMENTS = Pattern.compile("(<!--esi)|(-->)");
private final FutureParser parser = new FutureParser(PATTERN, IncludeElement.TYPE, CommentElement.TYPE,
RemoveElement.TYPE, VarsElement.TYPE, ChooseElement.TYPE, WhenElement.TYPE, OtherwiseElement.TYPE,
TryElement.TYPE, AttemptElement.TYPE, ExceptElement.TYPE, InlineElement.TYPE, ReplaceElement.TYPE,
FragmentElement.TYPE);
private final FutureParser parserComments = new FutureParser(PATTERN_COMMENTS, Comment.TYPE);
private Map<String, CharSequence> fragmentsToReplace;
private final String page;
private final String name;
private boolean write = true;
private boolean found = false;
private FutureAppendableAdapter futureOut;
private Executor executor;
public String getName() {
return name;
}
public void setWrite(boolean write) {
this.write = write;
}
/**
* Constructor used to render a complete page.
*
* @param executor
* Executor to use for background operations or null if single-thread operations.
*/
public EsiRenderer(Executor executor) {
this.page = null;
this.name = null;
this.executor = executor;
}
/**
* Constructor used to render a fragment Retrieves a fragment inside a page.<br>
*
* Extracts html between <code><esi:fragment name="myFragment"></code> and <code></esi:fragment></code>
*
* @param page
* @param name
* @param executor
* Executor to use for background operations or null if single-thread operations.
*/
public EsiRenderer(String page, String name, Executor executor) {
this.page = page;
this.name = name;
this.write = false;
this.executor = executor;
}
public Map<String, CharSequence> getFragmentsToReplace() {
return fragmentsToReplace;
}
public void setFragmentsToReplace(Map<String, CharSequence> fragmentsToReplace) {
this.fragmentsToReplace = fragmentsToReplace;
}
@Override
public void render(DriverRequest originalRequest, String content, Writer out) throws IOException, HttpErrorPage {
if (name != null) {
LOG.debug("Rendering fragment {} in page {}", name, page);
}
this.futureOut = new FutureAppendableAdapter(out);
if (content == null) {
return;
}
try {
// Pass 1. Remove esi comments
StringBuilderFutureAppendable contentWithoutComments = new StringBuilderFutureAppendable();
parserComments.setHttpRequest(originalRequest);
parserComments.setData(DATA_EXECUTOR, this.executor);
parserComments.parse(content, contentWithoutComments);
CharSequence contentWithoutCommentsResult;
contentWithoutCommentsResult = contentWithoutComments.get();
// Pass 2. Process ESI
parser.setHttpRequest(originalRequest);
parser.setData(DATA_EXECUTOR, this.executor);
parser.parse(contentWithoutCommentsResult, this);
if (name != null && !this.found) {
throw new HttpErrorPage(HttpStatus.SC_BAD_GATEWAY, "Fragment " + name + " not found", "Fragment "
+ name + " not found");
}
this.futureOut.performAppends();
} catch (ExecutionException e) {
throw new IOException(e);
}
}
@Override
public FutureAppendable enqueueAppend(Future<CharSequence> csq) {
if (this.write) {
this.futureOut.enqueueAppend(csq);
}
return this;
}
public boolean isWrite() {
return this.write;
}
public void setFound(boolean found) {
this.found = found;
}
@Override
public FutureAppendable performAppends() throws IOException, HttpErrorPage {
return this.futureOut.performAppends();
}
@Override
public boolean hasPending() {
return this.futureOut.hasPending();
}
@Override
public FutureAppendable performAppends(int timeout, TimeUnit unit) throws IOException, HttpErrorPage,
TimeoutException {
return this.futureOut.performAppends(timeout, unit);
}
}