org.jboss.remoting.transport.Connector Maven / Gradle / Ivy
/*
* JBoss, Home of Professional Open Source
* Copyright 2005, JBoss Inc., and individual contributors as indicated
* by the @authors tag. See the copyright.txt in the distribution for a
* full listing of individual contributors.
*
* This is free software; you can redistribute it and/or modify it
* under the terms of the GNU Lesser General Public License as
* published by the Free Software Foundation; either version 2.1 of
* the License, or (at your option) any later version.
*
* This software is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this software; if not, write to the Free
* Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
* 02110-1301 USA, or see the FSF site: http://www.fsf.org.
*/
package org.jboss.remoting.transport;
import org.jboss.logging.Logger;
import org.jboss.remoting.ConnectionListener;
import org.jboss.remoting.InvokerLocator;
import org.jboss.remoting.InvokerRegistry;
import org.jboss.remoting.ServerInvocationHandler;
import org.jboss.remoting.ServerInvoker;
import org.jboss.remoting.marshal.MarshalFactory;
import org.jboss.remoting.marshal.MarshallLoaderFactory;
import org.jboss.remoting.serialization.ClassLoaderUtility;
import org.jboss.remoting.util.LocalHostUtil;
import org.w3c.dom.Element;
import org.w3c.dom.NamedNodeMap;
import org.w3c.dom.Node;
import org.w3c.dom.NodeList;
import javax.management.MBeanRegistration;
import javax.management.MBeanServer;
import javax.management.MBeanServerInvocationHandler;
import javax.management.MalformedObjectNameException;
import javax.management.ObjectName;
import javax.net.ServerSocketFactory;
import javax.net.SocketFactory;
import java.net.InetAddress;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.StringTokenizer;
/**
* Connector is an implementation of the ConnectorMBean interface.
*
* The Connector is root component for the remoting server. It binds the server transport, marshaller,
* and handler together to form the remoting server instance.
*
* A transport connector is configured via *-service.xml such as:
*
*
*
*
*
*
*
*
*
*
*
*
*
*
*
*
*
*
*
* 1
* 303
* 304
* 60000
* ${jboss.bind.address}
* 6666
*
*
* false
* 200
*
*
* org.jboss.remoting.transport.mock.MockServerInvocationHandler
*
*
*
*
*
*
*
*
* 2410
*
*
*
*
*
* @author Jeff Haynie
* @author Adrian Brock
* @author David Jencks
* @author Juha Lindfors
* @author Tom Elrod
* @version $Revision: 5960 $
* @jmx.mbean description = "An MBean wrapper around a ServerInvoker."
* @jboss.xmbean
*/
public class Connector implements MBeanRegistration, ConnectorMBean
{
protected ServerInvoker invoker;
private String locatorURI;
private Element xml;
private Map configuration = new HashMap();
private MBeanServer server;
private ServerSocketFactory svrSocketFactory;
private SocketFactory socketFactory;
private Connector marshallerLoaderConnector = null;
private boolean isMarshallerLoader = false;
private boolean isStarted = false;
private boolean isCreated = false;
protected final Logger log = Logger.getLogger(getClass());
/**
* Empty constructor.
*/
public Connector()
{
}
/**
* Creates Connector with specified locator.
*
* @param locatorURI
*/
public Connector(String locatorURI)
{
this.locatorURI = locatorURI;
}
/**
* Creates Connector with specified locator.
*
* @param locator
*/
public Connector(InvokerLocator locator)
{
if (locator != null)
{
this.locatorURI = locator.getLocatorURI();
}
}
/**
* Constructs connector and populates configuration information.
*
* @param configuration
*/
public Connector(Map configuration)
{
this.configuration.putAll(configuration);
}
/**
* Constructs connector for given locator and configuration.
*
* @param locatorURI
* @param configuration
*/
public Connector(String locatorURI, Map configuration)
{
this.locatorURI = locatorURI;
this.configuration.putAll(configuration);
}
/**
* Constructs connector for given locator and configuration.
*
* @param locator
* @param configuration
*/
public Connector(InvokerLocator locator, Map configuration)
{
if (locator != null)
{
this.locatorURI = locator.getLocatorURI();
}
if (configuration != null)
{
this.configuration.putAll(configuration);
}
}
protected Connector(boolean isMarshallerConnector)
{
this();
this.isMarshallerLoader = isMarshallerConnector;
}
/**
* Indicates if the connector has been started yet.
*
* @return
*/
public boolean isStarted()
{
return isStarted;
}
/**
* This method is called by the MBeanServer before registration takes
* place. The MBean is passed a reference of the MBeanServer it is
* about to be registered with. The MBean must return the ObjectName it
* will be registered with. The MBeanServer can pass a suggested object
* depending upon how the MBean is registered.
*
* The MBean can stop the registration by throwing an exception.The
* exception is forwarded to the invoker wrapped in an
* MBeanRegistrationException.
*
* @param server the MBeanServer the MBean is about to be
* registered with.
* @param name the suggested ObjectName supplied by the
* MBeanServer.
* @return the actual ObjectName to register this MBean with.
* @throws Exception for any error, the MBean is not registered.
*/
public ObjectName preRegister(MBeanServer server, ObjectName name)
throws Exception
{
this.server = server;
return name;
}
/**
* This method is called by the MBeanServer after registration takes
* place or when registration fails.
*
* @param registrationDone the MBeanServer passes true when the
* MBean was registered, false otherwise.
*/
public void postRegister(Boolean registrationDone)
{
}
/**
* This method is called by the MBeanServer before deregistration takes
* place.
*
* The MBean can throw an exception, this will stop the deregistration.
* The exception is forwarded to the invoker wrapped in
* an MBeanRegistrationException.
*/
public void preDeregister()
throws Exception
{
}
/**
* This method is called by the MBeanServer after deregistration takes
* place.
*/
public void postDeregister()
{
}
/**
* Starts the connector. This is when configuration will be applied and server invoker created.
*
* @jmx.managed-operation description = "Start sets up the ServerInvoker we are wrapping."
* impact = "ACTION"
*/
public void start() throws Exception
{
if (!isStarted)
{
// doing this for those who use remoting outside of jboss container
// so don't have to call create() and then start()
if (!isCreated)
{
create();
}
// want to have handlers registered before starting, so if someone makes invocation,
// there is something to handle it.
configureHandlers();
// if marshaller loader not started, start it
if (!isMarshallerLoader)
{
if (marshallerLoaderConnector != null && !marshallerLoaderConnector.isStarted())
{
marshallerLoaderConnector.start();
}
}
// if invoker not started, start it
if (invoker.isStarted() == false)
{
try
{
invoker.start();
}
catch (Exception e)
{
if (marshallerLoaderConnector != null)
{
marshallerLoaderConnector.stop();
}
log.error("Error starting connector.", e);
throw e;
}
}
isStarted = true;
log.debug(this + " started");
}
}
/**
* Starts the connector.
*
* @param runAsNewThread indicates if should be started on new thread or the current one. If
* runAsNewThread is true, new thread will not be daemon thread.
* @throws Exception
*/
public void start(boolean runAsNewThread) throws Exception
{
Runnable r = new Runnable()
{
public void run()
{
try
{
start();
}
catch (Exception e)
{
log.error("Error starting Connector.", e);
}
}
};
Thread t = new Thread(r);
t.setDaemon(false);
t.start();
}
private void init()
throws Exception
{
Map invokerConfig = new HashMap();
if (locatorURI == null)
{
// nothing set for the InvokerLocator attribute, check to see if is part of Configuration attribute
if (xml != null)
{
getInvokerConfig(invokerConfig);
}
configuration.putAll(invokerConfig);
}
if (locatorURI == null)
{
throw new IllegalStateException("Connector not configured with LocatorURI.");
}
InvokerLocator locator = new InvokerLocator(locatorURI);
if (invoker == null)
{
// create the server invoker
invoker = InvokerRegistry.createServerInvoker(locator, configuration);
invoker.setMBeanServer(server);
// set the server socket factory if has been already set on the connector
invoker.setServerSocketFactory(svrSocketFactory);
// seting to null as don't want to keep reference in connector, but the server invoker
// see JBREM-367
this.svrSocketFactory = null;
// set the socket factory if has been already set on the connector
invoker.setSocketFactory(socketFactory);
this.socketFactory = null;
invoker.create();
}
// this will set the mbean server on the invoker and register it with mbean server
if (server != null)
{
try
{
ObjectName objName = new ObjectName(invoker.getMBeanObjectName());
if (!server.isRegistered(objName))
{
server.registerMBean(invoker, objName);
}
else
{
log.warn(objName + " is already registered with MBeanServer");
}
}
catch (Throwable e)
{
log.warn("Error registering invoker " + invoker + " with MBeanServer.", e);
}
}
// if using a generic locator (such as socket://localhost:0), the locator may change so
// keep the local cache in synch
locatorURI = invoker.getLocator().getLocatorURI();
if (!isMarshallerLoader)
{
// need to check if should create a marshaller loader on the server side
if (marshallerLoaderConnector == null)
{
marshallerLoaderConnector = createMarshallerLoader(invoker.getLocator());
}
}
}
private Connector createMarshallerLoader(InvokerLocator locator)
{
/**
* This is a bit of a hack, but have to bootstrap the marshaller/unmarshaller here because
* need them loaded and added to the MarshalFactory now, because is possible the first client
* to make a call on this connector may not have the marshaller/unmarshaller. Therefore, when
* the MarshallerLoaderHandler goes to load them for the client (MarshallerLoaderClient), they
* have to be there. Otherwise, would not be loaded until first client actually reaches the
* target server invoker, where they would otherwise be loaded.
*/
MarshalFactory.getMarshaller(locator, this.getClass().getClassLoader(), configuration);
Connector marshallerLoader = null;
InvokerLocator loaderLocator = MarshallLoaderFactory.convertLocator(locator);
// if loaderLocator is null, then probably not defined to have loader service (i.e. no loader port specified)
if (loaderLocator != null)
{
List repositories = getLoaderRepositories();
marshallerLoader = MarshallLoaderFactory.createMarshallLoader(loaderLocator, repositories, server);
}
return marshallerLoader;
}
private void getInvokerConfig(Map invokerConfig)
{
try
{
NodeList invokerNodes = xml.getElementsByTagName("invoker");
if (invokerNodes != null && invokerNodes.getLength() >= 1)
{
// only accept on invoker per connector at present
Node invokerNode = invokerNodes.item(0);
NamedNodeMap attributes = invokerNode.getAttributes();
Node transportNode = attributes.getNamedItem("transport");
if (transportNode != null)
{
String transport = transportNode.getNodeValue();
// need to log warning if there are more than one invoker elements
if (invokerNodes.getLength() > 1)
{
log.warn("Found more than one invokers defined in configuration. " +
"Will only be using the first one - " + transport);
}
// now create a map for all the sub attributes
Map paramConfig = new HashMap();
NodeList invokerAttributes = invokerNode.getChildNodes();
int len = invokerAttributes.getLength();
for (int x = 0; x < len; x++)
{
Node attr = invokerAttributes.item(x);
if ("attribute".equals(attr.getNodeName()))
{
String name = attr.getAttributes().getNamedItem("name").getNodeValue();
String value = attr.getFirstChild().getNodeValue();
invokerConfig.put(name, value);
Node isParamAttribute = attr.getAttributes().getNamedItem("isParam");
if (isParamAttribute != null && Boolean.valueOf(isParamAttribute.getNodeValue()).booleanValue())
{
paramConfig.put(name, value);
}
}
}
// should now have my map with all my attributes, now need to look for
// specific attributes that will impact the locator uri.
String clientConnectAddress = (String) invokerConfig.get("clientConnectAddress");
String clientConnectPort = (String) invokerConfig.get("clientConnectPort");
String serverBindAddress = (String) invokerConfig.get("serverBindAddress");
String serverBindPort = (String) invokerConfig.get("serverBindPort");
String path = (String) invokerConfig.get("path");
PortUtil.updateRange(invokerConfig);
String host = clientConnectAddress != null ? clientConnectAddress : serverBindAddress != null ? serverBindAddress : LocalHostUtil.getLocalHost().getHostAddress();
int port = clientConnectPort != null ? Integer.parseInt(clientConnectPort) : serverBindPort != null ? Integer.parseInt(serverBindPort) : PortUtil.findFreePort(serverBindAddress != null ? serverBindAddress : LocalHostUtil.getLocalHost().getHostAddress());
// finally, let's bild the invoker uri
String tempURI = transport + "://" + fixHostnameForURL(host) + ":" + port;
// append path if there is one
if (path != null)
{
tempURI += "/" + path;
}
// any params to add to the uri?
if (paramConfig.size() > 0)
{
tempURI += "/?";
Iterator keyItr = paramConfig.keySet().iterator();
if (keyItr.hasNext())
{
Object name = keyItr.next();
Object value = paramConfig.get(name);
tempURI += name + "=" + value;
}
while (keyItr.hasNext())
{
tempURI += "&";
Object name = keyItr.next();
Object value = paramConfig.get(name);
tempURI += name + "=" + value;
}
}
locatorURI = tempURI;
}
else
{
log.error("Invoker element within Configuration attribute does not contain a transport attribute.");
}
}
}
catch (Exception e)
{
log.error("Error configuring invoker for connector.", e);
throw new IllegalStateException("Error configuring invoker for connector. Can not continue without invoker.");
}
}
// adds enclosing brackets if an IPv6 literal address
private String fixHostnameForURL(String address)
{
if (address == null)
return address ;
if (address.indexOf(':') != -1 && address.indexOf("[") == -1)
return "[" + address + "]" ;
else
return address ;
}
private void configureHandlers()
throws Exception
{
if (xml != null)
{
NodeList handlersNodes = xml.getElementsByTagName("handler");
if ((handlersNodes == null || handlersNodes.getLength() <= 0) &&
(getInvocationHandlers() == null || getInvocationHandlers().length == 0))
{
throw new IllegalArgumentException("required 'handler' element not found and are no registered handlers found.");
}
int len = handlersNodes.getLength();
for (int c = 0; c < len; c++)
{
Node node = handlersNodes.item(c);
Node subNode = node.getAttributes().getNamedItem("subsystem");
if (subNode == null)
{
throw new IllegalArgumentException("Required 'subsystem' attribute on 'handler' element");
}
String handlerClass = node.getFirstChild().getNodeValue();
boolean isObjName = false;
ServerInvocationHandler handler = null;
//first check to see if this is an ObjectName
try
{
ObjectName objName = new ObjectName(handlerClass);
handler = createHandlerProxy(objName);
isObjName = true;
}
catch (MalformedObjectNameException e)
{
log.debug("Handler supplied is not an object name.");
}
if (!isObjName)
{
Class serverInvocationHandlerClass = ClassLoaderUtility.loadClass(handlerClass, Connector.class);
handler = (ServerInvocationHandler) serverInvocationHandlerClass.newInstance();
// handler = (ServerInvocationHandler) cl.loadClass(handlerClass).newInstance();
}
StringTokenizer tok = new StringTokenizer(subNode.getNodeValue(), ",");
while (tok.hasMoreTokens())
{
String subsystem = tok.nextToken();
addInvocationHandler(subsystem, handler);
}
}
}
}
private List getLoaderRepositories()
{
if (xml == null)
{
return null;
}
NodeList repositoryNodes = xml.getElementsByTagName("repository");
if (repositoryNodes.getLength() == 0)
{
return null;
}
List repositories = new ArrayList();
for (int i = 0; i < repositoryNodes.getLength(); i++)
{
Node node = repositoryNodes.item(i);
String repositoryName = node.getFirstChild().getNodeValue();
try
{
ObjectName objName = new ObjectName(repositoryName);
repositories.add(objName);
log.info("repository: " + repositoryName);
}
catch (MalformedObjectNameException e)
{
log.debug("repository name is not an object name: " + repositoryName);
continue;
}
}
return repositories;
}
private ServerInvocationHandler createHandlerProxy(ObjectName objName)
{
ServerInvocationHandler handler;
if (server != null)
{
handler = (ServerInvocationHandler)
MBeanServerInvocationHandler.newProxyInstance(server,
objName,
ServerInvocationHandler.class,
false);
}
else
{
throw new RuntimeException("Can not register MBean invocation handler as the Connector has not been registered with a MBeanServer.");
}
return handler;
}
/**
* Adds a connection listener to receive notification when a client connection
* is lost or disconnected. Will only be triggered for notifications when
* leasing is turned on (via the lease period attribute being set to > 0).
*
* @param listener
* @jmx.managed-operation description = "Add a connection listener to call when detect that a client has
* failed or disconnected."
* impact = "ACTION"
* @jmx.managed-parameter name = "listener"
* type = "org.jboss.remoting.ConnectionListener"
* description = "The connection listener to register"
*/
public void addConnectionListener(ConnectionListener listener)
{
if (invoker != null)
{
invoker.addConnectionListener(listener);
}
}
/**
* Removes connection listener from receiving client connection lost/disconnected
* notifications.
*
* @param listener
* @jmx.managed-operation description = "Remove a client connection listener."
* impact = "ACTION"
* @jmx.managed-parameter name = "listener"
* type = "org.jboss.remoting.ConnectionListener"
* description = "The client connection listener to remove."
*/
public void removeConnectionListener(ConnectionListener listener)
{
if (invoker != null)
{
invoker.removeConnectionListener(listener);
}
}
/**
* Sets the lease period for client connections.
* Value is in milliseconds.
*
* @param leasePeriodValue
* @jmx.managed-attribute description = "The number of milliseconds that should be used
* when establishing the client lease period (meaning client will need to update its lease
* within this amount of time or will be considered dead)."
* access = "read-write"
*/
public void setLeasePeriod(long leasePeriodValue)
{
if (invoker != null)
{
invoker.setLeasePeriod(leasePeriodValue);
}
}
/**
* Gets the lease period for client connections.
* Value in milliseconds.
*
* @return
* @jmx.managed-attribute
*/
public long getLeasePeriod()
{
if (invoker != null)
{
return invoker.getLeasePeriod();
}
else
{
return -1;
}
}
/**
* Stops the connector. Will also stop and destroy server invoker (transport)
*
* @jmx.managed-operation description = "Stop tears down the ServerInvoker we are wrapping."
* impact = "ACTION"
*/
public void stop()
{
if (isStarted)
{
if (invoker != null)
{
if (server != null)
{
try
{
ObjectName objName = new ObjectName(invoker.getMBeanObjectName());
server.unregisterMBean(objName);
}
catch (Exception e)
{
log.error("invalid Object Name", e);
}
}
invoker.stop();
invoker.destroy();
InvokerRegistry.destroyServerInvoker(invoker);
invoker = null;
}
if (marshallerLoaderConnector != null && marshallerLoaderConnector.isStarted)
{
marshallerLoaderConnector.stop();
marshallerLoaderConnector = null;
}
isStarted = false;
}
}
/**
* Creates the connector.
*
* @jmx.managed-operation
*/
public void create()
throws Exception
{
if (!isCreated)
{
try
{
init();
isCreated = true;
}
catch (Exception e)
{
// unwind create process
if (invoker != null)
{
invoker.stop();
invoker.destroy();
InvokerRegistry.destroyServerInvoker(invoker);
invoker = null;
}
isCreated = false;
throw e;
}
}
}
/**
* Destroys the connector.
*
* @jmx.managed-operation
*/
public void destroy()
{
if (isStarted)
{
stop();
}
if (invoker != null)
{
invoker.stop();
invoker.destroy();
InvokerRegistry.destroyServerInvoker(invoker);
invoker = null;
}
isCreated = false;
}
public ServerInvoker getServerInvoker()
{
return invoker;
}
/**
* Will get array of all the handlers registered with the connector's server invoker.
*
* @return
*/
public ServerInvocationHandler[] getInvocationHandlers()
{
ServerInvocationHandler[] handlers = null;
if (invoker != null)
{
handlers = invoker.getInvocationHandlers();
}
return handlers;
}
/**
* Returns the locator to the connector. Locator is the actual InvokerLocator
* object used to identify and get the ServerInvoker we are wrapping.
*
* @jmx.managed-attribute description = "Locator is the actual InvokerLocator object used to
* identify and get the ServerInvoker we are wrapping."
* access = "read-only"
*/
public InvokerLocator getLocator()
{
return invoker.getLocator();
}
/**
* Sets the invoker locator. InvokerLocator is the string URI representation
* of the InvokerLocator used to get and identify the ServerInvoker
* we are wrapping.
*
* @jmx.managed-attribute description = "InvokerLocator is the string URI representation of the
* InvokerLocator used to get and identify the ServerInvoker
* we are wrapping."
* access = "read-write"
*/
public void setInvokerLocator(String locator)
throws Exception
{
if (!isCreated)
{
locatorURI = locator;
}
else
{
throw new RuntimeException("Can not set the invoker locator on this Connector " +
"as has already been created with a different locator.");
}
}
/**
* Returns the invoker locator. InvokerLocator is the string URI representation
* of the InvokerLocator used to get and identify the ServerInvoker
* we are wrapping.
*
* @jmx.managed-attribute
*/
public String getInvokerLocator() throws Exception
{
return locatorURI;
}
/**
* Configuration is an xml element indicating subsystems to be registered
* with the ServerInvoker we wrap. Using mbean subsystems that call
* registerSubsystem is more flexible.
*
* @jmx.managed-attribute description = "Configuration is an xml element indicating subsystems
* to be registered with the ServerInvoker we wrap. Using
* mbean subsystems that call registerSubsystem is more
* flexible."
* access = "read-write"
*/
public void setConfiguration(Element xml)
throws Exception
{
this.xml = xml;
}
/**
* Configuration is an xml element indicating subsystems to be registered
* with the ServerInvoker we wrap. Using mbean subsystems that call
* registerSubsystem is more flexible.
*
* @jmx.managed-attribute
*/
public Element getConfiguration()
{
return xml;
}
/**
* Adds a handler to the connector via OjbectName. This will create a mbean proxy of
* type of ServerInvocationHandler for the MBean specified by object name passed (so has
* to implement ServerInvocationHandler interface).
*
* @param subsystem
* @param handlerObjectName
* @return Previous ServerInvocatioHandler with the same subsystem value (case insensitive) or null if one did not previously exist.
* @throws Exception
* @jmx.managed-operation description = "Add a subsystem invocation handler to the ServerInvoker
* we wrap, identified by the subsystem parameter."
* impact = "ACTION"
* @jmx.managed-parameter name = "subsystem"
* type = "java.lang.String"
* description = "The subsystem this handler is for."
* @jmx.managed-parameter name = "handlerObjectName"
* type = "javax.management.ObjectName"
* description = "The ServerInvocationHandler MBean we are registering
* for the subsystem"
*/
public ServerInvocationHandler addInvocationHandler(String subsystem, ObjectName handlerObjectName) throws Exception
{
ServerInvocationHandler invocationHandler = createHandlerProxy(handlerObjectName);
return addInvocationHandler(subsystem, invocationHandler);
}
/**
* Adds an invocation handler for the named subsystem to the invoker we
* manage, and sets the mbean server on the invocation handler.
*
* @return Previous ServerInvocatioHandler with the same subsystem value (case insensitive) or null if one did not previously exist.
* @jmx.managed-operation description = "Add a subsystem invocation handler to the ServerInvoker
* we wrap, identified by the subsystem parameter."
* impact = "ACTION"
* @jmx.managed-parameter name = "subsystem"
* type = "java.lang.String"
* description = "The subsystem this handler is for."
* @jmx.managed-parameter name = "handler"
* type = "org.jboss.remoting.ServerInvocationHandler"
* description = "The ServerInvocationHandler we are registering
* for the subsystem"
*/
public ServerInvocationHandler addInvocationHandler(String subsystem, ServerInvocationHandler handler)
throws Exception
{
if (invoker == null)
{
throw new IllegalStateException("You may only add handlers once the Connector is created (via create() method).");
}
handler.setMBeanServer(server);
return invoker.addInvocationHandler(subsystem, handler);
}
/**
* Removes an invocation handler for the supplied subsystem from the invoker
* we manage, and unsets the MBeanServer on the handler.
*
* @jmx.managed-operation description = "Remove a subsystem invocation handler to the
* ServerInvoker we wrap, identified by the subsystem
* parameter."
* impact = "ACTION"
* @jmx.managed-parameter name = "subsystem"
* type = "java.lang.String"
* description = "The subsystem this handler is for."
*/
public void removeInvocationHandler(String subsystem) throws Exception
{
ServerInvocationHandler handler = invoker.removeInvocationHandler(subsystem);
if (handler != null)
{
handler.setMBeanServer(null);
}
}
/**
* The server socket factory can only be set on the Connector before the create() method
* has been called. Otherwise, a runtime exception will be thrown.
* @param serverSocketFactory
*/
public void setServerSocketFactory(ServerSocketFactory serverSocketFactory)
{
if(isCreated)
{
throw new RuntimeException("Can not set server socket factory on Connector after the create() method has been called.");
}
if (invoker != null)
{
invoker.setServerSocketFactory(serverSocketFactory);
}
else
{
this.svrSocketFactory = serverSocketFactory;
}
}
public ServerSocketFactory getServerSocketFactory()
{
if (invoker != null)
{
return invoker.getServerSocketFactory();
}
else
{
return svrSocketFactory;
}
}
/**
* The socket factory (for callbacks) can only be set on the Connector before the
* create() method has been called. Otherwise, a runtime exception will be thrown.
* @param socketFactory
*/
public void setSocketFactory(SocketFactory socketFactory)
{
if(isCreated)
{
throw new RuntimeException("Can not set socket factory on Connector after the create() method has been called.");
}
if (invoker != null)
{
invoker.setSocketFactory(socketFactory);
}
else
{
this.socketFactory = socketFactory;
}
}
public SocketFactory getSocketFactory()
{
if (invoker != null)
{
return invoker.getSocketFactory();
}
else
{
return socketFactory;
}
}
}