一个ResourceAdapter可以提供不同类型的Connection:
- Outbound communication(由JavaEE应用服务器创建到EIS的连接):
- Inbound communication(由EIS创建到JavaEE应用服务器的连接):
- Bi-directional communication(双工通信连接):
连接器框架定义了下面几个应用服务器与EIS之间的接口:
- Connection Management 连接管理接口(连接管理,连接池)
- Transaction Management 事务管理接口(可以将连接纳入应用服务器的TransactionManager的管理范围)
- Security Contract安全接口
- ResourceAdapter Lifecycle Management Constract ResourceAdapter生命周期管理
- Work Management Constract 任务管理接口,ResourceAdapter可以向像线程池一样向任务管理接口提交任务,用于处理Inbound Connection等。
- 任务管理接口,ResourceAdapter可以管理自己管理的任务。
- 入站处理事务。(在事务上下文中处理Inbound Connection时)
- 安全上下文。(在安全上下文中运行提交的Work任务)
- 入站Message处理接口,ResourceAdapter可以异步将消息传递给Message Endpoint进行处理,消息可以是任何类型,而不必特定于jms(topic/queue)或者JAXM。
ResourceAdapter可以提供一套EIS-Specific的Client API
连接器框架支持在非受管环境下运行。
生命周期管理:
生命周期管理涉及的类如下图所示:
package javax.resource.spi; import javax.resource.spi.work.WorkManager; public interface ResourceAdapter { void start(BootstrapContext) // startup notification throws ResourceAdapterInternalException; void stop(); // shutdown notification ... // other operations } public interface BootstrapContext { WorkManager getWorkManager(); ... // other operations }
ResourceAdatpter以及ResourceAdapter的启动。
ResourceAdapter:
- ResourceAdapter实现类由部署描述符或者@Connector注解来指定,
- ResourceAdapter实现类必须是一个JavaBean,部署的时候Deployer需要对它配置并设置相应的属性。并且应用服务器可以使用Validator来对它进行验证。
ResourceAdapter的启动过程:
1、应用服务器实例化ResourceAdapter,并根据部署描述符设置相应的属性。(每次部署只允许实例化一个ResourceAdapter对象)
2、应用服务器调用ResourceAdapter的start方法,在参数中传入BootstrapContext对象,BootstrapContext对象向ResourceAdapter暴露应用服务器的组件服务比如WorkManager,ResourceAdapter可以使用这些组件。
ResourceAdapter在start方法中需要做下面的工作:
一些ResourceAdapter的初始化工作,因为我们的部署器只能给它设置相应的属性,初始化工作由ResourceAdapter在start方法中去做。初始化工作涉及
创建ResourceAdapter相关的对象
创建线程(Work Management)
初始化Endpoint(比如到EIS的连接或者监听端口)
在ResourceAdapter生命周期当中,ResourceAdapter可以包含一些多次创建或者销毁的对象,比如ManagedConnectionFactory,ActivationSpec,以及各种Connection,一些私有的对象以及一些暴露给应用的ResourceAdapter相关的对象。
ManagedConnectionFactory用于创建到EIS的Outbound Connection,一个ResourceAdapter可以有多个ManagedConnectionFactory对象。
ActivationSpec用于接收EIS的Inbound Connection,一个ResourceAdapter可以有多个ActionvationSpec对象。
ManagedConnectionFactory
当ManagedConnectionFactory创建的时候,它可以继承一些ResourceAdapter的配置,并覆盖一些默认的配置参数。
Output Communication由应用来发起并且与EIS通信的过程在应用的上下文中执行,ResourceAdapter也可以另起一个线程来处理通信过程。
在使用ManagedConnectionFactory之前,应用服务器必须将它与ResourceAdapter实例关联起来(通过调用ManagedConnectionFactory.setResourceAdapter方法)ManagedConnectionFactory.setResourceAdapter方法只允许调用一次,并且在ManagedConnectionFactory 的生命周期当中不能改变这个关联关系。
ActivationSpec
同ManagedConnectionFactory类似,在初始化的时候,ActivationSpec也可以继承来自ResourceAdapter的配置。
Inbound communication由EIS来发起,通信过程在ResourceAdapter线程上下文中执行,没有应用的线程参与、Resource Adapter可以使用Work Management contract接口来申请线程来处理Inbound Communication。
同ManagedConnectionFactory类似,使用ActivationSpec时,应用服务器必须将它与ResourceAdapter关联,ActivationSpec生命周期过程中不允许改变这个关联关系。
ResourceAdapter停止过程
ResourceAdapter的停止分为两个阶段:
阶段一:
在停止Resource之前(调用ResourceAdapter的stop方法),应用服务器必须保证所有使用此ResourceAdapter资源的相关应用都已经停止,包括停止Message endpoints。阶段一用于保证应用纯不再会使用到ResourceAdapter实例,这意味着所有相关应用的话动和事务活动都已经完成。因此阶段一保证了即使下一阶段中Resource Adapter不能正常停止,ResourceAdapter也不再会被使用。
阶段二:
应用服务器调用ResourceAdapter的stop方法,ResourceAdapter将在stop方法中执行停止操作以便被卸载。停止的操作包括以下几个方面:
1、关闭网络端口(Outbound and Inbound)
2、释放线程以及活动的Work任务。
3、允许ResourceAdapter内部正在 处理的事务完成提交,并将缓存的数据刷出到EIS中。
stop方法抛出任何异常都不会阻止应用服务器销毁ResourceAdapter
连接管理架构。
应用服务从部署描述符和注解中获取配置信息来配置ResourceAdapter实例,ResourceAdapter提供Connection和ConnectionFactory。比如javax.sql.DataSource和java.sql.Connection接口针对关系数据库提供的JDBC-base的接口。
相应地,CCI(Common Client Interface) 定义了javax.resource.cci.ConnectionFactory和javax.resource.cci.Connection。
应用组件通过jndi中的lookup操作来获取一个 Connection Factory,并通过这个Connection Factory获取一个连接到EIS的Connection实例。Connection Factory将connection 的创建请求委托给ConnectionManager来处理 。
应用服务器通过ConnectionManager来给受管应用提供QOS(quality-of-service),这里的quality-of-services包括:
- 事务管理
- 安全
- 错误日志记录及跟踪
- 连接池管理
应用服务器服用自己的实现方式去提供这些服务,连接器框架不指定应用服务器如何实现这些服务。
ConnectionManager实例接收到从Connection Factory过来的 Connection创建请求时,它从jndi上去获取应用服务器提供的连接池资源,如果连接池壁上没有能够满足相应请求的连接的话,应用服务器使用ManagedConnectionFactory接口(由ResourceAdapter来实现)来创建一个新的连接到EIS的物理连接。并将它加入到这个连接池当中。
应用服务器在ManagedConnection实例中注册一个ConnectionEventListener,这使用应用服务器可以从事件通知中获取到ManagedConnection实例的状态信息,进而利用这些事件通知去管理Connection Pooling(将连接加入池),管理事务,连接清理,以及处理一些错误。
应用服务器使用ManagedConnection实例去一个Connection实例,将它作为应用层到物理Connection的Handle(代理)。比如javax.resource.cci.Connection就是这样一个应用层的Connection Handler。
ResourceAdapter通过实现XAResource接口来提供事务管理,事务管理器也可以实现LocalTransaction接口,使得应用服务器可以管理ResourceManager内部的事务。
应用层面编码模型
受管应用应用:
应用集成者(Application Assembler)或者组件提供者通过部署描述符来指定需要引用的Connection Factory
■ res-ref-name: eis/MyEIS ■ res-type: javax.resource.cci.ConnectionFactory ■ res-auth: Application or Container <resource-ref> <res-ref-name>connectionFactory/testConnectionFactory</res-ref-name> <res-ref-type>javax.sql.DataSource</res-ref-type> <res-auth>Container</res-auth></resource-ref>
在ResourceAdapter部署过程中,deployer设置 ResourceAdapter的配置信息比如EIS的host, port等,然后应用服务器使用一个配置好的ResourceAdapter去创建到EIS的物理连接,
应用组件通过jndi获取ConnectionFactory,比如:
// obtain the initial JNDI Naming context Context initctx = new InitialContext(); // perform JNDI lookup to obtain the connection factory javax.resource.cci.ConnectionFactory cxf = (javax.resource.cci.ConnectionFactory) initctx.lookup(“java:comp/env/connectionFactory/testConnectionFactory”);
应用组件调用 Connection Factory的getConnection方法获取一个到EIS的连接javax.resource.cci.Connection
应用组件通过获取到的Connection实例对EIS进行操作。
操作完之后调用Connection实例的close方法。
连接管理涉及的接口/类UML图如下:
ConnectionFactory和Connection
ConnectionFactory用于获取到EIS实例的连接,Connection表示一个到EIS的连接。
public interface javax.resource.cci.ConnectionFactory extends java.io.Serializable, javax.resource.Referenceable { public javax.resource.cci.Connection getConnection() throws javax.resource.ResourceException; ... } public interface javax.resource.cci.Connection { public void close() throws javax.resource.ResourceException; ... }
ResourceAdapter被允许在Connection Factory接口额外的getConnection方法。
Connection的接口中应该有一个close方法。
ResourceAdapter必须提供Connection Factory的Connection接口的实现
ConnectionFactory的实现必须在应用组件的线程中去调用ConnectionManager.allocateConnection方法。
public interface javax.resource.spi.ConnectionManager extends java.io.Serializable { public Object allocateConnection( ManagedConnectionFactory mcf, ConnectionRequestInfo cxRequestInfo) throws ResourceException; } public interface javax.resource.spi.ConnectionRequestInfo { public boolean equals(Object other); public int hashCode(); }
ConnectionRequestInfo
ConnectionRequestInfo对象使得在ConnectionManager.allocateConnection时可以传递ResourceAdapter中的相关配置,资源甜酸器可以继承ConnectionRequestInfo来支持自定义的连接请求。
ManagedConnectionFactory必须持有全部用于创建物理连接的配置信息,这使得ConnectionManager可以管理这些连接。
资源适配器(Resource Adapter)必须实现ConnectionRequestInfo接口中的equals方法和hashCode方法。
ConnectionManager
javax.resource.spi.ConnectionManager接口为Resource Adapter向应用服务器传递连接创建请求提供了一个hook。allocateConnection方法由Connection Factory实例来调用,由些将创建请求传递给应用服务器的ConnectionManager实例。
public interface javax.resource.spi.ConnectionManager extends java.io.Serializable { public Object allocateConnection( ManagedConnectionFactory mcf, ConnectionRequestInfo cxRequestInfo) throws ResourceException; }
应用服务器必须提供ConnectionManager的实现,并且这个实现不能特定到某个Resource Adapter。ConnectionManager的将委托给应用服务器的部署机制,由些来提供比如安全,连接池管理,事务管理以及错误日志服务等。
ConnectionManager介入之后,连接创建请求被委托给ManagedConnectionFactory来进行处理,ManagedConnectionFactory将决定创建新请求还是从已经有的连接中取出一个返回。ConnectionManager必须实现Serializable接口。
Resource Adapter也可以提供一个ConnectionManager默认实现,用于非应用服务器管理的场合。
ManagedConnectionFactory
javax.resource.spi.ManagedConnectionFactory实现是ManagedConnection和Connection Factory的工厂类:
public interface javax.resource.spi.ManagedConnectionFactory extends java.io.Serializable { public Object createConnectionFactory( ConnectionManager connectionManager) throws ResourceException; public Object createConnectionFactory() throws ResourceException; public ManagedConnection createManagedConnection( javax.security.auth.Subject subject, ConnectionRequestInfo cxRequestInfo) throws ResourceException; public ManagedConnection matchManagedConnections( java.util.Set connectionSet, javax.security.auth.Subject subject, ConnectionRequestInfo cxRequestInfo) throws ResourceException; public boolean equals(Object other); public int hashCode(); }
createConnectionFactory方法用于创建Connection Factory实例,对于CCI来说,它就是javax.resource.cci.ConnectionFactory,然后使用应用服务器提供的ConnectionManager实例来对Connection Factory实例进行初始化。
createConnection方法用于创建一个物理连接。
matchConnection方法用于应用服务器从池中匹配一个连接。如果不能从池中匹配一个连接的话,这个方法直接返回null,然后应用服务器将请求Resource Adapter创建一个新的连接(上面的createConnection方法)
如果Resource Adapter不支持连接池的话,matchConnection方法它可以直接抛出NotSupportedException,这样子应用服务器就不会去缓存连接了。
ManagedConnectionFactory的实现类可以实现ValidatingManagedConnectionFactory接口,应用服务器可以使用该接口从池中移除无效的连接。
ManagedConnection
ManagedConnection实例表示一个物理连接,
public interface javax.resource.spi.ManagedConnection { public Object getConnection( javax.security.auth.Subject subject, ConnectionRequestInfo cxRequestInfo) throws ResourceException; public void destroy() throws ResourceException; public void cleanup() throws ResourceException; // Methods for Connection and transaction event notifications public void addConnectionEventListener(ConnectionEventListener listener); public void removeConnectionEventListener(ConnectionEventListener listener); public ManagedConnectionMetaData getMetaData() throws ResourceException; // Additional methods - specified in the other sections ... }
其中getConnection方法用于创建一个应用层面使用的Connection代理,对于CCI来说类型为javax.resource.cci.Connection
ManagedConnection实例可以使用getConnection方法并传递Subject和ConnectionRequestInfo参数来改变物理连接的状态。
addConnectionEventListener方法允许往ManagedConnection里面注册事件监听器,用于接收close/rrror或者事务相关的事件。
相反removeConnectionEventListener用于移除一个事件监听器。
Inbound Communication
这里涉及消息从EIS通过Resource Adapter投递到应用的EJB Container的整个过程。这里涉及到3个阶段:
1、Message Inflow
2、Ejb Invocation
3、Transaction Inflow
Message Inflow
Message Inflow定义了Resource Adapter异步地给应用中的endpoint传递Message的通用接口,这使得标准的Message Provider可以通过接入JavaEE应用服务器当中。这里的Endpoint是指Message Endpoint,比如MDB
ResourceAdapter接口提供了启用endpoint和禁用endpoint的方法,当需要启动一个Message Endpoint时候,应用服务器调用endpointActivation方法,当需要禁用一个endpoint时候,应用服务器去调用endpointDeactivation方法。
调用endpointActivation/endpointDeactivation方法时,应用服务器传递一个MessageEndpointFactory的实例和一个配置好的ActivationSpec实例。
ResourceAdapter使用MessageEndpointFactory实例来获取message endpoint实例,然后给把消息传递给它。可以使用MessageEndpointFactory来获取任意数量的message endpoint实例。
一个可以向message endpoint传递消息的ResourceAdapter必须为ActivationSpec提供它所支持的endpoint message listener类型(通过@Activation注解的messageListeners字段或者部署描述符<messagelistener-type>)。ActivationSpec实例是由message endpoint/application/deployer来配置,并为endpoint activation设置必要的属性配置信息。然后在endpoint部署的时候由应用服务器将ActivationSpec实例传递给resource adapter。
ResourceAdapter负责检测endpoint message listener的类型(通过使用ActivationSpec JavaBean的信息或者ActivationSpec JavaBean的类型,然后将从EIS接收到的message传递给这个endpoint),当创建一个“事务的”message的时,Resource Adapter可以选择传递一个XAResource实例给message endpoint。
endpoint的生命周期如下 :
1、Endpoint deployment
2、Message delivery(transacted and non-transacted)
3、Endpoint undeployment
Endpoint Deployment
Endpoint通常是一个message-driven bean应用。
MessageEndpoint负责提供它监听消息的类型(部署描述符<messaging-type>或者@MessageDriven的messageListeners字段)activation的配置信息(部署描述符<activation-config>或者@MessageDriven中的activationConfig字段)。
Resource adapter负责提供它支持的message listener类型,(message listener 接口的java全限定名)(resource adapter支持的message listener类型可以通过@Activation注解的messageListeners字段配置或者部署描述符<messagelistener-type>)。
比如下面的部署描述符中的例子:
CODE EXAMPLE 13-3 Message-Driven Bean Deployment Descriptor <!-- message-driven bean deployment descriptor --> ...<message-driven> <ejb-name>ExpenseProcessing</ejb-name> <ejb-class>com.wombat.empl.ExpenseProcessingBean</ejb-class> <messaging-type>javax.jms.MessageListener</messaging-type> ... <activation-config> <activation-config-property> <activation-config-property-name> destinationType </activation-config-property-name> <activation-config-property-value> javax.jms.Topic </activation-config-property-value> </activation-config-property> <activation-config-property> <activation-config-property-name> SubscriptionDurability </activation-config-property-name> <activation-config-property-value> Durable </activation-config-property-value> </activation-config-property> <activation-config-property> <activation-config-property-name> MessageSelector </activation-config-property-name> <activation-config-property-value> JMSType = ‘car‘ AND color = ‘blue‘ </activation-config-property-value> </activation-config-property> ... </activation-config></message-driven> ...
Application Server
应用服务器为message endpoint提供运行时环境。
应用服务器必须在启动应用组件的时候,将应用组件的ENC传递给resource adapter,resource adapter可以在endpointActivation和endpointDeactivation方便里面使用JNDI来获取jndi上面的资源。
resource adapter使用ManagedEndpointFactory来创建Message Endpoint实例来将消息传递给它。
应用服务器需要将配置好的Administrered object绑定到java:comp/env底下 。
部署:
Resource Adapter Provider
Resource Adapter 提供者负责指定ResourceAdapter的部署描述符,提供者可以在部署描述符中提供下面的信息
1、General information,通常是一些可读信息,比如Resource Adapter的名字,描述,License和版本信息等等 。
2、依赖的WorkContext类,ResourceAdapter可以提供一系列require-work-context元素用于指定要求应用服务器支持的WorkContext类型,这些类型必须是WorkContext的子类或者子接口。
3、javax.resource.spi.ResourceAdapter实现类,
4、ResourceAdapter的配置属性信息,这些信息用于配置ResourceAdapter JavaBean的属性。
5、Oubound resource adapter information
ManagedConnectionFactory实现类,与ResourceAdapter相同,它是遵循JavaBean规范。
ConnectionFactory接口和实现类
Connection接口及实现类。
Transaction Support事务支持情况(针对ResourceAdapter的实现情况),NoTransaction, LocalTransaction或者XATransaction。
ManagedConnectionFactory的默认配置信息(ManagedConnectionFactory可以有多个实例)
Authentication Mechanism认证机制。BasicPassword/Kerbv5
Reauthentication support
Extended Security Permission
6、Inbound Resource Adapter Informcation
Message Listener Type:Resource Adapter提供者必须指定一个或者多个支持的Message Listener类型(应用服务器根据@MessageDriven注解上messageListeners字段或者<messsaging-type>描述符元素对Inbound Resource Adapter进行匹配)。
ActivationSpec类:Resource Adapter提供者改组指定ActivationSpec的实现类名,和ResourceAdapter一样它也必须遵循JavaBean规范,ActivationSpec在部署过程中由message endpoint deployer来进行配置。
ActivationSpec要求的属性配置信息:
Administered object:Resource adapter提供者改必须指定Administered object的实现类 或者接口(应用服务器将查找接口的实现作为Administered Object),和ResourceAdapter实现类一样,必须遵循JavaBean规范,
Resource Definition Annotation
@ConnectionFactoryDefinition和@AdministeredObjectDefinition注解用于协助应用开 者定义和配置运行时要求的Resource Adapter相关资源。这些注解通过name来引用resource adapter。如果resource adapter打包在ear应用中的话,resource adapter的名字以#号开头,表示引用ear应用里面的resource adapter。
@ConnectionFactoryDefinition是一种Resource Definition Annotation,等同于<connection-factory>部署描述符元素,它用于定义一个connection factory并将它注册在jndi上,这个东西类似我们的@Resource/@PersistenceUnit等,这是JavaEE7引进来的annotation,类似于@DataSourceDefinition。在JavaEE7之前,ConnectionFactory只能通过私有部署描述符的方式创建 。
package javax.resource; import java.lang.annotation.Target; import java.lang.annotation.Retention; import java.lang.annotation.ElementType; import java.lang.annotation.RetentionPolicy; @Documented @Target({ElementType.TYPE}) @Retention(RetentionPolicy.RUNTIME) public @interface ConnectionFactoryDefinition { String name(); String description() default ""; String resourceAdapter(); String interfaceName(); TransactionSupport.TransactionSupportLevel transactionSupport() default TransactionSupport.TransactionSupportLevel.NoTransaction; int maxPoolSize() default -1; int minPoolSize() default -1; String[] properties() default {}; }
resourceAdapter字段用于指定resource adapter的名字()
interfaceName字段用于指定connection factory的全限定类名(必填),应用服务器将用它来查找对应的RAR中的ManagedConnectionFactory
@ConnectionFactoryDefinition(name="java:comp/eis/MyEISCF", interfaceName="com.eis.ConnectionFactory", resourceAdapter="MyEISRA", transactionSupport= TransactionSupport.TransactionSupportLevel.XATransaction) @Stateless public class TestBean { .... }