Commit e4a4fdd5 authored by Guus der Kinderen's avatar Guus der Kinderen

OF-1326: Improve sharing of the BOSH context

Intead of using a parent context, a collection of Jetty handlers can be used to dynamically add/remove functionality to the BOSH context.

This commit replaces the parent context instance with a ordered list of handlers, which will attempt to process a request by:

1. Checking if this is a BOSH request
2. Checking if this is a request for BOSH metadata
3. Check if an extension was provided that can handle the request
4. Try to serve static content as a last resort.

In step 3, a collection of handlers is used, that can be modified at runtime. This allows plugins to register/remove handlers.

The entire collection of handlers (1 to 4) is maintained with a lifecycle that's different from the embedded Jetty server that uses them. This
allows the collection to survive a server reconfiguration, as well as act independent of the 'enabled' state of the BOSH service.
parent 54696701
...@@ -20,33 +20,14 @@ ...@@ -20,33 +20,14 @@
package org.jivesoftware.openfire.http; package org.jivesoftware.openfire.http;
import java.io.File;
import java.security.KeyStore;
import java.security.cert.X509Certificate;
import java.util.*;
import javax.servlet.DispatcherType;
import javax.servlet.Filter;
import javax.servlet.FilterConfig;
import javax.servlet.ServletException;
import org.apache.jasper.servlet.JasperInitializer; import org.apache.jasper.servlet.JasperInitializer;
import org.apache.tomcat.InstanceManager; import org.apache.tomcat.InstanceManager;
import org.apache.tomcat.SimpleInstanceManager; import org.apache.tomcat.SimpleInstanceManager;
import org.eclipse.jetty.http.HttpMethod; import org.eclipse.jetty.http.HttpMethod;
import org.eclipse.jetty.plus.annotation.ContainerInitializer; import org.eclipse.jetty.plus.annotation.ContainerInitializer;
import org.eclipse.jetty.server.Connector; import org.eclipse.jetty.server.*;
import org.eclipse.jetty.server.ForwardedRequestCustomizer;
import org.eclipse.jetty.server.Handler;
import org.eclipse.jetty.server.HttpConfiguration;
import org.eclipse.jetty.server.HttpConnectionFactory;
import org.eclipse.jetty.server.SecureRequestCustomizer;
import org.eclipse.jetty.server.Server;
import org.eclipse.jetty.server.ServerConnector;
import org.eclipse.jetty.server.SslConnectionFactory;
import org.eclipse.jetty.server.handler.ContextHandlerCollection;
import org.eclipse.jetty.server.handler.DefaultHandler;
import org.eclipse.jetty.server.handler.HandlerCollection; import org.eclipse.jetty.server.handler.HandlerCollection;
import org.eclipse.jetty.server.handler.HandlerList;
import org.eclipse.jetty.servlet.FilterHolder; import org.eclipse.jetty.servlet.FilterHolder;
import org.eclipse.jetty.servlet.ServletContextHandler; import org.eclipse.jetty.servlet.ServletContextHandler;
import org.eclipse.jetty.servlet.ServletHolder; import org.eclipse.jetty.servlet.ServletHolder;
...@@ -63,18 +44,23 @@ import org.jivesoftware.openfire.spi.ConnectionConfiguration; ...@@ -63,18 +44,23 @@ import org.jivesoftware.openfire.spi.ConnectionConfiguration;
import org.jivesoftware.openfire.spi.ConnectionManagerImpl; import org.jivesoftware.openfire.spi.ConnectionManagerImpl;
import org.jivesoftware.openfire.spi.ConnectionType; import org.jivesoftware.openfire.spi.ConnectionType;
import org.jivesoftware.openfire.spi.EncryptionArtifactFactory; import org.jivesoftware.openfire.spi.EncryptionArtifactFactory;
import org.jivesoftware.util.CertificateEventListener; import org.jivesoftware.util.*;
import org.jivesoftware.util.CertificateManager;
import org.jivesoftware.util.JiveGlobals;
import org.jivesoftware.util.PropertyEventDispatcher;
import org.jivesoftware.util.PropertyEventListener;
import org.slf4j.Logger; import org.slf4j.Logger;
import org.slf4j.LoggerFactory; import org.slf4j.LoggerFactory;
import javax.servlet.DispatcherType;
import javax.servlet.Filter;
import javax.servlet.FilterConfig;
import javax.servlet.ServletException;
import java.io.File;
import java.security.KeyStore;
import java.security.cert.X509Certificate;
import java.util.*;
/** /**
* * Responsible for making available BOSH (functionality to the outside world, using an embedded web server.
*/ */
public final class HttpBindManager { public final class HttpBindManager implements CertificateEventListener, PropertyEventListener {
private static final Logger Log = LoggerFactory.getLogger(HttpBindManager.class); private static final Logger Log = LoggerFactory.getLogger(HttpBindManager.class);
...@@ -132,21 +118,33 @@ public final class HttpBindManager { ...@@ -132,21 +118,33 @@ public final class HttpBindManager {
private Server httpBindServer; private Server httpBindServer;
private int bindPort; private final HttpSessionManager httpSessionManager;
private int bindSecurePort;
private Connector httpConnector;
private Connector httpsConnector;
private CertificateListener certificateListener;
private HttpSessionManager httpSessionManager;
private ContextHandlerCollection contexts; /**
* An ordered collection of all handlers (that is created to include the #extensionHandlers).
*
* A reference to this collection is maintained outside of the Jetty server implementation ({@link #httpBindServer})
* as its lifecycle differs from that server: the server is recreated upon configuration changes, while the
* collection of handlers need not be.
*
* This collection is ordered, which ensures that:
* <ul>
* <li>The handlers providing BOSH functionality are tried first.</li>
* <li>Any handlers that are registered by external sources ({@link #extensionHandlers}) are tried in between.</li>
* <li>The 'catch-all' handler that maps to static content is tried last.</li>
* </ul>
*
* This collection should be regarded as immutable. When handlers are to be added/removed dynamically, this should
* occur in {@link #extensionHandlers}, to which a reference is stored in this list by the constructor of this class.
*/
private final HandlerList handlerList = new HandlerList();
// is all orgin allowed flag /**
private boolean allowAllOrigins; * Contains all Jetty handlers that are added as an extension.
*
* This collection is mutabl. Handlers can be added and removed at runtime.
*/
private final HandlerCollection extensionHandlers = new HandlerCollection( true );
public static HttpBindManager getInstance() { public static HttpBindManager getInstance() {
return instance; return instance;
...@@ -166,41 +164,78 @@ public final class HttpBindManager { ...@@ -166,41 +164,78 @@ public final class HttpBindManager {
JiveGlobals.migrateProperty(HTTP_BIND_CORS_ALLOW_ORIGIN); JiveGlobals.migrateProperty(HTTP_BIND_CORS_ALLOW_ORIGIN);
JiveGlobals.migrateProperty(HTTP_BIND_REQUEST_HEADER_SIZE); JiveGlobals.migrateProperty(HTTP_BIND_REQUEST_HEADER_SIZE);
PropertyEventDispatcher.addListener(new HttpServerPropertyListener()); PropertyEventDispatcher.addListener( this );
this.httpSessionManager = new HttpSessionManager(); this.httpSessionManager = new HttpSessionManager();
// we need to initialise contexts at constructor time in order for plugins to add their contexts before start()
contexts = new ContextHandlerCollection();
// setup the cache for the allowed origins // setup the cache for the allowed origins
this.setupAllowedOriginsMap(); this.setupAllowedOriginsMap();
// Setup the default handlers. Order is important here. First, evaluate if the 'standard' handlers can be used to fulfill requests.
this.handlerList.addHandler( createBoshHandler() );
this.handlerList.addHandler( createCrossDomainHandler() );
// When standard handling does not apply, see if any of the handlers in the extension pool of handlers applies to the request.
this.handlerList.addHandler( this.extensionHandlers );
// When everything else fails, use the static content handler. This one should be last, as it is mapping to the root context.
// This means that it will catch everything and prevent the invocation of later handlers.
this.handlerList.addHandler( createStaticContentHandler() );
} }
public void start() { public void start() {
certificateListener = new CertificateListener();
CertificateManager.addListener(certificateListener);
if (!isHttpBindServiceEnabled()) { if (!isHttpBindServiceEnabled()) {
return; return;
} }
bindPort = getHttpBindUnsecurePort();
bindSecurePort = getHttpBindSecurePort(); // this is the number of threads allocated to each connector/port
configureHttpBindServer(bindPort, bindSecurePort); final int processingThreads = JiveGlobals.getIntProperty(HTTP_BIND_THREADS, HTTP_BIND_THREADS_DEFAULT);
final QueuedThreadPool tp = new QueuedThreadPool(processingThreads);
tp.setName("Jetty-QTP-BOSH");
httpBindServer = new Server(tp);
if (JMXManager.isEnabled()) {
JMXManager jmx = JMXManager.getInstance();
httpBindServer.addBean(jmx.getContainer());
}
final Connector httpConnector = createConnector( httpBindServer );
final Connector httpsConnector = createSSLConnector( httpBindServer);
if (httpConnector == null && httpsConnector == null) {
httpBindServer = null;
return;
}
if (httpConnector != null) {
httpBindServer.addConnector(httpConnector);
}
if (httpsConnector != null) {
httpBindServer.addConnector(httpsConnector);
}
httpBindServer.setHandler( handlerList );
try { try {
httpBindServer.start(); httpBindServer.start();
handlerList.start();
CertificateManager.addListener(this);
Log.info("HTTP bind service started"); Log.info("HTTP bind service started");
} }
catch (Exception e) { catch (Exception e) {
Log.error("Error starting HTTP bind service", e); Log.error("Error starting HTTP bind service", e);
} }
} }
public void stop() { public void stop() {
CertificateManager.removeListener(certificateListener); CertificateManager.removeListener(this);
if (httpBindServer != null) { if (httpBindServer != null) {
try { try {
handlerList.stop();
httpBindServer.stop(); httpBindServer.stop();
Log.info("HTTP bind service stopped"); Log.info("HTTP bind service stopped");
} }
...@@ -219,8 +254,8 @@ public final class HttpBindManager { ...@@ -219,8 +254,8 @@ public final class HttpBindManager {
return JiveGlobals.getBooleanProperty(HTTP_BIND_ENABLED, HTTP_BIND_ENABLED_DEFAULT); return JiveGlobals.getBooleanProperty(HTTP_BIND_ENABLED, HTTP_BIND_ENABLED_DEFAULT);
} }
private void createConnector(int port) { private Connector createConnector( final Server httpBindServer ) {
httpConnector = null; final int port = getHttpBindUnsecurePort();
if (port > 0) { if (port > 0) {
HttpConfiguration httpConfig = new HttpConfiguration(); HttpConfiguration httpConfig = new HttpConfiguration();
configureProxiedConnector(httpConfig); configureProxiedConnector(httpConfig);
...@@ -229,12 +264,16 @@ public final class HttpBindManager { ...@@ -229,12 +264,16 @@ public final class HttpBindManager {
// Listen on a specific network interface if it has been set. // Listen on a specific network interface if it has been set.
connector.setHost(getBindInterface()); connector.setHost(getBindInterface());
connector.setPort(port); connector.setPort(port);
httpConnector = connector; return connector;
}
else
{
return null;
} }
} }
private void createSSLConnector(int securePort) { private Connector createSSLConnector( final Server httpBindServer ) {
httpsConnector = null; final int securePort = getHttpBindSecurePort();
try { try {
final IdentityStore identityStore = XMPPServer.getInstance().getCertificateStoreManager().getIdentityStore( ConnectionType.BOSH_C2S ); final IdentityStore identityStore = XMPPServer.getInstance().getCertificateStoreManager().getIdentityStore( ConnectionType.BOSH_C2S );
...@@ -266,12 +305,14 @@ public final class HttpBindManager { ...@@ -266,12 +305,14 @@ public final class HttpBindManager {
} }
sslConnector.setHost(getBindInterface()); sslConnector.setHost(getBindInterface());
sslConnector.setPort(securePort); sslConnector.setPort(securePort);
httpsConnector = sslConnector; return sslConnector;
} }
} }
catch (Exception e) { catch (Exception e) {
Log.error("Error creating SSL connector for Http bind", e); Log.error("Error creating SSL connector for Http bind", e);
} }
return null;
} }
private void configureProxiedConnector(HttpConfiguration httpConfig) { private void configureProxiedConnector(HttpConfiguration httpConfig) {
...@@ -330,8 +371,26 @@ public final class HttpBindManager { ...@@ -330,8 +371,26 @@ public final class HttpBindManager {
* *
* @return true if a listener on the HTTP binding port is running. * @return true if a listener on the HTTP binding port is running.
*/ */
public boolean isHttpBindActive() { public boolean isHttpBindActive()
return httpConnector != null && httpConnector.isRunning(); {
if ( isHttpBindEnabled() )
{
final int configuredPort = getHttpBindUnsecurePort();
for ( final Connector connector : httpBindServer.getConnectors() )
{
if ( !( connector instanceof ServerConnector ) )
{
continue;
}
final int activePort = ( (ServerConnector) connector ).getLocalPort();
if ( activePort == configuredPort )
{
return true;
}
}
}
return false;
} }
/** /**
...@@ -339,33 +398,45 @@ public final class HttpBindManager { ...@@ -339,33 +398,45 @@ public final class HttpBindManager {
* *
* @return true if a listener on the HTTPS binding port is running. * @return true if a listener on the HTTPS binding port is running.
*/ */
public boolean isHttpsBindActive() { public boolean isHttpsBindActive()
return httpsConnector != null && httpsConnector.isRunning(); {
if ( isHttpBindEnabled() )
{
final int configuredPort = getHttpBindSecurePort();
for ( final Connector connector : httpBindServer.getConnectors() )
{
if ( !( connector instanceof ServerConnector ) )
{
continue;
}
final int activePort = ( (ServerConnector) connector ).getLocalPort();
if ( activePort == configuredPort )
{
return true;
}
}
}
return false;
} }
public String getHttpBindUnsecureAddress() { public String getHttpBindUnsecureAddress() {
return "http://" + XMPPServer.getInstance().getServerInfo().getXMPPDomain() + ":" + return "http://" + XMPPServer.getInstance().getServerInfo().getXMPPDomain() + ":" + getHttpBindUnsecurePort() + "/http-bind/";
bindPort + "/http-bind/";
} }
public String getHttpBindSecureAddress() { public String getHttpBindSecureAddress() {
return "https://" + XMPPServer.getInstance().getServerInfo().getXMPPDomain() + ":" + return "https://" + XMPPServer.getInstance().getServerInfo().getXMPPDomain() + ":" + getHttpBindSecurePort() + "/http-bind/";
bindSecurePort + "/http-bind/";
} }
public String getJavaScriptUrl() { public String getJavaScriptUrl() {
return "http://" + XMPPServer.getInstance().getServerInfo().getXMPPDomain() + ":" + return "http://" + XMPPServer.getInstance().getServerInfo().getXMPPDomain() + ":" + getHttpBindUnsecurePort() + "/scripts/";
bindPort + "/scripts/";
} }
// http binding CORS support start // http binding CORS support start
private void setupAllowedOriginsMap() { private void setupAllowedOriginsMap() {
String originString = getCORSAllowOrigin(); final String originString = getCORSAllowOrigin();
if (originString.equals(HTTP_BIND_CORS_ALLOW_ORIGIN_DEFAULT)) { if (!originString.equals(HTTP_BIND_CORS_ALLOW_ORIGIN_DEFAULT)) {
allowAllOrigins = true;
} else {
allowAllOrigins = false;
String[] origins = originString.split(","); String[] origins = originString.split(",");
// reset the cache // reset the cache
HTTP_BIND_ALLOWED_ORIGINS.clear(); HTTP_BIND_ALLOWED_ORIGINS.clear();
...@@ -399,11 +470,11 @@ public final class HttpBindManager { ...@@ -399,11 +470,11 @@ public final class HttpBindManager {
} }
public boolean isAllOriginsAllowed() { public boolean isAllOriginsAllowed() {
return allowAllOrigins; return HTTP_BIND_CORS_ALLOW_ORIGIN_DEFAULT.equals( getCORSAllowOrigin() );
} }
public boolean isThisOriginAllowed(String origin) { public boolean isThisOriginAllowed(String origin) {
return HTTP_BIND_ALLOWED_ORIGINS.get(origin) != null; return isAllOriginsAllowed() || HTTP_BIND_ALLOWED_ORIGINS.get(origin) != null;
} }
// http binding CORS support end // http binding CORS support end
...@@ -491,72 +562,50 @@ public final class HttpBindManager { ...@@ -491,72 +562,50 @@ public final class HttpBindManager {
} }
/** /**
* Starts an HTTP Bind server on the specified port and secure port. * Creates a Jetty context handler that can be used to expose BOSH (HTTP-Bind) functionality.
* *
* @param port the port to start the normal (unsecured) HTTP Bind service on. * Note that an invocation of this method will not register the handler (and thus make the related functionality
* @param securePort the port to start the TLS (secure) HTTP Bind service on. * available to the end user). Instead, the created handler is returned by this method, and will need to be
* registered with the embedded Jetty webserver by the caller.
*
* @return A Jetty context handler (never null).
*/ */
private synchronized void configureHttpBindServer(int port, int securePort) { protected Handler createBoshHandler()
// this is the number of threads allocated to each connector/port {
final int processingThreads = JiveGlobals.getIntProperty(HTTP_BIND_THREADS, HTTP_BIND_THREADS_DEFAULT); final ServletContextHandler context = new ServletContextHandler( null, "/http-bind", ServletContextHandler.SESSIONS );
final QueuedThreadPool tp = new QueuedThreadPool(processingThreads);
tp.setName("Jetty-QTP-BOSH");
httpBindServer = new Server(tp);
if (JMXManager.isEnabled()) {
JMXManager jmx = JMXManager.getInstance();
httpBindServer.addBean(jmx.getContainer());
}
createConnector(port);
createSSLConnector(securePort);
if (httpConnector == null && httpsConnector == null) {
httpBindServer = null;
return;
}
if (httpConnector != null) {
httpBindServer.addConnector(httpConnector);
}
if (httpsConnector != null) {
httpBindServer.addConnector(httpsConnector);
}
//contexts = new ContextHandlerCollection(); // Ensure the JSP engine is initialized correctly (in order to be able to cope with Tomcat/Jasper precompiled JSPs).
// TODO implement a way to get plugins to add their their web services to contexts final List<ContainerInitializer> initializers = new ArrayList<>();
initializers.add( new ContainerInitializer( new JasperInitializer(), null ) );
context.setAttribute( "org.eclipse.jetty.containerInitializers", initializers );
context.setAttribute( InstanceManager.class.getName(), new SimpleInstanceManager() );
createBoshHandler(contexts, "/http-bind"); // Generic configuration of the context.
createCrossDomainHandler(contexts, "/crossdomain.xml"); context.setAllowNullPathInfo( true );
loadStaticDirectory(contexts);
HandlerCollection collection = new HandlerCollection(); // Add the functionality-providers.
httpBindServer.setHandler(collection); context.addServlet( new ServletHolder( new HttpBindServlet() ), "/*" );
collection.setHandlers(new Handler[] { contexts, new DefaultHandler() });
}
private void createBoshHandler(ContextHandlerCollection contexts, String boshPath) // Add compression filter when needed.
if ( isHttpCompressionEnabled() )
{
final Filter gzipFilter = new AsyncGzipFilter()
{ {
ServletContextHandler context = new ServletContextHandler(contexts, boshPath, ServletContextHandler.SESSIONS);
// Ensure the JSP engine is initialized correctly (in order to be able to cope with Tomcat/Jasper precompiled JSPs).
final List<ContainerInitializer> initializers = new ArrayList<>();
initializers.add(new ContainerInitializer(new JasperInitializer(), null));
context.setAttribute("org.eclipse.jetty.containerInitializers", initializers);
context.setAttribute(InstanceManager.class.getName(), new SimpleInstanceManager());
context.setAllowNullPathInfo(true);
context.addServlet(new ServletHolder(new HttpBindServlet()),"/*");
if (isHttpCompressionEnabled()) {
Filter gzipFilter = new AsyncGzipFilter() {
@Override @Override
public void init(FilterConfig config) throws ServletException { public void init( FilterConfig config ) throws ServletException
super.init(config); {
_methods.add(HttpMethod.POST.asString()); super.init( config );
Log.info("Installed response compression filter"); _methods.add( HttpMethod.POST.asString() );
Log.info( "Installed response compression filter" );
} }
}; };
FilterHolder filterHolder = new FilterHolder();
filterHolder.setFilter(gzipFilter); final FilterHolder filterHolder = new FilterHolder();
context.addFilter(filterHolder, "/*", EnumSet.of(DispatcherType.REQUEST)); filterHolder.setFilter( gzipFilter );
context.addFilter( filterHolder, "/*", EnumSet.of( DispatcherType.REQUEST ) );
} }
return context;
} }
// NOTE: enabled by default // NOTE: enabled by default
...@@ -566,34 +615,116 @@ public final class HttpBindManager { ...@@ -566,34 +615,116 @@ public final class HttpBindManager {
return configuration.getCompressionPolicy() == null || configuration.getCompressionPolicy().equals( Connection.CompressionPolicy.optional ); return configuration.getCompressionPolicy() == null || configuration.getCompressionPolicy().equals( Connection.CompressionPolicy.optional );
} }
private void createCrossDomainHandler(ContextHandlerCollection contexts, String crossPath) /**
* Creates a Jetty context handler that can be used to expose the cross-domain functionality as implemented by
* {@link FlashCrossDomainServlet}.
*
* Note that an invocation of this method will not register the handler (and thus make the related functionality
* available to the end user). Instead, the created handler is returned by this method, and will need to be
* registered with the embedded Jetty webserver by the caller.
*
* @return A Jetty context handler (never null).
*/
protected Handler createCrossDomainHandler()
{ {
ServletContextHandler context = new ServletContextHandler(contexts, crossPath, ServletContextHandler.SESSIONS); final ServletContextHandler context = new ServletContextHandler( null, "/crossdomain.xml", ServletContextHandler.SESSIONS );
// Ensure the JSP engine is initialized correctly (in order to be able to cope with Tomcat/Jasper precompiled JSPs). // Ensure the JSP engine is initialized correctly (in order to be able to cope with Tomcat/Jasper precompiled JSPs).
final List<ContainerInitializer> initializers = new ArrayList<>(); final List<ContainerInitializer> initializers = new ArrayList<>();
initializers.add(new ContainerInitializer(new JasperInitializer(), null)); initializers.add( new ContainerInitializer( new JasperInitializer(), null ) );
context.setAttribute("org.eclipse.jetty.containerInitializers", initializers); context.setAttribute( "org.eclipse.jetty.containerInitializers", initializers );
context.setAttribute(InstanceManager.class.getName(), new SimpleInstanceManager()); context.setAttribute( InstanceManager.class.getName(), new SimpleInstanceManager() );
context.setAllowNullPathInfo(true);
context.addServlet(new ServletHolder(new FlashCrossDomainServlet()),""); // Generic configuration of the context.
context.setAllowNullPathInfo( true );
// Add the functionality-providers.
context.addServlet( new ServletHolder( new FlashCrossDomainServlet() ), "" );
return context;
} }
private void loadStaticDirectory(ContextHandlerCollection contexts) { /**
File spankDirectory = new File(JiveGlobals.getHomeDirectory() + File.separator * Creates a Jetty context handler that can be used to expose static files.
+ "resources" + File.separator + "spank"); *
if (spankDirectory.exists()) { * Note that an invocation of this method will not register the handler (and thus make the related functionality
if (spankDirectory.canRead()) { * available to the end user). Instead, the created handler is returned by this method, and will need to be
WebAppContext context = new WebAppContext(contexts, spankDirectory.getPath(), "/"); * registered with the embedded Jetty webserver by the caller.
context.setWelcomeFiles(new String[]{"index.html"}); *
* @return A Jetty context handler, or null when the static content could not be accessed.
*/
protected Handler createStaticContentHandler()
{
final File spankDirectory = new File( JiveGlobals.getHomeDirectory() + File.separator + "resources" + File.separator + "spank" );
if ( spankDirectory.exists() )
{
if ( spankDirectory.canRead() )
{
final WebAppContext context = new WebAppContext( null, spankDirectory.getPath(), "/" );
context.setWelcomeFiles( new String[] { "index.html" } );
return context;
} }
else { else
Log.warn("Openfire cannot read the directory: " + spankDirectory); {
Log.warn( "Openfire cannot read the directory: " + spankDirectory );
} }
} }
return null;
}
/**
* Adds a Jetty handler to be added to the embedded web server that is used to expose BOSH (HTTP-bind)
* functionality.
*
* @param handler The handler (cannot be null).
*/
public void addJettyHandler( Handler handler )
{
if ( handler == null )
{
throw new IllegalArgumentException( "Argument 'handler' cannot be null." );
}
extensionHandlers.addHandler( handler );
if ( !handler.isStarted() && extensionHandlers.isStarted() )
{
try
{
handler.start();
}
catch ( Exception e )
{
Log.warn( "Unable to start handler {}", handler, e );
}
}
}
/**
* Removes a Jetty handler to be added to the embedded web server that is used to expose BOSH (HTTP-bind)
* functionality.
*
* Removing a handler, even when null, or non-existing, might have side-effects as introduced by the Jetty
* implementation. At the time of writing, Jetty will re
*
* @param handler The handler (should not be null).
*/
public void removeJettyHandler( Handler handler )
{
extensionHandlers.removeHandler( handler );
if ( handler.isStarted() )
{
try
{
handler.stop();
}
catch ( Exception e )
{
Log.warn( "Unable to stop the handler that was removed: {}", handler, e );
}
} }
public ContextHandlerCollection getContexts() {
return contexts;
} }
private void doEnableHttpBind(boolean shouldEnable) { private void doEnableHttpBind(boolean shouldEnable) {
...@@ -654,54 +785,35 @@ public final class HttpBindManager { ...@@ -654,54 +785,35 @@ public final class HttpBindManager {
} }
} }
private void setUnsecureHttpBindPort(int value) {
if (value == bindPort) {
return;
}
restartServer();
}
private void setSecureHttpBindPort(int value) {
if (value == bindSecurePort) {
return;
}
restartServer();
}
private synchronized void restartServer() { private synchronized void restartServer() {
stop(); stop();
start(); start();
} }
/** Listens for changes to Jive properties that affect the HTTP server manager. */
private class HttpServerPropertyListener implements PropertyEventListener {
@Override @Override
public void propertySet(String property, Map<String, Object> params) { public void propertySet(String property, Map<String, Object> params) {
if (property.equalsIgnoreCase(HTTP_BIND_ENABLED)) { if (property.equalsIgnoreCase(HTTP_BIND_ENABLED)) {
doEnableHttpBind(Boolean.valueOf(params.get("value").toString())); doEnableHttpBind(Boolean.valueOf(params.get("value").toString()));
} }
else if (property.equalsIgnoreCase(HTTP_BIND_PORT)) { else if (property.equalsIgnoreCase(HTTP_BIND_PORT)) {
int value;
try { try {
value = Integer.valueOf(params.get("value").toString()); Integer.valueOf(params.get("value").toString());
} }
catch (NumberFormatException ne) { catch (NumberFormatException ne) {
JiveGlobals.deleteProperty(HTTP_BIND_PORT); JiveGlobals.deleteProperty(HTTP_BIND_PORT);
return; return;
} }
setUnsecureHttpBindPort(value); restartServer();
} }
else if (property.equalsIgnoreCase(HTTP_BIND_SECURE_PORT)) { else if (property.equalsIgnoreCase(HTTP_BIND_SECURE_PORT)) {
int value;
try { try {
value = Integer.valueOf(params.get("value").toString()); Integer.valueOf(params.get("value").toString());
} }
catch (NumberFormatException ne) { catch (NumberFormatException ne) {
JiveGlobals.deleteProperty(HTTP_BIND_SECURE_PORT); JiveGlobals.deleteProperty(HTTP_BIND_SECURE_PORT);
return; return;
} }
setSecureHttpBindPort(value); restartServer();
} }
else if (HTTP_BIND_AUTH_PER_CLIENTCERT_POLICY.equalsIgnoreCase( property )) { else if (HTTP_BIND_AUTH_PER_CLIENTCERT_POLICY.equalsIgnoreCase( property )) {
restartServer(); restartServer();
...@@ -714,10 +826,10 @@ public final class HttpBindManager { ...@@ -714,10 +826,10 @@ public final class HttpBindManager {
doEnableHttpBind(HTTP_BIND_ENABLED_DEFAULT); doEnableHttpBind(HTTP_BIND_ENABLED_DEFAULT);
} }
else if (property.equalsIgnoreCase(HTTP_BIND_PORT)) { else if (property.equalsIgnoreCase(HTTP_BIND_PORT)) {
setUnsecureHttpBindPort(HTTP_BIND_PORT_DEFAULT); restartServer();
} }
else if (property.equalsIgnoreCase(HTTP_BIND_SECURE_PORT)) { else if (property.equalsIgnoreCase(HTTP_BIND_SECURE_PORT)) {
setSecureHttpBindPort(HTTP_BIND_SECURE_PORT_DEFAULT); restartServer();
} }
else if (HTTP_BIND_AUTH_PER_CLIENTCERT_POLICY.equalsIgnoreCase( property )) { else if (HTTP_BIND_AUTH_PER_CLIENTCERT_POLICY.equalsIgnoreCase( property )) {
restartServer(); restartServer();
...@@ -731,9 +843,6 @@ public final class HttpBindManager { ...@@ -731,9 +843,6 @@ public final class HttpBindManager {
@Override @Override
public void xmlPropertyDeleted(String property, Map<String, Object> params) { public void xmlPropertyDeleted(String property, Map<String, Object> params) {
} }
}
private class CertificateListener implements CertificateEventListener {
@Override @Override
public void certificateCreated(KeyStore keyStore, String alias, X509Certificate cert) { public void certificateCreated(KeyStore keyStore, String alias, X509Certificate cert) {
...@@ -756,5 +865,4 @@ public final class HttpBindManager { ...@@ -756,5 +865,4 @@ public final class HttpBindManager {
restartServer(); restartServer();
} }
} }
}
} }
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment