如何修复 Wildfly 20 下 sajdbc4 的 "Invalid ODBC handle"?

How fix "Invalid ODBC handle" for sajdbc4 under Wildfly 20?

我之前有 Wildfly 10 运行ning,刚刚升级到 Wildfly 20(低于 Ubuntu 20)。在让 Sybase SQL Anywhere 17 sajdbc4 驱动程序工作时,我过去的配置不再有效。

我在 sajdbc4.jar 和 /opt/wildfly-20.0.1.Final/modules/system/layers/base/com/mydomain/sybase/main/ 中安装了 sajdbc4.jar 和支持文件,这是对问题的快速总结。然后我 运行 一个简单的 Java 测试应用程序来确认它可以连接到测试数据库并且工作正常。我在 standalone.xml 和 运行 Wildfly 中配置了一个驱动程序和数据源,如下所示。当我测试数据源的连接时,它失败并显示“无效的 ODBC 句柄”:

export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:/opt/wildfly-20.0.1.Final/modules/system/layers/base/com/mydomain/sybase/main
export CLASSPATH=$CLASSPATH:/opt/wildfly-20.0.1.Final/modules/system/layers/base/com/mydomain/sybase/main
sudo ./standalone.sh

任何人都可以解释我还需要做什么才能让 sajdbc4 驱动程序工作吗?

这里有详细的注释:

*** 我使用 tar 在以下位置安装 Wildfly 20: /opt/wildfly-20.0.1.Final

*** 将 Sybase Sql Anywhere 17 sajdbc4.jar 和其他支持文件放在 /opt/wildfly-20.0.1.Final/modules/system/layers/base/com/mydomain/sybase/main

*** 我有一个简单的 Java 应用程序,用于测试我 运行 与:

的连接

导出 LD_LIBRARY_PATH=$LD_LIBRARY_PATH:/opt/wildfly-20.0.1.Final/modules/system/layers/base/com/mydomain/sybase/main

java -classpath .:/opt/wildfly-20.0.1.Final/modules/system/layers/base/com/mydomain/sybase/main/sajdbc4.jar sajdbc4DriverTest.java

这个简单的测试应用程序 运行s 和转储数据库 table 所以我得出结论,我需要工作的一切都在 /opt/wildfly-20.0.1.Final/modules/system/layers/base/com/mydomain/sybase/main

*** 这里是module.xml(在.sybase/main):

<?xml version="1.0" encoding="utf-8"?>
<module xmlns="urn:jboss:module:1.1" name="com.mydomain.sybase">
    <resources>
        <resource-root path="sajdbc4.jar"/>
    </resources>
    <dependencies>
        <module name="javax.api"/>
        <module name="javax.transaction.api"/>  
  </dependencies>
  </module>

*** 为了测试 Wildfly,我在 standalaone.xml 中添加了驱动程序定义:

<drivers>
    ...
    <driver name="sajdbc4" module="com.mydomain.sybase"/>
</drivers>

*** 然后我在 standalaone.xml 中添加以下数据源:

<datasource jndi-name="java:jboss/datasources/TestDB" pool-name="TestDB" spy="true" tracking="true">
    <connection-url>jdbc:sqlanywhere:Host=192.168.1.89:11111,192.168.1.89:11112;ServerName=TestDB;</connection-url>
    <driver>sajdbc4</driver>
    <pool>
        <min-pool-size>0</min-pool-size>
        <max-pool-size>30</max-pool-size>
    </pool>
    <security>
        <user-name>...</user-name>
        <password>...</password>
    </security>
</datasource>

*** 然后我 运行 在 /opt/wildfly-20.0.1.Final/bin

export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:/opt/wildfly-20.0.1.Final/modules/system/layers/base/com/mydomain/sybase/main
export CLASSPATH=$CLASSPATH:/opt/wildfly-20.0.1.Final/modules/system/layers/base/com/mydomain/sybase/main
sudo ./standalone.sh

在终端日志的顶部我们看到:

    ...
JAVA_OPTS:  -server -Xms64m -Xmx512m -XX:MetaspaceSize=96M -XX:MaxMetaspaceSize=256m -Djava.net.preferIPv4Stack=true -Djboss.modules.system.pkgs=org.jboss.byteman -Djava.awt.headless=true  --add-exports=java.base/sun.nio.ch=ALL-UNNAMED --add-exports=jdk.unsupported/sun.misc=ALL-UNNAMED --add-exports=jdk.unsupported/sun.reflect=ALL-UNNAMED

注意这两个导出被忽略了!

*** 如果我随后尝试测试数据源连接(在管理控制台中),我们会崩溃:

...

    Caused by: java.lang.UnsatisfiedLinkError: no dbjdbc17 in java.library.path: [/usr/java/packages/lib, /usr/lib/x86_64-linux-gnu/jni, /lib/x86_64-linux-gnu, /usr/lib/x86_64-linux-gnu, /usr/lib/jni, /lib, /usr/lib]
        at java.base/java.lang.ClassLoader.loadLibrary(ClassLoader.java:2670)
    ...
    
    ]) - failure description: "WFLYJCA0040: failed to invoke operation: WFLYJCA0047: Connection is not valid"

为了解决“导出失败”问题,我修改了 standalone.config(“# ADDED FOLLOWING HACK”之后的所有内容)

#
# Specify options to pass to the Java VM.
#

    if [ "x$JAVA_OPTS" = "x" ]; then
       JAVA_OPTS="-Xms64m -Xmx512m -XX:MetaspaceSize=96M -XX:MaxMetaspaceSize=256m -Djava.net.preferIPv4Stack=true"
       JAVA_OPTS="$JAVA_OPTS -Djboss.modules.system.pkgs=$JBOSS_MODULES_SYSTEM_PKGS -Djava.awt.headless=true"
       # ADDED FOLLOWING HACK
       JAVA_OPTS="$JAVA_OPTS -Djava.library.path=/opt/wildfly-20.0.1.Final/modules/system/layers/base/com/mydomain/sybase/main -cp .:/opt/wildfly-20.0.1.Final/modules/system/layers/base/com/mydomain/sybase/main/sajdbc4.jar"
    else
       echo "JAVA_OPTS already set in environment; overriding default settings with values: $JAVA_OPTS"
    fi

*** 我再次 运行 在 /opt/wildfly-20.0.1.Final/bin

sudo ./standalone.sh

在终端日志的顶部,我们现在看到:

JAVA_OPTS:  -server -Xms64m -Xmx512m -XX:MetaspaceSize=96M -XX:MaxMetaspaceSize=256m -Djava.net.preferIPv4Stack=true -Djboss.modules.system.pkgs=org.jboss.byteman -Djava.awt.headless=true -Djava.library.path=/opt/wildfly-20.0.1.Final/modules/system/layers/base/com/mydomain/sybase/main -cp .:/opt/wildfly-20.0.1.Final/modules/system/layers/base/com/mydomain/sybase/main/sajdbc4.jar  --add-exports=java.base/sun.nio.ch=ALL-UNNAMED --add-exports=jdk.unsupported/sun.misc=ALL-UNNAMED --add-exports=jdk.unsupported/sun.reflect=ALL-UNNAMED

所以这表明我们现在有 java.library.path 和所需的类路径。

*** 如果我再次尝试测试我们现在崩溃的数据源连接:

... 原因:java.sql.SQLException:无效的 ODBC 句柄 在 com.mydomain.sybase//sap.jdbc4.sqlanywhere.IDriver.makeODBCConnection(本机方法) 在 com.mydomain.sybase//sap.jdbc4.sqlanywhere.IDriver.connect(IDriver.java:809) 在 org.jboss.ironjacamar.jdbcadapters@1.4.22.Final//org.jboss.jca.adapters.jdbc.local.LocalManagedConnectionFactory.createLocalManagedConnection(LocalManagedConnectionFactory.java:321) ... 还有 35 个

]) - 故障描述:“WFLYJCA0040:调用操作失败:WFLYJCA0047:连接无效”

  1. 为什么导出 LD_LIBRARY_PATH 和导出 CLASSPATH 被忽略? (我预计这就是导致“java.library.path 中没有 dbjdbc17”错误的原因)。我如何将这些指定给 Wildfly?

  2. 即使 JAVA_OPTS 显示 java.library.path 并且设置了 cp,我们仍然会失败并显示“无效的 ODBC 句柄”。这是非常 st运行ge 因为我的简单 Java 应用程序测试表明当 java.library.path 和 cp 都设置为 ./sybase/main “一切正常”。请注意,测试应用程序使用的连接字符串与我在 standalone.xml.

    中使用的连接字符串相同

提前致谢。

问题原来与我使用 运行 Wildfly 即服务这一事实有关,显然我在上面设置 java.library.path 的努力失败了。我知道错误的原因,但我不知道如何在 运行 作为服务时设置路径。