net.jangaroo.apprunner.util.JettyWrapper Maven / Gradle / Ivy
package net.jangaroo.apprunner.util;
import net.jangaroo.apprunner.proxy.JangarooProxyServlet;
import org.apache.commons.lang3.Range;
import org.eclipse.jetty.server.Handler;
import org.eclipse.jetty.server.Server;
import org.eclipse.jetty.server.handler.HandlerCollection;
import org.eclipse.jetty.servlet.DefaultServlet;
import org.eclipse.jetty.servlet.ServletContextHandler;
import org.eclipse.jetty.servlet.ServletHolder;
import org.eclipse.jetty.util.resource.Resource;
import org.eclipse.jetty.util.resource.ResourceCollection;
import org.eclipse.jetty.webapp.WebAppContext;
import org.mitre.dsmiley.httpproxy.ProxyServlet;
import org.slf4j.Logger;
import javax.servlet.Servlet;
import java.io.File;
import java.io.IOException;
import java.net.InetSocketAddress;
import java.net.URI;
import java.nio.file.Path;
import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.stream.Collectors;
import static java.lang.invoke.MethodHandles.lookup;
import static java.util.Collections.addAll;
import static org.slf4j.LoggerFactory.getLogger;
/**
* Wrapper class for controlling a Jetty server.
*/
public class JettyWrapper {
private static final long WAIT_TIME_MILLIS = 1000L;
/**
* For the special port of zero, retry this given number of times
* to get a new random port, in case other services already blocked
* that port meanwhile.
*/
private static final int RANDOM_PORT_RETRY_LIMIT = 20;
private static final Integer RANDOM_PORT_IDENTIFIER = 0;
public static final String ROOT_PATH = "/";
public static final String ROOT_PATH_SPEC = "/*";
private static final StaticResourcesServletConfig DEFAULT_RESOURCES_SERVLET_CONFIG =
new StaticResourcesServletConfig(ROOT_PATH_SPEC, "/");
static {
Resource.setDefaultUseCaches(false);
}
private static final Logger LOG = getLogger(lookup().lookupClass());
private final Map configurationByPath = new HashMap<>();
private Server server;
private Class extends WebAppContext> webAppContextClass = WebAppContext.class;
/**
* Creates a Wrapper for controlling a Jetty server.
*
* @param baseDirs the base directories for serving static resources
*/
public JettyWrapper(Path... baseDirs) {
addBaseDirs(baseDirs);
}
public void addBaseDir(Path baseDir) {
addBaseDir(baseDir, ROOT_PATH);
}
@SuppressWarnings("WeakerAccess")
public void addBaseDirs(Path... baseDirs) {
addAll(getConfiguration(ROOT_PATH).baseDirs, baseDirs);
}
public void setWebAppContextClass(Class extends WebAppContext> webAppContextClass) {
this.webAppContextClass = webAppContextClass;
}
public void addBaseDir(Path baseDir, String path) {
getConfiguration(path).addBaseDir(baseDir);
}
public void addResourceJar(File resourceJar) {
addResourceJar(resourceJar, ROOT_PATH);
}
public void addResourceJar(File resourceJar, String path) {
addBaseDirInResourceJar(resourceJar, "META-INF/resources", path);
}
public void addBaseDirInResourceJar(File resourceJar, String relativePathInsideJar) {
addBaseDirInResourceJar(resourceJar, relativePathInsideJar, ROOT_PATH);
}
public void addBaseDirInResourceJar(File resourceJar, String relativePathInsideJar, String path) {
getConfiguration(path).addResourceJar(new JarFileWithRelativePath(resourceJar, relativePathInsideJar));
}
public void setStaticResourcesServletConfigs(List staticResourcesServletConfigs) {
setStaticResourcesServletConfigs(staticResourcesServletConfigs, ROOT_PATH);
}
public void setStaticResourcesServletConfigs(List staticResourcesServletConfigs, String path) {
getConfiguration(path).setStaticResourcesServletConfigs(staticResourcesServletConfigs);
}
public void setProxyServletConfigs(List proxyServletConfigs) {
setProxyServletConfigs(proxyServletConfigs, ROOT_PATH);
}
public void setProxyServletConfigs(List proxyServletConfigs, String path) {
getConfiguration(path).setProxyServletConfigs(proxyServletConfigs);
}
public void setAdditionalServlets(Map additionalServlets) {
setAdditionalServlets(additionalServlets, ROOT_PATH);
}
public void setAdditionalServlets(Map additionalServlets, String path) {
getConfiguration(path).setAdditionalServlets(additionalServlets);
}
public void start(String host, int port) throws JettyWrapperException {
start(host, Range.is(port));
}
public void start(String host, Range portRange) throws JettyWrapperException {
if (server != null) {
stop();
}
Exception lastException = null;
HandlerCollection handlerCollection = new HandlerCollection();
ArrayList paths = new ArrayList<>(configurationByPath.keySet());
// sort paths so that the deeper paths have precedence
paths.sort((path1, path2) -> {
String[] dirs1 = path1.split("/");
String[] dirs2 = path2.split("/");
return dirs2.length - dirs1.length;
});
for (String path : paths) {
Configuration configuration = getConfiguration(path);
WebAppContext handler = createHandler(
path,
configuration.baseDirs,
configuration.jarFilesWithRelativePath,
configuration.staticResourcesServletConfigs,
configuration.proxyServletConfigs,
configuration.additionalServlets
);
handlerCollection.addHandler(handler);
}
List shuffledPorts = shufflePortRange(portRange);
for (Integer port : shuffledPorts) {
try {
tryServerStart(host, port, handlerCollection);
break;
} catch (Exception e) {
getLog().debug("Could not start server", e);
lastException = e;
}
}
if (server == null || !server.isRunning()) {
throw new JettyWrapperException("Could not start server", lastException);
}
}
/**
* Tries to start the server.
*
* @param host host for server
* @param port port for server; zero will use a random port (and retry several times upon failure)
* @param handler server handler
* @throws Exception exception on failure; on retries, the last exception
*/
private void tryServerStart(String host, Integer port, Handler handler) throws Exception {
int retryLimit = RANDOM_PORT_IDENTIFIER.equals(port) ? RANDOM_PORT_RETRY_LIMIT : 1;
Exception lastException = null;
for (int i = 0; i < retryLimit; i++) {
try {
server = null;
server = createServer(host, port, handler);
server.start();
break;
} catch (Exception e) {
lastException = e;
}
}
if (lastException != null) {
throw lastException;
}
}
public void stop() {
if (server != null) {
try {
server.stop();
} catch (Exception e) {
getLog().warn("Could not stop server", e);
}
}
}
public URI getUri() {
return server.getURI().resolve(ROOT_PATH);
}
public void waitUntilStarted(int timeoutMillis) throws JettyWrapperException {
if (server == null) {
throw new JettyWrapperException("Server not started");
}
getLog().info(String.format("Waiting for Server startup (timeout = %d ms)", timeoutMillis));
int waitedMillis = 0;
try {
while (waitedMillis < timeoutMillis && !server.isStarted()) {
getLog().debug("Server not ready yet. Waiting ...");
Thread.sleep(WAIT_TIME_MILLIS);
waitedMillis += WAIT_TIME_MILLIS;
}
} catch (InterruptedException e) {
getLog().warn("Interrupted while waiting for startup", e);
Thread.currentThread().interrupt();
}
if (!server.isStarted()) {
throw new JettyWrapperException(String.format("Server did not start within %d ms", timeoutMillis));
}
}
public void blockUntilInterrupted() {
try {
server.join();
} catch (InterruptedException ignore) {
Thread.currentThread().interrupt();
}
}
private Configuration getConfiguration(String path) {
if (path.contains("\\")) {
getLog().warn("Path should not contain backslashes: " + path);
}
if (!configurationByPath.containsKey(path)) {
configurationByPath.put(path, new Configuration());
}
return configurationByPath.get(path);
}
private Server createServer(String host, int port, Handler handler) {
InetSocketAddress address = new InetSocketAddress(host, port);
getLog().info("Creating Jetty server at " + address);
Server server = new Server(address);
server.setHandler(handler);
return server;
}
private WebAppContext createHandler(String path, List baseDirs, List resourceJarsWithRelativePaths, List staticResourcesServletConfigs, List proxyServletConfigs, Map additionalServlets) throws JettyWrapperException {
// root path needs a root path servlet. make sure it is created.
boolean hasRootPathServlet = !(ROOT_PATH.equals(path));
try {
getLog().info("Setting up handler with context path " + path);
WebAppContext handler = webAppContextClass.newInstance();
handler.setContextPath(path);
handler.setInitParameter("org.eclipse.jetty.servlet.Default.useFileMappedBuffer", "false");
List baseResources = baseDirs.stream().map(Resource::newResource).collect(Collectors.toList());
baseResources = new ArrayList<>(baseResources);
if (!resourceJarsWithRelativePaths.isEmpty()) {
baseResources.addAll(
resourceJarsWithRelativePaths.stream()
.map(JarFileWithRelativePath::toResource)
.filter(Objects::nonNull)
.collect(Collectors.toList()));
}
handler.setBaseResource(new ResourceCollection(baseResources.stream().filter(Resource::exists).toArray(Resource[]::new)));
getLog().info(" Using base resources: " + baseResources);
if (staticResourcesServletConfigs != null && !staticResourcesServletConfigs.isEmpty()) {
for (StaticResourcesServletConfig config : staticResourcesServletConfigs) {
hasRootPathServlet = addDefaultServlet(handler, config) || hasRootPathServlet;
}
}
if (proxyServletConfigs != null && !proxyServletConfigs.isEmpty()) {
for (ProxyServletConfig config : proxyServletConfigs) {
hasRootPathServlet = addProxyServlet(handler, config) || hasRootPathServlet;
}
}
if (additionalServlets != null && !additionalServlets.isEmpty()) {
for (Map.Entry servletEntry : additionalServlets.entrySet()) {
hasRootPathServlet = addServlet(handler, new ServletHolder(servletEntry.getValue()), servletEntry.getKey()) || hasRootPathServlet;
}
}
if (!hasRootPathServlet) {
addDefaultServlet(handler, DEFAULT_RESOURCES_SERVLET_CONFIG);
}
return handler;
} catch (Exception e) {
throw new JettyWrapperException(e);
}
}
private boolean addDefaultServlet(ServletContextHandler webAppContext, StaticResourcesServletConfig config) {
String pathSpec = config.getPathSpec();
ServletHolder servletHolder = new ServletHolder(DefaultServlet.class);
servletHolder.setInitParameter("relativeResourceBase", config.getRelativeResourceBase());
servletHolder.setInitParameter("cacheControl", "no-store, no-cache, must-revalidate, max-age=0");
webAppContext.addAliasCheck(new AllowSymLinkAliasChecker());
boolean addedRootPathServlet = addServlet(webAppContext, servletHolder, pathSpec);
getLog().info(String.format(" Serving static resources: %s -> %s",
resolve(webAppContext.getContextPath(), pathSpec), config.getRelativeResourceBase()));
return addedRootPathServlet;
}
private boolean addProxyServlet(ServletContextHandler webAppContext, ProxyServletConfig config) {
String pathSpec = config.getPathSpec();
ServletHolder servletHolder = new ServletHolder(pathSpec, JangarooProxyServlet.class); // TODO: Is this really a good servlet name?
servletHolder.setInitParameter("targetUri", config.getTargetUri().replaceAll("/$", ""));
servletHolder.setInitParameter(ProxyServlet.P_FORWARDEDFOR, String.valueOf(config.isForwardedHeaderEnabled()));
servletHolder.setInitParameter(ProxyServlet.P_LOG, String.valueOf(config.isLoggingEnabled()));
servletHolder.setInitParameter(ProxyServlet.P_PRESERVECOOKIES, "true");
boolean addedRootPathServlet = addServlet(webAppContext, servletHolder, pathSpec);
getLog().info(String.format(" Proxying requests: %s -> %s",
resolve(webAppContext.getContextPath(), pathSpec), config.getTargetUri()));
return addedRootPathServlet;
}
private boolean addServlet(ServletContextHandler webAppContext, ServletHolder servletHolder, String pathSpec) {
webAppContext.addServlet(servletHolder, pathSpec);
return ROOT_PATH_SPEC.equals(pathSpec);
}
private List shufflePortRange(Range portRange) {
List shuffledPorts = new ArrayList<>();
for (int i = portRange.getMinimum(); i <= portRange.getMaximum(); i++) {
shuffledPorts.add(i);
}
Collections.shuffle(shuffledPorts);
return shuffledPorts;
}
protected Logger getLog() {
return LOG;
}
private static String resolve(String base, String relative) {
return (ROOT_PATH.equals(base) ? "" : base) + relative;
}
public static final class JettyWrapperException extends Exception {
JettyWrapperException(String message) {
super(message);
}
JettyWrapperException(String message, Throwable cause) {
super(message, cause);
}
JettyWrapperException(Throwable cause) {
super(cause);
}
}
private static class JarFileWithRelativePath {
public final File jarFile;
public final String relativePath;
private JarFileWithRelativePath(File jarFile, String relativePath) {
this.jarFile = jarFile;
this.relativePath = relativePath;
}
public Resource toResource() {
try {
return Resource.newResource("jar:" + Resource.toURL(jarFile).toString() + "!/" + relativePath);
} catch (IOException e) {
return null;
}
}
}
private static class Configuration {
private final List baseDirs = new ArrayList<>();
private final List jarFilesWithRelativePath = new ArrayList<>();
private List staticResourcesServletConfigs;
private List proxyServletConfigs;
private Map additionalServlets;
public void addBaseDir(Path baseDir) {
baseDirs.add(baseDir);
}
public void addBaseDirs(Path...baseDirs) {
addAll(this.baseDirs, baseDirs);
}
public void setStaticResourcesServletConfigs(List staticResourcesServletConfigs) {
this.staticResourcesServletConfigs = staticResourcesServletConfigs;
}
public void addResourceJar(JarFileWithRelativePath jarFileWithRelativePath) {
jarFilesWithRelativePath.add(jarFileWithRelativePath);
}
public void setProxyServletConfigs(List proxyServletConfigs) {
this.proxyServletConfigs = proxyServletConfigs;
}
public void setAdditionalServlets(Map additionalServlets) {
this.additionalServlets = additionalServlets;
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy