Spring Boot和Tomcat无法找到JNDI资源

时间:2017-01-06 19:47:07

标签: java spring tomcat spring-boot jndi

我正在使用Spring Boot 1.4.2.RELEASE。我还使用Tomcat 8.5.9(非嵌入式)和Java 8

在我的application.properties文件中,我可以使用以下方式手动配置我的数据源:

spring.datasource.url = jdbc:as400://blahblah....
spring.datasource.username = myuser
spring.datasource.password = password
spring.datasource.driver-class-name=com.ibm.as400.access.AS400JDBCDriver

这很好用。但是,我们使用Tomcat来托管我们的连接。在Tomcat中,我们有:

<GlobalNamingResources>
    <Resource name="jdbc/BLAH" ..... />
</GlobalNamingResources>

然后我将application.properties更改为:

spring.datasource.jndi-name=java:comp/env/jdbc/BLAH
spring.jpa.database-platform=org.hibernate.dialect.DB2400Dialect
spring.jpa.show-sql=true

现在,我甚至无法构建应用程序。我得到以下异常:

o.s.w.c.s.GenericWebApplicationContext   : 
Exception encountered during context initialization - cancelling refresh attempt: org.springframework.beans.factory.UnsatisfiedDependencyException: 
Error creating bean with name 'org.springframework.boot.autoconfigure.orm.jpa.HibernateJpaAutoConfiguration': 
Unsatisfied dependency expressed through constructor parameter 0; 
nested exception is org.springframework.beans.factory.BeanCreationException: 

Error creating bean with name 'dataSource' defined in class path resource 
[org/springframework/boot/autoconfigure/jdbc/JndiDataSourceAutoConfiguration.class]: 

Bean instantiation via factory method failed; 
nested exception is org.springframework.beans.BeanInstantiationException: 
Failed to instantiate [javax.sql.DataSource]: 

Factory method 'dataSource' threw exception; nested exception is 
org.springframework.jdbc.datasource.lookup.DataSourceLookupFailureException: 

Failed to look up JNDI DataSource with name 'java:comp/env/jdbc/BLAH'; 

nested exception is javax.naming.NoInitialContextException: 
Need to specify class name in environment or system property, or as an applet parameter, 
or in an application resource file:  java.naming.factory.initial

NoInitialContextException让我相信Spring Boot无法运行Tomcat,因此无法执行JNDI查找。

请记住,完全相同的设置适用于我们使用Spring Boot 1.2.5.RELEASE。我们有另一个使用1.2.5.RELEASE,Java 7和Tomcat 7的应用程序,它使用JNDI资源。但是使用最新的Spring Boot,Java 8和Tomcat 8并没有。

感谢任何帮助。

我的pom.xml中的一些片段:

<parent>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-parent</artifactId>
    <version>1.4.2.RELEASE</version>
    <relativePath/>
</parent>
...
<dependencies>
    <dependency>
        <groupId>net.sf.jt400</groupId>
        <artifactId>jt400</artifactId>
        <version>6.7</version>
        <scope>provided</scope>
    </dependency>
    <dependency>
        <groupId>com.vaadin</groupId>
        <artifactId>vaadin-spring-boot-starter</artifactId>
        <version>1.1.1</version>
    </dependency>
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-web</artifactId>
    </dependency>
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-data-jpa</artifactId>
    </dependency>
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-tomcat</artifactId>
        <scope>provided</scope>
    </dependency>
    ....
</dependencies>

其他代码段:

@SpringBootApplication
public class MyApplication {
    public static void main(String[] args) {
        SpringApplication.run(MyApplication.class, args);
    }
}


@SpringBootConfiguration
@EnableAutoConfiguration
@ComponentScan(basePackages = "com.company.myapp")
@EnableJpaRepositories(basePackages = "com.company.myapp.repository")
@EnableTransactionManagement
@EnableVaadin
@EnableVaadinServlet
public class AppConfiguration {
}

更新

我已使用GlobalNamingResources中的server.xml在Tomcat(外部)中创建了资源。 这适用于其他应用程序。

更新2

忘记提及我在src/main/webapp/META-INF/context.xml文件中有以下内容:

<?xml version="1.0" encoding="UTF-8"?>
<Context>
    <ResourceLink global="jdbc/BLAH"
                  name="jdbc/BLAH"
                  type="com.mchange.v2.c3p0.ComboPooledDataSource"/>
</Context>

更新3

另外,我想明确问题发生在maven package步骤。

1 个答案:

答案 0 :(得分:0)

如果要在独立的Tomcat(非嵌入式)上进行部署,我相当确定您需要在Tomcat上而不是在application.properties中定义JNDI资源,而是在server.xml中定义如下:

<GlobalNamingResources>
    <Resource name="jdbc/serverdb" auth="Container" type="javax.sql.DataSource"
              maxTotal="5" maxIdle="5" maxWaitMillis="30000"
              validationQuery="select 1" testOnBorrow="true"
              username="$AUTH_DB_USER" password="$AUTH_DB_PASSWORD" driverClassName="org.postgresql.Driver"
              url="jdbc:postgresql://$AUTH_DB_HOST:5432/$AUTH_DB_NAME"/>
</GlobalNamingResources>