如何修复 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:连接无效”
为什么导出 LD_LIBRARY_PATH 和导出 CLASSPATH 被忽略? (我预计这就是导致“java.library.path 中没有 dbjdbc17”错误的原因)。我如何将这些指定给 Wildfly?
即使 JAVA_OPTS 显示 java.library.path 并且设置了 cp,我们仍然会失败并显示“无效的 ODBC 句柄”。这是非常 st运行ge 因为我的简单 Java 应用程序测试表明当 java.library.path 和 cp 都设置为 ./sybase/main “一切正常”。请注意,测试应用程序使用的连接字符串与我在 standalone.xml.
中使用的连接字符串相同
提前致谢。
问题原来与我使用 运行 Wildfly 即服务这一事实有关,显然我在上面设置 java.library.path 的努力失败了。我知道错误的原因,但我不知道如何在 运行 作为服务时设置路径。
我之前有 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:连接无效”
为什么导出 LD_LIBRARY_PATH 和导出 CLASSPATH 被忽略? (我预计这就是导致“java.library.path 中没有 dbjdbc17”错误的原因)。我如何将这些指定给 Wildfly?
即使 JAVA_OPTS 显示 java.library.path 并且设置了 cp,我们仍然会失败并显示“无效的 ODBC 句柄”。这是非常 st运行ge 因为我的简单 Java 应用程序测试表明当 java.library.path 和 cp 都设置为 ./sybase/main “一切正常”。请注意,测试应用程序使用的连接字符串与我在 standalone.xml.
中使用的连接字符串相同
提前致谢。
问题原来与我使用 运行 Wildfly 即服务这一事实有关,显然我在上面设置 java.library.path 的努力失败了。我知道错误的原因,但我不知道如何在 运行 作为服务时设置路径。