2

我们在 WebSphere Liberty 上运行一个简单的 web 应用程序,它使用 Hibernate 作为持久性提供程序(作为库包含在 WAR 文件中)。

当应用程序启动时,Hibernate 被初始化,它将打开到 DB2 的连接并发出一些 SQL 语句。但是,当在 CICS 上运行并使用 JDBC Type 2 Driver DataSource 时,这会失败。记录以下消息(一些额外的换行符以提高可读性):

WARN  org.hibernate.engine.jdbc.env.internal.JdbcEnvironmentInitiator -
    HHH000342: Could not obtain connection to query metadata : [jcc][50053][12310][4.19.56]
    T2zOS exception: [jcc][T2zos]T2zosCicsApi.checkApiStatus:
       Thread is not CICS-DB2 compatible: CICS_REGION_BUT_API_DISALLOWED ERRORCODE=-4228, SQLSTATE=null
...
ERROR org.hibernate.hql.spi.id.IdTableHelper - Unable obtain JDBC Connection
com.ibm.db2.jcc.am.SqlException: [jcc][50053][12310][4.19.56] T2zOS exception: [jcc][T2zos]T2zosCicsApi.checkApiStatus:
       Thread is not CICS-DB2 compatible: CICS_REGION_BUT_API_DISALLOWED ERRORCODE=-4228, SQLSTATE=null
    at com.ibm.db2.jcc.am.kd.a(Unknown Source) ~[db2jcc4.jar:?]
    ...
    at com.ibm.db2.jcc.t2zos.T2zosConnection.a(Unknown Source) ~[db2jcc4.jar:?]
    ...
    at com.ibm.db2.jcc.DB2SimpleDataSource.getConnection(Unknown Source) ~[db2jcc4.jar:?]
    at com.ibm.cics.wlp.jdbc.internal.CICSDataSource.getConnection(CICSDataSource.java:176) ~[?:?]
    at org.hibernate.engine.jdbc.connections.internal.DatasourceConnectionProviderImpl.getConnection(DatasourceConnectionProviderImpl.java:122) ~[our-app.war:5.1.0.Final]
    at org.hibernate.internal.SessionFactoryImpl$3.obtainConnection(SessionFactoryImpl.java:643) ~[our-app.war:5.1.0.Final]
    at org.hibernate.hql.spi.id.IdTableHelper.executeIdTableCreationStatements(IdTableHelper.java:67) [our-app.war:5.1.0.Final]
    at org.hibernate.hql.spi.id.global.GlobalTemporaryTableBulkIdStrategy.finishPreparation(GlobalTemporaryTableBulkIdStrategy.java:125) [our-app.war:5.1.0.Final]
    at org.hibernate.hql.spi.id.global.GlobalTemporaryTableBulkIdStrategy.finishPreparation(GlobalTemporaryTableBulkIdStrategy.java:42) [our-app.war:5.1.0.Final]
    at org.hibernate.hql.spi.id.AbstractMultiTableBulkIdStrategyImpl.prepare(AbstractMultiTableBulkIdStrategyImpl.java:88) [our-app.war:5.1.0.Final]
    at org.hibernate.internal.SessionFactoryImpl.<init>(SessionFactoryImpl.java:451) [our-app.war:5.1.0.Final]

我目前的理解是,当在 CICS 上运行并使用 JDBC Type 2 驱动程序时,只有一些线程能够打开 DB2 连接。那将是应用程序线程(处理 HTTP 请求的线程)以及服务于CICSExecutorService.

目前的解决方案是:

  1. JdbcEnvironmentInitiator通过将hibernate.temp.use_jdbc_metadata_defaults属性设置为 禁用 JDBC 元数据查找false
  2. 将执行包装IdTableHelper#executeIdTableCreationStatements在 a 中Runnable并将其提交给CICSExecutorService.

您认为此解决方案是否足够且适合生产?或者,也许你使用一些不同的方法?

使用的版本:

  • 适用于 z/OS 5.3.0 的 CICS 事务服务器
  • WebSphere 应用服务器 8.5.5.8
  • 休眠 5.1.0

更新:澄清一下,一旦我们的应用程序启动,它就可以毫无问题地查询 DB2(在服务 HTTP 请求时)。该问题仅与启动有关。

4

2 回答 2

0

最近在服务刷新(2016 年 7 月)中提供了对 Liberty 中 JPA 功能的 CICS TS v5.3 支持。在该更新之前,尝试在应用程序中运行 JPA 会导致与您描述的问题非常相似的问题。

尽管您正在运行休眠并且您在启用 CICS 的线程上,但它没有 API 环境(这将允许类型 2 JDBC 调用成功)。新的检测逻辑是专门(但不是专门)为与 DB2 JDBC type 2 驱动程序和 JPA 一起使用而开发的。此更新是在最近的服务更新中发布的,可能会解决您看到的问题。

尝试申请: http ://www-01.ibm.com/support/docview.wss?crawler=1&uid=swg1PI58375

描述说它是为了“标准模式 Liberty”的支持,但它包含上面概述的其他发展。

于 2016-09-16T12:35:37.913 回答
0

以下解决方案经过测试可以正常工作。

这个想法是使用CICSExecutorService#runAsCICS. 以下扩展是通过hibernate.hql.bulk_id_strategy属性注册的。

package org.hibernate.hql.spi.id.global;

import java.util.concurrent.*;
import org.hibernate.boot.spi.MetadataImplementor;
import org.hibernate.engine.jdbc.connections.spi.JdbcConnectionAccess;
import org.hibernate.engine.jdbc.spi.JdbcServices;
import org.springframework.util.ClassUtils;
import com.ibm.cics.server.*;

public class CicsAwareGlobalTemporaryTableBulkIdStrategy extends GlobalTemporaryTableBulkIdStrategy {

    @Override
    protected void finishPreparation(JdbcServices jdbcServices, JdbcConnectionAccess connectionAccess, MetadataImplementor metadata, PreparationContextImpl context) {
        execute(() -> super.finishPreparation(jdbcServices, connectionAccess, metadata, context));
    }

    @Override
    public void release(JdbcServices jdbcServices, JdbcConnectionAccess connectionAccess) {
        execute(() -> super.release(jdbcServices, connectionAccess));
    }

    private void execute(Runnable runnable) {
        if (isCics() && IsCICS.getApiStatus() == IsCICS.CICS_REGION_BUT_API_DISALLOWED) {
            RunnableFuture<Void> task = new FutureTask<>(runnable, null);
            CICSExecutorService.runAsCICS(task);
            try {
                task.get();
            } catch (InterruptedException | ExecutionException e) {
                throw new RuntimeException("Failed to execute in a CICS API-enabled thread. " + e.getMessage(), e);
            }
        } else {
            runnable.run();
        }
    }

    private boolean isCics() {
        return ClassUtils.isPresent("com.ibm.cics.server.CICSExecutorService", null);
    }
}

请注意,较新的 JCICS API 版本对runAsCics接受 a 的方法有一个重载Callable,这可能有助于将该execute方法的 CICS 分支简化为如下所示:

CICSExecutorService.runAsCICS(() -> { runnable.run(); return null; }).get();

尝试了一些替代方案:

  1. 仅包装连接获取操作 ( org.hibernate.engine.jdbc.connections.internal.DatasourceConnectionProviderImpl#getConnection) 不起作用,因为在主线程中使用连接时已经关闭。
  2. 包装整个应用程序启动 ( org.springframework.web.context.ContextLoaderListener#contextInitialized) 会导致类加载问题。

编辑:最终使用了一个自定义的 HibernateMultiTableBulkIdStrategy实现,它在启动时不运行任何 SQL/DDL(参见 GitHub 上的项目页面)。

于 2016-09-19T07:44:09.180 回答