• Connector框架笔记


    一个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:

    1. ResourceAdapter实现类由部署描述符或者@Connector注解来指定,
    2. 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包括:

    1. 事务管理
    2. 安全
    3. 错误日志记录及跟踪
    4. 连接池管理

    应用服务器服用自己的实现方式去提供这些服务,连接器框架不指定应用服务器如何实现这些服务。

    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 {
        ....
    }
  • 相关阅读:
    MongoDB中_id(ObjectId)生成
    springMVC 拦截器源码解析
    初识设计模式(代理模式)
    vue VNode如何使用,是什么东西?
    AOP 怎么理解?
    Java集合框架——jdk 1.8 ArrayList 源码解析
    System.arraycopy 怎么使用的?
    Java集合框架——容器的快速报错机制 fail-fast 是什么?
    maven 如何依赖工程项目里面的 jar 包
    IDEA maven 项目如何上传到私服仓库
  • 原文地址:https://www.cnblogs.com/mosmith/p/6874080.html
Copyright © 2020-2023  润新知