我已经研究这个问题好几天了,它快把我逼疯了。
我有一段遗留代码,使用Jboss 4.2.3。我最近将项目转换为Maven进行依赖管理,但是为项目编写的单元测试需要数据源。当我禁用测试时,程序工作正常,但是当我启用测试时,它抛出以下内容:
java.lang.ClassNotFoundException: oracle.jdbc.driver.OracleDriver
我在pom文件中列出了jboss嵌入的依赖项。
<dependency>
<groupId>org.jboss.embedded</groupId>
<artifactId>jboss-embedded-all</artifactId>
<version>beta3.SP12</version>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.jboss.embedded</groupId>
<artifactId>jboss-embedded</artifactId>
<version>beta3.SP12</version>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.jboss.embedded</groupId>
<artifactId>hibernate-all</artifactId>
<version>beta3.SP12</version>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.jboss.embedded</groupId>
<artifactId>thirdparty-all</artifactId>
<version>beta3.SP12</version>
<scope>test</scope>
</dependency>
<plugin>
<artifactId>maven-surefire-plugin</artifactId>
<version>2.4.3</version>
<configuration>
<additionalClasspathElements>
<additionalClasspathElement>src/test/bootstrap</additionalClasspathElement>
</additionalClasspathElements>
<forkMode>once</forkMode>
<argLine>-Dsun.lang.ClassLoader.allowArraySyntax=true</argLine>
<includes>
<include>**/*Test.java</include>
</includes>
<skip>false</skip>
</configuration>
</plugin>
在src/test/bootstrap/deploy中,我放置了dev-ds.xml文件,它正在被容器接收。
Bound ConnectionManager 'jboss.jca:service=DataSourceBinding,name=jdbc/theDS' to JNDI name 'java:jdbc/theDS'
我的dev-ds文件看起来像这样(有一些混淆,请原谅任何拼写错误)
<datasources>
<local-tx-datasource>
<jndi-name>jdbc/theDS</jndi-name>
<connection-url>jdbc:oracle:thin:@xxx.xxx.xxx:1521:DEVDB</connection-url>
<driver-class>oracle.jdbc.driver.OracleDriver</driver-class>
<user-name>user</user-name>
<password>password</password>
</local-tx-datasource>
</datasources>
实际的测试本身使用Mockito来尝试获取数据源。
@BeforeClass
public static void setUpClass() throws Exception
{
if( ! containerRunning )
{
Bootstrap bootstrap = Bootstrap.getInstance();
bootstrap.bootstrap();
containerRunning = true;
}
}
@Before
public void setUp()
{
MyDAO dao = spy( new MyDAO() );
try
{
InitialContext ctx = new InitialContext();
dao.datasource = (DataSource) ctx.lookup( "java:jdbc/theDS" );
Connection conn = dao.datasource.getConnection(); //bang, we're dead
}
异常:Could not create connection; - nested throwable: (org.jboss.resource.JBossResourceException: Failed to register driver for: oracle.jdbc.driver.OracleDriver; - nested throwable: (java.lang.ClassNotFoundException: oracle.jdbc.driver.OracleDriver)); - nested throwable: (org.jboss.resource.JBossResourceException: Could not create connection; - nested throwable: (org.jboss.resource.JBossResourceException: Failed to register driver for: oracle.jdbc.driver.OracleDriver; - nested throwable: (java.lang.ClassNotFoundException: oracle.jdbc.driver.OracleDriver)
正如我所提到的,代码在运行时可以工作,所以我知道ojdbc6.jar位于${jboss.dir}/server/default/lib中。但为了确定起见,我尝试将其添加到构建路径中,并且尝试了Eclipse中的"订购和导出",但都无济于事。我没主意了。
这是一种变通,但它现在可以工作。在maven-surefire-plugin中,我添加了以下行:
<additionalClasspathElement>lib/ojdb6.jar</additionalClasspathElement>
我还为DAO对象创建了一个测试构造函数,因为数据源创建是在构造函数中完成的,并且Mockito不会覆盖它。所有这些工作都是为了发现这些测试可能从一开始就没有起作用。:)
为了帮助那些在这里绊倒的人,在引导中,它不允许你将InitialContext中的任何东西绑定到全局作用域。即
InitialContext ctx = new InitialContext();
ctx.createSubcontext( "jdbc/theDS" ); //Does not work
从我在网上搜集的资料来看,嵌入式jboss只知道环境作用域。