Atomikos 翻译文档(英文文档来源:下载安装包中START_HERE.html)
----译者:周枫
请尊重劳动成果,转载请标明,英语水平有限,如有不准确地方请在评论中指出,谢谢
官网地址:http://www.atomikos.com/Main/WebHome
使用版本:AtomikosTransactionsEssentials-3.7.2
感谢您使用Atomikos,下面的说明文档可以让您正确使用,如果您有任何问题或者反馈,请访问我们的帮助网页http://www.atomikos.com/Main/SupportOverview,或者给我们发送邮件sales@atomikos.com。
什么是Atomikos TransactionsEssentials
Atomikos TransactionsEssentials 是一个为Java平台提供增值服务的并且开源类事务管理器,以下是包括在这个开源版本中的一些功能:
l 全面崩溃 / 重启恢复
l 兼容标准的SUN公司JTA API
l 嵌套事务
l 为XA和非XA提供内置的JDBC适配器
注释:XA:XA协议由Tuxedo首先提出的,并交给X/Open组织,作为资源管理器(数据库)与事务管理器的接口标准。目前,Oracle、Informix、DB2和Sybase等各大数据库厂家都提供对XA的支持。XA协议采用两阶段提交方式来管理分布式事务。XA接口提供资源管理器与事务管理器之间进行通信的标准接口。XA协议包括两套函数,以xa_开头的及以ax_开头的。
以下的函数使事务管理器可以对资源管理器进行的操作:
1)xa_open,xa_close:建立和关闭与资源管理器的连接。
2)xa_start,xa_end:开始和结束一个本地事务。
3)xa_prepare,xa_commit,xa_rollback:预提交、提交和回滚一个本地事务。
4)xa_recover:回滚一个已进行预提交的事务。
5)ax_开头的函数使资源管理器可以动态地在事务管理器中进行注册,并可以对XID(TRANSACTION IDS)进行操作。
6)ax_reg,ax_unreg;允许一个资源管理器在一个TMS(TRANSACTION MANAGER SERVER)中动态注册或撤消注册。
l 内置的JMS适配器XA-capable JMS队列连接器
注释:JMS:jms即Java消息服务(Java Message Service)应用程序接口是一个Java平台中关于面向消息中间件(MOM)的API,用于在两个应用程序之间,或分布式系统中发送消息,进行异步通信。Java消息服务是一个与具体平台无关的API,绝大多数MOM提供商都对JMS提供支持。
l 通过XA API兼容第三方适配器
l 更好的整合您的项目
l 集成Hibernate
如何使用Atomikos TransactionsEssentials
Atomikos TransactionsEssentials 是一个可靠的库,可以加入到您的Java应用程序,也就是说为了使用这个产品,您必须添加一些jar文件(包括在dist和lib文件夹下)到您的应用程序或者应用程序服务器。
请注意:Atomikos TransactionsEssentials是一个非常快速的嵌入式事务管理器,这就意味着,您不需要另外启动一个单独的事务管理器进程(不要查找任何的bin文件夹)。相反,您的应用服务器将有它自己的intra-VM事务管理器。
配置需求:至少Java1.5 jdk,并且最少128M的内存
性能优化:尽管这个软件有着很大的优势,但是想要更好的发挥其作用,可以按以下的方法优化:
l 更高的内存,意味着更高的吞吐量(每秒的事务数目)
l 使连接池尽可能的大
l 一旦你不需要的连接请马上关闭它们。不要把你的应用程序放在缓存里,让内部连接池为你做这些,这将促使更高效的连接使用
l 不要让活动的事务闲置:终止所有情况下的事务,尤其是在异常报错情况下的事务。这将减少数据库的锁定时间,并且最大效率的处理启用的使用。
如果想获取这些细节的更多信息,也要参阅文档说明部分。
值得注意的是,在我们所有的压力测试中,Atomikos TransactionsEssentials比J2EE的web容器更高效的吞吐量。这些测量值包括日志记录的高效的事务状态,同样,在我们所有的测量中,包括XA和non-XA,高效的效率是一样的。
在J2SE中使用Atomikos Transactions Essentials,只需要按以下步骤
- 将idst和lib中的jar包全部放入的项目中
- 创建或者自定义你应用的transactions.properties(或者jta.properties)文件(事务管理器的配置),然后将它放入到classpath中,安装文件夹中包涵一个实例文件;在properties文件中注释(#)后面的是默认值,取消一行并且改变默认值。
# SAMPLE PROPERTIES FILE FOR THE TRANSACTION SERVICE
# THIS FILE ILLUSTRATES THE DIFFERENT SETTINGS FOR THE TRANSACTION MANAGER
# UNCOMMENT THE ASSIGNMENTS TO OVERRIDE DEFAULT VALUES;
# Required: factory implementation class of the transaction core.
# NOTE: there is no default for this, so it MUST be specified!
#
com.atomikos.icatch.service=com.atomikos.icatch.standalone.UserTransactionServiceFactory
# Set base name of file where messages are output
# (also known as the 'console file').
#
# com.atomikos.icatch.console_file_name = tm.out
# Size limit (in bytes) for the console file;
# negative means unlimited.
#
# com.atomikos.icatch.console_file_limit=-1
# For size-limited console files, this option
# specifies a number of rotating files to
# maintain.
#
# com.atomikos.icatch.console_file_count=1
# Set the number of log writes between checkpoints
#
# com.atomikos.icatch.checkpoint_interval=500
# Set output directory where console file and other files are to be put
# make sure this directory exists!
#
# com.atomikos.icatch.output_dir = ./
# Set directory of log files; make sure this directory exists!
#
# com.atomikos.icatch.log_base_dir = ./
# Set base name of log file
# this name will be used as the first part of
# the system-generated log file name
#
# com.atomikos.icatch.log_base_name = tmlog
# Set the max number of active local transactions
# or -1 for unlimited.
#
# com.atomikos.icatch.max_actives = 50
# Set the default timeout (in milliseconds) for local transactions
#
# com.atomikos.icatch.default_jta_timeout = 10000
# Set the max timeout (in milliseconds) for local transactions
#
# com.atomikos.icatch.max_timeout = 300000
# The globally unique name of this transaction manager process
# override this value with a globally unique name
#
# com.atomikos.icatch.tm_unique_name = tm
# Do we want to use parallel subtransactions? JTA's default
# is NO for J2EE compatibility
#
# com.atomikos.icatch.serial_jta_transactions=true
# If you want to do explicit resource registration then
# you need to set this value to false.
#
# com.atomikos.icatch.automatic_resource_registration=true
# Set this to WARN, INFO or DEBUG to control the granularity
# of output to the console file.
#
# com.atomikos.icatch.console_log_level=WARN
# Do you want transaction logging to be enabled or not?
# If set to false, then no logging overhead will be done
# at the risk of losing data after restart or crash.
#
# com.atomikos.icatch.enable_logging=true
# Should two-phase commit be done in (multi-)threaded mode or not?
# Set this to false if you want commits to be ordered according
# to the order in which resources are added to the transaction.
#
# NOTE: threads are reused on JDK 1.5 or higher.
# For JDK 1.4, thread reuse is enabled as soon as the
# concurrent backport is in the classpath - see
# http://mirrors.ibiblio.org/pub/mirrors/maven2/backport-util-concurrent/backport-util-concurrent/
#
# com.atomikos.icatch.threaded_2pc=false
# Should shutdown of the VM trigger shutdown of the transaction core too?
#
# com.atomikos.icatch.force_shutdown_on_vm_exit=false - 在你的应用程序中,创建一个实例com.atomikos.icatch.jta.UserTransactionImp或者com.atomikos.icatch.jta.UserTransactionManager(使用默认的无参数构造函数)
/**
* Copyright (C) 2000-2010 Atomikos <info@atomikos.com>
*
* This code ("Atomikos TransactionsEssentials"), by itself,
* is being distributed under the
* Apache License, Version 2.0 ("License"), a copy of which may be found at
* http://www.atomikos.com/licenses/apache-license-2.0.txt .
* You may not use this file except in compliance with the License.
*
* While the License grants certain patent license rights,
* those patent license rights only extend to the use of
* Atomikos TransactionsEssentials by itself.
*
* This code (Atomikos TransactionsEssentials) contains certain interfaces
* in package (namespace) com.atomikos.icatch
* (including com.atomikos.icatch.Participant) which, if implemented, may
* infringe one or more patents held by Atomikos.
* It should be appreciated that you may NOT implement such interfaces;
* licensing to implement these interfaces must be obtained separately from Atomikos.
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
*/
package com.atomikos.icatch.jta;
import java.io.Serializable;
import javax.naming.NamingException;
import javax.naming.Reference;
import javax.naming.Referenceable;
import javax.transaction.NotSupportedException;
import javax.transaction.SystemException;
import javax.transaction.TransactionManager;
import javax.transaction.UserTransaction;
import com.atomikos.icatch.admin.imp.SimpleLogAdministrator;
import com.atomikos.icatch.config.TSInitInfo;
import com.atomikos.icatch.config.UserTransactionService;
import com.atomikos.icatch.config.UserTransactionServiceImp;
import com.atomikos.util.SerializableObjectFactory;
/**
*
*
* Our UserTransaction implementation for J2SE transactions. This class is
* special in that it automatically starts up and recover the transaction
* service on first use. <b>Note: don't use this class in J2EE applications in
* order to avoid starting different transaction engines in the same application
* server! J2EE applications should use J2eeUserTransaction instead.</b>
*/
public class UserTransactionImp implements UserTransaction, Serializable,
Referenceable
{
private transient TransactionManager txmgr_;
/**
* No-argument constructor.
*/
public UserTransactionImp ()
{
}
/**
* Referenceable mechanism requires later setup of txmgr_, otherwise binding
* into JNDI already requires that TM is running.
*/
private void checkSetup ()
{
// REMOVED FOLLOWING IF CHECK: DON'T CACHE THE TXMGR TO MAKE INSTANCES
// RESILIENT TO RESTART IN TOMCAT. OTHERWISE, CLIENT APPS SEE THEIR
// USERTX REFERENCES INVALIDATED AND THIS IS INTOLERABLE
// if ( txmgr_ == null ) {
// txmgr_ = TransactionManagerImp.getTransactionManager();
synchronized ( TransactionManagerImp.class ) {
txmgr_ = TransactionManagerImp.getTransactionManager ();
// FOLLOWING COMMENTED OUT: NEW RECOVERY IN 2.0 ALLOWS US TO START
// THE TM
// IF NOT ALREADY RUNNING!!!
// if ( txmgr_ == null )
// throw new RuntimeException ( "No transaction monitor installed?"
// );
// NEW FROM 2.0: if TM is not running, just start it. Any resources
// can be registered later.
if ( txmgr_ == null ) {
UserTransactionService uts = new UserTransactionServiceImp ();
TSInitInfo info = uts.createTSInitInfo ();
uts.registerLogAdministrator ( SimpleLogAdministrator
.getInstance () );
uts.init ( info );
txmgr_ = TransactionManagerImp.getTransactionManager ();
}
}
// }
}
/**
* @see javax.transaction.UserTransaction
*/
public void begin () throws NotSupportedException, SystemException
{
checkSetup ();
txmgr_.begin ();
}
/**
* @see javax.transaction.UserTransaction
*/
public void commit () throws javax.transaction.RollbackException,
javax.transaction.HeuristicMixedException,
javax.transaction.HeuristicRollbackException,
javax.transaction.SystemException, java.lang.IllegalStateException,
java.lang.SecurityException
{
checkSetup ();
txmgr_.commit ();
}
/**
* @see javax.transaction.UserTransaction
*/
public void rollback () throws IllegalStateException, SystemException,
SecurityException
{
checkSetup ();
txmgr_.rollback ();
}
/**
* @see javax.transaction.UserTransaction
*/
public void setRollbackOnly () throws IllegalStateException,
SystemException
{
checkSetup ();
txmgr_.setRollbackOnly ();
}
/**
* @see javax.transaction.UserTransaction
*/
public int getStatus () throws SystemException
{
checkSetup ();
return txmgr_.getStatus ();
}
/**
* @see javax.transaction.UserTransaction
*/
public void setTransactionTimeout ( int seconds ) throws SystemException
{
checkSetup ();
txmgr_.setTransactionTimeout ( seconds );
}
//
//
// IMPLEMENTATION OF REFERENCEABLE
//
//
public Reference getReference () throws NamingException
{
return SerializableObjectFactory.createReference ( this );
}
}/**
* Copyright (C) 2000-2010 Atomikos <info@atomikos.com>
*
* This code ("Atomikos TransactionsEssentials"), by itself,
* is being distributed under the
* Apache License, Version 2.0 ("License"), a copy of which may be found at
* http://www.atomikos.com/licenses/apache-license-2.0.txt .
* You may not use this file except in compliance with the License.
*
* While the License grants certain patent license rights,
* those patent license rights only extend to the use of
* Atomikos TransactionsEssentials by itself.
*
* This code (Atomikos TransactionsEssentials) contains certain interfaces
* in package (namespace) com.atomikos.icatch
* (including com.atomikos.icatch.Participant) which, if implemented, may
* infringe one or more patents held by Atomikos.
* It should be appreciated that you may NOT implement such interfaces;
* licensing to implement these interfaces must be obtained separately from Atomikos.
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
*/
package com.atomikos.icatch.jta;
import java.io.Serializable;
import javax.naming.NamingException;
import javax.naming.Reference;
import javax.naming.Referenceable;
import javax.transaction.HeuristicMixedException;
import javax.transaction.HeuristicRollbackException;
import javax.transaction.InvalidTransactionException;
import javax.transaction.NotSupportedException;
import javax.transaction.RollbackException;
import javax.transaction.SystemException;
import javax.transaction.Transaction;
import javax.transaction.TransactionManager;
import javax.transaction.UserTransaction;
import com.atomikos.icatch.config.TSInitInfo;
import com.atomikos.icatch.config.UserTransactionService;
import com.atomikos.icatch.config.UserTransactionServiceImp;
import com.atomikos.util.SerializableObjectFactory;
/**
*
*
*
*
*
* A straightforward, zero-setup implementation of a transaction manager. J2SE
* applications can use an instance of this class to get a handle to the
* transaction manager, and automatically startup or recover the transaction
* service on first use. <b>J2EE applications should NOT use this class in order
* to avoid the concurrent use of different transaction services. For J2EE
* applications, we have the class J2eeTransactionManager instead.</b>
*/
public class UserTransactionManager implements TransactionManager,
Serializable, Referenceable, UserTransaction
{
private static final long serialVersionUID = -655789038710288096L;
private transient TransactionManagerImp tm;
private UserTransactionService uts;
private boolean forceShutdown;
private boolean startupTransactionService;
private boolean closed;
private void checkSetup () throws SystemException
{
if ( closed ) throw new SystemException ( "This UserTransactionManager instance was closed already. Call init() to reuse if desired." );
synchronized ( TransactionManagerImp.class ) {
tm = (TransactionManagerImp) TransactionManagerImp
.getTransactionManager ();
if ( tm == null ) {
// not initialized -> startup TM
// System.out.println ( "STARTING UP TM!!!!!!");
if ( getStartupTransactionService() ) {
uts = new UserTransactionServiceImp ();
TSInitInfo info = uts.createTSInitInfo ();
uts.init ( info );
tm = (TransactionManagerImp) TransactionManagerImp
.getTransactionManager ();
}
else {
throw new SystemException ( "Transaction service not running" );
}
}
}
}
public UserTransactionManager()
{
//startup by default, to have backward compatibility
this.startupTransactionService = true;
this.closed = false;
}
/**
* Sets whether the transaction service should be
* started if not already running.
* @param startup
*/
public void setStartupTransactionService ( boolean startup )
{
this.startupTransactionService = startup;
}
/**
* Returns true if the transaction service will
* be started if not already running.
* @return
*/
public boolean getStartupTransactionService()
{
return this.startupTransactionService;
}
/**
* Performs initialization if necessary.
* This will startup the TM (if not running)
* and perform recovery, unless <b>getStartupTransactionService</b>
* returns false.
*
* @throws SystemException
*/
public void init() throws SystemException
{
closed = false;
checkSetup();
}
/**
* @see javax.transaction.TransactionManager#begin()
*/
public void begin () throws NotSupportedException, SystemException
{
checkSetup ();
tm.begin ();
}
public boolean getForceShutdown()
{
return forceShutdown;
}
/**
* Sets the force shutdown mode to use during close.
* @param value
*/
public void setForceShutdown ( boolean value )
{
this.forceShutdown = value;
}
/**
* @see javax.transaction.TransactionManager#commit()
*/
public void commit () throws RollbackException, HeuristicMixedException,
HeuristicRollbackException, SecurityException,
IllegalStateException, SystemException
{
checkSetup ();
tm.commit ();
}
/**
* @see javax.transaction.TransactionManager#getStatus()
*/
public int getStatus () throws SystemException
{
checkSetup ();
return tm.getStatus ();
}
/**
* @see javax.transaction.TransactionManager#getTransaction()
*/
public Transaction getTransaction () throws SystemException
{
checkSetup ();
return tm.getTransaction ();
}
/**
* @see javax.transaction.TransactionManager#resume(javax.transaction.Transaction)
*/
public void resume ( Transaction tx ) throws InvalidTransactionException,
IllegalStateException, SystemException
{
checkSetup ();
tm.resume ( tx );
}
/**
* @see javax.transaction.TransactionManager#rollback()
*/
public void rollback () throws IllegalStateException, SecurityException,
SystemException
{
checkSetup ();
tm.rollback ();
}
/**
* @see javax.transaction.TransactionManager#setRollbackOnly()
*/
public void setRollbackOnly () throws IllegalStateException,
SystemException
{
checkSetup ();
tm.setRollbackOnly ();
}
/**
* @see javax.transaction.TransactionManager#setTransactionTimeout(int)
*/
public void setTransactionTimeout ( int secs ) throws SystemException
{
checkSetup ();
tm.setTransactionTimeout ( secs );
}
/**
* @see javax.transaction.TransactionManager#suspend()
*/
public Transaction suspend () throws SystemException
{
checkSetup ();
return tm.suspend ();
}
/**
* @see javax.naming.Referenceable#getReference()
*/
public Reference getReference () throws NamingException
{
return SerializableObjectFactory.createReference ( this );
}
/**
* Closes the transaction service, but only if it was
* implicitly started via this instance.
* In other words, if the transaction service was started
* in another way then this method will not do anything.
*
*/
public void close()
{
if ( uts != null ) {
uts.shutdown ( forceShutdown );
uts = null;
}
closed = true;
}
} - 对于JDBC,使用我们的一个实例com.atomikos.jdbc.AtomikosDataSourceBean或者,对于non-XA驱动,可以使用com.atomikos.jdbc.nonxa.AtomikosNonXADataSourceBean
/**
* Copyright (C) 2000-2010 Atomikos <info@atomikos.com>
*
* This code ("Atomikos TransactionsEssentials"), by itself,
* is being distributed under the
* Apache License, Version 2.0 ("License"), a copy of which may be found at
* http://www.atomikos.com/licenses/apache-license-2.0.txt .
* You may not use this file except in compliance with the License.
*
* While the License grants certain patent license rights,
* those patent license rights only extend to the use of
* Atomikos TransactionsEssentials by itself.
*
* This code (Atomikos TransactionsEssentials) contains certain interfaces
* in package (namespace) com.atomikos.icatch
* (including com.atomikos.icatch.Participant) which, if implemented, may
* infringe one or more patents held by Atomikos.
* It should be appreciated that you may NOT implement such interfaces;
* licensing to implement these interfaces must be obtained separately from Atomikos.
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
*/
package com.atomikos.jdbc;
import java.util.Enumeration;
import java.util.Properties;
import javax.sql.XADataSource;
import com.atomikos.beans.PropertyUtils;
import com.atomikos.datasource.RecoverableResource;
import com.atomikos.datasource.xa.jdbc.JdbcTransactionalResource;
import com.atomikos.icatch.system.Configuration;
import com.atomikos.jdbc.AbstractDataSourceBean;
import com.atomikos.util.ClassLoadingHelper;
/**
* The preferred class for using Atomikos connection pooling. Use an instance of
* this class if you want to use Atomikos JTA-enabled connection pooling. All
* you need to do is construct an instance and set the required properties as
* outlined below. The resulting bean will automatically register with the
* transaction service (for recovery) and take part in active transactions.
* All SQL done over connections (gotten from this class) will participate in JTA transactions.
*/
public class AtomikosDataSourceBean
extends AbstractDataSourceBean
{
private static final long serialVersionUID = 1L;
private Properties xaProperties = null;
private String xaDataSourceClassName;
private transient XADataSource xaDataSource;
public AtomikosDataSourceBean()
{
this.xaProperties = new Properties();
}
protected String printXaProperties()
{
StringBuffer ret = new StringBuffer();
if ( xaProperties != null ) {
Enumeration it = xaProperties.propertyNames();
ret.append ( "[" );
boolean first = true;
while ( it.hasMoreElements() ) {
if ( ! first ) ret.append ( "," );
String name = ( String ) it.nextElement();
String value = xaProperties.getProperty( name);
ret.append ( name ); ret.append ( "=" ); ret.append ( value );
first = false;
}
ret.append ( "]" );
}
return ret.toString();
}
/**
* Gets the properties used to
* configure the XADataSource.
*/
public Properties getXaProperties()
{
return xaProperties;
}
/**
* Sets the properties (name,value pairs) used to
* configure the XADataSource. Required, unless you call setXaDataSource directly.
*
* @param xaProperties
*
*
*/
public void setXaProperties ( Properties xaProperties )
{
this.xaProperties = xaProperties;
}
/**
* Get the XADataSource class name.
*/
public String getXaDataSourceClassName()
{
return xaDataSourceClassName;
}
/**
* Sets the fully qualified underlying XADataSource class name. Required, unless you
* call setXaDataSource directly.
*
* @param xaDataSourceClassName
*/
public void setXaDataSourceClassName ( String xaDataSourceClassName )
{
this.xaDataSourceClassName = xaDataSourceClassName;
}
/**
* Gets the configured XADataSource (if any).
* @return The instance, or null if none.
*/
public XADataSource getXaDataSource()
{
return xaDataSource;
}
/**
* Sets the XADataSource directly - instead of providing the xaDataSourceClassName and xaProperties.
* @param xaDataSource
*/
public void setXaDataSource(XADataSource xaDataSource)
{
this.xaDataSource = xaDataSource;
}
protected com.atomikos.datasource.pool.ConnectionFactory doInit() throws Exception
{
if (xaDataSource == null)
{
if (xaDataSourceClassName == null)
throwAtomikosSQLException("Property 'xaDataSourceClassName' cannot be null");
if (xaProperties == null)
throwAtomikosSQLException("Property 'xaProperties' cannot be null");
}
if ( Configuration.isInfoLoggingEnabled() ) Configuration.logInfo(
this + ": initializing with [" +
" xaDataSourceClassName=" + xaDataSourceClassName + "," +
" uniqueResourceName=" + getUniqueResourceName() + "," +
" maxPoolSize=" + getMaxPoolSize() + "," +
" minPoolSize=" + getMinPoolSize() + "," +
" borrowConnectionTimeout=" + getBorrowConnectionTimeout() + "," +
" maxIdleTime=" + getMaxIdleTime() + "," +
" reapTimeout=" + getReapTimeout() + "," +
" maintenanceInterval=" + getMaintenanceInterval() + "," +
" testQuery=" + getTestQuery() + "," +
" xaProperties=" + printXaProperties() +
" loginTimeout=" + getLoginTimeout() +
"]"
);
if (xaDataSource == null)
{
Class xadsClass = null;
try {
xadsClass = ClassLoadingHelper.loadClass ( getXaDataSourceClassName() );
} catch ( ClassNotFoundException nf ) {
AtomikosSQLException.throwAtomikosSQLException ( "The class '" + getXaDataSourceClassName() +
"' specified by property 'xaDataSourceClassName' could not be found in the classpath. Please make sure the spelling is correct, and that the required jar(s) are in the classpath." , nf );
}
Object driver = xadsClass.newInstance();
if ( ! ( driver instanceof XADataSource ) ) {
AtomikosSQLException.throwAtomikosSQLException (
"The class '" + getXaDataSourceClassName() +
"' specified by property 'xaDataSourceClassName' does not implement the required interface javax.jdbc.XADataSource. Please make sure the spelling is correct, and check your JDBC driver vendor's documentation."
);
}
xaDataSource = (XADataSource) driver;
xaDataSource.setLoginTimeout ( getLoginTimeout() );
xaDataSource.setLogWriter ( getLogWriter() );
PropertyUtils.setProperties(xaDataSource, xaProperties );
}
JdbcTransactionalResource tr = new JdbcTransactionalResource(getUniqueResourceName() , xaDataSource);
com.atomikos.datasource.pool.ConnectionFactory cf = new com.atomikos.jdbc.AtomikosXAConnectionFactory(xaDataSource, tr, this);
Configuration.addResource ( tr );
return cf;
}
protected void doClose()
{
RecoverableResource res = Configuration.getResource ( getUniqueResourceName() );
if ( res != null ) {
Configuration.removeResource ( getUniqueResourceName() );
//fix for case 26005
res.close();
}
}
public String toString()
{
String ret = "AtomikosDataSoureBean";
String name = getUniqueResourceName();
if ( name != null ) {
ret = ret + " '" + name + "'";
}
return ret;
}
}/**
* Copyright (C) 2000-2010 Atomikos <info@atomikos.com>
*
* This code ("Atomikos TransactionsEssentials"), by itself,
* is being distributed under the
* Apache License, Version 2.0 ("License"), a copy of which may be found at
* http://www.atomikos.com/licenses/apache-license-2.0.txt .
* You may not use this file except in compliance with the License.
*
* While the License grants certain patent license rights,
* those patent license rights only extend to the use of
* Atomikos TransactionsEssentials by itself.
*
* This code (Atomikos TransactionsEssentials) contains certain interfaces
* in package (namespace) com.atomikos.icatch
* (including com.atomikos.icatch.Participant) which, if implemented, may
* infringe one or more patents held by Atomikos.
* It should be appreciated that you may NOT implement such interfaces;
* licensing to implement these interfaces must be obtained separately from Atomikos.
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
*/
package com.atomikos.jdbc.nonxa;
import java.sql.Connection;
import java.sql.SQLException;
import com.atomikos.datasource.pool.ConnectionFactory;
import com.atomikos.icatch.HeuristicMessage;
import com.atomikos.icatch.system.Configuration;
import com.atomikos.jdbc.AbstractDataSourceBean;
import com.atomikos.util.DynamicProxy;
/**
*
* A Bean class for DataSource access to non-XA JDBC implementations.
* Instances are JTA transaction-aware and can rollback the work done
* over multiple connections (provided that all work was done in one and the same thread).
*
*
*/
public class AtomikosNonXADataSourceBean extends AbstractDataSourceBean
{
private static final long serialVersionUID = 1L;
private String url;
private String user;
private String password;
private String driverClassName;
private boolean readOnly;
/**
* Sets the URL to use for getting connections. Required.
*
* @param url
*/
public void setUrl ( String url )
{
this.url = url;
}
/**
* Gets the URL to connect.
*/
public String getUrl()
{
return url;
}
/**
* Marks this datasource as being used for read-only work. Optional.
*
* Setting this to true will avoid warnings/errors upon recovery. ReadOnly mode
* is intended to avoid XA configuration of databases where no updates are
* being done.
*
* @param readOnly Defaults to false.
*/
public void setReadOnly ( boolean readOnly )
{
this.readOnly = readOnly;
}
/**
* @return Whether or not this datasource is marked as readOnly.
*/
public boolean getReadOnly()
{
return readOnly;
}
/**
* @return The password.
*/
public String getPassword ()
{
return password;
}
/**
* Sets the password to use.
*
* @param string
*/
public void setPassword ( String string )
{
password = string;
}
/**
* Set the user name to get connections with.
*
* @param string
*/
public void setUser ( String string )
{
user = string;
}
/**
* @return The URL to connect with.
*/
public String getUser ()
{
return user;
}
/**
*
* @return The DriverManager class name.
*/
public String getDriverClassName ()
{
return driverClassName;
}
/**
* Sets the driver class name to be used by the DriverManager. Required.
*
* @param string
*/
public void setDriverClassName ( String string )
{
driverClassName = string;
}
protected void doClose()
{
//nothing to do
}
protected ConnectionFactory doInit() throws Exception
{
AtomikosNonXAConnectionFactory ret = null;
if ( Configuration.isInfoLoggingEnabled() ) Configuration.logInfo(
this + ": initializing with [" +
" uniqueResourceName=" + getUniqueResourceName() + "," +
" maxPoolSize=" + getMaxPoolSize() + "," +
" minPoolSize=" + getMinPoolSize() + "," +
" borrowConnectionTimeout=" + getBorrowConnectionTimeout() + "," +
" maxIdleTime=" + getMaxIdleTime() + "," +
" reapTimeout=" + getReapTimeout() + "," +
" maintenanceInterval=" + getMaintenanceInterval() + "," +
" testQuery=" + getTestQuery() + "," +
" driverClassName=" + getDriverClassName() + "," +
" user=" + getUser() + "," +
" url=" + getUrl() +
" loginTimeout=" + getLoginTimeout() +
"]"
);
ret = new com.atomikos.jdbc.nonxa.AtomikosNonXAConnectionFactory ( this , url , driverClassName , user , password , getLoginTimeout() , readOnly ) ;
ret.init();
return ret;
}
public synchronized Connection getConnection ( HeuristicMessage hmsg ) throws SQLException
{
if ( Configuration.isInfoLoggingEnabled() ) Configuration.logInfo ( this + ": getConnection ( " + hmsg + " )..." );
init();
//let pool take care of reusing an existing handle
Connection proxy = super.getConnection ( hmsg );
// here we are certain that proxy is not null -> increase the use count
DynamicProxy dproxy = ( DynamicProxy ) proxy;
com.atomikos.jdbc.nonxa.AtomikosThreadLocalConnection previous = (AtomikosThreadLocalConnection) dproxy.getInvocationHandler();
previous.incUseCount();
previous.addHeuristicMessage ( hmsg );
if ( Configuration.isDebugLoggingEnabled() ) Configuration.logDebug ( this + ": returning " + proxy );
return proxy;
}
public String toString()
{
String ret = "AtomikosNonXADataSourceBean";
String name = getUniqueResourceName();
if ( name != null ) {
ret = ret + " '" + name + "'";
}
return ret;
}
} - 对于JMS,可以使用我们的实例com.atomikos.jms.AtomikosConnectionFactoryBean, com.atomikos.jms.extra.AbstractJmsSenderTemplate(发送信息时使用)com.atomikos.jms.extra.MessageDrivenContainer(接收时使用)
/**
* Copyright (C) 2000-2010 Atomikos <info@atomikos.com>
*
* This code ("Atomikos TransactionsEssentials"), by itself,
* is being distributed under the
* Apache License, Version 2.0 ("License"), a copy of which may be found at
* http://www.atomikos.com/licenses/apache-license-2.0.txt .
* You may not use this file except in compliance with the License.
*
* While the License grants certain patent license rights,
* those patent license rights only extend to the use of
* Atomikos TransactionsEssentials by itself.
*
* This code (Atomikos TransactionsEssentials) contains certain interfaces
* in package (namespace) com.atomikos.icatch
* (including com.atomikos.icatch.Participant) which, if implemented, may
* infringe one or more patents held by Atomikos.
* It should be appreciated that you may NOT implement such interfaces;
* licensing to implement these interfaces must be obtained separately from Atomikos.
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
*/
package com.atomikos.jms;
import java.io.Serializable;
import javax.jms.ConnectionFactory;
import javax.jms.JMSException;
import javax.naming.NamingException;
import javax.naming.Reference;
import javax.naming.Referenceable;
import com.atomikos.util.SerializableObjectFactory;
/**
*
*
* Common logic for the connection factory beans.
*
*/
public abstract class AbstractConnectionFactoryBean
implements Serializable, Referenceable, ConnectionFactory
{
protected String resourceName_;
protected String xaFactoryJndiName_;
protected AbstractConnectionFactoryBean ( )
{
this.xaFactoryJndiName_ = "";
this.resourceName_ = "someUniqueName";
}
/**
* Sets the JNDI name of the underlying XAConnectionFactory (optional). This is
* optional and an alternative to directly supplying the required factory
* through setXaConnectionFactory().
*
* @param name
* The JNDI name where the XAConnectionFactory can be found.
* It is up to the client to make sure that the name exists and
* points to an existing XAConnectionFactory.
*/
public void setXaFactoryJndiName ( String name )
{
xaFactoryJndiName_ = name;
}
/**
* Retrieve the JNDI name where the XAConnectionFactory is expected.
*
* @return String the name or an empty String if not set.
*/
public String getXaFactoryJndiName()
{
return xaFactoryJndiName_;
}
/**
* Set the unique resource name for this factory (required). A unique
* resource name is needed by the transaction service in order to register
* and recover the underlying XA transactions.
* Note: the value you set here should not exceed 45 bytes in length.
*
* <p><b>MQSeries NOTE:</b> For
* IBM MQSeries, the name should include MQSeries_XA_RMI or the XA routines
* will not work properly!
*
* @param name
* The unique resource name.
*/
public void setResourceName ( String name )
{
resourceName_ = name;
}
/**
* Get the resource name.
*
* @return String the unique resource name as previously set.
*/
public String getResourceName()
{
return resourceName_;
}
public Reference getReference() throws NamingException
{
return SerializableObjectFactory.createReference ( this );
}
/**
* Initialization method to register the underlying resource for recovery
* and other init code.
*
* @throws JMSException
*/
public void init() throws JMSException
{
checkSetup();
}
protected abstract void checkSetup() throws JMSException;
}/**
* Copyright (C) 2000-2010 Atomikos <info@atomikos.com>
*
* This code ("Atomikos TransactionsEssentials"), by itself,
* is being distributed under the
* Apache License, Version 2.0 ("License"), a copy of which may be found at
* http://www.atomikos.com/licenses/apache-license-2.0.txt .
* You may not use this file except in compliance with the License.
*
* While the License grants certain patent license rights,
* those patent license rights only extend to the use of
* Atomikos TransactionsEssentials by itself.
*
* This code (Atomikos TransactionsEssentials) contains certain interfaces
* in package (namespace) com.atomikos.icatch
* (including com.atomikos.icatch.Participant) which, if implemented, may
* infringe one or more patents held by Atomikos.
* It should be appreciated that you may NOT implement such interfaces;
* licensing to implement these interfaces must be obtained separately from Atomikos.
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
*/
package com.atomikos.jms.extra;
import java.io.Serializable;
import java.util.Map;
import javax.jms.Connection;
import javax.jms.Destination;
import javax.jms.JMSException;
import javax.jms.Queue;
import javax.jms.Session;
import javax.jms.Topic;
import javax.transaction.Status;
import javax.transaction.SystemException;
import com.atomikos.icatch.jta.UserTransactionManager;
import com.atomikos.icatch.system.Configuration;
import com.atomikos.jms.AtomikosConnectionFactoryBean;
import com.atomikos.jms.AtomikosJMSException;
import com.atomikos.jms.AtomikosTransactionRequiredJMSException;
/**
* Common functionality for the sender templates.
*
*/
public abstract class AbstractJmsSenderTemplate
{
protected AtomikosConnectionFactoryBean connectionFactoryBean;
private String user;
protected String password;
protected Destination destination;
private String destinationName;
private Destination replyToDestination;
private String replyToDestinationName;
private int deliveryMode;
private int priority;
private long timeToLive;
protected boolean inited;
protected AbstractJmsSenderTemplate()
{
// set default values according to Sun's JMS javadocs
setTimeToLive ( 0 );
setDeliveryMode ( javax.jms.DeliveryMode.PERSISTENT );
setPriority ( 4 );
}
protected abstract Session getOrRefreshSession ( Connection c ) throws JMSException;
protected abstract Connection getOrReuseConnection() throws JMSException;
protected abstract void afterUseWithoutErrors ( Connection c , Session s ) throws JMSException;
protected void destroy ( Connection c , Session s)
throws JMSException {
try {
if ( s != null ) s.close();
} catch ( JMSException warn ) {
Configuration.logWarning ( this + ": error closing session" , warn);
}
try {
if ( c != null ) c.close();
} catch ( JMSException warn ) {
Configuration.logWarning ( this + ": error closing connection" , warn);
}
}
protected synchronized Connection refreshConnection() throws JMSException {
Connection connection = null;
if ( getDestinationName() == null )
throw new JMSException ( "Please call setDestination or setDestinationName first!" );
if ( user != null ) {
connection = connectionFactoryBean.createConnection (
user, password );
} else {
connection = connectionFactoryBean.createConnection ();
}
connection.start ();
return connection;
}
/**
* Initializes the session for sending.
* Call this method first.
*/
public void init() throws JMSException
{
if ( ! inited ) {
if ( connectionFactoryBean == null ) throw new IllegalStateException ( "Property 'atomikosConnectionFactoryBean' must be set first!" );
if ( getDestinationName() == null ) {
throw new IllegalStateException ( "Property 'destination' or 'destinationName' must be set first!" );
}
StringBuffer msg = new StringBuffer();
msg.append ( this + ":configured with [" );
msg.append ( "user=" ).append ( getUser() ).append ( ", " );
msg.append ( "password=" ).append ( password ).append ( ", " );
msg.append ( "deliveryMode=" ).append ( getDeliveryMode() ).append ( ", " );
msg.append ( "timeToLive=" ).append ( getTimeToLive() ).append ( ", " );
msg.append ( "priority=" ).append ( getPriority() ).append ( ", " );
msg.append ( "destination=" ).append( getDestinationName() ).append ( ", " );
msg.append ( "replyToDestination=" ).append ( getReplyToDestinationName() );
msg.append ( "]" );
if ( Configuration.isDebugLoggingEnabled() ) Configuration.logDebug ( msg.toString() );
inited = true;
}
}
private void retrieveDestinationIfNecessary() throws JMSException
{
if ( getDestination() == null ) {
String dName = getDestinationName();
RetrieveDestinationCallback cb = new RetrieveDestinationCallback ( dName );
executeCallbackInternal ( cb );
setDestination ( cb.getDestination() );
}
}
private void retrieveReplyToDestinationIfNecessary() throws JMSException
{
if ( getReplyToDestination() == null ) {
String dName = getReplyToDestinationName();
if ( dName != null ) {
RetrieveDestinationCallback cb = new RetrieveDestinationCallback ( dName );
executeCallbackInternal ( cb );
setReplyToDestination ( cb.getDestination() );
}
}
}
/**
* Sets the connection factory to use. Required.
* @param connectionFactory
*/
public void setAtomikosConnectionFactoryBean(AtomikosConnectionFactoryBean connectionFactory) {
this.connectionFactoryBean = connectionFactory;
}
public AtomikosConnectionFactoryBean getAtomikosConnectionFactoryBean() {
return connectionFactoryBean;
}
public Destination getDestination() {
return destination;
}
/**
* Sets the (provider-specific) destination name in order
* to lookup the destination (rather than providing one directly).
*
* Required, unless you set the destination directly.
*
* @param destinationName
*/
public void setDestinationName ( String destinationName )
{
this.destinationName = destinationName;
}
/**
* Sets the destination to send to. Required, unless
* you set the destinationName instead.
*
* @param destination
*/
public void setDestination(Destination destination) {
this.destination = destination;
}
private String getName(Destination d, String destinationName ) {
String ret = destinationName;
if ( ret == null ) {
if ( d instanceof Queue ) {
Queue q = ( Queue ) d;
try {
ret = q.getQueueName();
} catch ( JMSException e ) {
if ( Configuration.isDebugLoggingEnabled() ) Configuration.logDebug ( this + ": error retrieving queue name" , e );
}
} else if ( d instanceof Topic ) {
Topic t = ( Topic ) d;
try {
ret = t.getTopicName();
} catch ( JMSException e ) {
if ( Configuration.isDebugLoggingEnabled() ) Configuration.logDebug ( this + ": error retrieving topic name" , e );
}
}
}
return ret;
}
protected String getDestinationName() {
return getName ( getDestination() , destinationName );
}
protected String getReplyToDestinationName() {
return getName ( getReplyToDestination() , replyToDestinationName );
}
/**
* @return The user to connect with, or null if no explicit authentication
* is to be used.
*/
public String getUser() {
return user;
}
/**
* If this session is used for sending request/reply messages, then this
* property indicates the destination where the replies are to be sent (optional). The
* session uses this to set the JMSReplyTo header accordingly. This property
* can be omitted if no reply is needed.
*
* <p>
* The replyToDestination should be in the same JMS vendor domain as the send
* queue. To cross domains, configure a bridge for both the request and the
* reply channels.
*/
public void setReplyToDestination(Destination destination)
{
this.replyToDestination = destination;
}
/**
* Sets the provider-specific replyToDestinationName. Optional.
*
* @param replyToDestinationName
*/
public void setReplyToDestinationName ( String replyToDestinationName )
{
this.replyToDestinationName = replyToDestinationName;
}
/**
* Gets the replyToDestination.
*
* @return
*/
public Destination getReplyToDestination() {
return replyToDestination;
}
/**
* Set the password for explicit authentication (optional).
* This is only required if
* the user has also been set.
*
* @param password
* The password.
*/
public void setPassword(String password) {
this.password = password;
}
/**
* Set the user to use for explicit authentication (optional). If no explicit
* authentication is required then this method should not be called.
*
* @param user
*/
public void setUser(String user) {
this.user = user;
}
protected void executeCallbackInternal (
JmsSenderTemplateCallback callback ) throws JMSException {
init();
Session session = null;
Connection conn = null;
try {
conn = getOrReuseConnection();
session = getOrRefreshSession ( conn );
if ( Configuration.isDebugLoggingEnabled() ) Configuration.logDebug ( "Calling callback..." );
callback.doInJmsSession ( session );
if ( Configuration.isDebugLoggingEnabled() ) Configuration.logDebug ( "Callback done!" );
afterUseWithoutErrors ( conn , session );
} catch ( AtomikosTransactionRequiredJMSException notx ) {
destroy ( conn , session );
String msg = "The JMS session you are using requires a JTA transaction context for the calling thread and none was found." + "\n" +
"Please correct your code to do one of the following: " + "\n" +
"1. start a JTA transaction before sending any message, or" + "\n" +
"2. increase the maxPoolSize of the AtomikosConnectionFactoryBean to avoid transaction timeout while waiting for a connection.";
Configuration.logWarning ( msg );
AtomikosTransactionRequiredJMSException.throwAtomikosTransactionRequiredJMSException ( msg );
} catch ( JMSException e ) {
e.printStackTrace();
destroy ( conn , session );
String msg = this + ": error in sending JMS message";
AtomikosJMSException.throwAtomikosJMSException( msg , e );
}
}
/**
* Executes an application-level call-back within the managed session.
*
* @param callback
* @throws JMSException
*/
public void executeCallback(JmsSenderTemplateCallback callback) throws JMSException {
init();
retrieveDestinationIfNecessary();
retrieveReplyToDestinationIfNecessary();
UserTransactionManager tm = new UserTransactionManager ();
try {
if ( tm.getStatus () != Status.STATUS_ACTIVE )
throw new JMSException (
"This method requires an active transaction!" );
} catch ( SystemException e ) {
Configuration
.logWarning ( this +": error in getting transaction status", e );
throw new RuntimeException ( e.getMessage () );
}
executeCallbackInternal ( callback );
}
/**
* @return The deliverymode for messages sent in this session.
*/
public int getDeliveryMode() {
return deliveryMode;
}
/**
* @return The priority for messages sent in this session.
*/
public int getPriority() {
return priority;
}
/**
* @return The timeToLive for messages sent in this session.
*/
public long getTimeToLive() {
return timeToLive;
}
/**
*
* Set the deliverymode for messages sent in this session (optional). Defaults to
* persistent.
*
* @param
*/
public void setDeliveryMode(int i) {
deliveryMode = i;
}
/**
* Set the priority for messages sent in this session (optional). Defaults to 4.
*
* @param
*/
public void setPriority(int i) {
priority = i;
}
/**
* Set the time to live for messages sent in this session (optional). Defaults to 0.
*
* @param
*/
public void setTimeToLive(long l) {
timeToLive = l;
}
/**
* Sends a TextMessage.
*
* @param content The text as a string.
* @throws JMSException
*/
public void sendTextMessage(String content) throws JMSException {
retrieveDestinationIfNecessary();
retrieveReplyToDestinationIfNecessary();
SendTextMessageCallback cb = new SendTextMessageCallback ( content , getDestination() , getReplyToDestination() , getDeliveryMode() , getPriority() , getTimeToLive() );
executeCallback ( cb );
}
/**
* Sends a MapMessage.
*
* @param content The Map to get the content from.
*
* @throws JMSException
*/
public void sendMapMessage(Map content) throws JMSException {
retrieveDestinationIfNecessary();
retrieveReplyToDestinationIfNecessary();
SendMapMessageCallback cb = new SendMapMessageCallback ( content , getDestination() , getReplyToDestination() , getDeliveryMode() , getPriority() , getTimeToLive() );
executeCallback ( cb );
}
/**
* Sends an ObjectMessage.
*
* @param content The serializable object content.
* @throws JMSException
*/
public void sendObjectMessage(Serializable content) throws JMSException {
retrieveDestinationIfNecessary();
retrieveReplyToDestinationIfNecessary();
SendObjectMessageCallback cb = new SendObjectMessageCallback ( content , getDestination() , getReplyToDestination() , getDeliveryMode() , getPriority() , getTimeToLive() );
executeCallback ( cb );
}
/**
* Sends a ByteMessage.
*
* @param content The content as a byte array.
* @throws JMSException
*/
public void sendBytesMessage(byte[] content) throws JMSException {
retrieveDestinationIfNecessary();
retrieveReplyToDestinationIfNecessary();
SendBytesMessageCallback cb = new SendBytesMessageCallback ( content , getDestination() , getReplyToDestination() , getDeliveryMode() , getPriority() , getTimeToLive() );
executeCallback ( cb );
}
/**
* Closes all resources.
*/
public void close() {
try {
Connection c = getOrReuseConnection();
Session s = getOrRefreshSession(c);
destroy(c, s);
} catch (JMSException e) {
Configuration.logWarning ( this + ": error closing" , e );
}
connectionFactoryBean.close();
}
}/**
* Copyright (C) 2000-2010 Atomikos <info@atomikos.com>
*
* This code ("Atomikos TransactionsEssentials"), by itself,
* is being distributed under the
* Apache License, Version 2.0 ("License"), a copy of which may be found at
* http://www.atomikos.com/licenses/apache-license-2.0.txt .
* You may not use this file except in compliance with the License.
*
* While the License grants certain patent license rights,
* those patent license rights only extend to the use of
* Atomikos TransactionsEssentials by itself.
*
* This code (Atomikos TransactionsEssentials) contains certain interfaces
* in package (namespace) com.atomikos.icatch
* (including com.atomikos.icatch.Participant) which, if implemented, may
* infringe one or more patents held by Atomikos.
* It should be appreciated that you may NOT implement such interfaces;
* licensing to implement these interfaces must be obtained separately from Atomikos.
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
*/
package com.atomikos.jms.extra;
import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;
import javax.jms.Destination;
import javax.jms.ExceptionListener;
import javax.jms.JMSException;
import javax.jms.MessageListener;
import com.atomikos.icatch.system.Configuration;
import com.atomikos.jms.AtomikosConnectionFactoryBean;
/**
*
* A message-driven container for asynchronously receiving JMS messages
* from a topic or queue, within a managed JTA transaction context.
*
* Upon start, an instance of this class will create a number of
* concurrent sessions that listen for incoming messages on the same destination.
* MessageListener instances should be thread-safe if the pool size is larger
* than one. Note: in general, after start() any changed properties are only
* effective on the next start() event.
*
* <p>
* <b>IMPORTANT:</b> the transactional behaviour guarantees redelivery after failures.
* As a side-effect, this can lead to so-called <em>poison messages</em>: messages
* whose processing repeatedly fails due to some recurring error (for instance, a primary
* key violation in the database, a NullPointerException, ...). Poison messages are problematic
* because they can prevent other messages from being processed, and block the system.
*
* To avoid poison messages, make sure that your MessageListener implementation
* only throws a <b>RuntimeException</b> when the problem is <em>transient</em>. In that
* case, the system will perform rollback and the message will be redelivered
* facing a clean system state. All non-transient errors (i.e., those that happen
* each time a message is delivered) indicate problems at the application level
* and should be dealt with by writing better application code.
*/
public class MessageDrivenContainer
implements MessageConsumerSessionProperties
{
private static final int DEFAULT_TIMEOUT = 30;
private AtomikosConnectionFactoryBean connectionFactoryBean;
private MessageListener messageListener;
private String user;
private String password;
private Destination destination;
private String destinationName;
private int transactionTimeout;
private int poolSize;
private List sessions;
private boolean daemonThreads;
private boolean notifyListenerOnClose;
private String messageSelector;
private ExceptionListener exceptionListener;
private String subscriberName;
private boolean noLocal;
private boolean unsubscribeOnClose;
private String clientID;
private int receiveTimeout;
public MessageDrivenContainer()
{
sessions = new ArrayList ();
notifyListenerOnClose = false;
setPoolSize ( 1 );
setTransactionTimeout ( DEFAULT_TIMEOUT );
}
private MessageConsumerSession createSession()
{
return new MessageConsumerSession ( this );
}
/**
* Sets the clientID for durable subscriptions. Optional.
*
* @param clientID
*/
public void setClientID ( String clientID ) {
this.clientID = clientID;
}
/**
* Sets the connection factory to use. Required.
* @param bean
*/
public void setAtomikosConnectionFactoryBean ( AtomikosConnectionFactoryBean bean )
{
this.connectionFactoryBean = bean;
}
public AtomikosConnectionFactoryBean getAtomikosConnectionFactoryBean()
{
return connectionFactoryBean;
}
/**
* Gets the destination.
*
* @return The destination, or null if not set.
*/
public Destination getDestination()
{
return destination;
}
/**
* Sets the JMS destination to listen on (required unless the destinationName is set instead).
*
* @param dest
*/
public void setDestination ( Destination dest )
{
this.destination = dest;
}
/**
* Gets the destination name.
*
* @return The name, or null if not set.
*/
public String getDestinationName()
{
return destinationName;
}
/**
* Sets the JMS provider-specific destination name
* (required unless the destination is set directly).
*
* @param destinationName
*/
public void setDestinationName ( String destinationName )
{
this.destinationName = destinationName;
}
/**
* Sets whether threads should be daemon threads or not (optional).
* Default is false.
* @param value If true then threads will be daemon threads.
*/
public void setDaemonThreads ( boolean value )
{
this.daemonThreads = value;
}
/**
* Tests whether threads are daemon threads.
* @return True if threads are deamons.
*/
public boolean getDaemonThreads()
{
return daemonThreads;
}
/**
*
* Get the message listener if any.
*
* @return
*/
public MessageListener getMessageListener()
{
return messageListener;
}
/**
* Get the transaction timeout.
*
* @return
*/
public int getTransactionTimeout()
{
return transactionTimeout;
}
/**
* Get the user for connecting, or null if the default user should be used.
*
* @return
*/
public String getUser()
{
return user;
}
/**
* Set the message listener to use (required).
* The same instance will be used for each
* session in the pool, meaning that instances need to be thread-safe. Only
* one listener is allowed at a time. Call this method with a null argument
* to unset the listener.
*
* @param listener
*/
public void setMessageListener ( MessageListener listener )
{
messageListener = listener;
Iterator it = sessions.iterator ();
while ( it.hasNext () ) {
MessageConsumerSession s = (MessageConsumerSession) it.next ();
s.setMessageListener ( listener );
}
}
/**
* Set the password if explicit authentication is needed (optional).
* You need to set this if the user is also set.
*
* @param string
*/
public void setPassword ( String string )
{
password = string;
}
/**
* Set the transaction timeout in seconds (optional).
*
* @param seconds
*/
public void setTransactionTimeout ( int seconds )
{
transactionTimeout = seconds;
}
/**
* Set the user to use for explicit authentication (optional).
* Don't set this property
* if you want to use the default authentication.
*
* @param string
*/
public void setUser ( String string )
{
user = string;
}
/**
* Get the message selector (if any)
*
* @return The selector, or null if none.
*/
public String getMessageSelector()
{
return this.messageSelector;
}
/**
* Set the message selector to use (optional).
*
* @param selector
*/
public void setMessageSelector ( String selector )
{
this.messageSelector = selector;
}
/**
* Get the size of the pool.
*
* @return
*/
public int getPoolSize()
{
return poolSize;
}
/**
* Sets the size of the session pool (optional).
* Default is 1.
*
* @param size
*/
public void setPoolSize ( int size )
{
poolSize = size;
}
/**
* Gets the exception listener (if any).
* @return Null if no ExceptionListener was set.
*/
public ExceptionListener getExceptionListener()
{
return exceptionListener;
}
/**
* Sets the exception listener (optional). The listener will be
* notified of connection-level JMS errors.
*
* @param exceptionListener
*/
public void setExceptionListener ( ExceptionListener exceptionListener )
{
this.exceptionListener = exceptionListener;
}
/**
* Test if this instance will receive sends from the same connection.
*
* @return
*/
public boolean isNoLocal() {
return noLocal;
}
/**
* Sets whether or not this topic should receive sends from the
* same connection (optional).
*
* @param noLocal
*/
public void setNoLocal(boolean noLocal) {
this.noLocal = noLocal;
}
/**
* Gets the subscriber name (for durable subscribers).
* @return The name, or null if not set (no durable subscriber).
*/
public String getSubscriberName() {
return subscriberName;
}
/**
* Sets the name to use for durable subscriptions (optional).
* <br>
* <b>Note: this name will be appended with a suffix to ensure uniqueness
* among instances in the pool. Otherwise, the JMS back-end would see
* multiple instances subscribing with the same name - an error.</b>
*
* @param subscriberName
*/
public void setSubscriberName(String subscriberName) {
this.subscriberName = subscriberName;
}
protected boolean getNoLocal() {
return isNoLocal();
}
/**
* Start listening for messages.
*
* @throws JMSException
*/
public void start() throws JMSException
{
if ( destination == null && destinationName == null )
throw new JMSException (
"MessageDrivenContainer: destination not specified" );
if ( connectionFactoryBean == null )
throw new JMSException (
"MessageDrivenContainer: factory not set" );
if ( messageListener == null )
throw new JMSException (
"MessageDrivenContainer: messageListener not set" );
for ( int i = 0; i < poolSize; i++ ) {
MessageConsumerSession s = createSession();
s.setMessageListener ( messageListener );
s.setPassword ( password );
s.setUser ( user );
s.setDestination ( destination );
s.setDestinationName ( destinationName );
s.setAtomikosConnectionFactoryBean ( connectionFactoryBean );
s.setDaemonThreads ( daemonThreads );
s.setNotifyListenerOnClose ( notifyListenerOnClose );
s.setMessageSelector ( getMessageSelector () );
s.setExceptionListener ( exceptionListener );
s.setNoLocal( noLocal );
s.setSubscriberName( subscriberName );
//set subscriber name with suffix to ensure unique names
if ( getSubscriberName() != null ) s.setSubscriberName ( getSubscriberName() + "-" + i );
s.setNoLocal ( getNoLocal() );
s.setClientID(clientID);
try {
s.startListening ();
// System.out.println ( "MessageDrivenContainer: started
// session");
} catch ( Exception e ) {
Configuration.logWarning ( "Error starting pool", e );
}
sessions.add ( s );
}
// set listener again to trigger listening
setMessageListener ( messageListener );
}
/**
* Stop listening for messages. If <b>notifyListenerOnClose</b> is set then
* calling this method will notify the listener by calling its onMessage
* method with a null argument (and also without transaction context).
*
* This method will wait for all active receive operations to unblock, which may take
* up to <b>receiveTimeout</b> seconds per active thread.
*/
public void stop()
{
Iterator it = sessions.iterator ();
while ( it.hasNext () ) {
MessageConsumerSession s = (MessageConsumerSession) it.next ();
s.stopListening ();
}
}
/**
* Getter to check whether the listener is notified on close.
*
* @return
*/
public boolean getNotifyListenerOnClose()
{
return notifyListenerOnClose;
}
/**
* Set whether the listener should be notified of close events on the pool
* (optional). Default is false.
*
* @param b
* If true, then the listener will receive a null message if the
* pool is closed.
*/
public void setNotifyListenerOnClose ( boolean b )
{
notifyListenerOnClose = b;
Iterator it = sessions.iterator ();
while ( it.hasNext () ) {
MessageConsumerSession s = (MessageConsumerSession) it.next ();
s.setNotifyListenerOnClose ( b );
}
}
/**
* Sets whether unsubscribe should be done at closing time (optional). Default is false.
*
* @param b If true, then unsubscribe will be done at closing time. This only applies to
* durable subscribers (i.e., cases where subscriberName is set).
*/
public void setUnsubscribeOnClose ( boolean b )
{
this.unsubscribeOnClose = b;
}
/**
* Getter to test if unsubscribe should be called on close.
*/
public boolean getUnsubscribeOnClose()
{
return unsubscribeOnClose;
}
/**
* Gets the receive timeout in seconds.
*
* @return
*/
public int getReceiveTimeout() {
int ret = receiveTimeout;
if ( ret <=0 ) ret = getTransactionTimeout()/2;
return ret;
}
/**
* Sets the receive timeout in seconds,
* i.e. the number of seconds to wait for incoming messages in the message listener thread's event loop.
*
* This property is optional and defaults to half the transactionTimeout, but typically this should be lower
* because the time required to shutdown (stop) this container will be bound by this value multiplied by
* the number of threads (as indicated by <b>poolSize</b>).
*
* @param seconds
*/
public void setReceiveTimeout(int seconds) {
this.receiveTimeout = seconds;
}
} - 请检查我们完整的用户指南
注意:虽然这个版本包含了特定的第三方产品,如RDBMS软件和JMS代理软件,请注意,AtomikosTransactionsEssentials绝不是仅限制在这些特定的产品,除了RDBMS软件和JMS代理软件大多数的其他软件依然兼容。
J2SE实例:examples/jse这个文件夹包含带有源代码的各种各样的例子。可以执行这个脚本来运行这个程序(示例程序只有文本输出,而没有图形化界面输出)
运行示例前最好安装Ant,然后打开命令窗口在 examples/jse 文件夹并且输入“ant”。
Linux/Unix/Mac OSX Note:在这些系统,你必须在终端输入'chmod u+x *.sh'命令,否则,示例将无法运行。
解决问题:偶尔,一些示例可能不能再继续启动使用(尽管在开始阶段可以正常运行),这个通常是由于进程原因引起的(比如JMS代理后台环境),关闭这期间的所有进程便会解决这个问题,或者,重新启动计算机,也可以解决这个问题。
在Spring中使用:Atomikos TransactionEssentials可以很方便的于Spring相结合,运行您程序使用企业级的J2EE应用程序,而不需要EJB甚至EJB容器。另外,我们额外提供了一个强大的功能,JMS消息驱动添加到Spring的内置特性。
查看examples/spring文件夹,将展示Atomikos TransactionEssentials如何配置基于Spring的应用程序。
在J2EE的web容器中使用:在您的J2EE应用程序环境中使用Atomikos TransactionsEssentials最简单的方法是通过Spring作为pico-container(组件容器),如果想获取更多信息,请到Spring处查看。
Javadoc:
http://www.atomikos.com/downloads/transactions-essentials/com/atomikos/AtomikosTransactionsEssentials/javadoc/3.7/index.html