SipService
interface.
- * This class inherits from the Tomcat StandardService. It adds a SipApplicationDispatcher
- * that will be listen for sip messages received by the sip stacks started by
- * the sip connectors associated with this context.
- * This has one attribute which is the sipApplicationDispatcherClassName allowing one
- * to specify the class name of the sipApplicationDispacther to easily replace
- * the default sipApplicationDispatcher with a custom one.
- *
- * @author Jean Deruelle
- */
-public class SipStandardService extends StandardService implements CatalinaSipService {
- //the logger
- private static final Logger logger = Logger.getLogger(SipStandardService.class);
- public static final String DEFAULT_SIP_PATH_NAME = "gov.nist";
- public static final String PASS_INVITE_NON_2XX_ACK_TO_LISTENER = "gov.nist.javax.sip.PASS_INVITE_NON_2XX_ACK_TO_LISTENER";
- public static final String TCP_POST_PARSING_THREAD_POOL_SIZE = "gov.nist.javax.sip.TCP_POST_PARSING_THREAD_POOL_SIZE";
- public static final String AUTOMATIC_DIALOG_SUPPORT_STACK_PROP = "javax.sip.AUTOMATIC_DIALOG_SUPPORT";
- public static final String LOOSE_DIALOG_VALIDATION = "gov.nist.javax.sip.LOOSE_DIALOG_VALIDATION";
- public static final String SERVER_LOG_STACK_PROP = "gov.nist.javax.sip.SERVER_LOG";
- public static final String DEBUG_LOG_STACK_PROP = "gov.nist.javax.sip.DEBUG_LOG";
- public static final String SERVER_HEADER = "org.mobicents.servlet.sip.SERVER_HEADER";
- public static final String USER_AGENT_HEADER = "org.mobicents.servlet.sip.USER_AGENT_HEADER";
- public static final String JVM_ROUTE = "jvmRoute";
- /**
- * The descriptive information string for this implementation.
- */
- private static final String INFO =
- "org.mobicents.servlet.sip.startup.SipStandardService/1.0";
- //the sip application dispatcher class name defined in the server.xml
- protected String sipApplicationDispatcherClassName;
- //instatiated class from the sipApplicationDispatcherClassName of the sip application dispatcher
- protected SipApplicationDispatcher sipApplicationDispatcher;
- private boolean gatherStatistics = true;
- protected int sipMessageQueueSize = 1500;
- private int backToNormalSipMessageQueueSize = 1300;
- protected int memoryThreshold = 95;
- private int backToNormalMemoryThreshold = 90;
- protected OutboundProxy outboundProxy;
- protected String proxyTimerServiceImplementationType;
- protected String sasTimerServiceImplementationType;
- protected long congestionControlCheckingInterval = 30000;
- private int canceledTimerTasksPurgePeriod = 0;
- // base timer interval for jain sip tx
- private int baseTimerInterval = 500;
- private int t2Interval = 4000;
- private int t4Interval = 5000;
- private int timerDInterval = 32000;
- protected int dispatcherThreadPoolSize = 15;
- private boolean md5ContactUserPart = false;
-
- protected String concurrencyControlMode = ConcurrencyControlMode.SipApplicationSession.toString();
- protected String congestionControlPolicy = CongestionControlPolicy.ErrorResponse.toString();
- protected String additionalParameterableHeaders;
- protected boolean bypassResponseExecutor = true;
- protected boolean bypassRequestExecutor = true;
- //this should be made available to the application router as a system prop
- protected String darConfigurationFileLocation;
- protected boolean connectorsStartedExternally = false;
- protected boolean dialogPendingRequestChecking = false;
- protected int callIdMaxLength;
- protected int tagHashMaxLength;
-
- protected boolean httpFollowsSip = false;
- protected String jvmRoute;
- protected ReplicationStrategy replicationStrategy;
-
- /**
- * the sip stack path name. Since the sip factory is per classloader it should be set here for all underlying stacks
- */
- private String sipPathName;
- /**
- * use Pretty Encoding
- */
- private boolean usePrettyEncoding = true;
-
- private SipStack sipStack;
- // defining sip stack properties
- private Properties sipStackProperties;
- private String sipStackPropertiesFileLocation;
- @Deprecated
- private String addressResolverClass = null;
- private String dnsServerLocatorClass = DefaultDNSServerLocator.class.getName();
- private int dnsTimeout = 1;
- private String dnsResolverClass = MobicentsDNSResolver.class.getName();
- private String mobicentsSipServletMessageFactoryClassName = Servlet3SipServletMessageFactory.class.getName();
-
- //the balancers to send heartbeat to and our health info
- @Deprecated
- private String balancers;
- //
- private ScheduledFuture> gracefulStopFuture;
-
- @Override
- public String getInfo() {
- return (INFO);
- }
-
- @Override
- public void addConnector(Connector connector) {
- if (initialized) {
- MobicentsExtendedListeningPoint extendedListeningPoint = null;
- if (connector.getProtocolHandler() instanceof SipProtocolHandler) {
- extendedListeningPoint = (MobicentsExtendedListeningPoint)
- ((SipProtocolHandler)connector.getProtocolHandler()).getAttribute(ExtendedListeningPoint.class.getSimpleName());}
- if(extendedListeningPoint != null) {
- try {
- extendedListeningPoint.getSipProvider().addSipListener(sipApplicationDispatcher);
- sipApplicationDispatcher.getSipNetworkInterfaceManager().addExtendedListeningPoint(extendedListeningPoint);
- } catch (TooManyListenersException e) {
- logger.error("Connector.initialize", e);
- }
- }
- ProtocolHandler protocolHandler = connector.getProtocolHandler();
- if(protocolHandler instanceof SipProtocolHandler) {
- connector.setPort(((SipProtocolHandler)protocolHandler).getPort());
- ((SipProtocolHandler)protocolHandler).setSipStack(sipStack);
- ((SipProtocolHandler)protocolHandler).setAttribute(SipApplicationDispatcher.class.getSimpleName(), sipApplicationDispatcher);
- registerSipConnector(connector);
- }
- }
- else {
- }
- super.addConnector(connector);
- }
-
- /**
- * Register the sip connector under a different name than HTTP Connector and we add the transport to avoid clashing with 2 connectors having the same port and address
- * @param connector connector to register
- */
- protected void registerSipConnector(Connector connector) {
- try {
-
- ObjectName objectName = createSipConnectorObjectName(connector, getName(), "SipConnector");
- Registry.getRegistry(null, null)
- .registerComponent(connector, objectName, null);
-//TODO connector.setController(objectName);
- } catch (Exception e) {
- logger.error( "Error registering connector ", e);
- }
- if(logger.isDebugEnabled())
- logger.debug("Creating name for connector " + getObjectName());
- }
-
- @Override
- public void removeConnector(Connector connector) {
- MobicentsExtendedListeningPoint extendedListeningPoint = null;
- if (connector.getProtocolHandler() instanceof SipProtocolHandler){
- extendedListeningPoint = (MobicentsExtendedListeningPoint)
- ((SipProtocolHandler)connector.getProtocolHandler()).getAttribute(ExtendedListeningPoint.class.getSimpleName());}
- if(extendedListeningPoint != null) {
- extendedListeningPoint.getSipProvider().removeSipListener(sipApplicationDispatcher);
- sipApplicationDispatcher.getSipNetworkInterfaceManager().removeExtendedListeningPoint(extendedListeningPoint);
- }
- super.removeConnector(connector);
- }
-
- @Override
- public void initialize() throws LifecycleException {
- //load the sip application disptacher from the class name specified in the server.xml file
- //and initializes it
- StaticServiceHolder.sipStandardService = this;
- try {
- sipApplicationDispatcher = (SipApplicationDispatcher)
- Class.forName(sipApplicationDispatcherClassName).newInstance();
- } catch (InstantiationException e) {
- throw new LifecycleException("Impossible to load the Sip Application Dispatcher",e);
- } catch (IllegalAccessException e) {
- throw new LifecycleException("Impossible to load the Sip Application Dispatcher",e);
- } catch (ClassNotFoundException e) {
- throw new LifecycleException("Impossible to load the Sip Application Dispatcher",e);
- } catch (ClassCastException e) {
- throw new LifecycleException("Sip Application Dispatcher defined does not implement " + SipApplicationDispatcher.class.getName(),e);
- }
- if(logger.isInfoEnabled()) {
- logger.info("Pretty encoding of headers enabled ? " + usePrettyEncoding);
- }
- if(sipPathName == null) {
- sipPathName = DEFAULT_SIP_PATH_NAME;
- }
- if(logger.isInfoEnabled()) {
- logger.info("Sip Stack path name : " + sipPathName);
- }
- sipApplicationDispatcher.setSipService(this);
- sipApplicationDispatcher.getSipFactory().initialize(sipPathName, usePrettyEncoding);
-
- String catalinaBase = getCatalinaBase();
- if(darConfigurationFileLocation != null) {
- if(!darConfigurationFileLocation.startsWith("file:///")) {
- darConfigurationFileLocation = "file:///" + catalinaBase.replace(File.separatorChar, '/') + "/" + darConfigurationFileLocation;
- }
- System.setProperty("javax.servlet.sip.dar", darConfigurationFileLocation);
- }
- super.initialize();
- sipApplicationDispatcher.setDomain(this.getName());
- if(baseTimerInterval < 1) {
- throw new LifecycleException("It's forbidden to set the Base Timer Interval to a non positive value");
- }
- initSipStack();
- sipApplicationDispatcher.setBaseTimerInterval(baseTimerInterval);
- sipApplicationDispatcher.setT2Interval(t2Interval);
- sipApplicationDispatcher.setT4Interval(t4Interval);
- sipApplicationDispatcher.setTimerDInterval(timerDInterval);
- sipApplicationDispatcher.setMemoryThreshold(getMemoryThreshold());
- sipApplicationDispatcher.setBackToNormalMemoryThreshold(backToNormalMemoryThreshold);
- sipApplicationDispatcher.setCongestionControlCheckingInterval(getCongestionControlCheckingInterval());
- sipApplicationDispatcher.setCongestionControlPolicyByName(getCongestionControlPolicy());
- sipApplicationDispatcher.setQueueSize(getSipMessageQueueSize());
- sipApplicationDispatcher.setBackToNormalQueueSize(backToNormalSipMessageQueueSize);
- sipApplicationDispatcher.setGatherStatistics(gatherStatistics);
- sipApplicationDispatcher.setConcurrencyControlMode(ConcurrencyControlMode.valueOf(getConcurrencyControlMode()));
- sipApplicationDispatcher.setBypassRequestExecutor(bypassRequestExecutor);
- sipApplicationDispatcher.setBypassResponseExecutor(bypassResponseExecutor);
- sipApplicationDispatcher.setSipStack(sipStack);
- sipApplicationDispatcher.init();
- // Tomcat specific loading case where the connectors are added even before the service is initialized
- // so we need to set the sip stack before it starts
- synchronized (connectors) {
- for (Connector connector : connectors) {
- ProtocolHandler protocolHandler = connector.getProtocolHandler();
- if(protocolHandler instanceof SipProtocolHandler) {
- connector.setPort(((SipProtocolHandler)protocolHandler).getPort());
- ((SipProtocolHandler)protocolHandler).setSipStack(sipStack);
- ((SipProtocolHandler)protocolHandler).setAttribute(SipApplicationDispatcher.class.getSimpleName(), sipApplicationDispatcher);
- registerSipConnector(connector);
- }
- }
- }
- }
-
- @Override
- public void start() throws LifecycleException {
- super.start();
- synchronized (connectors) {
- for (Connector connector : connectors) {
- final ProtocolHandler protocolHandler = connector.getProtocolHandler();
- //Jboss sepcific loading case
- Boolean isSipConnector = false;
- if (protocolHandler instanceof SipProtocolHandler)
- isSipConnector = (Boolean) ((SipProtocolHandler)protocolHandler).getAttribute(SipProtocolHandler.IS_SIP_CONNECTOR);
- if(isSipConnector != null && isSipConnector) {
- if(logger.isDebugEnabled()) {
- logger.debug("Attaching the sip application dispatcher " +
- "as a sip listener to connector listening on port " +
- connector.getPort());
- }
- ((SipProtocolHandler)protocolHandler).setAttribute(SipApplicationDispatcher.class.getSimpleName(), sipApplicationDispatcher);
- ((SipProtocolHandler)protocolHandler).setSipStack(sipStack);
- connectorsStartedExternally = true;
- }
- //Tomcat specific loading case
- MobicentsExtendedListeningPoint extendedListeningPoint = null;
- if (protocolHandler instanceof SipProtocolHandler) {
- extendedListeningPoint = (MobicentsExtendedListeningPoint)
- ((SipProtocolHandler)protocolHandler).getAttribute(ExtendedListeningPoint.class.getSimpleName());}
- if(extendedListeningPoint != null && sipStack != null) {
- try {
- extendedListeningPoint.getSipProvider().addSipListener(sipApplicationDispatcher);
- sipApplicationDispatcher.getSipNetworkInterfaceManager().addExtendedListeningPoint(extendedListeningPoint);
- connectorsStartedExternally = false;
- } catch (TooManyListenersException e) {
- throw new LifecycleException("Couldn't add the sip application dispatcher "
- + sipApplicationDispatcher + " as a listener to the following listening point provider " + extendedListeningPoint, e);
- }
- }
- }
- }
- if(!connectorsStartedExternally) {
- sipApplicationDispatcher.start();
- }
-
- if(this.getSipMessageQueueSize() <= 0)
- throw new LifecycleException("Message queue size can not be 0 or less");
-
- if(logger.isDebugEnabled()) {
- logger.debug("SIP Standard Service Started.");
- }
- }
-
- public String getJvmRoute() {
- return this.jvmRoute;
- }
-
- public void setJvmRoute(String jvmRoute) {
- this.jvmRoute = jvmRoute;
- }
-
- protected void initSipStack() throws LifecycleException {
- try {
- if(logger.isDebugEnabled()) {
- logger.debug("Initializing SIP stack");
- }
-
- // This simply puts HTTP and SSL port numbers in JVM properties menat to be read by jsip ha when sending heart beats with Node description.
- initializeSystemPortProperties();
-
- String catalinaBase = getCatalinaBase();
- if(sipStackPropertiesFileLocation != null && !sipStackPropertiesFileLocation.startsWith("file:///")) {
- sipStackPropertiesFileLocation = "file:///" + catalinaBase.replace(File.separatorChar, '/') + "/" + sipStackPropertiesFileLocation;
- }
- boolean isPropsLoaded = false;
- if(sipStackProperties == null) {
- sipStackProperties = new Properties();
- } else {
- isPropsLoaded = true;
- }
-
- if (logger.isDebugEnabled()) {
- logger.debug("Loading SIP stack properties from following file : " + sipStackPropertiesFileLocation);
- }
- if(sipStackPropertiesFileLocation != null) {
- //hack to get around space char in path see http://weblogs.java.net/blog/kohsuke/archive/2007/04/how_to_convert.html,
- // we create a URL since it's permissive enough
- File sipStackPropertiesFile = null;
- URL url = null;
- try {
- url = new URL(sipStackPropertiesFileLocation);
- } catch (MalformedURLException e) {
- logger.fatal("Cannot find the sip stack properties file ! ",e);
- throw new IllegalArgumentException("The Default Application Router file Location : "+sipStackPropertiesFileLocation+" is not valid ! ",e);
- }
- try {
- sipStackPropertiesFile = new File(new URI(sipStackPropertiesFileLocation));
- } catch (URISyntaxException e) {
- //if the uri contains space this will fail, so getting the path will work
- sipStackPropertiesFile = new File(url.getPath());
- }
- FileInputStream sipStackPropertiesInputStream = null;
- try {
- sipStackPropertiesInputStream = new FileInputStream(sipStackPropertiesFile);
- sipStackProperties.load(sipStackPropertiesInputStream);
- } catch (Exception e) {
- logger.warn("Could not find or problem when loading the sip stack properties file : " + sipStackPropertiesFileLocation, e);
- } finally {
- if(sipStackPropertiesInputStream != null) {
- try {
- sipStackPropertiesInputStream.close();
- } catch (IOException e) {
- logger.error("fail to close the following file " + sipStackPropertiesFile.getAbsolutePath(), e);
- }
- }
- }
-
- String debugLog = sipStackProperties.getProperty(DEBUG_LOG_STACK_PROP);
- if(debugLog != null && debugLog.length() > 0 && !debugLog.startsWith("file:///")) {
- sipStackProperties.setProperty(DEBUG_LOG_STACK_PROP,
- catalinaBase + "/" + debugLog);
- }
- String serverLog = sipStackProperties.getProperty(SERVER_LOG_STACK_PROP);
- if(serverLog != null && serverLog.length() > 0 && !serverLog.startsWith("file:///")) {
- sipStackProperties.setProperty(SERVER_LOG_STACK_PROP,
- catalinaBase + "/" + serverLog);
- }
- // The whole MSS is built upon those assumptions, so those properties are not overrideable
- if (sipStackProperties.getProperty(AUTOMATIC_DIALOG_SUPPORT_STACK_PROP) == null) {
- //https://github.com/RestComm/sip-servlets/issues/143
- //set off if user didnt provided any value.
- sipStackProperties.setProperty(AUTOMATIC_DIALOG_SUPPORT_STACK_PROP, "off");
- }
- sipStackProperties.setProperty(LOOSE_DIALOG_VALIDATION, "true");
- sipStackProperties.setProperty(PASS_INVITE_NON_2XX_ACK_TO_LISTENER, "true");
- isPropsLoaded = true;
- } else {
- logger.warn("no sip stack properties file defined ");
- }
- if(!isPropsLoaded) {
- logger.warn("loading default Mobicents Sip Servlets sip stack properties");
- // Silently set default values
- sipStackProperties.setProperty("gov.nist.javax.sip.LOG_MESSAGE_CONTENT",
- "true");
- sipStackProperties.setProperty("gov.nist.javax.sip.TRACE_LEVEL",
- "LOG4J");
- sipStackProperties.setProperty(DEBUG_LOG_STACK_PROP,
- catalinaBase + "/" + "mss-jsip-" + getName() +"-debug.txt");
- sipStackProperties.setProperty(SERVER_LOG_STACK_PROP,
- catalinaBase + "/" + "mss-jsip-" + getName() +"-messages.xml");
- sipStackProperties.setProperty("javax.sip.STACK_NAME", "mss-" + getName());
- sipStackProperties.setProperty(AUTOMATIC_DIALOG_SUPPORT_STACK_PROP, "off");
- sipStackProperties.setProperty("gov.nist.javax.sip.DELIVER_UNSOLICITED_NOTIFY", "true");
- sipStackProperties.setProperty("gov.nist.javax.sip.THREAD_POOL_SIZE", "64");
- sipStackProperties.setProperty("gov.nist.javax.sip.REENTRANT_LISTENER", "true");
- sipStackProperties.setProperty("gov.nist.javax.sip.MAX_FORK_TIME_SECONDS", "0");
- sipStackProperties.setProperty(LOOSE_DIALOG_VALIDATION, "true");
- sipStackProperties.setProperty(PASS_INVITE_NON_2XX_ACK_TO_LISTENER, "true");
- sipStackProperties.setProperty("gov.nist.javax.sip.AUTOMATIC_DIALOG_ERROR_HANDLING", "false");
- sipStackProperties.setProperty("gov.nist.javax.sip.AGGRESSIVE_CLEANUP", "true");
- }
-
- if(sipStackProperties.get(TCP_POST_PARSING_THREAD_POOL_SIZE) == null) {
- sipStackProperties.setProperty(TCP_POST_PARSING_THREAD_POOL_SIZE, "30");
- }
-
- // set the DNSServerLocator allowing to support RFC 3263 and do DNS lookups to resolve uris
- if(dnsServerLocatorClass != null && dnsServerLocatorClass.trim().length() > 0) {
- if(logger.isDebugEnabled()) {
- logger.debug("Sip Stack " + sipStackProperties.getProperty("javax.sip.STACK_NAME") +" will be using " + dnsServerLocatorClass + " as DNSServerLocator");
- }
- try {
- // create parameters argument to identify constructor
- Class[] paramTypes = new Class[0];
- // get constructor of AddressResolver in order to instantiate
- Constructor dnsServerLocatorConstructor = Class.forName(dnsServerLocatorClass).getConstructor(
- paramTypes);
- // Wrap properties object in order to pass to constructor of AddressResolver
- Object[] conArgs = new Object[0];
- // Creates a new instance of AddressResolver Class with the supplied sipApplicationDispatcher.
- DNSServerLocator dnsServerLocator = (DNSServerLocator) dnsServerLocatorConstructor.newInstance(conArgs);
- sipApplicationDispatcher.setDNSServerLocator(dnsServerLocator);
- sipApplicationDispatcher.setDNSTimeout(dnsTimeout);
- if(sipStackProperties.getProperty("javax.sip.ROUTER_PATH") == null) {
- sipStackProperties.setProperty("javax.sip.ROUTER_PATH", DNSAwareRouter.class.getCanonicalName());
- }
- } catch (Exception e) {
- logger.error("Couldn't set the AddressResolver " + addressResolverClass, e);
- throw e;
- }
- } else {
- if(logger.isInfoEnabled()) {
- logger.info("no DNSServerLocator will be used since none has been specified.");
- }
- }
-
- String serverHeaderValue = sipStackProperties.getProperty(SERVER_HEADER);
- if(serverHeaderValue != null) {
- ListSipService
interface.
+ * This class inherits from the Tomcat StandardService. It adds a SipApplicationDispatcher
+ * that will be listen for sip messages received by the sip stacks started by
+ * the sip connectors associated with this context.
+ * This has one attribute which is the sipApplicationDispatcherClassName allowing one
+ * to specify the class name of the sipApplicationDispacther to easily replace
+ * the default sipApplicationDispatcher with a custom one.
+ *
+ * @author Jean Deruelle
+ */
+public class SipStandardService extends StandardService implements CatalinaSipService {
+ //the logger
+ private static final Logger logger = Logger.getLogger(SipStandardService.class);
+ public static final String DEFAULT_SIP_PATH_NAME = "gov.nist";
+ public static final String PASS_INVITE_NON_2XX_ACK_TO_LISTENER = "gov.nist.javax.sip.PASS_INVITE_NON_2XX_ACK_TO_LISTENER";
+ public static final String TCP_POST_PARSING_THREAD_POOL_SIZE = "gov.nist.javax.sip.TCP_POST_PARSING_THREAD_POOL_SIZE";
+ public static final String AUTOMATIC_DIALOG_SUPPORT_STACK_PROP = "javax.sip.AUTOMATIC_DIALOG_SUPPORT";
+ public static final String LOOSE_DIALOG_VALIDATION = "gov.nist.javax.sip.LOOSE_DIALOG_VALIDATION";
+ public static final String SERVER_LOG_STACK_PROP = "gov.nist.javax.sip.SERVER_LOG";
+ public static final String DEBUG_LOG_STACK_PROP = "gov.nist.javax.sip.DEBUG_LOG";
+ public static final String SERVER_HEADER = "org.mobicents.servlet.sip.SERVER_HEADER";
+ public static final String USER_AGENT_HEADER = "org.mobicents.servlet.sip.USER_AGENT_HEADER";
+ public static final String JVM_ROUTE = "jvmRoute";
+ /**
+ * The descriptive information string for this implementation.
+ */
+ private static final String INFO =
+ "org.mobicents.servlet.sip.startup.SipStandardService/1.0";
+ //the sip application dispatcher class name defined in the server.xml
+ protected String sipApplicationDispatcherClassName;
+ //instatiated class from the sipApplicationDispatcherClassName of the sip application dispatcher
+ protected SipApplicationDispatcher sipApplicationDispatcher;
+ private boolean gatherStatistics = true;
+ protected int sipMessageQueueSize = 1500;
+ private int backToNormalSipMessageQueueSize = 1300;
+ protected int memoryThreshold = 95;
+ private int backToNormalMemoryThreshold = 90;
+ protected OutboundProxy outboundProxy;
+ protected String proxyTimerServiceImplementationType;
+ protected String sasTimerServiceImplementationType;
+ protected long congestionControlCheckingInterval = 30000;
+ private int canceledTimerTasksPurgePeriod = 0;
+ // base timer interval for jain sip tx
+ private int baseTimerInterval = 500;
+ private int t2Interval = 4000;
+ private int t4Interval = 5000;
+ private int timerDInterval = 32000;
+ protected int dispatcherThreadPoolSize = 15;
+ private boolean md5ContactUserPart = false;
+
+ protected String concurrencyControlMode = ConcurrencyControlMode.SipApplicationSession.toString();
+ protected String congestionControlPolicy = CongestionControlPolicy.ErrorResponse.toString();
+ protected String additionalParameterableHeaders;
+ protected boolean bypassResponseExecutor = true;
+ protected boolean bypassRequestExecutor = true;
+ //this should be made available to the application router as a system prop
+ protected String darConfigurationFileLocation;
+ protected boolean connectorsStartedExternally = false;
+ protected boolean dialogPendingRequestChecking = false;
+ protected int callIdMaxLength;
+ protected int tagHashMaxLength;
+ private long gracefulInterval = 30000;
+
+ protected boolean httpFollowsSip = false;
+ protected String jvmRoute;
+ protected ReplicationStrategy replicationStrategy;
+
+ /**
+ * the sip stack path name. Since the sip factory is per classloader it should be set here for all underlying stacks
+ */
+ private String sipPathName;
+ /**
+ * use Pretty Encoding
+ */
+ private boolean usePrettyEncoding = true;
+
+ private SipStack sipStack;
+ // defining sip stack properties
+ private Properties sipStackProperties;
+ private String sipStackPropertiesFileLocation;
+ @Deprecated
+ private String addressResolverClass = null;
+ private String dnsServerLocatorClass = DefaultDNSServerLocator.class.getName();
+ private int dnsTimeout = 1;
+ private String dnsResolverClass = MobicentsDNSResolver.class.getName();
+ private String mobicentsSipServletMessageFactoryClassName = Servlet3SipServletMessageFactory.class.getName();
+
+ //the balancers to send heartbeat to and our health info
+ @Deprecated
+ private String balancers;
+ //
+ private ScheduledFuture> gracefulStopFuture;
+
+ @Override
+ public String getInfo() {
+ return (INFO);
+ }
+
+ @Override
+ public void addConnector(Connector connector) {
+ if (initialized) {
+ MobicentsExtendedListeningPoint extendedListeningPoint = null;
+ if (connector.getProtocolHandler() instanceof SipProtocolHandler) {
+ extendedListeningPoint = (MobicentsExtendedListeningPoint)
+ ((SipProtocolHandler)connector.getProtocolHandler()).getAttribute(ExtendedListeningPoint.class.getSimpleName());}
+ if(extendedListeningPoint != null) {
+ try {
+ extendedListeningPoint.getSipProvider().addSipListener(sipApplicationDispatcher);
+ sipApplicationDispatcher.getSipNetworkInterfaceManager().addExtendedListeningPoint(extendedListeningPoint);
+ } catch (TooManyListenersException e) {
+ logger.error("Connector.initialize", e);
+ }
+ }
+ ProtocolHandler protocolHandler = connector.getProtocolHandler();
+ if(protocolHandler instanceof SipProtocolHandler) {
+ connector.setPort(((SipProtocolHandler)protocolHandler).getPort());
+ ((SipProtocolHandler)protocolHandler).setSipStack(sipStack);
+ ((SipProtocolHandler)protocolHandler).setAttribute(SipApplicationDispatcher.class.getSimpleName(), sipApplicationDispatcher);
+ registerSipConnector(connector);
+ }
+ }
+ else {
+ }
+ super.addConnector(connector);
+ }
+
+ /**
+ * Register the sip connector under a different name than HTTP Connector and we add the transport to avoid clashing with 2 connectors having the same port and address
+ * @param connector connector to register
+ */
+ protected void registerSipConnector(Connector connector) {
+ try {
+ ObjectName objectName = createSipConnectorObjectName(connector, getName(), "SipConnector");
+ if(logger.isDebugEnabled()) {
+ logger.debug("registering sip connector : " + objectName);
+ }
+ Registry.getRegistry(null, null)
+ .registerComponent(connector, objectName, null);
+//TODO connector.setController(objectName);
+ } catch (Exception e) {
+ logger.error( "Error registering connector ", e);
+ }
+ if(logger.isDebugEnabled())
+ logger.debug("Creating name for connector " + getObjectName());
+ }
+
+ /**
+ * Register the sip connector under a different name than HTTP Connector and we add the transport to avoid clashing with 2 connectors having the same port and address
+ * @param connector connector to register
+ */
+ protected void unregisterSipConnector(Connector connector) {
+ try {
+
+ ObjectName objectName = createSipConnectorObjectName(connector, getName(), "SipConnector");
+ if(logger.isDebugEnabled()) {
+ logger.debug("unregistering sip connector : " + objectName);
+ }
+ Registry.getRegistry(null, null)
+ .unregisterComponent(objectName);
+ } catch (Exception e) {
+ logger.error( "Error registering connector ", e);
+ }
+ if(logger.isDebugEnabled())
+ logger.debug("Creating name for connector " + getObjectName());
+ }
+
+ @Override
+ public void removeConnector(Connector connector) {
+ MobicentsExtendedListeningPoint extendedListeningPoint = null;
+ if (connector.getProtocolHandler() instanceof SipProtocolHandler) {
+ extendedListeningPoint = (MobicentsExtendedListeningPoint)
+ ((SipProtocolHandler)connector.getProtocolHandler()).getAttribute(ExtendedListeningPoint.class.getSimpleName());
+ unregisterSipConnector(connector);
+ }
+ if(extendedListeningPoint != null) {
+ extendedListeningPoint.getSipProvider().removeSipListener(sipApplicationDispatcher);
+ sipApplicationDispatcher.getSipNetworkInterfaceManager().removeExtendedListeningPoint(extendedListeningPoint);
+ }
+ super.removeConnector(connector);
+ }
+
+ @Override
+ public void initialize() throws LifecycleException {
+ //load the sip application disptacher from the class name specified in the server.xml file
+ //and initializes it
+ StaticServiceHolder.sipStandardService = this;
+ try {
+ sipApplicationDispatcher = (SipApplicationDispatcher)
+ Class.forName(sipApplicationDispatcherClassName).newInstance();
+ } catch (InstantiationException e) {
+ throw new LifecycleException("Impossible to load the Sip Application Dispatcher",e);
+ } catch (IllegalAccessException e) {
+ throw new LifecycleException("Impossible to load the Sip Application Dispatcher",e);
+ } catch (ClassNotFoundException e) {
+ throw new LifecycleException("Impossible to load the Sip Application Dispatcher",e);
+ } catch (ClassCastException e) {
+ throw new LifecycleException("Sip Application Dispatcher defined does not implement " + SipApplicationDispatcher.class.getName(),e);
+ }
+ if(logger.isInfoEnabled()) {
+ logger.info("Pretty encoding of headers enabled ? " + usePrettyEncoding);
+ }
+ if(sipPathName == null) {
+ sipPathName = DEFAULT_SIP_PATH_NAME;
+ }
+ if(logger.isInfoEnabled()) {
+ logger.info("Sip Stack path name : " + sipPathName);
+ }
+ sipApplicationDispatcher.setSipService(this);
+ sipApplicationDispatcher.getSipFactory().initialize(sipPathName, usePrettyEncoding);
+
+ String catalinaBase = getCatalinaBase();
+ if(darConfigurationFileLocation != null) {
+ if(!darConfigurationFileLocation.startsWith("file:///")) {
+ darConfigurationFileLocation = "file:///" + catalinaBase.replace(File.separatorChar, '/') + "/" + darConfigurationFileLocation;
+ }
+ System.setProperty("javax.servlet.sip.dar", darConfigurationFileLocation);
+ }
+ super.initialize();
+ sipApplicationDispatcher.setDomain(this.getName());
+ if(baseTimerInterval < 1) {
+ throw new LifecycleException("It's forbidden to set the Base Timer Interval to a non positive value");
+ }
+ initSipStack();
+ sipApplicationDispatcher.setBaseTimerInterval(baseTimerInterval);
+ sipApplicationDispatcher.setT2Interval(t2Interval);
+ sipApplicationDispatcher.setT4Interval(t4Interval);
+ sipApplicationDispatcher.setTimerDInterval(timerDInterval);
+ sipApplicationDispatcher.setMemoryThreshold(getMemoryThreshold());
+ sipApplicationDispatcher.setBackToNormalMemoryThreshold(backToNormalMemoryThreshold);
+ sipApplicationDispatcher.setCongestionControlCheckingInterval(getCongestionControlCheckingInterval());
+ sipApplicationDispatcher.setCongestionControlPolicyByName(getCongestionControlPolicy());
+ sipApplicationDispatcher.setQueueSize(getSipMessageQueueSize());
+ sipApplicationDispatcher.setBackToNormalQueueSize(backToNormalSipMessageQueueSize);
+ sipApplicationDispatcher.setGatherStatistics(gatherStatistics);
+ sipApplicationDispatcher.setConcurrencyControlMode(ConcurrencyControlMode.valueOf(getConcurrencyControlMode()));
+ sipApplicationDispatcher.setBypassRequestExecutor(bypassRequestExecutor);
+ sipApplicationDispatcher.setBypassResponseExecutor(bypassResponseExecutor);
+ sipApplicationDispatcher.setSipStack(sipStack);
+ sipApplicationDispatcher.init();
+ // Tomcat specific loading case where the connectors are added even before the service is initialized
+ // so we need to set the sip stack before it starts
+ synchronized (connectors) {
+ for (Connector connector : connectors) {
+ ProtocolHandler protocolHandler = connector.getProtocolHandler();
+ if(protocolHandler instanceof SipProtocolHandler) {
+ connector.setPort(((SipProtocolHandler)protocolHandler).getPort());
+ ((SipProtocolHandler)protocolHandler).setSipStack(sipStack);
+ ((SipProtocolHandler)protocolHandler).setAttribute(SipApplicationDispatcher.class.getSimpleName(), sipApplicationDispatcher);
+ registerSipConnector(connector);
+ }
+ }
+ }
+ }
+
+ @Override
+ public void start() throws LifecycleException {
+ super.start();
+ synchronized (connectors) {
+ for (Connector connector : connectors) {
+ final ProtocolHandler protocolHandler = connector.getProtocolHandler();
+ //Jboss sepcific loading case
+ Boolean isSipConnector = false;
+ if (protocolHandler instanceof SipProtocolHandler)
+ isSipConnector = (Boolean) ((SipProtocolHandler)protocolHandler).getAttribute(SipProtocolHandler.IS_SIP_CONNECTOR);
+ if(isSipConnector != null && isSipConnector) {
+ if(logger.isDebugEnabled()) {
+ logger.debug("Attaching the sip application dispatcher " +
+ "as a sip listener to connector listening on port " +
+ connector.getPort());
+ }
+ ((SipProtocolHandler)protocolHandler).setAttribute(SipApplicationDispatcher.class.getSimpleName(), sipApplicationDispatcher);
+ ((SipProtocolHandler)protocolHandler).setSipStack(sipStack);
+ connectorsStartedExternally = true;
+ }
+ //Tomcat specific loading case
+ MobicentsExtendedListeningPoint extendedListeningPoint = null;
+ if (protocolHandler instanceof SipProtocolHandler) {
+ extendedListeningPoint = (MobicentsExtendedListeningPoint)
+ ((SipProtocolHandler)protocolHandler).getAttribute(ExtendedListeningPoint.class.getSimpleName());}
+ if(extendedListeningPoint != null && sipStack != null) {
+ try {
+ extendedListeningPoint.getSipProvider().addSipListener(sipApplicationDispatcher);
+ sipApplicationDispatcher.getSipNetworkInterfaceManager().addExtendedListeningPoint(extendedListeningPoint);
+ connectorsStartedExternally = false;
+ } catch (TooManyListenersException e) {
+ throw new LifecycleException("Couldn't add the sip application dispatcher "
+ + sipApplicationDispatcher + " as a listener to the following listening point provider " + extendedListeningPoint, e);
+ }
+ }
+ }
+ }
+ if(!connectorsStartedExternally) {
+ sipApplicationDispatcher.start();
+ }
+
+ if(this.getSipMessageQueueSize() <= 0)
+ throw new LifecycleException("Message queue size can not be 0 or less");
+
+ if(logger.isDebugEnabled()) {
+ logger.debug("SIP Standard Service Started.");
+ }
+ }
+
+ public String getJvmRoute() {
+ return this.jvmRoute;
+ }
+
+ public void setJvmRoute(String jvmRoute) {
+ this.jvmRoute = jvmRoute;
+ }
+
+ protected void initSipStack() throws LifecycleException {
+ try {
+ if(logger.isDebugEnabled()) {
+ logger.debug("Initializing SIP stack");
+ }
+
+ // This simply puts HTTP and SSL port numbers in JVM properties menat to be read by jsip ha when sending heart beats with Node description.
+ initializeSystemPortProperties();
+
+ String catalinaBase = getCatalinaBase();
+ if(sipStackPropertiesFileLocation != null && !sipStackPropertiesFileLocation.startsWith("file:///")) {
+ sipStackPropertiesFileLocation = "file:///" + catalinaBase.replace(File.separatorChar, '/') + "/" + sipStackPropertiesFileLocation;
+ }
+ boolean isPropsLoaded = false;
+ if(sipStackProperties == null) {
+ sipStackProperties = new Properties();
+ } else {
+ isPropsLoaded = true;
+ }
+
+ if (logger.isDebugEnabled()) {
+ logger.debug("Loading SIP stack properties from following file : " + sipStackPropertiesFileLocation);
+ }
+ if(sipStackPropertiesFileLocation != null) {
+ //hack to get around space char in path see http://weblogs.java.net/blog/kohsuke/archive/2007/04/how_to_convert.html,
+ // we create a URL since it's permissive enough
+ File sipStackPropertiesFile = null;
+ URL url = null;
+ try {
+ url = new URL(sipStackPropertiesFileLocation);
+ } catch (MalformedURLException e) {
+ logger.fatal("Cannot find the sip stack properties file ! ",e);
+ throw new IllegalArgumentException("The Default Application Router file Location : "+sipStackPropertiesFileLocation+" is not valid ! ",e);
+ }
+ try {
+ sipStackPropertiesFile = new File(new URI(sipStackPropertiesFileLocation));
+ } catch (URISyntaxException e) {
+ //if the uri contains space this will fail, so getting the path will work
+ sipStackPropertiesFile = new File(url.getPath());
+ }
+ FileInputStream sipStackPropertiesInputStream = null;
+ try {
+ sipStackPropertiesInputStream = new FileInputStream(sipStackPropertiesFile);
+ sipStackProperties.load(sipStackPropertiesInputStream);
+ } catch (Exception e) {
+ logger.warn("Could not find or problem when loading the sip stack properties file : " + sipStackPropertiesFileLocation, e);
+ } finally {
+ if(sipStackPropertiesInputStream != null) {
+ try {
+ sipStackPropertiesInputStream.close();
+ } catch (IOException e) {
+ logger.error("fail to close the following file " + sipStackPropertiesFile.getAbsolutePath(), e);
+ }
+ }
+ }
+
+ String debugLog = sipStackProperties.getProperty(DEBUG_LOG_STACK_PROP);
+ if(debugLog != null && debugLog.length() > 0 && !debugLog.startsWith("file:///")) {
+ sipStackProperties.setProperty(DEBUG_LOG_STACK_PROP,
+ catalinaBase + "/" + debugLog);
+ }
+ String serverLog = sipStackProperties.getProperty(SERVER_LOG_STACK_PROP);
+ if(serverLog != null && serverLog.length() > 0 && !serverLog.startsWith("file:///")) {
+ sipStackProperties.setProperty(SERVER_LOG_STACK_PROP,
+ catalinaBase + "/" + serverLog);
+ }
+ // The whole MSS is built upon those assumptions, so those properties are not overrideable
+ if (sipStackProperties.getProperty(AUTOMATIC_DIALOG_SUPPORT_STACK_PROP) == null) {
+ //https://github.com/RestComm/sip-servlets/issues/143
+ //set off if user didnt provided any value.
+ sipStackProperties.setProperty(AUTOMATIC_DIALOG_SUPPORT_STACK_PROP, "off");
+ }
+ sipStackProperties.setProperty(LOOSE_DIALOG_VALIDATION, "true");
+ sipStackProperties.setProperty(PASS_INVITE_NON_2XX_ACK_TO_LISTENER, "true");
+ isPropsLoaded = true;
+ } else {
+ logger.warn("no sip stack properties file defined ");
+ }
+ if(!isPropsLoaded) {
+ logger.warn("loading default Mobicents Sip Servlets sip stack properties");
+ // Silently set default values
+ sipStackProperties.setProperty("gov.nist.javax.sip.LOG_MESSAGE_CONTENT",
+ "true");
+ sipStackProperties.setProperty("gov.nist.javax.sip.TRACE_LEVEL",
+ "LOG4J");
+ sipStackProperties.setProperty(DEBUG_LOG_STACK_PROP,
+ catalinaBase + "/" + "mss-jsip-" + getName() +"-debug.txt");
+ sipStackProperties.setProperty(SERVER_LOG_STACK_PROP,
+ catalinaBase + "/" + "mss-jsip-" + getName() +"-messages.xml");
+ sipStackProperties.setProperty("javax.sip.STACK_NAME", "mss-" + getName());
+ sipStackProperties.setProperty(AUTOMATIC_DIALOG_SUPPORT_STACK_PROP, "off");
+ sipStackProperties.setProperty("gov.nist.javax.sip.DELIVER_UNSOLICITED_NOTIFY", "true");
+ sipStackProperties.setProperty("gov.nist.javax.sip.THREAD_POOL_SIZE", "64");
+ sipStackProperties.setProperty("gov.nist.javax.sip.REENTRANT_LISTENER", "true");
+ sipStackProperties.setProperty("gov.nist.javax.sip.MAX_FORK_TIME_SECONDS", "0");
+ sipStackProperties.setProperty(LOOSE_DIALOG_VALIDATION, "true");
+ sipStackProperties.setProperty(PASS_INVITE_NON_2XX_ACK_TO_LISTENER, "true");
+ sipStackProperties.setProperty("gov.nist.javax.sip.AUTOMATIC_DIALOG_ERROR_HANDLING", "false");
+ sipStackProperties.setProperty("gov.nist.javax.sip.AGGRESSIVE_CLEANUP", "true");
+ }
+
+ if(sipStackProperties.get(TCP_POST_PARSING_THREAD_POOL_SIZE) == null) {
+ sipStackProperties.setProperty(TCP_POST_PARSING_THREAD_POOL_SIZE, "30");
+ }
+
+ // set the DNSServerLocator allowing to support RFC 3263 and do DNS lookups to resolve uris
+ if(dnsServerLocatorClass != null && dnsServerLocatorClass.trim().length() > 0) {
+ if(logger.isDebugEnabled()) {
+ logger.debug("Sip Stack " + sipStackProperties.getProperty("javax.sip.STACK_NAME") +" will be using " + dnsServerLocatorClass + " as DNSServerLocator");
+ }
+ try {
+ // create parameters argument to identify constructor
+ Class[] paramTypes = new Class[0];
+ // get constructor of AddressResolver in order to instantiate
+ Constructor dnsServerLocatorConstructor = Class.forName(dnsServerLocatorClass).getConstructor(
+ paramTypes);
+ // Wrap properties object in order to pass to constructor of AddressResolver
+ Object[] conArgs = new Object[0];
+ // Creates a new instance of AddressResolver Class with the supplied sipApplicationDispatcher.
+ DNSServerLocator dnsServerLocator = (DNSServerLocator) dnsServerLocatorConstructor.newInstance(conArgs);
+ sipApplicationDispatcher.setDNSServerLocator(dnsServerLocator);
+ sipApplicationDispatcher.setDNSTimeout(dnsTimeout);
+ if(sipStackProperties.getProperty("javax.sip.ROUTER_PATH") == null) {
+ sipStackProperties.setProperty("javax.sip.ROUTER_PATH", DNSAwareRouter.class.getCanonicalName());
+ }
+ } catch (Exception e) {
+ logger.error("Couldn't set the AddressResolver " + addressResolverClass, e);
+ throw e;
+ }
+ } else {
+ if(logger.isInfoEnabled()) {
+ logger.info("no DNSServerLocator will be used since none has been specified.");
+ }
+ }
+
+ String serverHeaderValue = sipStackProperties.getProperty(SERVER_HEADER);
+ if(serverHeaderValue != null) {
+ ListSipStandardContext
container implementation.
- *
- * USAGE CONSTRAINT: This implementation is likely to be useful only
- * when processing HTTP requests.
- *
- * The code was copy pasted from tomcat 5.5.20 source and some specific code was added on invoke
- * method to store the htpp session in the threadlocal of the sipfactoryfacade and check if the request
- * has an application key associated with it
- *
- * @author Jean Deruelle
- *
- */
-final class SipStandardContextValve extends org.apache.catalina.valves.ValveBase {
-
- // ----------------------------------------------------- Instance Variables
-
-
- /**
- * The descriptive information related to this implementation.
- */
- private static final String info =
- "org.apache.catalina.core.StandardContextValve/1.0";
-
-
- /**
- * The string manager for this package (not supported in AS-7.2.0)
- */
-// private static final StringManager sm =
-// StringManager.getManager(Constants.Package);
-
-
- private static transient final Logger logger = Logger.getLogger(SipStandardContextValve.class);
-
-
- private SipStandardContext context = null;
-
-
- // ------------------------------------------------------------- Properties
-
-
- /**
- * Return descriptive information about this Valve implementation.
- */
- public String getInfo() {
-
- return (info);
-
- }
-
-
- // --------------------------------------------------------- Public Methods
-
-
- /**
- * Cast to a StandardContext right away, as it will be needed later.
- *
- * @see org.apache.catalina.Contained#setContainer(org.apache.catalina.Container)
- */
- public void setContainer(Container container) {
- super.setContainer(container);
- context = (SipStandardContext) container;
- }
-
-
- /**
- * Select the appropriate child Wrapper to process this request,
- * based on the specified request URI. If no matching Wrapper can
- * be found, return an appropriate HTTP error.
- *
- * @param request Request to be processed
- * @param response Response to be produced
- * @param valveContext Valve context used to forward to the next Valve
- *
- * @exception IOException if an input/output error occurred
- * @exception ServletException if a servlet error occurred
- */
- public final void invoke(Request request, Response response)
- throws IOException, ServletException {
-
- // Disallow any direct access to resources under WEB-INF or META-INF
- MessageBytes requestPathMB = request.getRequestPathMB();
- if ((requestPathMB.startsWithIgnoreCase("/META-INF/", 0))
- || (requestPathMB.equalsIgnoreCase("/META-INF"))
- || (requestPathMB.startsWithIgnoreCase("/WEB-INF/", 0))
- || (requestPathMB.equalsIgnoreCase("/WEB-INF"))) {
- notFound(response);
- return;
- }
-
- // Wait if we are reloading
- boolean reloaded = false;
- while (context.getPaused()) {
- reloaded = true;
- try {
- Thread.sleep(1000);
- } catch (InterruptedException e) {
- ;
- }
- }
-
- // Reloading will have stopped the old webappclassloader and
- // created a new one
- if (reloaded &&
- context.getLoader() != null &&
- context.getLoader().getClassLoader() != null) {
- Thread.currentThread().setContextClassLoader(
- context.getLoader().getClassLoader());
- }
-
- // Select the Wrapper to be used for this Request
- Wrapper wrapper = request.getWrapper();
- if (wrapper == null) {
- notFound(response);
- return;
- } else if (wrapper.isUnavailable()) {
- // May be as a result of a reload, try and find the new wrapper
- wrapper = (Wrapper) container.findChild(wrapper.getName());
- if (wrapper == null) {
- notFound(response);
- return;
- }
- }
-
- // Normal request processing
- Object instances[] = context.getApplicationEventListeners();
-
- ServletRequestEvent event = null;
-
- if ((instances != null)
- && (instances.length > 0)) {
- event = new ServletRequestEvent
- (((StandardContext) container).getServletContext(),
- request.getRequest());
- // create pre-service event
- for (int i = 0; i < instances.length; i++) {
- if (instances[i] == null)
- continue;
- if (!(instances[i] instanceof ServletRequestListener))
- continue;
- ServletRequestListener listener =
- (ServletRequestListener) instances[i];
- try {
- listener.requestInitialized(event);
- } catch (Throwable t) {
- container.getLogger().error(MESSAGES.requestListenerInitException(instances[i].getClass().getName()), t);
- ServletRequest sreq = request.getRequest();
- sreq.setAttribute(RequestDispatcher.ERROR_EXCEPTION,t);
- return;
- }
- }
- }
- boolean batchStarted = context.enterSipAppHa(false);
-
- //the line below was replaced by the whole bunch of code because getting the parameter from the request is causing
- //JRuby-Rails persistence to fail, go figure...
-// String sipApplicationKey = request.getParameter(MobicentsSipApplicationSession.SIP_APPLICATION_KEY_PARAM_NAME);
-
- String sipApplicationKey = null;
- String queryString = request.getQueryString();
- if(queryString != null) {
- int indexOfSipAppKey = queryString.indexOf(MobicentsSipApplicationSession.SIP_APPLICATION_KEY_PARAM_NAME);
-
- if(indexOfSipAppKey != -1) {
- // +1 to remove the = sign also
- String sipAppKeyParam = queryString.substring(indexOfSipAppKey + MobicentsSipApplicationSession.SIP_APPLICATION_KEY_PARAM_NAME.length() + 1);
- int indexOfPoundSign = sipAppKeyParam.indexOf("&");
- if(indexOfPoundSign != -1) {
- sipAppKeyParam = sipAppKeyParam.substring(0, indexOfPoundSign);
- }
- sipApplicationKey = sipAppKeyParam;
- }
- }
-
- MobicentsSipApplicationSession sipApplicationSessionImpl = null;
- if(sipApplicationKey != null && sipApplicationKey.length() > 0) {
- try {
- SipApplicationSessionKey sipApplicationSessionKey =
- SessionManagerUtil.parseSipApplicationSessionKey(sipApplicationKey);
- sipApplicationSessionImpl =
- ((SipManager)context.getManager()).getSipApplicationSession(sipApplicationSessionKey, false);
- sipApplicationSessionImpl.addHttpSession(request.getSession());
- } catch (ParseException pe) {
- logger.error("Unexpected exception while parsing the sip application session key" + sipApplicationKey, pe);
- }
- } else {
- // Fix for Issue 882 : HTTP requests to a SIP application always create an HTTP session, even for static resources
- // Don't create an http session if not already created
- final HttpSession httpSession = request.getSession(false);
- if(httpSession != null) {
- context.getSipFactoryFacade().storeHttpSession(httpSession);
- ConvergedSession convergedSession = (ConvergedSession) httpSession;
- sipApplicationSessionImpl = convergedSession.getApplicationSession(false);
- }
- }
- // Fix for http://code.google.com/p/mobicents/issues/detail?id=1386 :
- // Ensure SipApplicationSession concurrency control on converged HTTP apps
- context.enterSipApp(sipApplicationSessionImpl, null, false, true);
- try {
- wrapper.getPipeline().getFirst().invoke(request, response);
- } finally {
- context.exitSipApp(sipApplicationSessionImpl, null);
- context.exitSipAppHa(null, null, batchStarted);
- }
- // Fix for Issue 882 :remove the http session from the thread local to avoid any leaking of the session
- context.getSipFactoryFacade().removeHttpSession();
-
- if ((instances !=null ) &&
- (instances.length > 0)) {
- // create post-service event
- for (int i = 0; i < instances.length; i++) {
- if (instances[i] == null)
- continue;
- if (!(instances[i] instanceof ServletRequestListener))
- continue;
- ServletRequestListener listener =
- (ServletRequestListener) instances[i];
- try {
- listener.requestDestroyed(event);
- } catch (Throwable t) {
- container.getLogger().error(MESSAGES.requestListenerDestroyException(instances[i].getClass().getName()), t);
- ServletRequest sreq = request.getRequest();
- sreq.setAttribute(RequestDispatcher.ERROR_EXCEPTION,t);
- }
- }
- }
-
- }
-
- /**
- * Select the appropriate child Wrapper to process this request,
- * based on the specified request URI. If no matching Wrapper can
- * be found, return an appropriate HTTP error.
- *
- * @param request Request to be processed
- * @param response Response to be produced
- * @param valveContext Valve context used to forward to the next Valve
- *
- * @exception IOException if an input/output error occurred
- * @exception ServletException if a servlet error occurred
- */
- public final void event(Request request, Response response, HttpEvent event)
- throws IOException, ServletException {
-
- // Select the Wrapper to be used for this Request
- Wrapper wrapper = request.getWrapper();
-
- // Normal request processing
- // FIXME: This could be an addition to the core API too
- /*
- Object instances[] = context.getApplicationEventListeners();
-
- ServletRequestEvent event = null;
-
- if ((instances != null)
- && (instances.length > 0)) {
- event = new ServletRequestEvent
- (((StandardContext) container).getServletContext(),
- request.getRequest());
- // create pre-service event
- for (int i = 0; i < instances.length; i++) {
- if (instances[i] == null)
- continue;
- if (!(instances[i] instanceof ServletRequestListener))
- continue;
- ServletRequestListener listener =
- (ServletRequestListener) instances[i];
- try {
- listener.requestInitialized(event);
- } catch (Throwable t) {
- container.getLogger().error(sm.getString("requestListenerValve.requestInit",
- instances[i].getClass().getName()), t);
- ServletRequest sreq = request.getRequest();
- sreq.setAttribute(Globals.EXCEPTION_ATTR,t);
- return;
- }
- }
- }
- */
-
- wrapper.getPipeline().getFirst().event(request, response, event);
-
- /*
- if ((instances !=null ) &&
- (instances.length > 0)) {
- // create post-service event
- for (int i = 0; i < instances.length; i++) {
- if (instances[i] == null)
- continue;
- if (!(instances[i] instanceof ServletRequestListener))
- continue;
- ServletRequestListener listener =
- (ServletRequestListener) instances[i];
- try {
- listener.requestDestroyed(event);
- } catch (Throwable t) {
- container.getLogger().error(sm.getString("requestListenerValve.requestDestroy",
- instances[i].getClass().getName()), t);
- ServletRequest sreq = request.getRequest();
- sreq.setAttribute(Globals.EXCEPTION_ATTR,t);
- }
- }
- }
- */
-
- }
-
- // -------------------------------------------------------- Private Methods
-
-
-
- /**
- * Report a "not found" error for the specified resource. FIXME: We
- * should really be using the error reporting settings for this web
- * application, but currently that code runs at the wrapper level rather
- * than the context level.
- *
- * @param response The response we are creating
- */
- protected void notFound(HttpServletResponse response) {
-
- try {
- response.sendError(HttpServletResponse.SC_NOT_FOUND);
- } catch (IllegalStateException e) {
- ;
- } catch (IOException e) {
- ;
- }
-
- }
-}
+/*
+ * TeleStax, Open Source Cloud Communications
+ * Copyright 2011-2014, Telestax Inc and individual contributors
+ * by the @authors tag.
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * under the terms of the GNU Affero General Public License as
+ * published by the Free Software Foundation; either version 3 of
+ * the License, or (at your option) any later version.
+ *
+ * This program 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 Affero General Public License for more details.
+ *
+ * You should have received a copy of the GNU Affero General Public License
+ * along with this program. If not, see SipStandardContext
container implementation.
+ *
+ * USAGE CONSTRAINT: This implementation is likely to be useful only
+ * when processing HTTP requests.
+ *
+ * The code was copy pasted from tomcat 5.5.20 source and some specific code was added on invoke
+ * method to store the htpp session in the threadlocal of the sipfactoryfacade and check if the request
+ * has an application key associated with it
+ *
+ * @author Jean Deruelle
+ *
+ */
+final class SipStandardContextValve extends org.apache.catalina.valves.ValveBase {
+
+ // ----------------------------------------------------- Instance Variables
+
+
+ /**
+ * The descriptive information related to this implementation.
+ */
+ private static final String info =
+ "org.apache.catalina.core.StandardContextValve/1.0";
+
+
+ /**
+ * The string manager for this package (not supported in AS-7.2.0)
+ */
+// private static final StringManager sm =
+// StringManager.getManager(Constants.Package);
+
+
+ private static transient final Logger logger = Logger.getLogger(SipStandardContextValve.class);
+
+
+ private SipStandardContext context = null;
+
+
+ // ------------------------------------------------------------- Properties
+
+
+ /**
+ * Return descriptive information about this Valve implementation.
+ */
+ public String getInfo() {
+
+ return (info);
+
+ }
+
+
+ // --------------------------------------------------------- Public Methods
+
+
+ /**
+ * Cast to a StandardContext right away, as it will be needed later.
+ *
+ * @see org.apache.catalina.Contained#setContainer(org.apache.catalina.Container)
+ */
+ public void setContainer(Container container) {
+ super.setContainer(container);
+ context = (SipStandardContext) container;
+ }
+
+
+ /**
+ * Select the appropriate child Wrapper to process this request,
+ * based on the specified request URI. If no matching Wrapper can
+ * be found, return an appropriate HTTP error.
+ *
+ * @param request Request to be processed
+ * @param response Response to be produced
+ * @param valveContext Valve context used to forward to the next Valve
+ *
+ * @exception IOException if an input/output error occurred
+ * @exception ServletException if a servlet error occurred
+ */
+ public final void invoke(Request request, Response response)
+ throws IOException, ServletException {
+
+ // Disallow any direct access to resources under WEB-INF or META-INF
+ MessageBytes requestPathMB = request.getRequestPathMB();
+ if ((requestPathMB.startsWithIgnoreCase("/META-INF/", 0))
+ || (requestPathMB.equalsIgnoreCase("/META-INF"))
+ || (requestPathMB.startsWithIgnoreCase("/WEB-INF/", 0))
+ || (requestPathMB.equalsIgnoreCase("/WEB-INF"))) {
+ notFound(response);
+ return;
+ }
+
+ // Wait if we are reloading
+ boolean reloaded = false;
+ while (context.getPaused()) {
+ reloaded = true;
+ try {
+ Thread.sleep(1000);
+ } catch (InterruptedException e) {
+ ;
+ }
+ }
+
+ // Reloading will have stopped the old webappclassloader and
+ // created a new one
+ if (reloaded &&
+ context.getLoader() != null &&
+ context.getLoader().getClassLoader() != null) {
+ Thread.currentThread().setContextClassLoader(
+ context.getLoader().getClassLoader());
+ }
+
+ // Select the Wrapper to be used for this Request
+ Wrapper wrapper = request.getWrapper();
+ if (wrapper == null) {
+ notFound(response);
+ return;
+ } else if (wrapper.isUnavailable()) {
+ // May be as a result of a reload, try and find the new wrapper
+ wrapper = (Wrapper) container.findChild(wrapper.getName());
+ if (wrapper == null) {
+ notFound(response);
+ return;
+ }
+ }
+
+ // Acknowledge the request
+ try {
+ response.sendAcknowledgement();
+ } catch (IOException ioe) {
+ container.getLogger().error(MESSAGES.errorAcknowledgingRequest("standardContextValve.acknowledgeException"), ioe);
+ request.setAttribute(RequestDispatcher.ERROR_EXCEPTION, ioe);
+ response.sendError(HttpServletResponse.SC_INTERNAL_SERVER_ERROR);
+ return;
+ }
+
+ // Normal request processing
+ Object instances[] = context.getApplicationEventListeners();
+
+ ServletRequestEvent event = null;
+
+ if ((instances != null)
+ && (instances.length > 0)) {
+ event = new ServletRequestEvent
+ (((StandardContext) container).getServletContext(),
+ request.getRequest());
+ // create pre-service event
+ for (int i = 0; i < instances.length; i++) {
+ if (instances[i] == null)
+ continue;
+ if (!(instances[i] instanceof ServletRequestListener))
+ continue;
+ ServletRequestListener listener =
+ (ServletRequestListener) instances[i];
+ try {
+ listener.requestInitialized(event);
+ } catch (Throwable t) {
+ container.getLogger().error(MESSAGES.requestListenerInitException(instances[i].getClass().getName()), t);
+ ServletRequest sreq = request.getRequest();
+ sreq.setAttribute(RequestDispatcher.ERROR_EXCEPTION,t);
+ return;
+ }
+ }
+ }
+ boolean batchStarted = context.enterSipAppHa(false);
+
+ //the line below was replaced by the whole bunch of code because getting the parameter from the request is causing
+ //JRuby-Rails persistence to fail, go figure...
+// String sipApplicationKey = request.getParameter(MobicentsSipApplicationSession.SIP_APPLICATION_KEY_PARAM_NAME);
+
+ String sipApplicationKey = null;
+ String queryString = request.getQueryString();
+ if(queryString != null) {
+ int indexOfSipAppKey = queryString.indexOf(MobicentsSipApplicationSession.SIP_APPLICATION_KEY_PARAM_NAME);
+
+ if(indexOfSipAppKey != -1) {
+ // +1 to remove the = sign also
+ String sipAppKeyParam = queryString.substring(indexOfSipAppKey + MobicentsSipApplicationSession.SIP_APPLICATION_KEY_PARAM_NAME.length() + 1);
+ int indexOfPoundSign = sipAppKeyParam.indexOf("&");
+ if(indexOfPoundSign != -1) {
+ sipAppKeyParam = sipAppKeyParam.substring(0, indexOfPoundSign);
+ }
+ sipApplicationKey = sipAppKeyParam;
+ }
+ }
+
+ MobicentsSipApplicationSession sipApplicationSessionImpl = null;
+ if(sipApplicationKey != null && sipApplicationKey.length() > 0) {
+ try {
+ SipApplicationSessionKey sipApplicationSessionKey =
+ SessionManagerUtil.parseSipApplicationSessionKey(sipApplicationKey);
+ sipApplicationSessionImpl =
+ ((SipManager)context.getManager()).getSipApplicationSession(sipApplicationSessionKey, false);
+ sipApplicationSessionImpl.addHttpSession(request.getSession());
+ } catch (ParseException pe) {
+ logger.error("Unexpected exception while parsing the sip application session key" + sipApplicationKey, pe);
+ }
+ } else {
+ // Fix for Issue 882 : HTTP requests to a SIP application always create an HTTP session, even for static resources
+ // Don't create an http session if not already created
+ final HttpSession httpSession = request.getSession(false);
+ if(httpSession != null) {
+ context.getSipFactoryFacade().storeHttpSession(httpSession);
+ ConvergedSession convergedSession = (ConvergedSession) httpSession;
+ sipApplicationSessionImpl = convergedSession.getApplicationSession(false);
+ }
+ }
+ // Fix for http://code.google.com/p/mobicents/issues/detail?id=1386 :
+ // Ensure SipApplicationSession concurrency control on converged HTTP apps
+ context.enterSipApp(sipApplicationSessionImpl, null, false, true);
+ try {
+ wrapper.getPipeline().getFirst().invoke(request, response);
+ } finally {
+ context.exitSipApp(sipApplicationSessionImpl, null);
+ context.exitSipAppHa(null, null, batchStarted);
+ }
+ // Fix for Issue 882 :remove the http session from the thread local to avoid any leaking of the session
+ context.getSipFactoryFacade().removeHttpSession();
+
+ if ((instances !=null ) &&
+ (instances.length > 0)) {
+ // create post-service event
+ for (int i = 0; i < instances.length; i++) {
+ if (instances[i] == null)
+ continue;
+ if (!(instances[i] instanceof ServletRequestListener))
+ continue;
+ ServletRequestListener listener =
+ (ServletRequestListener) instances[i];
+ try {
+ listener.requestDestroyed(event);
+ } catch (Throwable t) {
+ container.getLogger().error(MESSAGES.requestListenerDestroyException(instances[i].getClass().getName()), t);
+ ServletRequest sreq = request.getRequest();
+ sreq.setAttribute(RequestDispatcher.ERROR_EXCEPTION,t);
+ }
+ }
+ }
+
+ }
+
+ /**
+ * Select the appropriate child Wrapper to process this request,
+ * based on the specified request URI. If no matching Wrapper can
+ * be found, return an appropriate HTTP error.
+ *
+ * @param request Request to be processed
+ * @param response Response to be produced
+ * @param valveContext Valve context used to forward to the next Valve
+ *
+ * @exception IOException if an input/output error occurred
+ * @exception ServletException if a servlet error occurred
+ */
+ public final void event(Request request, Response response, HttpEvent event)
+ throws IOException, ServletException {
+
+ // Select the Wrapper to be used for this Request
+ Wrapper wrapper = request.getWrapper();
+
+ // Normal request processing
+ // FIXME: This could be an addition to the core API too
+ /*
+ Object instances[] = context.getApplicationEventListeners();
+
+ ServletRequestEvent event = null;
+
+ if ((instances != null)
+ && (instances.length > 0)) {
+ event = new ServletRequestEvent
+ (((StandardContext) container).getServletContext(),
+ request.getRequest());
+ // create pre-service event
+ for (int i = 0; i < instances.length; i++) {
+ if (instances[i] == null)
+ continue;
+ if (!(instances[i] instanceof ServletRequestListener))
+ continue;
+ ServletRequestListener listener =
+ (ServletRequestListener) instances[i];
+ try {
+ listener.requestInitialized(event);
+ } catch (Throwable t) {
+ container.getLogger().error(sm.getString("requestListenerValve.requestInit",
+ instances[i].getClass().getName()), t);
+ ServletRequest sreq = request.getRequest();
+ sreq.setAttribute(Globals.EXCEPTION_ATTR,t);
+ return;
+ }
+ }
+ }
+ */
+
+ wrapper.getPipeline().getFirst().event(request, response, event);
+
+ /*
+ if ((instances !=null ) &&
+ (instances.length > 0)) {
+ // create post-service event
+ for (int i = 0; i < instances.length; i++) {
+ if (instances[i] == null)
+ continue;
+ if (!(instances[i] instanceof ServletRequestListener))
+ continue;
+ ServletRequestListener listener =
+ (ServletRequestListener) instances[i];
+ try {
+ listener.requestDestroyed(event);
+ } catch (Throwable t) {
+ container.getLogger().error(sm.getString("requestListenerValve.requestDestroy",
+ instances[i].getClass().getName()), t);
+ ServletRequest sreq = request.getRequest();
+ sreq.setAttribute(Globals.EXCEPTION_ATTR,t);
+ }
+ }
+ }
+ */
+
+ }
+
+ // -------------------------------------------------------- Private Methods
+
+
+
+ /**
+ * Report a "not found" error for the specified resource. FIXME: We
+ * should really be using the error reporting settings for this web
+ * application, but currently that code runs at the wrapper level rather
+ * than the context level.
+ *
+ * @param response The response we are creating
+ */
+ protected void notFound(HttpServletResponse response) {
+
+ try {
+ response.sendError(HttpServletResponse.SC_NOT_FOUND);
+ } catch (IllegalStateException e) {
+ ;
+ } catch (IOException e) {
+ ;
+ }
+
+ }
+}
diff --git a/containers/sip-servlets-as8-drop-in/build-restcomm-modules/build.xml b/containers/sip-servlets-as8-drop-in/build-restcomm-modules/build.xml
index 32920fed3c..bf8140c49a 100644
--- a/containers/sip-servlets-as8-drop-in/build-restcomm-modules/build.xml
+++ b/containers/sip-servlets-as8-drop-in/build-restcomm-modules/build.xml
@@ -63,9 +63,17 @@
SipService
interface.
- * This class inherits from the Tomcat StandardService. It adds a SipApplicationDispatcher
- * that will be listen for sip messages received by the sip stacks started by
- * the sip connectors associated with this context.
- * This has one attribute which is the sipApplicationDispatcherClassName allowing one
- * to specify the class name of the sipApplicationDispacther to easily replace
- * the default sipApplicationDispatcher with a custom one.
- *
- * @author Jean Deruelle
- */
-public class SipStandardService extends StandardService implements CatalinaSipService {
- //the logger
- private static final Logger logger = Logger.getLogger(SipStandardService.class);
- public static final String DEFAULT_SIP_PATH_NAME = "gov.nist";
- public static final String PASS_INVITE_NON_2XX_ACK_TO_LISTENER = "gov.nist.javax.sip.PASS_INVITE_NON_2XX_ACK_TO_LISTENER";
- public static final String TCP_POST_PARSING_THREAD_POOL_SIZE = "gov.nist.javax.sip.TCP_POST_PARSING_THREAD_POOL_SIZE";
- public static final String AUTOMATIC_DIALOG_SUPPORT_STACK_PROP = "javax.sip.AUTOMATIC_DIALOG_SUPPORT";
- public static final String LOOSE_DIALOG_VALIDATION = "gov.nist.javax.sip.LOOSE_DIALOG_VALIDATION";
- public static final String SERVER_LOG_STACK_PROP = "gov.nist.javax.sip.SERVER_LOG";
- public static final String DEBUG_LOG_STACK_PROP = "gov.nist.javax.sip.DEBUG_LOG";
- public static final String SERVER_HEADER = "org.mobicents.servlet.sip.SERVER_HEADER";
- public static final String USER_AGENT_HEADER = "org.mobicents.servlet.sip.USER_AGENT_HEADER";
- public static final String JVM_ROUTE = "jvmRoute";
- /**
- * The descriptive information string for this implementation.
- */
- private static final String INFO =
- "org.mobicents.servlet.sip.startup.SipStandardService/1.0";
- //the sip application dispatcher class name defined in the server.xml
- protected String sipApplicationDispatcherClassName;
- //instatiated class from the sipApplicationDispatcherClassName of the sip application dispatcher
- protected SipApplicationDispatcher sipApplicationDispatcher;
- private boolean gatherStatistics = true;
- protected int sipMessageQueueSize = 1500;
- private int backToNormalSipMessageQueueSize = 1300;
- protected int memoryThreshold = 95;
- private int backToNormalMemoryThreshold = 90;
- protected OutboundProxy outboundProxy;
- protected String proxyTimerServiceImplementationType;
- protected String sasTimerServiceImplementationType;
- protected long congestionControlCheckingInterval = 30000;
- private int canceledTimerTasksPurgePeriod = 0;
- // base timer interval for jain sip tx
- private int baseTimerInterval = 500;
- private int t2Interval = 4000;
- private int t4Interval = 5000;
- private int timerDInterval = 32000;
- protected int dispatcherThreadPoolSize = 15;
- private boolean md5ContactUserPart = false;
-
- protected String concurrencyControlMode = ConcurrencyControlMode.SipApplicationSession.toString();
- protected String congestionControlPolicy = CongestionControlPolicy.ErrorResponse.toString();
- protected String additionalParameterableHeaders;
- protected boolean bypassResponseExecutor = true;
- protected boolean bypassRequestExecutor = true;
- //this should be made available to the application router as a system prop
- protected String darConfigurationFileLocation;
- protected boolean connectorsStartedExternally = false;
- protected boolean dialogPendingRequestChecking = false;
- protected int callIdMaxLength;
- protected int tagHashMaxLength;
-
- protected boolean httpFollowsSip = false;
- protected String jvmRoute;
- protected ReplicationStrategy replicationStrategy;
-
- /**
- * the sip stack path name. Since the sip factory is per classloader it should be set here for all underlying stacks
- */
- private String sipPathName;
- /**
- * use Pretty Encoding
- */
- private boolean usePrettyEncoding = true;
-
- private SipStack sipStack;
- // defining sip stack properties
- private Properties sipStackProperties;
- private String sipStackPropertiesFileLocation;
- @Deprecated
- private String addressResolverClass = null;
- private String dnsServerLocatorClass = DefaultDNSServerLocator.class.getName();
- private int dnsTimeout = 1;
- private String dnsResolverClass = MobicentsDNSResolver.class.getName();
- private String mobicentsSipServletMessageFactoryClassName = Servlet3SipServletMessageFactory.class.getName();
-
- //the balancers to send heartbeat to and our health info
- @Deprecated
- private String balancers;
- //
- private ScheduledFuture> gracefulStopFuture;
-
- @Override
- public String getInfo() {
- return (INFO);
- }
-
- @Override
- public void addConnector(Connector connector) {
- MobicentsExtendedListeningPoint extendedListeningPoint = null;
- if (connector.getProtocolHandler() instanceof SipProtocolHandler) {
- extendedListeningPoint = (MobicentsExtendedListeningPoint)
- ((SipProtocolHandler)connector.getProtocolHandler()).getAttribute(ExtendedListeningPoint.class.getSimpleName());}
- if(extendedListeningPoint != null) {
- try {
- extendedListeningPoint.getSipProvider().addSipListener(sipApplicationDispatcher);
- sipApplicationDispatcher.getSipNetworkInterfaceManager().addExtendedListeningPoint(extendedListeningPoint);
- } catch (TooManyListenersException e) {
- logger.error("Connector.initialize", e);
- }
- }
- ProtocolHandler protocolHandler = connector.getProtocolHandler();
- if(protocolHandler instanceof SipProtocolHandler) {
- connector.setPort(((SipProtocolHandler)protocolHandler).getPort());
- ((SipProtocolHandler)protocolHandler).setSipStack(sipStack);
- ((SipProtocolHandler)protocolHandler).setAttribute(SipApplicationDispatcher.class.getSimpleName(), sipApplicationDispatcher);
- registerSipConnector(connector);
- }
- super.addConnector(connector);
- }
-
- /**
- * Register the sip connector under a different name than HTTP Connector and we add the transport to avoid clashing with 2 connectors having the same port and address
- * @param connector connector to register
- */
- protected void registerSipConnector(Connector connector) {
- try {
-
- ObjectName objectName = createSipConnectorObjectName(connector, getName(), "SipConnector");
- Registry.getRegistry(null, null)
- .registerComponent(connector, objectName, null);
-//TODO connector.setController(objectName);
- } catch (Exception e) {
- logger.error( "Error registering connector ", e);
- }
- if(logger.isDebugEnabled())
- logger.debug("Creating name for connector " + getObjectName());
- }
-
- @Override
- public void removeConnector(Connector connector) {
- MobicentsExtendedListeningPoint extendedListeningPoint = null;
- if (connector.getProtocolHandler() instanceof SipProtocolHandler){
- extendedListeningPoint = (MobicentsExtendedListeningPoint)
- ((SipProtocolHandler)connector.getProtocolHandler()).getAttribute(ExtendedListeningPoint.class.getSimpleName());}
- if(extendedListeningPoint != null) {
- extendedListeningPoint.getSipProvider().removeSipListener(sipApplicationDispatcher);
- sipApplicationDispatcher.getSipNetworkInterfaceManager().removeExtendedListeningPoint(extendedListeningPoint);
- }
- super.removeConnector(connector);
- }
-
- @Override
- public void initInternal() throws LifecycleException {
- //load the sip application disptacher from the class name specified in the server.xml file
- //and initializes it
- StaticServiceHolder.sipStandardService = this;
- try {
- sipApplicationDispatcher = (SipApplicationDispatcher)
- Class.forName(sipApplicationDispatcherClassName).newInstance();
- } catch (InstantiationException e) {
- throw new LifecycleException("Impossible to load the Sip Application Dispatcher",e);
- } catch (IllegalAccessException e) {
- throw new LifecycleException("Impossible to load the Sip Application Dispatcher",e);
- } catch (ClassNotFoundException e) {
- throw new LifecycleException("Impossible to load the Sip Application Dispatcher",e);
- } catch (ClassCastException e) {
- throw new LifecycleException("Sip Application Dispatcher defined does not implement " + SipApplicationDispatcher.class.getName(),e);
- }
- if(logger.isInfoEnabled()) {
- logger.info("Pretty encoding of headers enabled ? " + usePrettyEncoding);
- }
- if(sipPathName == null) {
- sipPathName = DEFAULT_SIP_PATH_NAME;
- }
- if(logger.isInfoEnabled()) {
- logger.info("Sip Stack path name : " + sipPathName);
- }
- sipApplicationDispatcher.setSipService(this);
- sipApplicationDispatcher.getSipFactory().initialize(sipPathName, usePrettyEncoding);
-
- String catalinaBase = getCatalinaBase();
- if(darConfigurationFileLocation != null) {
- if(!darConfigurationFileLocation.startsWith("file:///")) {
- darConfigurationFileLocation = "file:///" + catalinaBase.replace(File.separatorChar, '/') + "/" + darConfigurationFileLocation;
- }
- System.setProperty("javax.servlet.sip.dar", darConfigurationFileLocation);
- }
- super.initInternal();
- sipApplicationDispatcher.setDomain(this.getName());
- if(baseTimerInterval < 1) {
- throw new LifecycleException("It's forbidden to set the Base Timer Interval to a non positive value");
- }
- initSipStack();
- // https://telestax.atlassian.net/browse/MSS-148 make sure we have a default to bin direction for better out of the box experience
- if(System.getProperty("telscale.license.dir") == null) {
- System.setProperty("telscale.license.dir", getCatalinaBase() + File.separatorChar + "bin");
- if(logger.isDebugEnabled()) {
- logger.debug("Setting telscale.license.dir directory to : " + getCatalinaBase() + File.separatorChar + "bin");
- }
- }
- if(System.getProperty("telscale.license.key.location") == null) {
- System.setProperty("telscale.license.key.location", getCatalinaBase() + File.separatorChar + "bin");
- if(logger.isDebugEnabled()) {
- logger.debug("Setting telscale.license.key.location directory to : " + getCatalinaBase() + File.separatorChar + "bin");
- }
- }
-
- sipApplicationDispatcher.setBaseTimerInterval(baseTimerInterval);
- sipApplicationDispatcher.setT2Interval(t2Interval);
- sipApplicationDispatcher.setT4Interval(t4Interval);
- sipApplicationDispatcher.setTimerDInterval(timerDInterval);
- sipApplicationDispatcher.setMemoryThreshold(getMemoryThreshold());
- sipApplicationDispatcher.setBackToNormalMemoryThreshold(backToNormalMemoryThreshold);
- sipApplicationDispatcher.setCongestionControlCheckingInterval(getCongestionControlCheckingInterval());
- sipApplicationDispatcher.setCongestionControlPolicyByName(getCongestionControlPolicy());
- sipApplicationDispatcher.setQueueSize(getSipMessageQueueSize());
- sipApplicationDispatcher.setBackToNormalQueueSize(backToNormalSipMessageQueueSize);
- sipApplicationDispatcher.setGatherStatistics(gatherStatistics);
- sipApplicationDispatcher.setConcurrencyControlMode(ConcurrencyControlMode.valueOf(getConcurrencyControlMode()));
- sipApplicationDispatcher.setBypassRequestExecutor(bypassRequestExecutor);
- sipApplicationDispatcher.setBypassResponseExecutor(bypassResponseExecutor);
- sipApplicationDispatcher.setSipStack(sipStack);
- sipApplicationDispatcher.init();
- // Tomcat specific loading case where the connectors are added even before the service is initialized
- // so we need to set the sip stack before it starts
- synchronized (connectors) {
- for (Connector connector : connectors) {
- ProtocolHandler protocolHandler = connector.getProtocolHandler();
- if(protocolHandler instanceof SipProtocolHandler) {
- connector.setPort(((SipProtocolHandler)protocolHandler).getPort());
- ((SipProtocolHandler)protocolHandler).setSipStack(sipStack);
- ((SipProtocolHandler)protocolHandler).setAttribute(SipApplicationDispatcher.class.getSimpleName(), sipApplicationDispatcher);
- registerSipConnector(connector);
- }
- }
- }
- }
-
- @Override
- public void startInternal() throws LifecycleException {
- super.startInternal();
- synchronized (connectors) {
- for (Connector connector : connectors) {
- final ProtocolHandler protocolHandler = connector.getProtocolHandler();
- //Jboss sepcific loading case
- Boolean isSipConnector = false;
- if (protocolHandler instanceof SipProtocolHandler)
- isSipConnector = (Boolean) ((SipProtocolHandler)protocolHandler).getAttribute(SipProtocolHandler.IS_SIP_CONNECTOR);
- if(isSipConnector != null && isSipConnector) {
- if(logger.isDebugEnabled()) {
- logger.debug("Attaching the sip application dispatcher " +
- "as a sip listener to connector listening on port " +
- connector.getPort());
- }
- ((SipProtocolHandler)protocolHandler).setAttribute(SipApplicationDispatcher.class.getSimpleName(), sipApplicationDispatcher);
- ((SipProtocolHandler)protocolHandler).setSipStack(sipStack);
- connectorsStartedExternally = true;
- }
- //Tomcat specific loading case
- MobicentsExtendedListeningPoint extendedListeningPoint = null;
- if (protocolHandler instanceof SipProtocolHandler) {
- extendedListeningPoint = (MobicentsExtendedListeningPoint)
- ((SipProtocolHandler)protocolHandler).getAttribute(ExtendedListeningPoint.class.getSimpleName());}
- if(extendedListeningPoint != null && sipStack != null) {
- try {
- extendedListeningPoint.getSipProvider().addSipListener(sipApplicationDispatcher);
- sipApplicationDispatcher.getSipNetworkInterfaceManager().addExtendedListeningPoint(extendedListeningPoint);
- connectorsStartedExternally = false;
- } catch (TooManyListenersException e) {
- throw new LifecycleException("Couldn't add the sip application dispatcher "
- + sipApplicationDispatcher + " as a listener to the following listening point provider " + extendedListeningPoint, e);
- }
- }
- }
- }
- if(!connectorsStartedExternally) {
- sipApplicationDispatcher.start();
- }
-
- if(this.getSipMessageQueueSize() <= 0)
- throw new LifecycleException("Message queue size can not be 0 or less");
-
- if(logger.isDebugEnabled()) {
- logger.debug("SIP Standard Service Started.");
- }
- }
-
- public String getJvmRoute() {
- return this.jvmRoute;
- }
-
- public void setJvmRoute(String jvmRoute) {
- this.jvmRoute = jvmRoute;
- }
-
- protected void initSipStack() throws LifecycleException {
- try {
- if(logger.isDebugEnabled()) {
- logger.debug("Initializing SIP stack");
- }
-
- // This simply puts HTTP and SSL port numbers in JVM properties menat to be read by jsip ha when sending heart beats with Node description.
- initializeSystemPortProperties();
-
- String catalinaBase = getCatalinaBase();
- if(sipStackPropertiesFileLocation != null && !sipStackPropertiesFileLocation.startsWith("file:///")) {
- sipStackPropertiesFileLocation = "file:///" + catalinaBase.replace(File.separatorChar, '/') + "/" + sipStackPropertiesFileLocation;
- }
- boolean isPropsLoaded = false;
- if(sipStackProperties == null) {
- sipStackProperties = new Properties();
- } else {
- isPropsLoaded = true;
- }
-
- if (logger.isDebugEnabled()) {
- logger.debug("Loading SIP stack properties from following file : " + sipStackPropertiesFileLocation);
- }
- if(sipStackPropertiesFileLocation != null) {
- //hack to get around space char in path see http://weblogs.java.net/blog/kohsuke/archive/2007/04/how_to_convert.html,
- // we create a URL since it's permissive enough
- File sipStackPropertiesFile = null;
- URL url = null;
- try {
- url = new URL(sipStackPropertiesFileLocation);
- } catch (MalformedURLException e) {
- logger.fatal("Cannot find the sip stack properties file ! ",e);
- throw new IllegalArgumentException("The Default Application Router file Location : "+sipStackPropertiesFileLocation+" is not valid ! ",e);
- }
- try {
- sipStackPropertiesFile = new File(new URI(sipStackPropertiesFileLocation));
- } catch (URISyntaxException e) {
- //if the uri contains space this will fail, so getting the path will work
- sipStackPropertiesFile = new File(url.getPath());
- }
- FileInputStream sipStackPropertiesInputStream = null;
- try {
- sipStackPropertiesInputStream = new FileInputStream(sipStackPropertiesFile);
- sipStackProperties.load(sipStackPropertiesInputStream);
- } catch (Exception e) {
- logger.warn("Could not find or problem when loading the sip stack properties file : " + sipStackPropertiesFileLocation, e);
- } finally {
- if(sipStackPropertiesInputStream != null) {
- try {
- sipStackPropertiesInputStream.close();
- } catch (IOException e) {
- logger.error("fail to close the following file " + sipStackPropertiesFile.getAbsolutePath(), e);
- }
- }
- }
-
- String debugLog = sipStackProperties.getProperty(DEBUG_LOG_STACK_PROP);
- if(debugLog != null && debugLog.length() > 0 && !debugLog.startsWith("file:///")) {
- sipStackProperties.setProperty(DEBUG_LOG_STACK_PROP,
- catalinaBase + "/" + debugLog);
- }
- String serverLog = sipStackProperties.getProperty(SERVER_LOG_STACK_PROP);
- if(serverLog != null && serverLog.length() > 0 && !serverLog.startsWith("file:///")) {
- sipStackProperties.setProperty(SERVER_LOG_STACK_PROP,
- catalinaBase + "/" + serverLog);
- }
- // The whole MSS is built upon those assumptions, so those properties are not overrideable
- if (sipStackProperties.getProperty(AUTOMATIC_DIALOG_SUPPORT_STACK_PROP) == null) {
- //https://github.com/RestComm/sip-servlets/issues/143
- //set off if user didnt provided any value.
- sipStackProperties.setProperty(AUTOMATIC_DIALOG_SUPPORT_STACK_PROP, "off");
- }
- sipStackProperties.setProperty(LOOSE_DIALOG_VALIDATION, "true");
- sipStackProperties.setProperty(PASS_INVITE_NON_2XX_ACK_TO_LISTENER, "true");
- isPropsLoaded = true;
- } else {
- logger.warn("no sip stack properties file defined ");
- }
- if(!isPropsLoaded) {
- logger.warn("loading default Mobicents Sip Servlets sip stack properties");
- // Silently set default values
- sipStackProperties.setProperty("gov.nist.javax.sip.LOG_MESSAGE_CONTENT",
- "true");
- sipStackProperties.setProperty("gov.nist.javax.sip.TRACE_LEVEL",
- "32");
- sipStackProperties.setProperty(DEBUG_LOG_STACK_PROP,
- catalinaBase + "/" + "mss-jsip-" + getName() +"-debug.txt");
- sipStackProperties.setProperty(SERVER_LOG_STACK_PROP,
- catalinaBase + "/" + "mss-jsip-" + getName() +"-messages.xml");
- sipStackProperties.setProperty("javax.sip.STACK_NAME", "mss-" + getName());
- sipStackProperties.setProperty(AUTOMATIC_DIALOG_SUPPORT_STACK_PROP, "off");
- sipStackProperties.setProperty("gov.nist.javax.sip.DELIVER_UNSOLICITED_NOTIFY", "true");
- sipStackProperties.setProperty("gov.nist.javax.sip.THREAD_POOL_SIZE", "64");
- sipStackProperties.setProperty("gov.nist.javax.sip.REENTRANT_LISTENER", "true");
- sipStackProperties.setProperty("gov.nist.javax.sip.MAX_FORK_TIME_SECONDS", "0");
- sipStackProperties.setProperty(LOOSE_DIALOG_VALIDATION, "true");
- sipStackProperties.setProperty(PASS_INVITE_NON_2XX_ACK_TO_LISTENER, "true");
- sipStackProperties.setProperty("gov.nist.javax.sip.AUTOMATIC_DIALOG_ERROR_HANDLING", "false");
- sipStackProperties.setProperty("gov.nist.javax.sip.AGGRESSIVE_CLEANUP", "true");
- }
-
- if(sipStackProperties.get(TCP_POST_PARSING_THREAD_POOL_SIZE) == null) {
- sipStackProperties.setProperty(TCP_POST_PARSING_THREAD_POOL_SIZE, "30");
- }
-
- // set the DNSServerLocator allowing to support RFC 3263 and do DNS lookups to resolve uris
- if(dnsServerLocatorClass != null && dnsServerLocatorClass.trim().length() > 0) {
- if(logger.isDebugEnabled()) {
- logger.debug("Sip Stack " + sipStackProperties.getProperty("javax.sip.STACK_NAME") +" will be using " + dnsServerLocatorClass + " as DNSServerLocator");
- }
- try {
- // create parameters argument to identify constructor
- Class[] paramTypes = new Class[0];
- // get constructor of AddressResolver in order to instantiate
- Constructor dnsServerLocatorConstructor = Class.forName(dnsServerLocatorClass).getConstructor(
- paramTypes);
- // Wrap properties object in order to pass to constructor of AddressResolver
- Object[] conArgs = new Object[0];
- // Creates a new instance of AddressResolver Class with the supplied sipApplicationDispatcher.
- DNSServerLocator dnsServerLocator = (DNSServerLocator) dnsServerLocatorConstructor.newInstance(conArgs);
- sipApplicationDispatcher.setDNSServerLocator(dnsServerLocator);
- sipApplicationDispatcher.setDNSTimeout(dnsTimeout);
- if(sipStackProperties.getProperty("javax.sip.ROUTER_PATH") == null) {
- sipStackProperties.setProperty("javax.sip.ROUTER_PATH", DNSAwareRouter.class.getCanonicalName());
- }
- } catch (Exception e) {
- logger.error("Couldn't set the AddressResolver " + addressResolverClass, e);
- throw e;
- }
- } else {
- if(logger.isInfoEnabled()) {
- logger.info("no DNSServerLocator will be used since none has been specified.");
- }
- }
-
- String serverHeaderValue = sipStackProperties.getProperty(SERVER_HEADER);
- if(serverHeaderValue != null) {
- ListSipService
interface.
+ * This class inherits from the Tomcat StandardService. It adds a SipApplicationDispatcher
+ * that will be listen for sip messages received by the sip stacks started by
+ * the sip connectors associated with this context.
+ * This has one attribute which is the sipApplicationDispatcherClassName allowing one
+ * to specify the class name of the sipApplicationDispacther to easily replace
+ * the default sipApplicationDispatcher with a custom one.
+ *
+ * @author Jean Deruelle
+ */
+public class SipStandardService extends StandardService implements CatalinaSipService {
+ //the logger
+ private static final Logger logger = Logger.getLogger(SipStandardService.class);
+ public static final String DEFAULT_SIP_PATH_NAME = "gov.nist";
+ public static final String PASS_INVITE_NON_2XX_ACK_TO_LISTENER = "gov.nist.javax.sip.PASS_INVITE_NON_2XX_ACK_TO_LISTENER";
+ public static final String TCP_POST_PARSING_THREAD_POOL_SIZE = "gov.nist.javax.sip.TCP_POST_PARSING_THREAD_POOL_SIZE";
+ public static final String AUTOMATIC_DIALOG_SUPPORT_STACK_PROP = "javax.sip.AUTOMATIC_DIALOG_SUPPORT";
+ public static final String LOOSE_DIALOG_VALIDATION = "gov.nist.javax.sip.LOOSE_DIALOG_VALIDATION";
+ public static final String SERVER_LOG_STACK_PROP = "gov.nist.javax.sip.SERVER_LOG";
+ public static final String DEBUG_LOG_STACK_PROP = "gov.nist.javax.sip.DEBUG_LOG";
+ public static final String SERVER_HEADER = "org.mobicents.servlet.sip.SERVER_HEADER";
+ public static final String USER_AGENT_HEADER = "org.mobicents.servlet.sip.USER_AGENT_HEADER";
+ public static final String JVM_ROUTE = "jvmRoute";
+ /**
+ * The descriptive information string for this implementation.
+ */
+ private static final String INFO =
+ "org.mobicents.servlet.sip.startup.SipStandardService/1.0";
+ //the sip application dispatcher class name defined in the server.xml
+ protected String sipApplicationDispatcherClassName;
+ //instatiated class from the sipApplicationDispatcherClassName of the sip application dispatcher
+ protected SipApplicationDispatcher sipApplicationDispatcher;
+ private boolean gatherStatistics = true;
+ protected int sipMessageQueueSize = 1500;
+ private int backToNormalSipMessageQueueSize = 1300;
+ protected int memoryThreshold = 95;
+ private int backToNormalMemoryThreshold = 90;
+ protected OutboundProxy outboundProxy;
+ protected String proxyTimerServiceImplementationType;
+ protected String sasTimerServiceImplementationType;
+ protected long congestionControlCheckingInterval = 30000;
+ private int canceledTimerTasksPurgePeriod = 0;
+ // base timer interval for jain sip tx
+ private int baseTimerInterval = 500;
+ private int t2Interval = 4000;
+ private int t4Interval = 5000;
+ private int timerDInterval = 32000;
+ protected int dispatcherThreadPoolSize = 15;
+ private boolean md5ContactUserPart = false;
+
+ protected String concurrencyControlMode = ConcurrencyControlMode.SipApplicationSession.toString();
+ protected String congestionControlPolicy = CongestionControlPolicy.ErrorResponse.toString();
+ protected String additionalParameterableHeaders;
+ protected boolean bypassResponseExecutor = true;
+ protected boolean bypassRequestExecutor = true;
+ //this should be made available to the application router as a system prop
+ protected String darConfigurationFileLocation;
+ protected boolean connectorsStartedExternally = false;
+ protected boolean dialogPendingRequestChecking = false;
+ protected int callIdMaxLength;
+ protected int tagHashMaxLength;
+ private long gracefulInterval = 30000;
+
+ protected boolean httpFollowsSip = false;
+ protected String jvmRoute;
+ protected ReplicationStrategy replicationStrategy;
+
+ /**
+ * the sip stack path name. Since the sip factory is per classloader it should be set here for all underlying stacks
+ */
+ private String sipPathName;
+ /**
+ * use Pretty Encoding
+ */
+ private boolean usePrettyEncoding = true;
+
+ private SipStack sipStack;
+ // defining sip stack properties
+ private Properties sipStackProperties;
+ private String sipStackPropertiesFileLocation;
+ @Deprecated
+ private String addressResolverClass = null;
+ private String dnsServerLocatorClass = DefaultDNSServerLocator.class.getName();
+ private int dnsTimeout = 1;
+ private String dnsResolverClass = MobicentsDNSResolver.class.getName();
+ private String mobicentsSipServletMessageFactoryClassName = Servlet3SipServletMessageFactory.class.getName();
+
+ //the balancers to send heartbeat to and our health info
+ @Deprecated
+ private String balancers;
+ //
+ private ScheduledFuture> gracefulStopFuture;
+
+ @Override
+ public String getInfo() {
+ return (INFO);
+ }
+
+ @Override
+ public void addConnector(Connector connector) {
+ MobicentsExtendedListeningPoint extendedListeningPoint = null;
+ if (connector.getProtocolHandler() instanceof SipProtocolHandler) {
+ extendedListeningPoint = (MobicentsExtendedListeningPoint)
+ ((SipProtocolHandler)connector.getProtocolHandler()).getAttribute(ExtendedListeningPoint.class.getSimpleName());}
+ if(extendedListeningPoint != null) {
+ try {
+ extendedListeningPoint.getSipProvider().addSipListener(sipApplicationDispatcher);
+ sipApplicationDispatcher.getSipNetworkInterfaceManager().addExtendedListeningPoint(extendedListeningPoint);
+ } catch (TooManyListenersException e) {
+ logger.error("Connector.initialize", e);
+ }
+ }
+ ProtocolHandler protocolHandler = connector.getProtocolHandler();
+ if(protocolHandler instanceof SipProtocolHandler) {
+ connector.setPort(((SipProtocolHandler)protocolHandler).getPort());
+ ((SipProtocolHandler)protocolHandler).setSipStack(sipStack);
+ ((SipProtocolHandler)protocolHandler).setAttribute(SipApplicationDispatcher.class.getSimpleName(), sipApplicationDispatcher);
+ registerSipConnector(connector);
+ }
+ super.addConnector(connector);
+ }
+
+ /**
+ * Register the sip connector under a different name than HTTP Connector and we add the transport to avoid clashing with 2 connectors having the same port and address
+ * @param connector connector to register
+ */
+ protected void registerSipConnector(Connector connector) {
+ try {
+
+ ObjectName objectName = createSipConnectorObjectName(connector, getName(), "SipConnector");
+ Registry.getRegistry(null, null)
+ .registerComponent(connector, objectName, null);
+//TODO connector.setController(objectName);
+ } catch (Exception e) {
+ logger.error( "Error registering connector ", e);
+ }
+ if(logger.isDebugEnabled())
+ logger.debug("Creating name for connector " + getObjectName());
+ }
+
+ @Override
+ public void removeConnector(Connector connector) {
+ MobicentsExtendedListeningPoint extendedListeningPoint = null;
+ if (connector.getProtocolHandler() instanceof SipProtocolHandler){
+ extendedListeningPoint = (MobicentsExtendedListeningPoint)
+ ((SipProtocolHandler)connector.getProtocolHandler()).getAttribute(ExtendedListeningPoint.class.getSimpleName());}
+ if(extendedListeningPoint != null) {
+ extendedListeningPoint.getSipProvider().removeSipListener(sipApplicationDispatcher);
+ sipApplicationDispatcher.getSipNetworkInterfaceManager().removeExtendedListeningPoint(extendedListeningPoint);
+ }
+ super.removeConnector(connector);
+ }
+
+ @Override
+ public void initInternal() throws LifecycleException {
+ //load the sip application disptacher from the class name specified in the server.xml file
+ //and initializes it
+ StaticServiceHolder.sipStandardService = this;
+ try {
+ sipApplicationDispatcher = (SipApplicationDispatcher)
+ Class.forName(sipApplicationDispatcherClassName).newInstance();
+ } catch (InstantiationException e) {
+ throw new LifecycleException("Impossible to load the Sip Application Dispatcher",e);
+ } catch (IllegalAccessException e) {
+ throw new LifecycleException("Impossible to load the Sip Application Dispatcher",e);
+ } catch (ClassNotFoundException e) {
+ throw new LifecycleException("Impossible to load the Sip Application Dispatcher",e);
+ } catch (ClassCastException e) {
+ throw new LifecycleException("Sip Application Dispatcher defined does not implement " + SipApplicationDispatcher.class.getName(),e);
+ }
+ if(logger.isInfoEnabled()) {
+ logger.info("Pretty encoding of headers enabled ? " + usePrettyEncoding);
+ }
+ if(sipPathName == null) {
+ sipPathName = DEFAULT_SIP_PATH_NAME;
+ }
+ if(logger.isInfoEnabled()) {
+ logger.info("Sip Stack path name : " + sipPathName);
+ }
+ sipApplicationDispatcher.setSipService(this);
+ sipApplicationDispatcher.getSipFactory().initialize(sipPathName, usePrettyEncoding);
+
+ String catalinaBase = getCatalinaBase();
+ if(darConfigurationFileLocation != null) {
+ if(!darConfigurationFileLocation.startsWith("file:///")) {
+ darConfigurationFileLocation = "file:///" + catalinaBase.replace(File.separatorChar, '/') + "/" + darConfigurationFileLocation;
+ }
+ System.setProperty("javax.servlet.sip.dar", darConfigurationFileLocation);
+ }
+ super.initInternal();
+ sipApplicationDispatcher.setDomain(this.getName());
+ if(baseTimerInterval < 1) {
+ throw new LifecycleException("It's forbidden to set the Base Timer Interval to a non positive value");
+ }
+ initSipStack();
+ // https://telestax.atlassian.net/browse/MSS-148 make sure we have a default to bin direction for better out of the box experience
+ if(System.getProperty("telscale.license.dir") == null) {
+ System.setProperty("telscale.license.dir", getCatalinaBase() + File.separatorChar + "bin");
+ if(logger.isDebugEnabled()) {
+ logger.debug("Setting telscale.license.dir directory to : " + getCatalinaBase() + File.separatorChar + "bin");
+ }
+ }
+ if(System.getProperty("telscale.license.key.location") == null) {
+ System.setProperty("telscale.license.key.location", getCatalinaBase() + File.separatorChar + "bin");
+ if(logger.isDebugEnabled()) {
+ logger.debug("Setting telscale.license.key.location directory to : " + getCatalinaBase() + File.separatorChar + "bin");
+ }
+ }
+
+ sipApplicationDispatcher.setBaseTimerInterval(baseTimerInterval);
+ sipApplicationDispatcher.setT2Interval(t2Interval);
+ sipApplicationDispatcher.setT4Interval(t4Interval);
+ sipApplicationDispatcher.setTimerDInterval(timerDInterval);
+ sipApplicationDispatcher.setMemoryThreshold(getMemoryThreshold());
+ sipApplicationDispatcher.setBackToNormalMemoryThreshold(backToNormalMemoryThreshold);
+ sipApplicationDispatcher.setCongestionControlCheckingInterval(getCongestionControlCheckingInterval());
+ sipApplicationDispatcher.setCongestionControlPolicyByName(getCongestionControlPolicy());
+ sipApplicationDispatcher.setQueueSize(getSipMessageQueueSize());
+ sipApplicationDispatcher.setBackToNormalQueueSize(backToNormalSipMessageQueueSize);
+ sipApplicationDispatcher.setGatherStatistics(gatherStatistics);
+ sipApplicationDispatcher.setConcurrencyControlMode(ConcurrencyControlMode.valueOf(getConcurrencyControlMode()));
+ sipApplicationDispatcher.setBypassRequestExecutor(bypassRequestExecutor);
+ sipApplicationDispatcher.setBypassResponseExecutor(bypassResponseExecutor);
+ sipApplicationDispatcher.setSipStack(sipStack);
+ sipApplicationDispatcher.init();
+ // Tomcat specific loading case where the connectors are added even before the service is initialized
+ // so we need to set the sip stack before it starts
+ synchronized (connectors) {
+ for (Connector connector : connectors) {
+ ProtocolHandler protocolHandler = connector.getProtocolHandler();
+ if(protocolHandler instanceof SipProtocolHandler) {
+ connector.setPort(((SipProtocolHandler)protocolHandler).getPort());
+ ((SipProtocolHandler)protocolHandler).setSipStack(sipStack);
+ ((SipProtocolHandler)protocolHandler).setAttribute(SipApplicationDispatcher.class.getSimpleName(), sipApplicationDispatcher);
+ registerSipConnector(connector);
+ }
+ }
+ }
+ }
+
+ @Override
+ public void startInternal() throws LifecycleException {
+ super.startInternal();
+ synchronized (connectors) {
+ for (Connector connector : connectors) {
+ final ProtocolHandler protocolHandler = connector.getProtocolHandler();
+ //Jboss sepcific loading case
+ Boolean isSipConnector = false;
+ if (protocolHandler instanceof SipProtocolHandler)
+ isSipConnector = (Boolean) ((SipProtocolHandler)protocolHandler).getAttribute(SipProtocolHandler.IS_SIP_CONNECTOR);
+ if(isSipConnector != null && isSipConnector) {
+ if(logger.isDebugEnabled()) {
+ logger.debug("Attaching the sip application dispatcher " +
+ "as a sip listener to connector listening on port " +
+ connector.getPort());
+ }
+ ((SipProtocolHandler)protocolHandler).setAttribute(SipApplicationDispatcher.class.getSimpleName(), sipApplicationDispatcher);
+ ((SipProtocolHandler)protocolHandler).setSipStack(sipStack);
+ connectorsStartedExternally = true;
+ }
+ //Tomcat specific loading case
+ MobicentsExtendedListeningPoint extendedListeningPoint = null;
+ if (protocolHandler instanceof SipProtocolHandler) {
+ extendedListeningPoint = (MobicentsExtendedListeningPoint)
+ ((SipProtocolHandler)protocolHandler).getAttribute(ExtendedListeningPoint.class.getSimpleName());}
+ if(extendedListeningPoint != null && sipStack != null) {
+ try {
+ extendedListeningPoint.getSipProvider().addSipListener(sipApplicationDispatcher);
+ sipApplicationDispatcher.getSipNetworkInterfaceManager().addExtendedListeningPoint(extendedListeningPoint);
+ connectorsStartedExternally = false;
+ } catch (TooManyListenersException e) {
+ throw new LifecycleException("Couldn't add the sip application dispatcher "
+ + sipApplicationDispatcher + " as a listener to the following listening point provider " + extendedListeningPoint, e);
+ }
+ }
+ }
+ }
+ if(!connectorsStartedExternally) {
+ sipApplicationDispatcher.start();
+ sipApplicationDispatcher.putInService();
+ }
+
+ if(this.getSipMessageQueueSize() <= 0)
+ throw new LifecycleException("Message queue size can not be 0 or less");
+
+ if(logger.isDebugEnabled()) {
+ logger.debug("SIP Standard Service Started.");
+ }
+ }
+
+ public String getJvmRoute() {
+ return this.jvmRoute;
+ }
+
+ public void setJvmRoute(String jvmRoute) {
+ this.jvmRoute = jvmRoute;
+ }
+
+ protected void initSipStack() throws LifecycleException {
+ try {
+ if(logger.isDebugEnabled()) {
+ logger.debug("Initializing SIP stack");
+ }
+
+ // This simply puts HTTP and SSL port numbers in JVM properties menat to be read by jsip ha when sending heart beats with Node description.
+ initializeSystemPortProperties();
+
+ String catalinaBase = getCatalinaBase();
+ if(sipStackPropertiesFileLocation != null && !sipStackPropertiesFileLocation.startsWith("file:///")) {
+ sipStackPropertiesFileLocation = "file:///" + catalinaBase.replace(File.separatorChar, '/') + "/" + sipStackPropertiesFileLocation;
+ }
+ boolean isPropsLoaded = false;
+ if(sipStackProperties == null) {
+ sipStackProperties = new Properties();
+ } else {
+ isPropsLoaded = true;
+ }
+
+ if (logger.isDebugEnabled()) {
+ logger.debug("Loading SIP stack properties from following file : " + sipStackPropertiesFileLocation);
+ }
+ if(sipStackPropertiesFileLocation != null) {
+ //hack to get around space char in path see http://weblogs.java.net/blog/kohsuke/archive/2007/04/how_to_convert.html,
+ // we create a URL since it's permissive enough
+ File sipStackPropertiesFile = null;
+ URL url = null;
+ try {
+ url = new URL(sipStackPropertiesFileLocation);
+ } catch (MalformedURLException e) {
+ logger.fatal("Cannot find the sip stack properties file ! ",e);
+ throw new IllegalArgumentException("The Default Application Router file Location : "+sipStackPropertiesFileLocation+" is not valid ! ",e);
+ }
+ try {
+ sipStackPropertiesFile = new File(new URI(sipStackPropertiesFileLocation));
+ } catch (URISyntaxException e) {
+ //if the uri contains space this will fail, so getting the path will work
+ sipStackPropertiesFile = new File(url.getPath());
+ }
+ FileInputStream sipStackPropertiesInputStream = null;
+ try {
+ sipStackPropertiesInputStream = new FileInputStream(sipStackPropertiesFile);
+ sipStackProperties.load(sipStackPropertiesInputStream);
+ } catch (Exception e) {
+ logger.warn("Could not find or problem when loading the sip stack properties file : " + sipStackPropertiesFileLocation, e);
+ } finally {
+ if(sipStackPropertiesInputStream != null) {
+ try {
+ sipStackPropertiesInputStream.close();
+ } catch (IOException e) {
+ logger.error("fail to close the following file " + sipStackPropertiesFile.getAbsolutePath(), e);
+ }
+ }
+ }
+
+ String debugLog = sipStackProperties.getProperty(DEBUG_LOG_STACK_PROP);
+ if(debugLog != null && debugLog.length() > 0 && !debugLog.startsWith("file:///")) {
+ sipStackProperties.setProperty(DEBUG_LOG_STACK_PROP,
+ catalinaBase + "/" + debugLog);
+ }
+ String serverLog = sipStackProperties.getProperty(SERVER_LOG_STACK_PROP);
+ if(serverLog != null && serverLog.length() > 0 && !serverLog.startsWith("file:///")) {
+ sipStackProperties.setProperty(SERVER_LOG_STACK_PROP,
+ catalinaBase + "/" + serverLog);
+ }
+ // The whole MSS is built upon those assumptions, so those properties are not overrideable
+ if (sipStackProperties.getProperty(AUTOMATIC_DIALOG_SUPPORT_STACK_PROP) == null) {
+ //https://github.com/RestComm/sip-servlets/issues/143
+ //set off if user didnt provided any value.
+ sipStackProperties.setProperty(AUTOMATIC_DIALOG_SUPPORT_STACK_PROP, "off");
+ }
+ sipStackProperties.setProperty(LOOSE_DIALOG_VALIDATION, "true");
+ sipStackProperties.setProperty(PASS_INVITE_NON_2XX_ACK_TO_LISTENER, "true");
+ isPropsLoaded = true;
+ } else {
+ logger.warn("no sip stack properties file defined ");
+ }
+ if(!isPropsLoaded) {
+ logger.warn("loading default Mobicents Sip Servlets sip stack properties");
+ // Silently set default values
+ sipStackProperties.setProperty("gov.nist.javax.sip.LOG_MESSAGE_CONTENT",
+ "true");
+ sipStackProperties.setProperty("gov.nist.javax.sip.TRACE_LEVEL",
+ "32");
+ sipStackProperties.setProperty(DEBUG_LOG_STACK_PROP,
+ catalinaBase + "/" + "mss-jsip-" + getName() +"-debug.txt");
+ sipStackProperties.setProperty(SERVER_LOG_STACK_PROP,
+ catalinaBase + "/" + "mss-jsip-" + getName() +"-messages.xml");
+ sipStackProperties.setProperty("javax.sip.STACK_NAME", "mss-" + getName());
+ sipStackProperties.setProperty(AUTOMATIC_DIALOG_SUPPORT_STACK_PROP, "off");
+ sipStackProperties.setProperty("gov.nist.javax.sip.DELIVER_UNSOLICITED_NOTIFY", "true");
+ sipStackProperties.setProperty("gov.nist.javax.sip.THREAD_POOL_SIZE", "64");
+ sipStackProperties.setProperty("gov.nist.javax.sip.REENTRANT_LISTENER", "true");
+ sipStackProperties.setProperty("gov.nist.javax.sip.MAX_FORK_TIME_SECONDS", "0");
+ sipStackProperties.setProperty(LOOSE_DIALOG_VALIDATION, "true");
+ sipStackProperties.setProperty(PASS_INVITE_NON_2XX_ACK_TO_LISTENER, "true");
+ sipStackProperties.setProperty("gov.nist.javax.sip.AUTOMATIC_DIALOG_ERROR_HANDLING", "false");
+ sipStackProperties.setProperty("gov.nist.javax.sip.AGGRESSIVE_CLEANUP", "true");
+ }
+
+ if(sipStackProperties.get(TCP_POST_PARSING_THREAD_POOL_SIZE) == null) {
+ sipStackProperties.setProperty(TCP_POST_PARSING_THREAD_POOL_SIZE, "30");
+ }
+
+ // set the DNSServerLocator allowing to support RFC 3263 and do DNS lookups to resolve uris
+ if(dnsServerLocatorClass != null && dnsServerLocatorClass.trim().length() > 0) {
+ if(logger.isDebugEnabled()) {
+ logger.debug("Sip Stack " + sipStackProperties.getProperty("javax.sip.STACK_NAME") +" will be using " + dnsServerLocatorClass + " as DNSServerLocator");
+ }
+ try {
+ // create parameters argument to identify constructor
+ Class[] paramTypes = new Class[0];
+ // get constructor of AddressResolver in order to instantiate
+ Constructor dnsServerLocatorConstructor = Class.forName(dnsServerLocatorClass).getConstructor(
+ paramTypes);
+ // Wrap properties object in order to pass to constructor of AddressResolver
+ Object[] conArgs = new Object[0];
+ // Creates a new instance of AddressResolver Class with the supplied sipApplicationDispatcher.
+ DNSServerLocator dnsServerLocator = (DNSServerLocator) dnsServerLocatorConstructor.newInstance(conArgs);
+ sipApplicationDispatcher.setDNSServerLocator(dnsServerLocator);
+ sipApplicationDispatcher.setDNSTimeout(dnsTimeout);
+ if(sipStackProperties.getProperty("javax.sip.ROUTER_PATH") == null) {
+ sipStackProperties.setProperty("javax.sip.ROUTER_PATH", DNSAwareRouter.class.getCanonicalName());
+ }
+ } catch (Exception e) {
+ logger.error("Couldn't set the AddressResolver " + addressResolverClass, e);
+ throw e;
+ }
+ } else {
+ if(logger.isInfoEnabled()) {
+ logger.info("no DNSServerLocator will be used since none has been specified.");
+ }
+ }
+
+ String serverHeaderValue = sipStackProperties.getProperty(SERVER_HEADER);
+ if(serverHeaderValue != null) {
+ ListSipService
interface.
- * This class inherits from the Tomcat StandardService. It adds a SipApplicationDispatcher
- * that will be listen for sip messages received by the sip stacks started by
- * the sip connectors associated with this context.
- * This has one attribute which is the sipApplicationDispatcherClassName allowing one
- * to specify the class name of the sipApplicationDispacther to easily replace
- * the default sipApplicationDispatcher with a custom one.
- *
- * @author Jean Deruelle
- */
-public class SipStandardService extends StandardService implements CatalinaSipService {
- //the logger
- private static final Logger logger = Logger.getLogger(SipStandardService.class);
- public static final String DEFAULT_SIP_PATH_NAME = "gov.nist";
- public static final String PASS_INVITE_NON_2XX_ACK_TO_LISTENER = "gov.nist.javax.sip.PASS_INVITE_NON_2XX_ACK_TO_LISTENER";
- public static final String TCP_POST_PARSING_THREAD_POOL_SIZE = "gov.nist.javax.sip.TCP_POST_PARSING_THREAD_POOL_SIZE";
- public static final String AUTOMATIC_DIALOG_SUPPORT_STACK_PROP = "javax.sip.AUTOMATIC_DIALOG_SUPPORT";
- public static final String LOOSE_DIALOG_VALIDATION = "gov.nist.javax.sip.LOOSE_DIALOG_VALIDATION";
- public static final String SERVER_LOG_STACK_PROP = "gov.nist.javax.sip.SERVER_LOG";
- public static final String DEBUG_LOG_STACK_PROP = "gov.nist.javax.sip.DEBUG_LOG";
- public static final String SERVER_HEADER = "org.mobicents.servlet.sip.SERVER_HEADER";
- public static final String USER_AGENT_HEADER = "org.mobicents.servlet.sip.USER_AGENT_HEADER";
- public static final String JVM_ROUTE = "jvmRoute";
- /**
- * The descriptive information string for this implementation.
- */
- private static final String INFO =
- "org.mobicents.servlet.sip.startup.SipStandardService/1.0";
- //the sip application dispatcher class name defined in the server.xml
- protected String sipApplicationDispatcherClassName;
- //instatiated class from the sipApplicationDispatcherClassName of the sip application dispatcher
- protected SipApplicationDispatcher sipApplicationDispatcher;
- private boolean gatherStatistics = true;
- protected int sipMessageQueueSize = 1500;
- private int backToNormalSipMessageQueueSize = 1300;
- protected int memoryThreshold = 95;
- private int backToNormalMemoryThreshold = 90;
- protected OutboundProxy outboundProxy;
- protected String proxyTimerServiceImplementationType;
- protected String sasTimerServiceImplementationType;
- protected long congestionControlCheckingInterval = 30000;
- private int canceledTimerTasksPurgePeriod = 0;
- // base timer interval for jain sip tx
- private int baseTimerInterval = 500;
- private int t2Interval = 4000;
- private int t4Interval = 5000;
- private int timerDInterval = 32000;
- protected int dispatcherThreadPoolSize = 15;
- private boolean md5ContactUserPart = false;
-
- protected String concurrencyControlMode = ConcurrencyControlMode.SipApplicationSession.toString();
- protected String congestionControlPolicy = CongestionControlPolicy.ErrorResponse.toString();
- protected String additionalParameterableHeaders;
- protected boolean bypassResponseExecutor = true;
- protected boolean bypassRequestExecutor = true;
- //this should be made available to the application router as a system prop
- protected String darConfigurationFileLocation;
- protected boolean connectorsStartedExternally = false;
- protected boolean dialogPendingRequestChecking = false;
- protected int callIdMaxLength;
- protected int tagHashMaxLength;
-
- protected boolean httpFollowsSip = false;
- protected String jvmRoute;
- protected ReplicationStrategy replicationStrategy;
-
- /**
- * the sip stack path name. Since the sip factory is per classloader it should be set here for all underlying stacks
- */
- private String sipPathName;
- /**
- * use Pretty Encoding
- */
- private boolean usePrettyEncoding = true;
-
- private SipStack sipStack;
- // defining sip stack properties
- private Properties sipStackProperties;
- private String sipStackPropertiesFileLocation;
- @Deprecated
- private String addressResolverClass = null;
- private String dnsServerLocatorClass = DefaultDNSServerLocator.class.getName();
- private int dnsTimeout = 1;
- private String dnsResolverClass = MobicentsDNSResolver.class.getName();
- private String mobicentsSipServletMessageFactoryClassName = Servlet3SipServletMessageFactory.class.getName();
-
- //the balancers to send heartbeat to and our health info
- @Deprecated
- private String balancers;
- //
- private ScheduledFuture> gracefulStopFuture;
-
- @Override
- public void addConnector(Connector connector) {
- MobicentsExtendedListeningPoint extendedListeningPoint = null;
- if (connector.getProtocolHandler() instanceof SipProtocolHandler) {
- extendedListeningPoint = (MobicentsExtendedListeningPoint)
- ((SipProtocolHandler)connector.getProtocolHandler()).getAttribute(ExtendedListeningPoint.class.getSimpleName());}
- if(extendedListeningPoint != null) {
- try {
- extendedListeningPoint.getSipProvider().addSipListener(sipApplicationDispatcher);
- sipApplicationDispatcher.getSipNetworkInterfaceManager().addExtendedListeningPoint(extendedListeningPoint);
- } catch (TooManyListenersException e) {
- logger.error("Connector.initialize", e);
- }
- }
- ProtocolHandler protocolHandler = connector.getProtocolHandler();
- if(protocolHandler instanceof SipProtocolHandler) {
- connector.setPort(((SipProtocolHandler)protocolHandler).getPort());
- ((SipProtocolHandler)protocolHandler).setSipStack(sipStack);
- ((SipProtocolHandler)protocolHandler).setAttribute(SipApplicationDispatcher.class.getSimpleName(), sipApplicationDispatcher);
- registerSipConnector(connector);
- }
- super.addConnector(connector);
- }
-
- /**
- * Register the sip connector under a different name than HTTP Connector and we add the transport to avoid clashing with 2 connectors having the same port and address
- * @param connector connector to register
- */
- protected void registerSipConnector(Connector connector) {
- try {
-
- ObjectName objectName = createSipConnectorObjectName(connector, getName(), "SipConnector");
- Registry.getRegistry(null, null)
- .registerComponent(connector, objectName, null);
-//TODO connector.setController(objectName);
- } catch (Exception e) {
- logger.error( "Error registering connector ", e);
- }
- if(logger.isDebugEnabled())
- logger.debug("Creating name for connector " + getObjectName());
- }
-
- @Override
- public void removeConnector(Connector connector) {
- MobicentsExtendedListeningPoint extendedListeningPoint = null;
- if (connector.getProtocolHandler() instanceof SipProtocolHandler){
- extendedListeningPoint = (MobicentsExtendedListeningPoint)
- ((SipProtocolHandler)connector.getProtocolHandler()).getAttribute(ExtendedListeningPoint.class.getSimpleName());}
- if(extendedListeningPoint != null) {
- extendedListeningPoint.getSipProvider().removeSipListener(sipApplicationDispatcher);
- sipApplicationDispatcher.getSipNetworkInterfaceManager().removeExtendedListeningPoint(extendedListeningPoint);
- }
- super.removeConnector(connector);
- }
-
- @Override
- public void initInternal() throws LifecycleException {
- //load the sip application disptacher from the class name specified in the server.xml file
- //and initializes it
- StaticServiceHolder.sipStandardService = this;
- try {
- sipApplicationDispatcher = (SipApplicationDispatcher)
- Class.forName(sipApplicationDispatcherClassName).newInstance();
- } catch (InstantiationException e) {
- throw new LifecycleException("Impossible to load the Sip Application Dispatcher",e);
- } catch (IllegalAccessException e) {
- throw new LifecycleException("Impossible to load the Sip Application Dispatcher",e);
- } catch (ClassNotFoundException e) {
- throw new LifecycleException("Impossible to load the Sip Application Dispatcher",e);
- } catch (ClassCastException e) {
- throw new LifecycleException("Sip Application Dispatcher defined does not implement " + SipApplicationDispatcher.class.getName(),e);
- }
- if(logger.isInfoEnabled()) {
- logger.info("Pretty encoding of headers enabled ? " + usePrettyEncoding);
- }
- if(sipPathName == null) {
- sipPathName = DEFAULT_SIP_PATH_NAME;
- }
- if(logger.isInfoEnabled()) {
- logger.info("Sip Stack path name : " + sipPathName);
- }
- sipApplicationDispatcher.setSipService(this);
- sipApplicationDispatcher.getSipFactory().initialize(sipPathName, usePrettyEncoding);
-
- String catalinaBase = getCatalinaBase();
- if(darConfigurationFileLocation != null) {
- if(!darConfigurationFileLocation.startsWith("file:///")) {
- darConfigurationFileLocation = "file:///" + catalinaBase.replace(File.separatorChar, '/') + "/" + darConfigurationFileLocation;
- }
- System.setProperty("javax.servlet.sip.dar", darConfigurationFileLocation);
- }
- super.initInternal();
- sipApplicationDispatcher.setDomain(this.getName());
- if(baseTimerInterval < 1) {
- throw new LifecycleException("It's forbidden to set the Base Timer Interval to a non positive value");
- }
- initSipStack();
- // https://telestax.atlassian.net/browse/MSS-148 make sure we have a default to bin direction for better out of the box experience
- if(System.getProperty("telscale.license.dir") == null) {
- System.setProperty("telscale.license.dir", getCatalinaBase() + File.separatorChar + "bin");
- if(logger.isDebugEnabled()) {
- logger.debug("Setting telscale.license.dir directory to : " + getCatalinaBase() + File.separatorChar + "bin");
- }
- }
- if(System.getProperty("telscale.license.key.location") == null) {
- System.setProperty("telscale.license.key.location", getCatalinaBase() + File.separatorChar + "bin");
- if(logger.isDebugEnabled()) {
- logger.debug("Setting telscale.license.key.location directory to : " + getCatalinaBase() + File.separatorChar + "bin");
- }
- }
-
- sipApplicationDispatcher.setBaseTimerInterval(baseTimerInterval);
- sipApplicationDispatcher.setT2Interval(t2Interval);
- sipApplicationDispatcher.setT4Interval(t4Interval);
- sipApplicationDispatcher.setTimerDInterval(timerDInterval);
- sipApplicationDispatcher.setMemoryThreshold(getMemoryThreshold());
- sipApplicationDispatcher.setBackToNormalMemoryThreshold(backToNormalMemoryThreshold);
- sipApplicationDispatcher.setCongestionControlCheckingInterval(getCongestionControlCheckingInterval());
- sipApplicationDispatcher.setCongestionControlPolicyByName(getCongestionControlPolicy());
- sipApplicationDispatcher.setQueueSize(getSipMessageQueueSize());
- sipApplicationDispatcher.setBackToNormalQueueSize(backToNormalSipMessageQueueSize);
- sipApplicationDispatcher.setGatherStatistics(gatherStatistics);
- sipApplicationDispatcher.setConcurrencyControlMode(ConcurrencyControlMode.valueOf(getConcurrencyControlMode()));
- sipApplicationDispatcher.setBypassRequestExecutor(bypassRequestExecutor);
- sipApplicationDispatcher.setBypassResponseExecutor(bypassResponseExecutor);
- sipApplicationDispatcher.setSipStack(sipStack);
- sipApplicationDispatcher.init();
- // Tomcat specific loading case where the connectors are added even before the service is initialized
- // so we need to set the sip stack before it starts
- synchronized (connectors) {
- for (Connector connector : connectors) {
- ProtocolHandler protocolHandler = connector.getProtocolHandler();
- if(protocolHandler instanceof SipProtocolHandler) {
- connector.setPort(((SipProtocolHandler)protocolHandler).getPort());
- ((SipProtocolHandler)protocolHandler).setSipStack(sipStack);
- ((SipProtocolHandler)protocolHandler).setAttribute(SipApplicationDispatcher.class.getSimpleName(), sipApplicationDispatcher);
- registerSipConnector(connector);
- }
- }
- }
- }
-
- @Override
- public void startInternal() throws LifecycleException {
- super.startInternal();
- synchronized (connectors) {
- for (Connector connector : connectors) {
- final ProtocolHandler protocolHandler = connector.getProtocolHandler();
- //Jboss sepcific loading case
- Boolean isSipConnector = false;
- if (protocolHandler instanceof SipProtocolHandler)
- isSipConnector = (Boolean) ((SipProtocolHandler)protocolHandler).getAttribute(SipProtocolHandler.IS_SIP_CONNECTOR);
- if(isSipConnector != null && isSipConnector) {
- if(logger.isDebugEnabled()) {
- logger.debug("Attaching the sip application dispatcher " +
- "as a sip listener to connector listening on port " +
- connector.getPort());
- }
- ((SipProtocolHandler)protocolHandler).setAttribute(SipApplicationDispatcher.class.getSimpleName(), sipApplicationDispatcher);
- ((SipProtocolHandler)protocolHandler).setSipStack(sipStack);
- connectorsStartedExternally = true;
- }
- //Tomcat specific loading case
- MobicentsExtendedListeningPoint extendedListeningPoint = null;
- if (protocolHandler instanceof SipProtocolHandler) {
- extendedListeningPoint = (MobicentsExtendedListeningPoint)
- ((SipProtocolHandler)protocolHandler).getAttribute(ExtendedListeningPoint.class.getSimpleName());}
- if(extendedListeningPoint != null && sipStack != null) {
- try {
- extendedListeningPoint.getSipProvider().addSipListener(sipApplicationDispatcher);
- sipApplicationDispatcher.getSipNetworkInterfaceManager().addExtendedListeningPoint(extendedListeningPoint);
- connectorsStartedExternally = false;
- } catch (TooManyListenersException e) {
- throw new LifecycleException("Couldn't add the sip application dispatcher "
- + sipApplicationDispatcher + " as a listener to the following listening point provider " + extendedListeningPoint, e);
- }
- }
- }
- }
- if(!connectorsStartedExternally) {
- sipApplicationDispatcher.start();
- }
-
- if(this.getSipMessageQueueSize() <= 0)
- throw new LifecycleException("Message queue size can not be 0 or less");
-
- if(logger.isDebugEnabled()) {
- logger.debug("SIP Standard Service Started.");
- }
- }
-
- public String getJvmRoute() {
- return this.jvmRoute;
- }
-
- public void setJvmRoute(String jvmRoute) {
- this.jvmRoute = jvmRoute;
- }
-
- protected void initSipStack() throws LifecycleException {
- try {
- if(logger.isDebugEnabled()) {
- logger.debug("Initializing SIP stack");
- }
-
- // This simply puts HTTP and SSL port numbers in JVM properties menat to be read by jsip ha when sending heart beats with Node description.
- initializeSystemPortProperties();
-
- String catalinaBase = getCatalinaBase();
- if(sipStackPropertiesFileLocation != null && !sipStackPropertiesFileLocation.startsWith("file:///")) {
- sipStackPropertiesFileLocation = "file:///" + catalinaBase.replace(File.separatorChar, '/') + "/" + sipStackPropertiesFileLocation;
- }
- boolean isPropsLoaded = false;
- if(sipStackProperties == null) {
- sipStackProperties = new Properties();
- } else {
- isPropsLoaded = true;
- }
-
- if (logger.isDebugEnabled()) {
- logger.debug("Loading SIP stack properties from following file : " + sipStackPropertiesFileLocation);
- }
- if(sipStackPropertiesFileLocation != null) {
- //hack to get around space char in path see http://weblogs.java.net/blog/kohsuke/archive/2007/04/how_to_convert.html,
- // we create a URL since it's permissive enough
- File sipStackPropertiesFile = null;
- URL url = null;
- try {
- url = new URL(sipStackPropertiesFileLocation);
- } catch (MalformedURLException e) {
- logger.fatal("Cannot find the sip stack properties file ! ",e);
- throw new IllegalArgumentException("The Default Application Router file Location : "+sipStackPropertiesFileLocation+" is not valid ! ",e);
- }
- try {
- sipStackPropertiesFile = new File(new URI(sipStackPropertiesFileLocation));
- } catch (URISyntaxException e) {
- //if the uri contains space this will fail, so getting the path will work
- sipStackPropertiesFile = new File(url.getPath());
- }
- FileInputStream sipStackPropertiesInputStream = null;
- try {
- sipStackPropertiesInputStream = new FileInputStream(sipStackPropertiesFile);
- sipStackProperties.load(sipStackPropertiesInputStream);
- } catch (Exception e) {
- logger.warn("Could not find or problem when loading the sip stack properties file : " + sipStackPropertiesFileLocation, e);
- } finally {
- if(sipStackPropertiesInputStream != null) {
- try {
- sipStackPropertiesInputStream.close();
- } catch (IOException e) {
- logger.error("fail to close the following file " + sipStackPropertiesFile.getAbsolutePath(), e);
- }
- }
- }
-
- String debugLog = sipStackProperties.getProperty(DEBUG_LOG_STACK_PROP);
- if(debugLog != null && debugLog.length() > 0 && !debugLog.startsWith("file:///")) {
- sipStackProperties.setProperty(DEBUG_LOG_STACK_PROP,
- catalinaBase + "/" + debugLog);
- }
- String serverLog = sipStackProperties.getProperty(SERVER_LOG_STACK_PROP);
- if(serverLog != null && serverLog.length() > 0 && !serverLog.startsWith("file:///")) {
- sipStackProperties.setProperty(SERVER_LOG_STACK_PROP,
- catalinaBase + "/" + serverLog);
- }
- // The whole MSS is built upon those assumptions, so those properties are not overrideable
- if (sipStackProperties.getProperty(AUTOMATIC_DIALOG_SUPPORT_STACK_PROP) == null) {
- //https://github.com/RestComm/sip-servlets/issues/143
- //set off if user didnt provided any value.
- sipStackProperties.setProperty(AUTOMATIC_DIALOG_SUPPORT_STACK_PROP, "off");
- }
- sipStackProperties.setProperty(LOOSE_DIALOG_VALIDATION, "true");
- sipStackProperties.setProperty(PASS_INVITE_NON_2XX_ACK_TO_LISTENER, "true");
- isPropsLoaded = true;
- } else {
- logger.warn("no sip stack properties file defined ");
- }
- if(!isPropsLoaded) {
- logger.warn("loading default Mobicents Sip Servlets sip stack properties");
- // Silently set default values
- sipStackProperties.setProperty("gov.nist.javax.sip.LOG_MESSAGE_CONTENT",
- "true");
- sipStackProperties.setProperty("gov.nist.javax.sip.TRACE_LEVEL",
- "32");
- sipStackProperties.setProperty(DEBUG_LOG_STACK_PROP,
- catalinaBase + "/" + "mss-jsip-" + getName() +"-debug.txt");
- sipStackProperties.setProperty(SERVER_LOG_STACK_PROP,
- catalinaBase + "/" + "mss-jsip-" + getName() +"-messages.xml");
- sipStackProperties.setProperty("javax.sip.STACK_NAME", "mss-" + getName());
- sipStackProperties.setProperty(AUTOMATIC_DIALOG_SUPPORT_STACK_PROP, "off");
- sipStackProperties.setProperty("gov.nist.javax.sip.DELIVER_UNSOLICITED_NOTIFY", "true");
- sipStackProperties.setProperty("gov.nist.javax.sip.THREAD_POOL_SIZE", "64");
- sipStackProperties.setProperty("gov.nist.javax.sip.REENTRANT_LISTENER", "true");
- sipStackProperties.setProperty("gov.nist.javax.sip.MAX_FORK_TIME_SECONDS", "0");
- sipStackProperties.setProperty(LOOSE_DIALOG_VALIDATION, "true");
- sipStackProperties.setProperty(PASS_INVITE_NON_2XX_ACK_TO_LISTENER, "true");
- sipStackProperties.setProperty("gov.nist.javax.sip.AUTOMATIC_DIALOG_ERROR_HANDLING", "false");
- sipStackProperties.setProperty("gov.nist.javax.sip.AGGRESSIVE_CLEANUP", "true");
- }
-
- if(sipStackProperties.get(TCP_POST_PARSING_THREAD_POOL_SIZE) == null) {
- sipStackProperties.setProperty(TCP_POST_PARSING_THREAD_POOL_SIZE, "30");
- }
-
- // set the DNSServerLocator allowing to support RFC 3263 and do DNS lookups to resolve uris
- if(dnsServerLocatorClass != null && dnsServerLocatorClass.trim().length() > 0) {
- if(logger.isDebugEnabled()) {
- logger.debug("Sip Stack " + sipStackProperties.getProperty("javax.sip.STACK_NAME") +" will be using " + dnsServerLocatorClass + " as DNSServerLocator");
- }
- try {
- // create parameters argument to identify constructor
- Class[] paramTypes = new Class[0];
- // get constructor of AddressResolver in order to instantiate
- Constructor dnsServerLocatorConstructor = Class.forName(dnsServerLocatorClass).getConstructor(
- paramTypes);
- // Wrap properties object in order to pass to constructor of AddressResolver
- Object[] conArgs = new Object[0];
- // Creates a new instance of AddressResolver Class with the supplied sipApplicationDispatcher.
- DNSServerLocator dnsServerLocator = (DNSServerLocator) dnsServerLocatorConstructor.newInstance(conArgs);
- sipApplicationDispatcher.setDNSServerLocator(dnsServerLocator);
- sipApplicationDispatcher.setDNSTimeout(dnsTimeout);
- if(sipStackProperties.getProperty("javax.sip.ROUTER_PATH") == null) {
- sipStackProperties.setProperty("javax.sip.ROUTER_PATH", DNSAwareRouter.class.getCanonicalName());
- }
- } catch (Exception e) {
- logger.error("Couldn't set the AddressResolver " + addressResolverClass, e);
- throw e;
- }
- } else {
- if(logger.isInfoEnabled()) {
- logger.info("no DNSServerLocator will be used since none has been specified.");
- }
- }
-
- String serverHeaderValue = sipStackProperties.getProperty(SERVER_HEADER);
- if(serverHeaderValue != null) {
- ListSipService
interface.
+ * This class inherits from the Tomcat StandardService. It adds a SipApplicationDispatcher
+ * that will be listen for sip messages received by the sip stacks started by
+ * the sip connectors associated with this context.
+ * This has one attribute which is the sipApplicationDispatcherClassName allowing one
+ * to specify the class name of the sipApplicationDispacther to easily replace
+ * the default sipApplicationDispatcher with a custom one.
+ *
+ * @author Jean Deruelle
+ */
+public class SipStandardService extends StandardService implements CatalinaSipService {
+ //the logger
+ private static final Logger logger = Logger.getLogger(SipStandardService.class);
+ public static final String DEFAULT_SIP_PATH_NAME = "gov.nist";
+ public static final String PASS_INVITE_NON_2XX_ACK_TO_LISTENER = "gov.nist.javax.sip.PASS_INVITE_NON_2XX_ACK_TO_LISTENER";
+ public static final String TCP_POST_PARSING_THREAD_POOL_SIZE = "gov.nist.javax.sip.TCP_POST_PARSING_THREAD_POOL_SIZE";
+ public static final String AUTOMATIC_DIALOG_SUPPORT_STACK_PROP = "javax.sip.AUTOMATIC_DIALOG_SUPPORT";
+ public static final String LOOSE_DIALOG_VALIDATION = "gov.nist.javax.sip.LOOSE_DIALOG_VALIDATION";
+ public static final String SERVER_LOG_STACK_PROP = "gov.nist.javax.sip.SERVER_LOG";
+ public static final String DEBUG_LOG_STACK_PROP = "gov.nist.javax.sip.DEBUG_LOG";
+ public static final String SERVER_HEADER = "org.mobicents.servlet.sip.SERVER_HEADER";
+ public static final String USER_AGENT_HEADER = "org.mobicents.servlet.sip.USER_AGENT_HEADER";
+ public static final String JVM_ROUTE = "jvmRoute";
+ /**
+ * The descriptive information string for this implementation.
+ */
+ private static final String INFO =
+ "org.mobicents.servlet.sip.startup.SipStandardService/1.0";
+ //the sip application dispatcher class name defined in the server.xml
+ protected String sipApplicationDispatcherClassName;
+ //instatiated class from the sipApplicationDispatcherClassName of the sip application dispatcher
+ protected SipApplicationDispatcher sipApplicationDispatcher;
+ private boolean gatherStatistics = true;
+ protected int sipMessageQueueSize = 1500;
+ private int backToNormalSipMessageQueueSize = 1300;
+ protected int memoryThreshold = 95;
+ private int backToNormalMemoryThreshold = 90;
+ protected OutboundProxy outboundProxy;
+ protected String proxyTimerServiceImplementationType;
+ protected String sasTimerServiceImplementationType;
+ protected long congestionControlCheckingInterval = 30000;
+ private int canceledTimerTasksPurgePeriod = 0;
+ // base timer interval for jain sip tx
+ private int baseTimerInterval = 500;
+ private int t2Interval = 4000;
+ private int t4Interval = 5000;
+ private int timerDInterval = 32000;
+ protected int dispatcherThreadPoolSize = 15;
+ private boolean md5ContactUserPart = false;
+
+ protected String concurrencyControlMode = ConcurrencyControlMode.SipApplicationSession.toString();
+ protected String congestionControlPolicy = CongestionControlPolicy.ErrorResponse.toString();
+ protected String additionalParameterableHeaders;
+ protected boolean bypassResponseExecutor = true;
+ protected boolean bypassRequestExecutor = true;
+ //this should be made available to the application router as a system prop
+ protected String darConfigurationFileLocation;
+ protected boolean connectorsStartedExternally = false;
+ protected boolean dialogPendingRequestChecking = false;
+ protected int callIdMaxLength;
+ protected int tagHashMaxLength;
+
+ protected boolean httpFollowsSip = false;
+ protected String jvmRoute;
+ protected ReplicationStrategy replicationStrategy;
+
+ /**
+ * the sip stack path name. Since the sip factory is per classloader it should be set here for all underlying stacks
+ */
+ private String sipPathName;
+ /**
+ * use Pretty Encoding
+ */
+ private boolean usePrettyEncoding = true;
+
+ private SipStack sipStack;
+ // defining sip stack properties
+ private Properties sipStackProperties;
+ private String sipStackPropertiesFileLocation;
+ @Deprecated
+ private String addressResolverClass = null;
+ private String dnsServerLocatorClass = DefaultDNSServerLocator.class.getName();
+ private int dnsTimeout = 1;
+ private String dnsResolverClass = MobicentsDNSResolver.class.getName();
+ private String mobicentsSipServletMessageFactoryClassName = Servlet3SipServletMessageFactory.class.getName();
+
+ //the balancers to send heartbeat to and our health info
+ @Deprecated
+ private String balancers;
+ //
+ private ScheduledFuture> gracefulStopFuture;
+
+ @Override
+ public void addConnector(Connector connector) {
+ MobicentsExtendedListeningPoint extendedListeningPoint = null;
+ if (connector.getProtocolHandler() instanceof SipProtocolHandler) {
+ extendedListeningPoint = (MobicentsExtendedListeningPoint)
+ ((SipProtocolHandler)connector.getProtocolHandler()).getAttribute(ExtendedListeningPoint.class.getSimpleName());}
+ if(extendedListeningPoint != null) {
+ try {
+ extendedListeningPoint.getSipProvider().addSipListener(sipApplicationDispatcher);
+ sipApplicationDispatcher.getSipNetworkInterfaceManager().addExtendedListeningPoint(extendedListeningPoint);
+ } catch (TooManyListenersException e) {
+ logger.error("Connector.initialize", e);
+ }
+ }
+ ProtocolHandler protocolHandler = connector.getProtocolHandler();
+ if(protocolHandler instanceof SipProtocolHandler) {
+ connector.setPort(((SipProtocolHandler)protocolHandler).getPort());
+ ((SipProtocolHandler)protocolHandler).setSipStack(sipStack);
+ ((SipProtocolHandler)protocolHandler).setAttribute(SipApplicationDispatcher.class.getSimpleName(), sipApplicationDispatcher);
+ registerSipConnector(connector);
+ }
+ super.addConnector(connector);
+ }
+
+ /**
+ * Register the sip connector under a different name than HTTP Connector and we add the transport to avoid clashing with 2 connectors having the same port and address
+ * @param connector connector to register
+ */
+ protected void registerSipConnector(Connector connector) {
+ try {
+
+ ObjectName objectName = createSipConnectorObjectName(connector, getName(), "SipConnector");
+ Registry.getRegistry(null, null)
+ .registerComponent(connector, objectName, null);
+//TODO connector.setController(objectName);
+ } catch (Exception e) {
+ logger.error( "Error registering connector ", e);
+ }
+ if(logger.isDebugEnabled())
+ logger.debug("Creating name for connector " + getObjectName());
+ }
+
+ @Override
+ public void removeConnector(Connector connector) {
+ MobicentsExtendedListeningPoint extendedListeningPoint = null;
+ if (connector.getProtocolHandler() instanceof SipProtocolHandler){
+ extendedListeningPoint = (MobicentsExtendedListeningPoint)
+ ((SipProtocolHandler)connector.getProtocolHandler()).getAttribute(ExtendedListeningPoint.class.getSimpleName());}
+ if(extendedListeningPoint != null) {
+ extendedListeningPoint.getSipProvider().removeSipListener(sipApplicationDispatcher);
+ sipApplicationDispatcher.getSipNetworkInterfaceManager().removeExtendedListeningPoint(extendedListeningPoint);
+ }
+ super.removeConnector(connector);
+ }
+
+ @Override
+ public void initInternal() throws LifecycleException {
+ //load the sip application disptacher from the class name specified in the server.xml file
+ //and initializes it
+ StaticServiceHolder.sipStandardService = this;
+ try {
+ sipApplicationDispatcher = (SipApplicationDispatcher)
+ Class.forName(sipApplicationDispatcherClassName).newInstance();
+ } catch (InstantiationException e) {
+ throw new LifecycleException("Impossible to load the Sip Application Dispatcher",e);
+ } catch (IllegalAccessException e) {
+ throw new LifecycleException("Impossible to load the Sip Application Dispatcher",e);
+ } catch (ClassNotFoundException e) {
+ throw new LifecycleException("Impossible to load the Sip Application Dispatcher",e);
+ } catch (ClassCastException e) {
+ throw new LifecycleException("Sip Application Dispatcher defined does not implement " + SipApplicationDispatcher.class.getName(),e);
+ }
+ if(logger.isInfoEnabled()) {
+ logger.info("Pretty encoding of headers enabled ? " + usePrettyEncoding);
+ }
+ if(sipPathName == null) {
+ sipPathName = DEFAULT_SIP_PATH_NAME;
+ }
+ if(logger.isInfoEnabled()) {
+ logger.info("Sip Stack path name : " + sipPathName);
+ }
+ sipApplicationDispatcher.setSipService(this);
+ sipApplicationDispatcher.getSipFactory().initialize(sipPathName, usePrettyEncoding);
+
+ String catalinaBase = getCatalinaBase();
+ if(darConfigurationFileLocation != null) {
+ if(!darConfigurationFileLocation.startsWith("file:///")) {
+ darConfigurationFileLocation = "file:///" + catalinaBase.replace(File.separatorChar, '/') + "/" + darConfigurationFileLocation;
+ }
+ System.setProperty("javax.servlet.sip.dar", darConfigurationFileLocation);
+ }
+ super.initInternal();
+ sipApplicationDispatcher.setDomain(this.getName());
+ if(baseTimerInterval < 1) {
+ throw new LifecycleException("It's forbidden to set the Base Timer Interval to a non positive value");
+ }
+ initSipStack();
+ // https://telestax.atlassian.net/browse/MSS-148 make sure we have a default to bin direction for better out of the box experience
+ if(System.getProperty("telscale.license.dir") == null) {
+ System.setProperty("telscale.license.dir", getCatalinaBase() + File.separatorChar + "bin");
+ if(logger.isDebugEnabled()) {
+ logger.debug("Setting telscale.license.dir directory to : " + getCatalinaBase() + File.separatorChar + "bin");
+ }
+ }
+ if(System.getProperty("telscale.license.key.location") == null) {
+ System.setProperty("telscale.license.key.location", getCatalinaBase() + File.separatorChar + "bin");
+ if(logger.isDebugEnabled()) {
+ logger.debug("Setting telscale.license.key.location directory to : " + getCatalinaBase() + File.separatorChar + "bin");
+ }
+ }
+
+ sipApplicationDispatcher.setBaseTimerInterval(baseTimerInterval);
+ sipApplicationDispatcher.setT2Interval(t2Interval);
+ sipApplicationDispatcher.setT4Interval(t4Interval);
+ sipApplicationDispatcher.setTimerDInterval(timerDInterval);
+ sipApplicationDispatcher.setMemoryThreshold(getMemoryThreshold());
+ sipApplicationDispatcher.setBackToNormalMemoryThreshold(backToNormalMemoryThreshold);
+ sipApplicationDispatcher.setCongestionControlCheckingInterval(getCongestionControlCheckingInterval());
+ sipApplicationDispatcher.setCongestionControlPolicyByName(getCongestionControlPolicy());
+ sipApplicationDispatcher.setQueueSize(getSipMessageQueueSize());
+ sipApplicationDispatcher.setBackToNormalQueueSize(backToNormalSipMessageQueueSize);
+ sipApplicationDispatcher.setGatherStatistics(gatherStatistics);
+ sipApplicationDispatcher.setConcurrencyControlMode(ConcurrencyControlMode.valueOf(getConcurrencyControlMode()));
+ sipApplicationDispatcher.setBypassRequestExecutor(bypassRequestExecutor);
+ sipApplicationDispatcher.setBypassResponseExecutor(bypassResponseExecutor);
+ sipApplicationDispatcher.setSipStack(sipStack);
+ sipApplicationDispatcher.init();
+ // Tomcat specific loading case where the connectors are added even before the service is initialized
+ // so we need to set the sip stack before it starts
+ synchronized (connectors) {
+ for (Connector connector : connectors) {
+ ProtocolHandler protocolHandler = connector.getProtocolHandler();
+ if(protocolHandler instanceof SipProtocolHandler) {
+ connector.setPort(((SipProtocolHandler)protocolHandler).getPort());
+ ((SipProtocolHandler)protocolHandler).setSipStack(sipStack);
+ ((SipProtocolHandler)protocolHandler).setAttribute(SipApplicationDispatcher.class.getSimpleName(), sipApplicationDispatcher);
+ registerSipConnector(connector);
+ }
+ }
+ }
+ }
+
+ @Override
+ public void startInternal() throws LifecycleException {
+ super.startInternal();
+ synchronized (connectors) {
+ for (Connector connector : connectors) {
+ final ProtocolHandler protocolHandler = connector.getProtocolHandler();
+ //Jboss sepcific loading case
+ Boolean isSipConnector = false;
+ if (protocolHandler instanceof SipProtocolHandler)
+ isSipConnector = (Boolean) ((SipProtocolHandler)protocolHandler).getAttribute(SipProtocolHandler.IS_SIP_CONNECTOR);
+ if(isSipConnector != null && isSipConnector) {
+ if(logger.isDebugEnabled()) {
+ logger.debug("Attaching the sip application dispatcher " +
+ "as a sip listener to connector listening on port " +
+ connector.getPort());
+ }
+ ((SipProtocolHandler)protocolHandler).setAttribute(SipApplicationDispatcher.class.getSimpleName(), sipApplicationDispatcher);
+ ((SipProtocolHandler)protocolHandler).setSipStack(sipStack);
+ connectorsStartedExternally = true;
+ }
+ //Tomcat specific loading case
+ MobicentsExtendedListeningPoint extendedListeningPoint = null;
+ if (protocolHandler instanceof SipProtocolHandler) {
+ extendedListeningPoint = (MobicentsExtendedListeningPoint)
+ ((SipProtocolHandler)protocolHandler).getAttribute(ExtendedListeningPoint.class.getSimpleName());}
+ if(extendedListeningPoint != null && sipStack != null) {
+ try {
+ extendedListeningPoint.getSipProvider().addSipListener(sipApplicationDispatcher);
+ sipApplicationDispatcher.getSipNetworkInterfaceManager().addExtendedListeningPoint(extendedListeningPoint);
+ connectorsStartedExternally = false;
+ } catch (TooManyListenersException e) {
+ throw new LifecycleException("Couldn't add the sip application dispatcher "
+ + sipApplicationDispatcher + " as a listener to the following listening point provider " + extendedListeningPoint, e);
+ }
+ }
+ }
+ }
+ if(!connectorsStartedExternally) {
+ sipApplicationDispatcher.start();
+ sipApplicationDispatcher.putInService();
+ }
+
+ if(this.getSipMessageQueueSize() <= 0)
+ throw new LifecycleException("Message queue size can not be 0 or less");
+
+ if(logger.isDebugEnabled()) {
+ logger.debug("SIP Standard Service Started.");
+ }
+ }
+
+ public String getJvmRoute() {
+ return this.jvmRoute;
+ }
+
+ public void setJvmRoute(String jvmRoute) {
+ this.jvmRoute = jvmRoute;
+ }
+
+ protected void initSipStack() throws LifecycleException {
+ try {
+ if(logger.isDebugEnabled()) {
+ logger.debug("Initializing SIP stack");
+ }
+
+ // This simply puts HTTP and SSL port numbers in JVM properties menat to be read by jsip ha when sending heart beats with Node description.
+ initializeSystemPortProperties();
+
+ String catalinaBase = getCatalinaBase();
+ if(sipStackPropertiesFileLocation != null && !sipStackPropertiesFileLocation.startsWith("file:///")) {
+ sipStackPropertiesFileLocation = "file:///" + catalinaBase.replace(File.separatorChar, '/') + "/" + sipStackPropertiesFileLocation;
+ }
+ boolean isPropsLoaded = false;
+ if(sipStackProperties == null) {
+ sipStackProperties = new Properties();
+ } else {
+ isPropsLoaded = true;
+ }
+
+ if (logger.isDebugEnabled()) {
+ logger.debug("Loading SIP stack properties from following file : " + sipStackPropertiesFileLocation);
+ }
+ if(sipStackPropertiesFileLocation != null) {
+ //hack to get around space char in path see http://weblogs.java.net/blog/kohsuke/archive/2007/04/how_to_convert.html,
+ // we create a URL since it's permissive enough
+ File sipStackPropertiesFile = null;
+ URL url = null;
+ try {
+ url = new URL(sipStackPropertiesFileLocation);
+ } catch (MalformedURLException e) {
+ logger.fatal("Cannot find the sip stack properties file ! ",e);
+ throw new IllegalArgumentException("The Default Application Router file Location : "+sipStackPropertiesFileLocation+" is not valid ! ",e);
+ }
+ try {
+ sipStackPropertiesFile = new File(new URI(sipStackPropertiesFileLocation));
+ } catch (URISyntaxException e) {
+ //if the uri contains space this will fail, so getting the path will work
+ sipStackPropertiesFile = new File(url.getPath());
+ }
+ FileInputStream sipStackPropertiesInputStream = null;
+ try {
+ sipStackPropertiesInputStream = new FileInputStream(sipStackPropertiesFile);
+ sipStackProperties.load(sipStackPropertiesInputStream);
+ } catch (Exception e) {
+ logger.warn("Could not find or problem when loading the sip stack properties file : " + sipStackPropertiesFileLocation, e);
+ } finally {
+ if(sipStackPropertiesInputStream != null) {
+ try {
+ sipStackPropertiesInputStream.close();
+ } catch (IOException e) {
+ logger.error("fail to close the following file " + sipStackPropertiesFile.getAbsolutePath(), e);
+ }
+ }
+ }
+
+ String debugLog = sipStackProperties.getProperty(DEBUG_LOG_STACK_PROP);
+ if(debugLog != null && debugLog.length() > 0 && !debugLog.startsWith("file:///")) {
+ sipStackProperties.setProperty(DEBUG_LOG_STACK_PROP,
+ catalinaBase + "/" + debugLog);
+ }
+ String serverLog = sipStackProperties.getProperty(SERVER_LOG_STACK_PROP);
+ if(serverLog != null && serverLog.length() > 0 && !serverLog.startsWith("file:///")) {
+ sipStackProperties.setProperty(SERVER_LOG_STACK_PROP,
+ catalinaBase + "/" + serverLog);
+ }
+ // The whole MSS is built upon those assumptions, so those properties are not overrideable
+ if (sipStackProperties.getProperty(AUTOMATIC_DIALOG_SUPPORT_STACK_PROP) == null) {
+ //https://github.com/RestComm/sip-servlets/issues/143
+ //set off if user didnt provided any value.
+ sipStackProperties.setProperty(AUTOMATIC_DIALOG_SUPPORT_STACK_PROP, "off");
+ }
+ sipStackProperties.setProperty(LOOSE_DIALOG_VALIDATION, "true");
+ sipStackProperties.setProperty(PASS_INVITE_NON_2XX_ACK_TO_LISTENER, "true");
+ isPropsLoaded = true;
+ } else {
+ logger.warn("no sip stack properties file defined ");
+ }
+ if(!isPropsLoaded) {
+ logger.warn("loading default Mobicents Sip Servlets sip stack properties");
+ // Silently set default values
+ sipStackProperties.setProperty("gov.nist.javax.sip.LOG_MESSAGE_CONTENT",
+ "true");
+ sipStackProperties.setProperty("gov.nist.javax.sip.TRACE_LEVEL",
+ "32");
+ sipStackProperties.setProperty(DEBUG_LOG_STACK_PROP,
+ catalinaBase + "/" + "mss-jsip-" + getName() +"-debug.txt");
+ sipStackProperties.setProperty(SERVER_LOG_STACK_PROP,
+ catalinaBase + "/" + "mss-jsip-" + getName() +"-messages.xml");
+ sipStackProperties.setProperty("javax.sip.STACK_NAME", "mss-" + getName());
+ sipStackProperties.setProperty(AUTOMATIC_DIALOG_SUPPORT_STACK_PROP, "off");
+ sipStackProperties.setProperty("gov.nist.javax.sip.DELIVER_UNSOLICITED_NOTIFY", "true");
+ sipStackProperties.setProperty("gov.nist.javax.sip.THREAD_POOL_SIZE", "64");
+ sipStackProperties.setProperty("gov.nist.javax.sip.REENTRANT_LISTENER", "true");
+ sipStackProperties.setProperty("gov.nist.javax.sip.MAX_FORK_TIME_SECONDS", "0");
+ sipStackProperties.setProperty(LOOSE_DIALOG_VALIDATION, "true");
+ sipStackProperties.setProperty(PASS_INVITE_NON_2XX_ACK_TO_LISTENER, "true");
+ sipStackProperties.setProperty("gov.nist.javax.sip.AUTOMATIC_DIALOG_ERROR_HANDLING", "false");
+ sipStackProperties.setProperty("gov.nist.javax.sip.AGGRESSIVE_CLEANUP", "true");
+ }
+
+ if(sipStackProperties.get(TCP_POST_PARSING_THREAD_POOL_SIZE) == null) {
+ sipStackProperties.setProperty(TCP_POST_PARSING_THREAD_POOL_SIZE, "30");
+ }
+
+ // set the DNSServerLocator allowing to support RFC 3263 and do DNS lookups to resolve uris
+ if(dnsServerLocatorClass != null && dnsServerLocatorClass.trim().length() > 0) {
+ if(logger.isDebugEnabled()) {
+ logger.debug("Sip Stack " + sipStackProperties.getProperty("javax.sip.STACK_NAME") +" will be using " + dnsServerLocatorClass + " as DNSServerLocator");
+ }
+ try {
+ // create parameters argument to identify constructor
+ Class[] paramTypes = new Class[0];
+ // get constructor of AddressResolver in order to instantiate
+ Constructor dnsServerLocatorConstructor = Class.forName(dnsServerLocatorClass).getConstructor(
+ paramTypes);
+ // Wrap properties object in order to pass to constructor of AddressResolver
+ Object[] conArgs = new Object[0];
+ // Creates a new instance of AddressResolver Class with the supplied sipApplicationDispatcher.
+ DNSServerLocator dnsServerLocator = (DNSServerLocator) dnsServerLocatorConstructor.newInstance(conArgs);
+ sipApplicationDispatcher.setDNSServerLocator(dnsServerLocator);
+ sipApplicationDispatcher.setDNSTimeout(dnsTimeout);
+ if(sipStackProperties.getProperty("javax.sip.ROUTER_PATH") == null) {
+ sipStackProperties.setProperty("javax.sip.ROUTER_PATH", DNSAwareRouter.class.getCanonicalName());
+ }
+ } catch (Exception e) {
+ logger.error("Couldn't set the AddressResolver " + addressResolverClass, e);
+ throw e;
+ }
+ } else {
+ if(logger.isInfoEnabled()) {
+ logger.info("no DNSServerLocator will be used since none has been specified.");
+ }
+ }
+
+ String serverHeaderValue = sipStackProperties.getProperty(SERVER_HEADER);
+ if(serverHeaderValue != null) {
+ List--- To see the contents of the file -my_next_bestselling_novel in your current working directory, enter thecat my_next_bestselling_novel command at the shell prompt and pressEnter to execute the command. -
--- Press -Enter to execute the command. -- Press -to switch to the first virtual terminal. Press Ctrl Alt F1 to return to your X-Windows session. - Ctrl Alt F7
--- File-related classes include -filesystem for file systems,file for files, anddir for directories. Each class has its own associated set of permissions. -
--- Choose -System > Preferences > Mouse from the main menu bar to launchMouse Preferences . In theButtons tab, click theLeft-handed mouse check box and clickClose to switch the primary mouse button from the left to the right (making the mouse suitable for use in the left hand). -- To insert a special character into a -gedit file, chooseApplications > Accessories > Character Map from the main menu bar. Next, chooseSearch > Find… from theCharacter Map menu bar, type the name of the character in theSearch field and clickNext . The character you sought will be highlighted in theCharacter Table . Double-click this highlighted character to place it in theText to copy field and then click theCopy button. Now switch back to your document and chooseEdit > Paste from thegedit menu bar. -
--- To connect to a remote machine using ssh, type -ssh at a shell prompt. If the remote machine isusername @domain.name example.com and your username on that machine is john, typessh john@example.com . -- The -mount -o remount command remounts the named file system. For example, to remount thefile-system /home file system, the command ismount -o remount /home . -- To see the version of a currently installed package, use the -rpm -q command. It will return a result as follows:package . - package-version-release
--- When the Apache HTTP Server accepts requests, it dispatches child processes or threads to handle them. This group of child processes or threads is known as a -server-pool . Under Apache HTTP Server 2.0, the responsibility for creating and maintaining these server-pools has been abstracted to a group of modules calledMulti-Processing Modules (MPMs ). Unlike other modules, only one module from the MPM group can be loaded by the Apache HTTP Server. -
over top ofif you prefer) a clean
Started in 23s:648ms):
resource lists), policy data, and many others.
entityattribute.
presenceevent package.
over top ofif you prefer) a clean
any-configurationphone, make a call to -
appsessionHTTP GET parameter of the request. If the desired behaviour group these requests, we can just make sure the affinity values (user name and GET parameter) are the same.
org.mobicents.servlet.sip.example.SimpleApplication. From the Sip Servlets Management Console you can make sure it is subscribed to receive
-{
- /**
- * Given a {@link Property} and its {@link PropertyDefinition}, as well as a {@link MetaValue}, populate that
- * MetaValue so it corresponds to the Property. If the Property is a list or a map, all descendant Property's should
- * be added as descendants of the MetaValue.
- *
- * @param property the property to be copied
- * @param metaValue the MetaValue to be populated; should not be null
- * @param propertyDefinition the property's definition
- */
- public void populateMetaValueFromProperty(P property, MetaValue metaValue, D propertyDefinition);
-
- /**
- * Given a {@link Property} and its {@link PropertyDefinition}, create and return a corresponding {@link MetaValue}
- * with the specified {@link MetaType}. If the Property is a list or a map, all descendant Property's should
- * be represented as descendants of the returned MetaValue. Generally this method can simply create an empty
- * MetaValue object and then delegate the population of the guts of the object to
- * {@link #populateMetaValueFromProperty(Property, MetaValue, PropertyDefinition)}.
- *
- * @param property the property to be converted
- * @param propertyDefinition the property's definition
- * @param metaType the type of MetaValue that should be created and returned
- * @return the MetaValue representation of the given Property
- */
- public MetaValue convertToMetaValue(P property, D propertyDefinition, MetaType metaType);
-
- /**
- * Given a {@link Property} and its {@link PropertyDefinition}, as well as a {@link MetaValue}, populate the
- * Property so it corresponds to the MetaValue. If the MetaValue is a from of list or map, all descendant
- * MetaValue's should be added as descendants of the Property.
- *
- * @param property the property to be populated; should not be null
- * @param metaValue the MetaValue to be copied
- * @param propertyDefinition the property's definition
- */
- public void populatePropertyFromMetaValue(P property, MetaValue metaValue, D propertyDefinition);
-
- /**
- * Given a {@link MetaValue}, create and return a corresponding {@link Property}
- * with the specified {@link PropertyDefinition}. If the MetaValue is a form of list or map, all descendant
- * MetaValue's should be represented as descendants of the returned Property. Generally this method can simply
- * create an empty Property object and then delegate the population of the guts of the object to
- * {@link #populatePropertyFromMetaValue(Property, MetaValue, PropertyDefinition)}.
- *
- * @param metaValue the metaValue to be converted
- * @param propertyDefinition the definition of the property to be created and returned
- * @return the Property representation of the given MetaValue
- */
- public P convertToProperty(MetaValue metaValue, D propertyDefinition);
-}
diff --git a/management/jopr-plugin-as-5/src/main/java/org/rhq/plugins/jbossas5/adapter/api/PropertyAdapterFactory.java b/management/jopr-plugin-as-5/src/main/java/org/rhq/plugins/jbossas5/adapter/api/PropertyAdapterFactory.java
deleted file mode 100644
index 78f6c35712..0000000000
--- a/management/jopr-plugin-as-5/src/main/java/org/rhq/plugins/jbossas5/adapter/api/PropertyAdapterFactory.java
+++ /dev/null
@@ -1,126 +0,0 @@
- /*
- * Jopr Management Platform
- * Copyright (C) 2005-2009 Red Hat, Inc.
- * All rights reserved.
- *
- * This program is free software; you can redistribute it and/or modify
- * it under the terms of the GNU General Public License, version 2, as
- * published by the Free Software Foundation, and/or the GNU Lesser
- * General Public License, version 2.1, also as published by the Free
- * Software Foundation.
- *
- * This program 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 General Public License and the GNU Lesser General Public License
- * for more details.
- *
- * You should have received a copy of the GNU General Public License
- * and the GNU Lesser General Public License along with this program;
- * if not, write to the Free Software Foundation, Inc.,
- * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
- */
-package org.rhq.plugins.jbossas5.adapter.api;
-
- import org.apache.commons.logging.Log;
- import org.apache.commons.logging.LogFactory;
-
- import org.jboss.metatype.api.types.MetaType;
- import org.jboss.metatype.api.types.MapCompositeMetaType;
- import org.jboss.metatype.api.types.PropertiesMetaType;
- import org.jboss.metatype.api.values.MetaValue;
-
- import org.rhq.core.domain.configuration.PropertySimple;
- import org.rhq.plugins.jbossas5.adapter.impl.configuration.PropertyListToArrayValueAdapter;
- import org.rhq.plugins.jbossas5.adapter.impl.configuration.PropertyListToCollectionValueAdapter;
- import org.rhq.plugins.jbossas5.adapter.impl.configuration.PropertyMapToMapCompositeValueSupportAdapter;
- import org.rhq.plugins.jbossas5.adapter.impl.configuration.PropertyMapToTableValueAdapter;
- import org.rhq.plugins.jbossas5.adapter.impl.configuration.PropertySimpleToSimpleValueAdapter;
- import org.rhq.plugins.jbossas5.adapter.impl.configuration.PropertyMapToGenericValueAdapter;
- import org.rhq.plugins.jbossas5.adapter.impl.configuration.PropertySimpleToEnumValueAdapter;
- import org.rhq.plugins.jbossas5.adapter.impl.configuration.PropertyMapToCompositeValueSupportAdapter;
- import org.rhq.plugins.jbossas5.adapter.impl.configuration.PropertyMapToPropertiesValueAdapter;
- import org.jetbrains.annotations.Nullable;
-
- /**
- * @author Mark Spritzler
- */
-public class PropertyAdapterFactory
-{
- private static final Log LOG = LogFactory.getLog(PropertyAdapterFactory.class);
-
- public static PropertyAdapter getPropertyAdapter(MetaValue metaValue)
- {
- if (metaValue == null)
- {
- LOG.debug("The MetaValue passed in is null.");
- return null;
- }
- MetaType metaType = metaValue.getMetaType();
- return getPropertyAdapter(metaType);
- }
-
- public static PropertyAdapter getPropertyAdapter(MetaType metaType)
- {
- PropertyAdapter propertyAdapter = null;
- if (metaType.isSimple())
- {
- propertyAdapter = new PropertySimpleToSimpleValueAdapter();
- }
- else if (metaType.isGeneric())
- {
- propertyAdapter = new PropertyMapToGenericValueAdapter();
- }
- else if (metaType.isComposite())
- {
- if (metaType instanceof MapCompositeMetaType)
- propertyAdapter = new PropertyMapToMapCompositeValueSupportAdapter();
- else
- propertyAdapter = new PropertyMapToCompositeValueSupportAdapter();
- }
- else if (metaType.isTable())
- {
- propertyAdapter = new PropertyMapToTableValueAdapter();
- }
- else if (metaType.isCollection())
- {
- propertyAdapter = new PropertyListToCollectionValueAdapter();
- }
- else if (metaType.isArray())
- {
- propertyAdapter = new PropertyListToArrayValueAdapter();
- }
- else if (metaType.isEnum())
- {
- propertyAdapter = new PropertySimpleToEnumValueAdapter();
- }
- else if (metaType instanceof PropertiesMetaType)
- {
- propertyAdapter = new PropertyMapToPropertiesValueAdapter();
- }
- else
- {
- LOG.warn("Unsupported MetaType: " + metaType);
- }
- return propertyAdapter;
- }
-
- @Nullable
- public static PropertyAdapter getCustomPropertyAdapter(PropertySimple customProp)
- {
- if (customProp == null)
- return null;
- String adapterClassName = customProp.getStringValue();
- PropertyAdapter propertyAdapter = null;
- try
- {
- Class adapterClass = Class.forName(adapterClassName);
- propertyAdapter = (PropertyAdapter) adapterClass.newInstance();
- }
- catch (Exception e)
- {
- LOG.error("Unable to create custom adapter class for " + customProp + ".", e);
- }
- return propertyAdapter;
- }
-}
diff --git a/management/jopr-plugin-as-5/src/main/java/org/rhq/plugins/jbossas5/adapter/impl/configuration/AbstractPropertyMapToCompositeValueAdapter.java b/management/jopr-plugin-as-5/src/main/java/org/rhq/plugins/jbossas5/adapter/impl/configuration/AbstractPropertyMapToCompositeValueAdapter.java
deleted file mode 100644
index 339264556c..0000000000
--- a/management/jopr-plugin-as-5/src/main/java/org/rhq/plugins/jbossas5/adapter/impl/configuration/AbstractPropertyMapToCompositeValueAdapter.java
+++ /dev/null
@@ -1,114 +0,0 @@
-/*
-* Jopr Management Platform
-* Copyright (C) 2005-2009 Red Hat, Inc.
-* All rights reserved.
-*
-* This program is free software; you can redistribute it and/or modify
-* it under the terms of the GNU General Public License, version 2, as
-* published by the Free Software Foundation, and/or the GNU Lesser
-* General Public License, version 2.1, also as published by the Free
-* Software Foundation.
-*
-* This program 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 General Public License and the GNU Lesser General Public License
-* for more details.
-*
-* You should have received a copy of the GNU General Public License
-* and the GNU Lesser General Public License along with this program;
-* if not, write to the Free Software Foundation, Inc.,
-* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
-*/
-package org.rhq.plugins.jbossas5.adapter.impl.configuration;
-
-import java.util.Set;
-
-import org.apache.commons.logging.Log;
-import org.apache.commons.logging.LogFactory;
-
-import org.jboss.metatype.api.types.MetaType;
-import org.jboss.metatype.api.types.SimpleMetaType;
-import org.jboss.metatype.api.values.CompositeValue;
-import org.jboss.metatype.api.values.MetaValue;
-
-import org.rhq.core.domain.configuration.Property;
-import org.rhq.core.domain.configuration.PropertyMap;
-import org.rhq.core.domain.configuration.PropertySimple;
-import org.rhq.core.domain.configuration.definition.PropertyDefinition;
-import org.rhq.core.domain.configuration.definition.PropertyDefinitionMap;
-import org.rhq.plugins.jbossas5.adapter.api.AbstractPropertyMapAdapter;
-import org.rhq.plugins.jbossas5.adapter.api.PropertyAdapter;
-import org.rhq.plugins.jbossas5.adapter.api.PropertyAdapterFactory;
-
-/**
- * This class provides code that maps back and forth between a {@link PropertyMap} and
- * a {@link CompositeValue}. Subclasses exist for the two different implementations
- * of If the connection has already been established, return the object reference to it. If not, attempt to make
- * a live connection to the JMX server. If the connection could not be made in the {@link #start(org.rhq.core.pluginapi.inventory.ResourceContext)}
- * method, this method will effectively try to load the connection on each attempt to use it. As such, multiple
- * threads may attempt to access the connection through this means at a time. Therefore, the method has been
- * made synchronized on instances of the class. If any errors are encountered, this method will log the error, taking into account logic to prevent spamming
- * the log file. Calling methods should take care to not redundantly log the exception thrown by this method.CompositeValue
.
- *
- * @author Mark Spritzler
- * @author Ian Springer
- */
-public abstract class AbstractPropertyMapToCompositeValueAdapter extends AbstractPropertyMapAdapter
- implements PropertyAdapterCompositeValueSupport
is a
- * {@link CompositeValue} implementation that contains items that may be of different
- * types.
- *
- * @author Ian Springer
- */
-public class PropertyMapToCompositeValueSupportAdapter extends AbstractPropertyMapToCompositeValueAdapter
- implements PropertyAdapterMapCompositeValueSupport
is a
- * {@link CompositeValue} implementation that contains items that are all of the same
- * type.
- *
- * @author Ian Springer
- */
-public class PropertyMapToMapCompositeValueSupportAdapter extends AbstractPropertyMapToCompositeValueAdapter implements PropertyAdapterPropertiesMetaValue
extends
- * java.util.Properties. Its associated PropertiesMetaType may or may not define
- * member property names; since the RHQ Configuration system assumes a
- * PropertyDefinitionMap either defines none of its member properties (i.e. an "open
- * map") or defines all of them, we just ignore the PropertiesMetaType and always map
- * to an open map.
- *
- * @author Ian Springer
- */
-public class PropertyMapToPropertiesValueAdapter extends AbstractPropertyMapAdapter
- implements PropertyAdapter
- * <c:list-property name="securityConfig" min="2" max="4"
- * displayName="Security Configurations"
- * description="This element specifies a XML fragment which describes the access control list to be
- * used by the SecurityManager to authorize client operations against the destination.
- * The content model is the same as for the SecurityManager SecurityConf attribute.">
- * <c:map-property name="role"
- * displayName="Security Configuration Attributes"
- * description="These are the attributes that define the role name, and if the role is allowed to
- * read, write or create Messages on this Queue.">
- * <c:simple-property name="name"
- * description="Name of the Security Role. e.g. Guest"
- * summary="true" required="true"/>
- * <c:simple-property name="read"
- * description="Is this role allowed to read messages?"
- * summary="true" required="false" type="boolean"/>
- * <c:simple-property name="write"
- * description="Is this role allowed to write messages?"
- * summary="true" required="false" type="boolean"/>
- * <c:simple-property name="create"
- * description="Is this role allowed to create messages?"
- * summary="true" required="false" type="boolean"/>
- * </c:map-property>
- * </c:list-property>
- *
- *
- * @author Ian Springer
- */
-public class JMSSecurityConfigAdapter extends AbstractPropertyListAdapter
-{
- public MetaValue convertToMetaValue(PropertyList propertyList, PropertyDefinitionList propertyDefinitionList,
- MetaType metaType)
- {
- MapCompositeMetaType securityConfigCompositeMetaType = (MapCompositeMetaType)metaType;
- MapCompositeValueSupport securityConfigCompositeValue =
- new MapCompositeValueSupport(securityConfigCompositeMetaType.getValueType());
- populateMetaValueFromProperty(propertyList, securityConfigCompositeValue, propertyDefinitionList);
- return securityConfigCompositeValue;
- }
-
- public void populateMetaValueFromProperty(PropertyList propertyList, MetaValue metaValue,
- PropertyDefinitionList propertyDefinitionList)
- {
- MapCompositeValueSupport securityConfigCompositeValue = (MapCompositeValueSupport)metaValue;
- PropertyDefinitionMap memberPropDefMap = (PropertyDefinitionMap)propertyDefinitionList.getMemberDefinition();
- for (Property memberProperty : propertyList.getList()) {
- PropertyMap memberPropMap = (PropertyMap)memberProperty;
- String roleName = memberPropMap.getSimple("name").getStringValue();
- CompositeValueSupport roleCompositeValue = (CompositeValueSupport)createCompositeValue(memberPropDefMap);
- populateMetaValueFromProperty(memberPropMap, roleCompositeValue, memberPropDefMap);
- securityConfigCompositeValue.put(roleName, roleCompositeValue);
- }
- }
-
- public void populatePropertyFromMetaValue(PropertyList propertyList, MetaValue metaValue,
- PropertyDefinitionList propertyDefinitionList)
- {
- CompositeValue compositeValue = (CompositeValue)metaValue;
- CompositeMetaType compositeMetaType = compositeValue.getMetaType();
- PropertyDefinitionMap memberPropertyDefinitionMap = (PropertyDefinitionMap)propertyDefinitionList.getMemberDefinition();
- PropertyMapToMapCompositeValueSupportAdapter mapToMapCompositeValueAdapter = new PropertyMapToMapCompositeValueSupportAdapter();
- for (String memberName : compositeMetaType.itemSet()) {
- MetaValue memberMetaValue = compositeValue.get(memberName);
- PropertyMap memberPropertyMap = mapToMapCompositeValueAdapter.convertToProperty(memberMetaValue, memberPropertyDefinitionMap);
- memberPropertyMap.put(new PropertySimple("name", memberName)); // add a simple for the role name to the map
- propertyList.add(memberPropertyMap);
- }
- }
-
- // NOTE: We can't just leverage PropertyMapToCompositeValueSupportAdapter, because we have to skip the "name" map member.
- private CompositeValue createCompositeValue(PropertyDefinitionMap propDefMap) {
- String name = (propDefMap != null) ?
- propDefMap.getName() : "CompositeMetaType";
- String desc = (propDefMap != null && propDefMap.getDescription() != null) ?
- propDefMap.getDescription() : "none";
- MutableCompositeMetaType compositeMetaType = new MutableCompositeMetaType(name, desc);
- if (propDefMap != null) {
- for (PropertyDefinition mapMemberPropDef : propDefMap.getPropertyDefinitions().values()) {
- if (mapMemberPropDef.getName().equals("name"))
- continue;
- String mapMemberDesc = (propDefMap.getDescription() != null) ? propDefMap.getDescription() : "none";
- MetaType mapMemberMetaType = ConversionUtils.convertPropertyDefinitionToMetaType(mapMemberPropDef);
- compositeMetaType.addItem(mapMemberPropDef.getName(), mapMemberDesc, mapMemberMetaType);
- }
- }
- return new CompositeValueSupport(compositeMetaType);
- }
-
- // NOTE: We can't just leverage PropertyMapToCompositeValueSupportAdapter, because we have to skip the "name" map member.
- private void populateMetaValueFromProperty(PropertyMap propMap, MetaValue metaValue, PropertyDefinitionMap propDefMap) {
- CompositeValueSupport compositeValue = (CompositeValueSupport)metaValue;
- for (String mapMemberPropName : propMap.getMap().keySet()) {
- if (mapMemberPropName.equals("name"))
- continue;
- Property mapMemberProp = propMap.get(mapMemberPropName);
- PropertyDefinition mapMemberPropDef = propDefMap.get(mapMemberPropName);
- MetaType mapMemberMetaType = compositeValue.getMetaType().getType(mapMemberPropName);
- if (mapMemberMetaType == null) {
- // this will occur when new map properties are added since they are not present
- // in the original metaValue which we are using
- mapMemberMetaType = SimpleMetaType.STRING;
- }
- PropertyAdapter adapter = PropertyAdapterFactory.getPropertyAdapter(mapMemberMetaType);
- MetaValue mapMemberMetaValue = adapter.convertToMetaValue(mapMemberProp, mapMemberPropDef, mapMemberMetaType);
- compositeValue.set(mapMemberPropName, mapMemberMetaValue);
- }
- }
-}
diff --git a/management/jopr-plugin-as-5/src/main/java/org/rhq/plugins/jbossas5/adapter/impl/measurement/SimpleMetaValueMeasurementAdapter.java b/management/jopr-plugin-as-5/src/main/java/org/rhq/plugins/jbossas5/adapter/impl/measurement/SimpleMetaValueMeasurementAdapter.java
deleted file mode 100644
index 6a4ac814e8..0000000000
--- a/management/jopr-plugin-as-5/src/main/java/org/rhq/plugins/jbossas5/adapter/impl/measurement/SimpleMetaValueMeasurementAdapter.java
+++ /dev/null
@@ -1,67 +0,0 @@
-/*
-* Jopr Management Platform
-* Copyright (C) 2005-2009 Red Hat, Inc.
-* All rights reserved.
-*
-* This program is free software; you can redistribute it and/or modify
-* it under the terms of the GNU General Public License, version 2, as
-* published by the Free Software Foundation, and/or the GNU Lesser
-* General Public License, version 2.1, also as published by the Free
-* Software Foundation.
-*
-* This program 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 General Public License and the GNU Lesser General Public License
-* for more details.
-*
-* You should have received a copy of the GNU General Public License
-* and the GNU Lesser General Public License along with this program;
-* if not, write to the Free Software Foundation, Inc.,
-* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
-*/
-package org.rhq.plugins.jbossas5.adapter.impl.measurement;
-
-import org.apache.commons.logging.Log;
-import org.apache.commons.logging.LogFactory;
-
-import org.jboss.metatype.api.values.MetaValue;
-import org.jboss.metatype.api.values.SimpleValueSupport;
-
-import org.rhq.core.domain.measurement.DataType;
-import org.rhq.core.domain.measurement.MeasurementDataNumeric;
-import org.rhq.core.domain.measurement.MeasurementDataTrait;
-import org.rhq.core.domain.measurement.MeasurementDefinition;
-import org.rhq.core.domain.measurement.MeasurementReport;
-import org.rhq.core.domain.measurement.MeasurementScheduleRequest;
-import org.rhq.plugins.jbossas5.adapter.api.MeasurementAdapter;
-
-/**
- * TODO
- */
-public class SimpleMetaValueMeasurementAdapter implements MeasurementAdapter {
- private final Log LOG = LogFactory.getLog(SimpleMetaValueMeasurementAdapter.class);
-
- public void setMeasurementData(MeasurementReport report, MetaValue metaValue, MeasurementScheduleRequest request, MeasurementDefinition measurementDefinition) {
- SimpleValueSupport simpleValue = (SimpleValueSupport)metaValue;
- DataType dataType = measurementDefinition.getDataType();
- switch (dataType) {
- case MEASUREMENT:
- try {
- MeasurementDataNumeric dataNumeric = new MeasurementDataNumeric(request, new Double(simpleValue.getValue().toString()));
- report.addData(dataNumeric);
- }
- catch (NumberFormatException e) {
- LOG.warn("Measurement request: " + request.getName() + " did not return a numeric value from the Profile Service", e);
- }
- break;
- case TRAIT:
- MeasurementDataTrait dataTrait = new MeasurementDataTrait(request, String.valueOf(simpleValue.getValue()));
- report.addData(dataTrait);
- break;
- default:
- throw new IllegalStateException("Unsupported measurement data type: " + dataType);
-
- }
- }
-}
diff --git a/management/jopr-plugin-as-5/src/main/java/org/rhq/plugins/jbossas5/adapter/impl/measurement/custom/JMSMessageCounterAdapter.java b/management/jopr-plugin-as-5/src/main/java/org/rhq/plugins/jbossas5/adapter/impl/measurement/custom/JMSMessageCounterAdapter.java
deleted file mode 100644
index 4b2563d5fc..0000000000
--- a/management/jopr-plugin-as-5/src/main/java/org/rhq/plugins/jbossas5/adapter/impl/measurement/custom/JMSMessageCounterAdapter.java
+++ /dev/null
@@ -1,57 +0,0 @@
- /*
- * Jopr Management Platform
- * Copyright (C) 2005-2009 Red Hat, Inc.
- * All rights reserved.
- *
- * This program is free software; you can redistribute it and/or modify
- * it under the terms of the GNU General Public License, version 2, as
- * published by the Free Software Foundation, and/or the GNU Lesser
- * General Public License, version 2.1, also as published by the Free
- * Software Foundation.
- *
- * This program 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 General Public License and the GNU Lesser General Public License
- * for more details.
- *
- * You should have received a copy of the GNU General Public License
- * and the GNU Lesser General Public License along with this program;
- * if not, write to the Free Software Foundation, Inc.,
- * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
- */
- package org.rhq.plugins.jbossas5.adapter.impl.measurement.custom;
-
-import org.jboss.metatype.api.values.CompositeValueSupport;
-import org.jboss.metatype.api.values.MetaValue;
-import org.jboss.metatype.api.values.SimpleValueSupport;
-import org.rhq.core.domain.measurement.DataType;
-import org.rhq.core.domain.measurement.MeasurementDataNumeric;
-import org.rhq.core.domain.measurement.MeasurementDataTrait;
-import org.rhq.core.domain.measurement.MeasurementDefinition;
-import org.rhq.core.domain.measurement.MeasurementReport;
-import org.rhq.core.domain.measurement.MeasurementScheduleRequest;
-import org.rhq.plugins.jbossas5.adapter.api.MeasurementAdapter;
-
-public class JMSMessageCounterAdapter implements MeasurementAdapter
-{
- public void setMeasurementData(MeasurementReport report, MetaValue metaValue, MeasurementScheduleRequest request, MeasurementDefinition measurementDefinition)
- {
- // TODO: fix this
- CompositeValueSupport compositeValue = (CompositeValueSupport) metaValue;
- DataType dataType = measurementDefinition.getDataType();
- String metricName = request.getName();
- if (dataType.equals(DataType.MEASUREMENT))
- {
- //@todo break out the getting the value out of the ValueSupport object
- MeasurementDataNumeric dataNumeric = new MeasurementDataNumeric(request, (Double) ((SimpleValueSupport) (compositeValue.get(metricName))).getValue());
- report.addData(dataNumeric);
- }
- else if (dataType.equals(DataType.TRAIT))
- {
- //@todo break out the getting the value out of the ValueSupport object
- MeasurementDataTrait dataTrait = new MeasurementDataTrait(request, (String) ((SimpleValueSupport) (compositeValue.get(metricName))).getValue());
- report.addData(dataTrait);
- }
- }
-}
diff --git a/management/jopr-plugin-as-5/src/main/java/org/rhq/plugins/jbossas5/adapter/impl/measurement/custom/JMSMessageStatisticsAdapter.java b/management/jopr-plugin-as-5/src/main/java/org/rhq/plugins/jbossas5/adapter/impl/measurement/custom/JMSMessageStatisticsAdapter.java
deleted file mode 100644
index b3cc172dd0..0000000000
--- a/management/jopr-plugin-as-5/src/main/java/org/rhq/plugins/jbossas5/adapter/impl/measurement/custom/JMSMessageStatisticsAdapter.java
+++ /dev/null
@@ -1,37 +0,0 @@
- /*
- * Jopr Management Platform
- * Copyright (C) 2005-2009 Red Hat, Inc.
- * All rights reserved.
- *
- * This program is free software; you can redistribute it and/or modify
- * it under the terms of the GNU General Public License, version 2, as
- * published by the Free Software Foundation, and/or the GNU Lesser
- * General Public License, version 2.1, also as published by the Free
- * Software Foundation.
- *
- * This program 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 General Public License and the GNU Lesser General Public License
- * for more details.
- *
- * You should have received a copy of the GNU General Public License
- * and the GNU Lesser General Public License along with this program;
- * if not, write to the Free Software Foundation, Inc.,
- * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
- */
- package org.rhq.plugins.jbossas5.adapter.impl.measurement.custom;
-
-import org.rhq.plugins.jbossas5.adapter.api.MeasurementAdapter;
-import org.rhq.core.domain.measurement.MeasurementReport;
-import org.rhq.core.domain.measurement.MeasurementScheduleRequest;
-import org.rhq.core.domain.measurement.MeasurementDefinition;
-import org.jboss.metatype.api.values.MetaValue;
-
-public class JMSMessageStatisticsAdapter implements MeasurementAdapter
-{
- public void setMeasurementData(MeasurementReport report, MetaValue metaValue, MeasurementScheduleRequest request, MeasurementDefinition measurementDefinition)
- {
- // TODO
- }
-}
diff --git a/management/jopr-plugin-as-5/src/main/java/org/rhq/plugins/jbossas5/factory/ProfileServiceFactory.java b/management/jopr-plugin-as-5/src/main/java/org/rhq/plugins/jbossas5/factory/ProfileServiceFactory.java
deleted file mode 100644
index f48210d568..0000000000
--- a/management/jopr-plugin-as-5/src/main/java/org/rhq/plugins/jbossas5/factory/ProfileServiceFactory.java
+++ /dev/null
@@ -1,227 +0,0 @@
- /*
- * Jopr Management Platform
- * Copyright (C) 2005-2009 Red Hat, Inc.
- * All rights reserved.
- *
- * This program is free software; you can redistribute it and/or modify
- * it under the terms of the GNU General Public License, version 2, as
- * published by the Free Software Foundation, and/or the GNU Lesser
- * General Public License, version 2.1, also as published by the Free
- * Software Foundation.
- *
- * This program 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 General Public License and the GNU Lesser General Public License
- * for more details.
- *
- * You should have received a copy of the GNU General Public License
- * and the GNU Lesser General Public License along with this program;
- * if not, write to the Free Software Foundation, Inc.,
- * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
- */
-package org.rhq.plugins.jbossas5.factory;
-
-import java.util.Set;
-
-import javax.naming.InitialContext;
-import javax.naming.NamingException;
-
-import org.apache.commons.logging.Log;
-import org.apache.commons.logging.LogFactory;
-import org.jetbrains.annotations.NotNull;
-
-import org.jboss.deployers.spi.management.ManagementView;
-import org.jboss.deployers.spi.management.deploy.DeploymentManager;
-import org.jboss.managed.api.ComponentType;
-import org.jboss.managed.api.ManagedComponent;
-import org.jboss.profileservice.spi.ProfileKey;
-import org.jboss.profileservice.spi.ProfileService;
-
-/**
- * Factory class to get the ProfileService and Profile Service related objects directly from the
- * ProfileService object.
- *
- * @author Mark Spritzler
- */
-public class ProfileServiceFactory
-{
- private static final Log LOG = LogFactory.getLog(ProfileServiceFactory.class);
-
- private static final String PROFILE_SERVICE_JNDI_NAME = "ProfileService";
-
- private static ProfileService profileService;
-
- private static ManagementView currentProfileView;
-
- private static final ProfileKey DEFAULT_PROFILE_KEY = new ProfileKey(ProfileKey.DEFAULT);
-
- /**
- * Returns the profile service from the JBoss server through JNDI
- *
- * @return ProfileService
- */
- @NotNull
- public static ProfileService getProfileService()
- {
- if (profileService == null)
- {
- InitialContext initialContext;
- try
- {
- initialContext = new InitialContext();
- }
- catch (NamingException e)
- {
- throw new RuntimeException("Failed to create JNDI InitialContext.", e);
- }
- try
- {
- profileService = (ProfileService) initialContext.lookup(PROFILE_SERVICE_JNDI_NAME);
- }
- catch (NamingException e)
- {
- throw new RuntimeException("Failed to lookup JNDI name '" + PROFILE_SERVICE_JNDI_NAME
- + "' from InitialContext.", e);
- }
- }
- return profileService;
- }
-
- /**
- * Get the current profile's Management view. This will get the domains from the profile service
- * and return the first one in the list.
- *
- * @return ManagementView the management view of the first domain
- */
- public static ManagementView getCurrentProfileView()
- {
- if (currentProfileView == null)
- {
- currentProfileView = getProfileService().getViewManager();
- refreshCurrentProfileView();
- }
- return currentProfileView;
- }
-
- /**
- * Refresh the current profile's ManagementView so it contains all the latest data.
- * Use {@link #getCurrentProfileView()} to obtain the ManagementView.
- */
- public static void refreshCurrentProfileView()
- {
- try
- {
- loadProfile(getCurrentProfileView());
- }
- catch (Exception e)
- {
- LOG.error("Could not load default profile from current management view.", e);
- }
- }
-
- public static DeploymentManager getDeploymentManager() throws Exception {
- DeploymentManager deploymentManager = getProfileService().getDeploymentManager();
- // Load and associate the given profile with the DeploymentManager for future operations. This is mandatory
- // in order for us to be able to successfully invoke the various DeploymentManager methods.
- loadProfile(deploymentManager);
- return deploymentManager;
- }
-
- private static void loadProfile(ManagementView managementView)
- {
- try
- {
- LOG.trace("Loading profile via ManagementView...");
- long startTime = System.currentTimeMillis();
- managementView.load();
- long elapsedTime = System.currentTimeMillis() - startTime;
- LOG.trace("Loaded profile via Management View in " + elapsedTime + " ms.");
- }
- catch (Exception e)
- {
- LOG.error("Failed to load profile via Management View.", e);
- }
- }
-
- private static void loadProfile(DeploymentManager deploymentManager)
- {
- try
- {
- LOG.trace("Loading profile '" + DEFAULT_PROFILE_KEY + "' via Deployment Manager...");
- long startTime = System.currentTimeMillis();
- deploymentManager.loadProfile(DEFAULT_PROFILE_KEY);
- long elapsedTime = System.currentTimeMillis() - startTime;
- LOG.trace("Loaded profile '" + DEFAULT_PROFILE_KEY + "' via Deployment Manager in " + elapsedTime + " ms.");
- }
- catch (Exception e)
- {
- LOG.error("Failed to load profile '" + DEFAULT_PROFILE_KEY + "' via Deployment Manager.", e);
- }
- }
-
- /**
- * Locate the given ComponentType with the given component name.
- *
- * @param type ComponentType of the component to get
- * @param name String name of the component
- * @return the matching ManagedComponent if found, null otherwise
- * @throws Exception on error
- */
- public static ManagedComponent getManagedComponent(ComponentType type, String name)
- throws Exception
- {
- ManagementView managementView = getCurrentProfileView();
- return getManagedComponent(managementView, type, name);
- }
-
- /**
- * Locate the given ComponentType with the given component name.
- *
- * @param managementView
- * @param type
- * @param name
- * @return the matching ManagedComponent if found, null otherwise
- * @throws Exception on error
- */
- public static ManagedComponent getManagedComponent(ManagementView managementView,
- ComponentType type, String name)
- throws Exception
- {
- Setnull
- *
- * @throws Exception if there are any issues at all connecting to the server
- */
- private synchronized EmsConnection loadConnection() throws Exception {
- if (this.connection == null) {
- try {
- Configuration pluginConfig = resourceContext.getPluginConfiguration();
- String jbossHomeDir = pluginConfig.getSimpleValue(JBOSS_HOME_DIR_CONFIG_PROP, null);
-
- ConnectionSettings connectionSettings = new ConnectionSettings();
-
- String connectionTypeDescriptorClass = pluginConfig.getSimple(JMXDiscoveryComponent.CONNECTION_TYPE)
- .getStringValue();
- connectionSettings.initializeConnectionType((ConnectionTypeDescriptor) Class.forName(
- connectionTypeDescriptorClass).newInstance());
- boolean runningEmbedded = runningEmbedded();
- if (runningEmbedded) {
- connectionSettings.setServerUrl(System.getProperty("jboss.bind.address", "127.0.0.1"));
- } else {
- connectionSettings.setServerUrl(pluginConfig.getSimpleValue(NAMING_URL_CONFIG_PROP, null));
- }
-
- connectionSettings.setPrincipal(pluginConfig.getSimpleValue(PRINCIPAL_CONFIG_PROP, null));
- connectionSettings.setCredentials(pluginConfig.getSimpleValue(CREDENTIALS_CONFIG_PROP, null));
- connectionSettings.setLibraryURI(jbossHomeDir);
-
- ConnectionFactory connectionFactory = new ConnectionFactory();
- connectionFactory.discoverServerClasses(connectionSettings);
-
- if (connectionSettings.getAdvancedProperties() == null) {
- connectionSettings.setAdvancedProperties(new Properties());
- }
-
- connectionSettings.getAdvancedProperties().setProperty(JNP_DISABLE_DISCOVERY_JNP_INIT_PROP, "true");
-
- // Make sure the timeout always happens, even if the JBoss server is hung.
- connectionSettings.getAdvancedProperties().setProperty("jnp.timeout", String.valueOf(JNP_TIMEOUT));
- connectionSettings.getAdvancedProperties().setProperty("jnp.sotimeout", String.valueOf(JNP_SO_TIMEOUT));
-
- // Tell EMS to make copies of jar files so that the ems classloader doesn't lock
- // application files (making us unable to update them) Bug: JBNADM-670
- // TODO GH: turn this off in the embedded case
- connectionSettings.getControlProperties().setProperty(ConnectionFactory.COPY_JARS_TO_TEMP,
- String.valueOf(Boolean.TRUE));
-
- // But tell it to put them in a place that we clean up when shutting down the agent
- connectionSettings.getControlProperties().setProperty(ConnectionFactory.JAR_TEMP_DIR,
- resourceContext.getTemporaryDirectory().getAbsolutePath());
-
- connectionSettings.getAdvancedProperties().setProperty(InternalVMTypeDescriptor.DEFAULT_DOMAIN_SEARCH,
- "jboss");
-
- log.info("Loading JBoss connection [" + connectionSettings.getServerUrl() + "] with install path ["
- + connectionSettings.getLibraryURI() + "]...");
-
- ConnectionProvider connectionProvider = connectionFactory.getConnectionProvider(connectionSettings);
- this.connection = connectionProvider.connect();
-
- this.connection.loadSynchronous(false); // this loads all the MBeans
-
- this.consecutiveConnectionErrors = 0;
-
- try {
- this.mainDeployer = new MainDeployer(this.connection);
- } catch (Exception e) {
- log.error("Unable to access MainDeployer MBean required for creation and deletion of managed "
- + "resources - this should never happen. Cause: " + e);
- }
-
- if (log.isDebugEnabled())
- log.debug("Successfully made connection to the AS instance for resource ["
- + this.resourceContext.getResourceKey() + "]");
- } catch (Exception e) {
-
- // The connection will be established even in the case that the principal cannot be authenticated,
- // but the connection will not work. That failure seems to come from the call to loadSynchronous after
- // the connection is established. If we get to this point that an exception was thrown, close any
- // connection that was made and null it out so we can try to establish it again.
- if (connection != null) {
- if (log.isDebugEnabled())
- log.debug("Connection created but an exception was thrown. Closing the connection.", e);
- connection.close();
- connection = null;
- }
-
- // Since the connection is attempted each time it's used, failure to connect could result in log
- // file spamming. Log it once for every 10 consecutive times it's encountered.
- if (consecutiveConnectionErrors % 10 == 0) {
- log.warn("Could not establish connection to the JBoss AS instance ["
- + (consecutiveConnectionErrors + 1) + "] times for resource ["
- + resourceContext.getResourceKey() + "]", e);
- }
-
- if (log.isDebugEnabled())
- log.debug("Could not connect to the JBoss AS instance for resource ["
- + resourceContext.getResourceKey() + "]", e);
-
- consecutiveConnectionErrors++;
-
- throw e;
- }
- }
-
- return connection;
- }
-
- public Listnull
- */
- private PackageVersions loadPackageVersions() {
- if (this.versions == null) {
- ResourceType resourceType = getResourceContext().getResourceType();
- String pluginName = resourceType.getPlugin();
- File dataDirectoryFile = getResourceContext().getDataDirectory();
- dataDirectoryFile.mkdirs();
- String dataDirectory = dataDirectoryFile.getAbsolutePath();
- log.trace("Creating application versions store with plugin name [" + pluginName +
- "] and data directory [" + dataDirectory + "]");
- this.versions = new PackageVersions(pluginName, dataDirectory);
- this.versions.loadFromDisk();
- }
-
- return this.versions;
- }
-}
diff --git a/management/jopr-plugin-as-5/src/main/java/org/rhq/plugins/mobicents/servlet/sip/jboss5/StandaloneManagedDeploymentDiscoveryComponent.java b/management/jopr-plugin-as-5/src/main/java/org/rhq/plugins/mobicents/servlet/sip/jboss5/StandaloneManagedDeploymentDiscoveryComponent.java
deleted file mode 100644
index 8a60a42e73..0000000000
--- a/management/jopr-plugin-as-5/src/main/java/org/rhq/plugins/mobicents/servlet/sip/jboss5/StandaloneManagedDeploymentDiscoveryComponent.java
+++ /dev/null
@@ -1,37 +0,0 @@
-/*
- * RHQ Management Platform
- * Copyright (C) 2005-2009 Red Hat, Inc.
- * All rights reserved.
- *
- * This program is free software; you can redistribute it and/or modify
- * it under the terms of the GNU General Public License as published by
- * the Free Software Foundation version 2 of the License.
- *
- * This program 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 General Public License for more details.
- *
- * You should have received a copy of the GNU General Public License
- * along with this program; if not, write to the Free Software
- * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
- */
-package org.rhq.plugins.mobicents.servlet.sip.jboss5;
-
-import org.apache.commons.logging.Log;
-import org.apache.commons.logging.LogFactory;
-import org.jboss.managed.api.ManagedDeployment;
-import org.rhq.core.pluginapi.inventory.ResourceDiscoveryContext;
-
-/**
- * @author Ian Springer
- */
-public class StandaloneManagedDeploymentDiscoveryComponent extends ManagedDeploymentDiscoveryComponent
-{
- private final Log log = LogFactory.getLog(this.getClass());
-
- protected boolean accept(ManagedDeployment managedDeployment, ResourceDiscoveryContextEmsOperation
object that matches the specified
- * name
and parameterTypes
- *
- * @throws NoSuchMethodException if a matching operation is not found
- */
- public static EmsOperation getOperation(EmsBean mbean, String name, Class>... parameterTypes) throws NoSuchMethodException {
- if (mbean == null || name == null || parameterTypes == null)
- throw new IllegalArgumentException("All parameters must be non-null.");
-
- String[] parameterTypeNames = new String[parameterTypes.length];
- int i = 0;
- for (Class> paramType : parameterTypes) {
- parameterTypeNames[i] = paramType.getName();
- }
-
- return getOperation(mbean, name, parameterTypeNames);
- }
-
- public static EmsOperation getOperation(EmsBean mbean, String name, String... parameterTypeNames) throws NoSuchMethodException {
- if (mbean == null || name == null || parameterTypeNames == null)
- throw new IllegalArgumentException("All parameters must be non-null.");
- SortedSet