/*******************************************************************************
*
* Copyright (c) 2010-2011 Sonatype, Inc.
*
* All rights reserved. This program and the accompanying materials
* are made available under the terms of the Eclipse Public License v1.0
* which accompanies this distribution, and is available at
* http://www.eclipse.org/legal/epl-v10.html
*
* Contributors:
*
*
*
*
*******************************************************************************/
package org.hudsonci.servlets.internal;
import org.hudsonci.servlets.ServletContainer;
import org.hudsonci.servlets.ServletContainerAware;
import org.hudsonci.servlets.ServletRegistration;
import hudson.util.PluginServletFilter;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import javax.inject.Inject;
import javax.inject.Named;
import javax.inject.Singleton;
import javax.servlet.Filter;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import static com.google.common.base.Preconditions.checkNotNull;
/**
* Default {@link ServletContainer} implementation.
*
* @author <a href="mailto:jason@planet57.com">Jason Dillon</a>
* @since 2.1.0
*/
@Named
@Singleton
public class ServletContainerImpl
implements ServletContainer {
private static final Logger log = LoggerFactory.getLogger(ServletContainerImpl.class);
private final Map<ServletRegistration, Filter> registrations = new LinkedHashMap<ServletRegistration, Filter>();
private final List<ServletContainerAware> concerned;
@Inject
public ServletContainerImpl(final List<ServletContainerAware> concerned) {
this.concerned = checkNotNull(concerned);
}
private ServletContainerAware[] getConcerned() {
return concerned.toArray(new ServletContainerAware[concerned.size()]);
}
public void start() throws Exception {
log.debug("Starting");
// When a ServletContainerAware ext gets executed, we need to set the TCL so that it loads properly, else crazy things will happen
final ClassLoader cl = Thread.currentThread().getContextClassLoader();
for (ServletContainerAware target : getConcerned()) {
Thread.currentThread().setContextClassLoader(target.getClass().getClassLoader());
try {
target.setServletContainer(this);
} catch (Exception e) {
log.error("Exception while exposing servlet container to: " + target, e);
} finally {
Thread.currentThread().setContextClassLoader(cl);
}
}
}
public void stop() throws Exception {
log.debug("Stopping");
for (Filter filter : registrations.values()) {
PluginServletFilter.removeFilter(filter);
}
registrations.clear();
}
public ServletRegistration.Handle register(ServletRegistration registration) throws Exception {
checkNotNull(registration);
log.debug("Registering: {}", registration);
registration = registration.clone();
ServletRegistrationFilterAdapter filter = new ServletRegistrationFilterAdapter(registration);
PluginServletFilter.addFilter(filter);
registrations.put(registration, filter);
return new HandleImpl(registration, filter);
}
private class HandleImpl
implements ServletRegistration.Handle {
private final ServletRegistration registration;
private final ServletRegistrationFilterAdapter filterAdapter;
private HandleImpl(final ServletRegistration registration, final ServletRegistrationFilterAdapter filterAdapter) {
this.registration = checkNotNull(registration);
this.filterAdapter = checkNotNull(filterAdapter);
}
public ServletRegistration getRegistration() {
return registration;
}
public void setEnabled(final boolean enabled) {
filterAdapter.setEnabled(enabled);
}
public boolean isEnabled() {
return filterAdapter.isEnabled();
}
}
}