This project is for the Generic JMS JCA Resource Adapter for JBoss AS. As the name suggests, this JCA RA provides the ability to integrate with any JMS broker which allows remote clients to look-up connection factories and destinations via JNDI (as outlined in section 4.2 of the JMS 1.1 specification). It currently is only verified to work in JBoss AS7 and supports, for example, consuming messages with an MDB and sending messages with a JCA-base JMS connection factory to 3rd-party brokers. It is based on the generic JMS JCA RA found in previous versions of JBoss AS (e.g. 4, 5, and 6). However, unlike those versions this is a stand-alone project now and no longer supports internal dead-letter processing since every modern JMS broker supports this already.
To be clear, the Generic JMS JCA Resource Adapter for JBoss AS should only be used if the JMS provider with which you are integrating does not have a JCA Resource Adapter of its own. Most enterprise JMS providers have their own JCA RA, but for whatever reason there are still a few who are lacking this essential integration component.
Any questions, etc. can be posted on the "Generic JMS JCA Resource Adapter for JBoss AS" Google Group.
Note: this is a community project and has no official support from Red Hat in any capacity. I will certainly work with the community to address any issues, but I cannot guarantee any particular SLA.
The project consists of three Maven modules:
- The parent module
- The "generic-jms-ra-jar" module to create the library which goes inside the RAR.
- The "generic-jms-ra-rar" module to create the actual resource adapter archive which is deployed within the Java EE application server (e.g. JBoss AS7).
FYI - Pre-built versions of the resource adapter archive used to be available in the downloads section, but GitHub has deprecated this feature.
- Download the source via any of the methods which GitHub provides (e.g. the tags page).
- Execute 'mvn install' to build the code.
- Execute 'mvn -Prelease install' to generate the deployable resource adapter.
- 1.0.BETA: The initial release. Basically a raw copy of the generic JMS JCA RA from the old JBoss AS code-base. Minimal code clean-up and refactoring done. The main change was eliminating the dependency on org.jboss.jms.jndi.JMSProviderAdapter.
- 1.0.RC1: Refactored code style. Refactored activation configuration properties to simplify and eliminate legacy code. Refactored transaction handling so simplify and eliminate legacy code (hacks) where JMS transactions were used alongside JTA transactions to "fake" real XA semantics.
JTA transactions are very commonly used with MDBs since it is easy to treat a JMS message as a unit of work which should be performed atomically. For example, an MDB might consume a message, update a table in one or more databases, and then send another JMS message. In this kind of use-case it's extremely common to require all this work be done atomically so that if any individual part fails then the whole unit of work fails which then usually re-delivers the original message or moves it to a DLQ of some kind.
To enable this behavior an MDB needs to be configured appropriately. For example, it would need these annotations ( Note: these are added by default in JBoss AS7 and every other Java EE 6 compliant application server):
@TransactionManagement(TransactionManagementType.CONTAINER)
@TransactionAttribute(TransactionAttributeType.REQUIRED)
This tells the container to start a JTA transaction when it delivers a message to the MDB's onMessage method. You can read more about the semantics of these annotations in this tutorial from Oracle.
Behind the scenes, the Java EE server and the JCA RA use the JTA API to enlist and handle all the various javax.transaction.xa.XAResource implementations from the resource managers involved in the transaction (e.g. JMS providers, JDBC datasources, etc.). Section 8 of the JMS 1.1 specification entitled "JMS Application Server Facilities" describes all the interfaces which a JMS provider must implement in order to support this transactional use-case. However, JMS providers are not required to implement these interfaces which means even though you may want this kind of behavior, you are at the mercy of the JMS provider with whom you are integrating.
When the Java EE server activates an MDB endpoint using the "Generic JMS JCA Resource Adapter for JBoss AS" the RA sets up the JMS sessions that will be used for consuming messages. During the setup of these sessions the RA looks at the implementation of the connection factory object which the MDB is configured to use (via the connectionFactory
activation configuration property) to see if it implements javax.jms.XAConnectionFactory. If it does implement javax.jms.XAConnectionFactory then everything will be set up so that the aforementioned transactional use-case is possible. If it doesn't implement javax.jms.XAConnectionFactory then the aforementioned transactional use-case will not be possible. Instead you will have to annotate (or otherwise configure) your MDB with something like:
@TransactionManagement(TransactionManagementType.CONTAINER)
@TransactionAttribute(TransactionAttributeType.NOT_SUPPORTED)
Or perhaps:
@TransactionManagement(TransactionManagementType.BEAN)
Otherwise an exception will be thrown and the MDB will not deploy.
Since this is a generic JMS JCA RA, the user must supply it with the proper client classes to actually make a physical connection to a 3rd party JMS broker. Since AS7 uses a modular classload this requires the user to:
- Create a module with the proper integration classes
- Modify the manifest.mf of the RAR to use the aforementioned module
If you don't want to use the @ResourceAdapter annotation on your EJB3 MDB(s) then you can change the default resource adapter which all MDBs in the system will use:
<subsystem xmlns="urn:jboss:domain:ejb3:1.2">
...
<mdb>
<resource-adapter-ref resource-adapter-name="generic-jms-rar.rar"/>
<bean-instance-pool-ref pool-name="mdb-strict-max-pool"/>
</mdb>
...
</subsystem>
For example, to integrate with JBoss Messaging running in JBoss AS 5 create a module.xml like this (jar files copied from <JBOSS_5_HOME>/client):
<module xmlns="urn:jboss:module:1.1" name="org.jboss.jboss-5-client">
<resources>
<resource-root path="concurrent.jar"/>
<resource-root path="javassist.jar"/>
<resource-root path="jboss-aop-client.jar"/>
<resource-root path="jboss-common-core.jar"/>
<resource-root path="jboss-logging-log4j.jar"/>
<resource-root path="jboss-logging-spi.jar"/>
<resource-root path="jboss-mdr.jar"/>
<resource-root path="jboss-messaging-client.jar"/>
<resource-root path="jboss-remoting.jar"/>
<resource-root path="jboss-serialization.jar"/>
<resource-root path="jnp-client.jar"/>
<resource-root path="log4j.jar"/>
<resource-root path="trove.jar"/>
</resources>
<dependencies>
<module name="javax.api"/>
<module name="javax.jms.api"/>
</dependencies>
</module>
Note the "name" of the <module> - "org.jboss.jboss-5-client". This name must match the path of the module in <JBOSS_7_HOME>/modules, therefore all the related jar files would need to be placed in <JBOSS_7_HOME>/modules/org/jboss/jboss-5-client/main.
The next step is to modify the generic JMS JCA RA to use this module so it has access to all the proper integration classes when it interacts with the remote JBoss Messaging broker. To do this, simply add this line to the generic-jms-rar.rar/META-INF/manifest.mf:
Dependencies: org.jboss.jboss-5-client
Once the proper dependencies have been configured for the RAR, copy it to the "deployments" directory (e.g. <JBOSS_HOME>/standalone/deployments).
To create an outbound connection factory, use a deployment descriptor like this in your standalone*.xml.
<subsystem xmlns="urn:jboss:domain:resource-adapters:1.0">
<resource-adapters>
<resource-adapter>
<archive>
generic-jms-ra-<VERSION>.rar
</archive>
<transaction-support>XATransaction</transaction-support>
<connection-definitions>
<connection-definition class-name="org.jboss.resource.adapter.jms.JmsManagedConnectionFactory" jndi-name="java:/GenericJmsXA" enabled="true" use-java-context="true" pool-name="GenericJmsXA" use-ccm="true">
<config-property name="JndiParameters">
java.naming.factory.initial=org.jnp.interfaces.NamingContextFactory;java.naming.provider.url=JBM_HOST:1099;java.naming.factory.url.pkgs=org.jboss.naming:org.jnp.interfaces
</config-property>
<config-property name="ConnectionFactory">
XAConnectionFactory
</config-property>
<xa-pool>
<min-pool-size>0</min-pool-size>
<max-pool-size>10</max-pool-size>
<prefill>false</prefill>
<use-strict-min>false</use-strict-min>
<flush-strategy>FailingConnectionOnly</flush-strategy>
<pad-xid>false</pad-xid>
<wrap-xa-resource>true</wrap-xa-resource>
</xa-pool>
<security>
<application/>
</security>
</connection-definition>
</connection-definitions>
</resource-adapter>
</resource-adapters>
</subsystem>
This particular configuration binds a JMS connection factory to "java:/GenericJmsXA". Under the covers it looks up the "XAConnectionFactory" via JNDI from JBM_HOST. The "JndiParameters" are, of course, specific to JBoss AS 5 since that is the JNDI implementation to which we are connecting here. To connect to a different kind of server you'll need to specify its specific JNDI properties as appropriate.
This MDB will connect to JBM_HOST using the "XAConnectionFactory" and consume messages from the "queue/source" destination. It's important to note that the RA will use the "jndiParameters" activation configuration property to lookup the "connectionFactory" and the "destination."
Once a message is received the MDB will use the "java:/GenericJmsXA" connection factory (defined above) to send a message to the "target" destination hosted on JBM_HOST. Notice here that the GenericJmsXA connection factory is looked up via JNDI, but the "target" destination is not looked up via JNDI but rather instantiated with javax.jms.Session.createQueue(String) where the Sting parameter is the actual name of the destination (i.e. not necessarily where it is bound in JNDI). This is done because we want to avoid a full JNDI look-up of the destination on the remote server, and because there is currently no way to make a local JNDI look-up go to a remote server (a la the ExternalContext MBean from JBoss AS 4, 5, and 6). The reason it is typically good to avoid a full JNDI lookup of the destination on the remote server is because it saves the developer from having to specify the same JNDI lookup parameters both in the code and the activation configuration.
The consumption and production will be done atomically because the underlying connection factories used to consume and produce the messages support XA and also because the way the MDB is coded to rollback the transaction when production fails for any reason.
import javax.ejb.ActivationConfigProperty;
import javax.ejb.MessageDriven;
import javax.jms.Connection;
import javax.jms.ConnectionFactory;
import javax.jms.Destination;
import javax.jms.Message;
import javax.jms.MessageListener;
import javax.jms.MessageProducer;
import javax.jms.Session;
import javax.naming.Context;
import javax.naming.InitialContext;
import org.jboss.ejb3.annotation.ResourceAdapter;
@MessageDriven(activationConfig = {
@ActivationConfigProperty(propertyName = "destinationType", propertyValue = "javax.jms.Queue"),
@ActivationConfigProperty(propertyName = "destination", propertyValue = "/queue/source"),
@ActivationConfigProperty(propertyName = "jndiParameters", propertyValue = "java.naming.factory.initial=org.jnp.interfaces.NamingContextFactory;java.naming.provider.url=JBM_HOST:1099;java.naming.factory.url.pkgs=org.jboss.naming:org.jnp.interfaces"),
@ActivationConfigProperty(propertyName = "connectionFactory", propertyValue = "XAConnectionFactory")
})
@ResourceAdapter("generic-jms-ra-<VERSION>.rar")
public class ExampleMDB implements MessageListener
{
@Resource
private MessageDrivenContext context;
public void onMessage(final Message message)
{
Connection connection = null;
try
{
Context context = new InitialContext();
ConnectionFactory cf = (ConnectionFactory) context.lookup("java:/GenericJmsXA");
context.close();
connection = cf.createConnection();
Session session = connection.createSession(false, Session.AUTO_ACKNOWLEDGE);
Destination destination = session.createQueue("target");
MessageProducer producer = session.createProducer(destination);
Message msg = session.createTextMessage("example text");
producer.send(msg);
}
catch (Exception e)
{
context.setRollbackOnly();
}
finally
{
if (connection != null)
{
connection.close();
}
}
}
}
If you don't want to specify your MDB configuration in the code via annotations then you can use the traditional ejb-jar.xml deployment descriptor. Furthermore, in JBoss AS7 you can use system property substitution in ejb-jar.xml so that you can change the configuration of the MDB without opening the archive in which the MDB is deployed (e.g. JAR or EAR). This is helpful when for example you need to move an application from a development environment to a QA or production environment. To enable system property substitution in ejb-jar.xml in JBoss AS7 set the <spec-descriptor-property-replacement>
to true
, e.g.:
<subsystem xmlns="urn:jboss:domain:ee:1.1">
<spec-descriptor-property-replacement>true</spec-descriptor-property-replacement>
<jboss-descriptor-property-replacement>true</jboss-descriptor-property-replacement>
</subsystem>
As I understand it, <spec-descriptor-property-replacement>
is set to false
by default because of the way the Java EE TCK works.
Here is an example ejb-jar.xml that would replace the @MessageDriven configuration above:
<?xml version="1.0" encoding="UTF-8"?>
<ejb-jar version="3.0" xmlns="http://java.sun.com/xml/ns/javaee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/ejb-jar_3_0.xsd">
<enterprise-beans>
<message-driven>
<ejb-name>ExampleMDB</ejb-name>
<ejb-class>ExampleMDB</ejb-class>
<activation-config>
<activation-config-property>
<activation-config-property-name>destinationType</activation-config-property-name>
<activation-config-property-value>javax.jms.Queue</activation-config-property-value>
</activation-config-property>
<activation-config-property>
<activation-config-property-name>destination</activation-config-property-name>
<activation-config-property-value>/queue/source</activation-config-property-value>
</activation-config-property>
<activation-config-property>
<activation-config-property-name>jndiParameters</activation-config-property-name>
<activation-config-property-value>java.naming.factory.initial=org.jnp.interfaces.NamingContextFactory;java.naming.provider.url=${myJmsProvider};java.naming.factory.url.pkgs=org.jboss.naming:org.jnp.interfaces</activation-config-property-value>
</activation-config-property>
<activation-config-property>
<activation-config-property-name>connectionFactory</activation-config-property-name>
<activation-config-property-value>${myConnectionFactory}</activation-config-property-value>
</activation-config-property>
</activation-config>
</message-driven>
</enterprise-beans>
</ejb-jar>
Notice that both the jndiParameters
and connectionFactory
activation configuration properties use a special ${}
syntax in their values. That is where the substitution would take place. You can specify the value of those substitutions on the command line when you start JBoss AS7, e.g.:
./standalone.sh -c standalone-full.xml -DmyJmsProvider=hostname:1099 -DmyConnectionFactory=XAConnectionFactory
Lastly, when deploying an MDB which depends on a non-default RA it is customary to modify the MDB's deployment so that it is not deployed until the RA it needs has been deployed. To do this in JBoss AS7 simply add this line to the META-INF/manifest.mf of your deployment:
Dependencies: deployment.generic-jms-rar.rar
You can set up this kind of dependency for any application that needs to use the RA (e.g. a servlet sending a JMS message).
This information was provided by community members as I don't have access to a Tibco instance. I have received module definitions from 2 different versions of Tibco EMS. I imagine the same configuration could be used for other Tibco versions as well.
<?xml version='1.0' encoding='UTF-8'?>
<module xmlns="urn:jboss:module:1.1" name="com.tibco.tibjms">
<resources>
<resource-root path="slf4j-api-1.4.2.jar"></resource-root>
<resource-root path="slf4j-simple-1.4.2.jar"></resource-root>
<resource-root path="tibcrypt.jar"></resource-root>
<resource-root path="tibjms.jar"></resource-root>
<resource-root path="tibjmsadmin.jar"></resource-root>
<resource-root path="tibjmsufo.jar"></resource-root>
</resources>
<dependencies>
<module name="javax.api"/>
<module name="javax.jms.api"/>
</dependencies>
</module>
<?xml version='1.0' encoding='UTF-8'?>
<module xmlns="urn:jboss:module:1.1" name="com.tibco.tibjms">
<resources>
<resource-root path="tibjms.jar"/>
<resource-root path="tibcrypt.jar"/>
</resources>
<dependencies>
<module name="javax.api"/>
<module name="javax.jms.api"/>
</dependencies>
</module>
<subsystem xmlns="urn:jboss:domain:resource-adapters:1.0">
<resource-adapters>
<resource-adapter>
<archive>
generic-jms-ra-<VERSION>.rar
</archive>
<transaction-support>NoTransaction</transaction-support>
<connection-definitions>
<connection-definition class-name="org.jboss.resource.adapter.jms.JmsManagedConnectionFactory" jndi-name="java:/GenericJmsXA" enabled="true" use-java-context="true" pool-name="GenericJmsXA" use-ccm="true">
<config-property name="JndiParameters">
java.naming.factory.initial=com.tibco.tibjms.naming.TibjmsInitialContextFactory;java.naming.provider.url=tcp://TIBCO_HOST:7222
</config-property>
<config-property name="ConnectionFactory">
QueueConnectionFactory
</config-property>
<pool>
<min-pool-size>0</min-pool-size>
<max-pool-size>10</max-pool-size>
<prefill>false</prefill>
<use-strict-min>false</use-strict-min>
<flush-strategy>FailingConnectionOnly</flush-strategy>
</pool>
<security>
<application></application>
</security>
</connection-definition>
</connection-definitions>
</resource-adapter>
</resource-adapters>
</subsystem>
Notice the "JndiParameters" are Tibco specific.
- destination - the JNDI name of JMS destination from which the MDB will consume messages; this is required
- destinationType - the type of JMS destination from which to consume messages; valid values are
javax.jms.Queue
,javax.jms.Topic
, orjavax.jms.Destination
; default isjavax.jms.Destination
- jndiParameters - the JNDI parameters used to perform the lookup of the destination and the connectionFactory; each parameter consists of a "name=value" pair; parameters are separated with a semi-colon (';'); if no parameters are specified then an empty InitialContext will be used (i.e. the lookup will be local)
- connectionFactory - the JNDI name of connection factory which the RA will use to consume the messages; this is normally a connection factory which supports XA; this is required
- messageSelector - the JMS selector to use when consuming messages; default is null
- acknowledgeMode - the acknowledgement mode used when consuming messages; only applicable when using Bean-Managed transactions; valid values are
DUPS_OK_ACKNOWLEDGE
andAUTO_ACKNOWLEDGE
; default isAUTO_ACKNOWLEDGE
; when Container-Managed transactions are used the acknowledgement of the message is performed by the Java EE application server in accordance with the outcome of the MDB's transaction (assuming such a transaction exists) - subscriptionDurability - the durability of the topic subscription; default is non-durable; the value "Durable" makes the subscription durable, anything else makes it non-durable
- clientId - the client ID to use for a topic subscription
- subscriptionName - the name of the topic subscription
- reconnectInterval - how long to wait between reconnectAttempts; value is measured in seconds; default is 10
- reconnectAttempts - how many times to try to reconnect if the connection to the JMS broker is lost; default is -1 (i.e. infinite attempts)
- user - the name of the user used when connecting to the JMS provider
- pass - the password used when connecting to the JMS provider
- minSession - the minimum number of JMS sessions to create; default is 1
- maxSession - the maximum number of JMS sessions to create; default is 15
- maxMessages - the value passed to
javax.jms.ConnectionConsumer.createConnectionConsumer(..)
; see section 8.2.4 of the JMS 1.1 specification for further details; default is 1 - transactionTimeout - the value used for the JTA transaction timeout when using Container-Managed transactions; default is 0 (i.e. use the system default timeout)
- forceClearOnShutdown - whether or not to wait for MDB processing to complete before shutting down the internal JMS ServerSession pool; default is false (i.e. wait for MDB processing to complete)
- forceClearOnShutdownInterval - how long to wait between attempts to shutdown the internal JMS ServerSession pool; value is measured in milliseconds; default is 1000
- forceClearAttempts - how many times to attempt shutting down the internal JMS ServerSession pool; default is 0