
com.agapsys.sevlet.container.ServletContextHandlerBuilder Maven / Gradle / Ivy
/*
* Copyright 2015 Agapsys Tecnologia Ltda-ME.
*
* 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 com.agapsys.sevlet.container;
import java.util.EnumSet;
import java.util.EventListener;
import java.util.LinkedHashMap;
import java.util.LinkedHashSet;
import java.util.LinkedList;
import java.util.List;
import java.util.Objects;
import java.util.Set;
import javax.servlet.DispatcherType;
import javax.servlet.Filter;
import javax.servlet.annotation.WebFilter;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import org.eclipse.jetty.server.handler.ErrorHandler;
import org.eclipse.jetty.servlet.ErrorPageErrorHandler;
import org.eclipse.jetty.servlet.ServletContextHandler;
/**
* @author Leandro Oliveira ([email protected])
*/
class ServletContextHandlerBuilder {
// STATIC SCOPE =============================================================
private class UrlMapping extends LinkedHashMap {}
private static class ClassMapping {
public final Class extends T> mappedClass;
public final String urlPattern;
public ClassMapping(String urlPattern, Class extends T> mappedClass) {
if (urlPattern == null || urlPattern.trim().isEmpty()) {
throw new IllegalArgumentException("Null/Empty URL pattern");
}
if (mappedClass == null) {
throw new IllegalArgumentException("Original class cannot be null");
}
this.urlPattern = urlPattern;
this.mappedClass = mappedClass;
}
@Override
public int hashCode() {
int hash = 7;
return hash;
}
@Override
public boolean equals(Object obj) {
if (obj == null) {
return false;
}
if (getClass() != obj.getClass()) {
return false;
}
final ClassMapping> other = (ClassMapping>) obj;
if (!Objects.equals(this.urlPattern, other.urlPattern))
return false;
if (!Objects.equals(this.mappedClass, other.mappedClass))
return false;
return true;
}
}
private static class FilterMapping extends ClassMapping {
public FilterMapping(Class extends Filter> filterClass, String urlPattern) {
super(urlPattern, filterClass);
}
}
private static class ServletMapping extends ClassMapping {
public ServletMapping(Class extends HttpServlet> servletClass, String urlPattern) {
super(urlPattern, servletClass);
}
}
// =========================================================================
// INSTANCE SCOPE ==========================================================
private final UrlMapping urlFilterMapping = new UrlMapping<>();
private final UrlMapping urlServletMapping = new UrlMapping<>();
private final Set> eventListenerSet = new LinkedHashSet<>();
private final List filterMappingList = new LinkedList<>();
private final List servletMappingList = new LinkedList<>();
private final ServletContainerBuilder servletContainerBuilder;
private final String contextPath;
private ErrorHandler errorHandler = null;
ServletContextHandlerBuilder(ServletContainerBuilder servletContainerBuilder, String contextPath) {
this.servletContainerBuilder = servletContainerBuilder;
this.contextPath = contextPath;
}
/**
* Registers an event listener with this context handler builder
*
* @param eventListener event listener to be registered
* @param append boolean indicating if given listener shall be appended. If
* false, given listener will be prepended.
* @return this
*/
public ServletContextHandlerBuilder registerEventListener(Class extends EventListener> eventListener, boolean append) {
if (eventListener == null) {
throw new IllegalArgumentException("Event listener cannot be null");
}
if (!eventListenerSet.add(eventListener)) {
throw new IllegalArgumentException(String.format("Event listener is already registered: %s", eventListener.getName()));
}
return this;
}
/**
* Convenience method for registerEventListener(eventListener, true).
*
* @param eventListener event listener to be registered
* @return this
*/
public final ServletContextHandlerBuilder registerEventListener(Class extends EventListener> eventListener) {
return registerEventListener(eventListener, true);
}
/**
* Registers a filter with this context handler builder
*
* @param filterClass filter class to be registered.
* @param urlPattern URL pattern associated with given filter
* @param append boolean indicating if given filter shall be appended. If
* false, given filter will be prepended.
* @return this
*/
public ServletContextHandlerBuilder registerFilter(Class extends Filter> filterClass, String urlPattern, boolean append) {
FilterMapping filterMapping = urlFilterMapping.get(urlPattern);
if (filterMapping != null && filterMapping.mappedClass == filterClass) {
throw new IllegalArgumentException(String.format("URL pattern is already associated with given filter class: %s => %s", urlPattern, filterClass.getName()));
}
filterMapping = new FilterMapping(filterClass, urlPattern);
urlFilterMapping.put(urlPattern, filterMapping);
filterMappingList.add(append ? filterMappingList.size() : 0, filterMapping);
return this;
}
/**
* Convenience method for registerFilter(filterClass, urlPattern, true)
*
* @param filterClass filter class to be registered
* @param urlPattern URL pattern associated with given filter
* @return this
*/
public final ServletContextHandlerBuilder registerFilter(Class extends Filter> filterClass, String urlPattern) {
return registerFilter(filterClass, urlPattern, true);
}
/**
* Convenience method for registerFilter(filterClass).
*
* @param filterClass filter class to be registered. Informed class must be
* annotated with {@linkplain WebFilter}.
* @return this
*/
public final ServletContextHandlerBuilder registerFilter(Class extends Filter> filterClass) {
WebFilter[] annotations = filterClass.getAnnotationsByType(WebFilter.class);
if (annotations.length == 0) {
throw new IllegalArgumentException("Filter class does not have a WebFilter annotation");
}
for (WebFilter annotation : annotations) {
String[] urlPatterns = annotation.value();
if (urlPatterns.length == 0) {
urlPatterns = annotation.urlPatterns();
}
if (urlPatterns.length == 0) {
throw new IllegalArgumentException("Missing urlPatterns");
}
for (String urlPattern : urlPatterns) {
registerFilter(filterClass, urlPattern);
}
}
return this;
}
/**
* Registers a servlet with this context handler builder.
*
* @param servletClass servlet class to be registered.
* @param urlPattern URL pattern associated with given servlet
* @return this
*/
public ServletContextHandlerBuilder registerServlet(Class extends HttpServlet> servletClass, String urlPattern) {
ServletMapping servletMapping = urlServletMapping.get(urlPattern);
if (servletMapping != null) {
throw new IllegalArgumentException(String.format("URL pattern is already associated with a servlet: %s => %s", urlPattern, servletClass.getName()));
}
servletMapping = new ServletMapping(servletClass, urlPattern);
urlServletMapping.put(urlPattern, servletMapping);
servletMappingList.add(servletMapping);
return this;
}
/**
* Convenience method for registerServlet(servletClass).
*
* @param servletClass servlet class to be registered. Informed class must
* be annotated with {@linkplain WebServlet}.
* @return this
*/
public final ServletContextHandlerBuilder registerServlet(Class extends HttpServlet> servletClass) {
WebServlet[] annotations = servletClass.getAnnotationsByType(WebServlet.class);
if (annotations.length == 0) {
throw new IllegalArgumentException("Servlet class does not have a WebServlet annotation");
}
for (WebServlet annotation : annotations) {
String[] urlPatterns = annotation.value();
if (urlPatterns.length == 0) {
urlPatterns = annotation.urlPatterns();
}
if (urlPatterns.length == 0) {
throw new IllegalArgumentException("Missing urlPatterns");
}
for (String urlPattern : urlPatterns) {
registerServlet(servletClass, urlPattern);
}
}
return this;
}
public ServletContextHandlerBuilder registerErrorPage(int code, String url) {
if (url == null || url.trim().isEmpty()) {
throw new IllegalArgumentException("Null/Empty URL");
}
if (errorHandler == null) {
errorHandler = new ErrorPageErrorHandler();
}
if (!(errorHandler instanceof ErrorPageErrorHandler)) {
throw new IllegalStateException("An custom error handler is already defined");
}
((ErrorPageErrorHandler) errorHandler).addErrorPage(code, url);
return this;
}
public ServletContextHandlerBuilder setErrorHandler(ErrorHandler errorHandler) {
if (errorHandler == null) {
throw new IllegalArgumentException("Error handler cannot be null");
}
if (this.errorHandler != null) {
throw new IllegalArgumentException("Error handler is already defined");
}
this.errorHandler = errorHandler;
return this;
}
public ServletContainerBuilder endContext() {
servletContainerBuilder.contextBuilders.put(contextPath, this);
return servletContainerBuilder;
}
ServletContextHandler build() {
ServletContextHandler servletContextHandler = new ServletContextHandler(ServletContextHandler.SESSIONS);
for (FilterMapping filterMapping : filterMappingList) {
Class extends Filter> filterClass = filterMapping.mappedClass;
String urlPattern = filterMapping.urlPattern;
servletContextHandler.addFilter(filterClass, urlPattern, EnumSet.of(DispatcherType.REQUEST));
}
for (ServletMapping servletMapping : servletMappingList) {
Class extends HttpServlet> servletClass = servletMapping.mappedClass;
String urlPattern = servletMapping.urlPattern;
servletContextHandler.addServlet(servletClass, urlPattern);
}
for (Class extends EventListener> eventListenerClass : eventListenerSet) {
try {
EventListener eventListener = eventListenerClass.newInstance();
servletContextHandler.addEventListener(eventListener);
} catch (IllegalAccessException | InstantiationException ex) {
throw new RuntimeException(ex);
}
}
if (errorHandler != null) {
servletContextHandler.setErrorHandler(errorHandler);
}
return servletContextHandler;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy