Ошибка при использовании com.zaxxer.hikari.HikariDataSource

Я столкнулся с ошибкой при интеграции HikariCP с Spring JdbcTemplate. Я использую Spring 3.2.2, HikariCP 2.3.8 и sybase jconn4 версии 7.0.0.

Конфигурация пружины:

<bean id="hikariConfig" class="com.zaxxer.hikari.HikariConfig">
    <property name="dataSourceClassName" value="com.sybase.jdbc4.jdbc.SybDataSource" />

    <property name="connectionTestQuery" value="SELECT 1" />
    <property name="maximumPoolSize" value="100" />
    <property name="idleTimeout" value="60000" />
    <property name="jdbcUrl"
        value="jdbc:sybase:Tds:${hostname}:${port}/${dbname}" />
    <property name="username" value="${username}" />
    <property name="password" value="${password}" />
</bean>


<bean id="ds" class="com.zaxxer.hikari.HikariDataSource"
    destroy-method="close">
    <constructor-arg ref="hikariConfig" />
</bean>

<bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
    <property name="dataSource">
        <ref bean="ds" />
    </property>
</bean>

Каждый раз, когда контекст загружается, он выдает эту ошибку, и источник данных не может быть создан.

Caused by: org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'ds' defined in class path resource [commons/config/datasourceTest.xml]: Instantiation of bean failed; nested exception is org.springframework.beans.BeanInstantiationException: Could not instantiate bean class [com.zaxxer.hikari.HikariDataSource]: Constructor threw exception; nested exception is com.zaxxer.hikari.pool.PoolInitializationException: Exception during pool initialization
    at org.springframework.beans.factory.support.ConstructorResolver.autowireConstructor(ConstructorResolver.java:288)
    at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.autowireConstructor(AbstractAutowireCapableBeanFactory.java:1051)
    at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.createBeanInstance(AbstractAutowireCapableBeanFactory.java:955)
    at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.doCreateBean(AbstractAutowireCapableBeanFactory.java:490)
    at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.createBean(AbstractAutowireCapableBeanFactory.java:461)
    at org.springframework.beans.factory.support.AbstractBeanFactory$1.getObject(AbstractBeanFactory.java:295)
    at org.springframework.beans.factory.support.DefaultSingletonBeanRegistry.getSingleton(DefaultSingletonBeanRegistry.java:223)
    at org.springframework.beans.factory.support.AbstractBeanFactory.doGetBean(AbstractBeanFactory.java:292)
    at org.springframework.beans.factory.support.AbstractBeanFactory.getBean(AbstractBeanFactory.java:194)
    at org.springframework.beans.factory.support.DefaultListableBeanFactory.preInstantiateSingletons(DefaultListableBeanFactory.java:626)
    at org.springframework.context.support.AbstractApplicationContext.finishBeanFactoryInitialization(AbstractApplicationContext.java:932)
    at org.springframework.context.support.AbstractApplicationContext.refresh(AbstractApplicationContext.java:479)
    at org.springframework.test.context.support.AbstractGenericContextLoader.loadContext(AbstractGenericContextLoader.java:120)
    at org.springframework.test.context.support.AbstractGenericContextLoader.loadContext(AbstractGenericContextLoader.java:60)
    at org.springframework.test.context.support.AbstractDelegatingSmartContextLoader.delegateLoading(AbstractDelegatingSmartContextLoader.java:100)
    at org.springframework.test.context.support.AbstractDelegatingSmartContextLoader.loadContext(AbstractDelegatingSmartContextLoader.java:248)
    at org.springframework.test.context.CacheAwareContextLoaderDelegate.loadContextInternal(CacheAwareContextLoaderDelegate.java:64)
    at org.springframework.test.context.CacheAwareContextLoaderDelegate.loadContext(CacheAwareContextLoaderDelegate.java:91)
    ... 25 more
Caused by: org.springframework.beans.BeanInstantiationException: Could not instantiate bean class [com.zaxxer.hikari.HikariDataSource]: Constructor threw exception; nested exception is com.zaxxer.hikari.pool.PoolInitializationException: Exception during pool initialization
    at org.springframework.beans.BeanUtils.instantiateClass(BeanUtils.java:163)
    at org.springframework.beans.factory.support.SimpleInstantiationStrategy.instantiate(SimpleInstantiationStrategy.java:121)
    at org.springframework.beans.factory.support.ConstructorResolver.autowireConstructor(ConstructorResolver.java:280)
    ... 42 more
Caused by: com.zaxxer.hikari.pool.PoolInitializationException: Exception during pool initialization
    at com.zaxxer.hikari.pool.BaseHikariPool.initializeConnections(BaseHikariPool.java:544)
    at com.zaxxer.hikari.pool.BaseHikariPool.<init>(BaseHikariPool.java:171)
    at com.zaxxer.hikari.pool.HikariPool.<init>(HikariPool.java:60)
    at com.zaxxer.hikari.pool.HikariPool.<init>(HikariPool.java:48)
    at com.zaxxer.hikari.HikariDataSource.<init>(HikariDataSource.java:80)
    at sun.reflect.NativeConstructorAccessorImpl.newInstance0(Native Method)
    at sun.reflect.NativeConstructorAccessorImpl.newInstance(NativeConstructorAccessorImpl.java:57)
    at sun.reflect.DelegatingConstructorAccessorImpl.newInstance(DelegatingConstructorAccessorImpl.java:45)
    at java.lang.reflect.Constructor.newInstance(Constructor.java:526)
    at org.springframework.beans.BeanUtils.instantiateClass(BeanUtils.java:148)
    ... 44 more
Caused by: java.sql.SQLException: JZ0PN: Specified port number of -1 was out of range. Port numbers must meet the following conditions: 0 <= portNumber <= 65535
    at com.sybase.jdbc4.jdbc.SybConnection.getAllExceptions(Unknown Source)
    at com.sybase.jdbc4.jdbc.SybConnection.handleSQLE(Unknown Source)
    at com.sybase.jdbc4.jdbc.SybConnection.a(Unknown Source)
    at com.sybase.jdbc4.jdbc.SybConnection.handleHAFailover(Unknown Source)
    at com.sybase.jdbc4.jdbc.SybConnection.<init>(Unknown Source)
    at com.sybase.jdbc4.jdbc.SybDriver.createConnection(Unknown Source)
    at com.sybase.jdbc4.jdbc.SybDriver.connect(Unknown Source)
    at com.sybase.jdbc4.jdbc.SybDriver.connect(Unknown Source)
    at com.sybase.jdbc4.jdbc.SybDataSource.getConnection(Unknown Source)
    at com.zaxxer.hikari.pool.BaseHikariPool.addConnection(BaseHikariPool.java:438)
    at com.zaxxer.hikari.pool.BaseHikariPool.initializeConnections(BaseHikariPool.java:542)
    ... 53 more

Теперь интересно то, что это работает отлично, если я делаю это с помощью простого кода.

public class JdbcTemplateFactoryTest {

    @Autowired
    HikariConfig hikariConfig;

    //@Autowired
    //HikariDataSource ds;


    @Test
    public void hikaritest(){

        HikariDataSource ds = new HikariDataSource(hikariConfig);
        JdbcTemplate jdbcTemplate = new JdbcTemplate(ds);

        assertNotNull(jdbcTemplate);
    }
}

Странная часть, если я раскомментирую ds для загрузки весной), это не удастся.

Целый день потратил на это. Любое предложение приветствуется


person Amey Jadiye    schedule 24.06.2015    source источник
comment
Я вижу Указанный номер порта -1 был вне допустимого диапазона в трассировке стека. Подозреваю, что конфигурация еще не загружена. Ваш вариант, вероятно, происходит после получения конфигурации.   -  person OldCurmudgeon    schedule 24.06.2015
comment
Как определяется ${port}? Вы проверили, правильно ли он заменен реальным номером порта?   -  person Eric Darchis    schedule 24.06.2015
comment
я тоже подозреваю, но я думаю, что Spring позаботится об этом, прежде чем инициализировать bean-компонент, должны быть загружены зависимые bean-компоненты/конфигурации. и да, все конфигурации, определенные в ${}, загружаются правильно, я проверил это.   -  person Amey Jadiye    schedule 24.06.2015
comment
Покажите определение компонента HikariDataSource, который вы пытаетесь связать автоматически.   -  person Roman    schedule 24.06.2015


Ответы (2)


Свойства dataSourceClassName и jdbcUrl являются взаимоисключающими (см. документ). У вас есть два варианта...

1) Используя dataSourceClassName:

<bean id="hikariConfig" class="com.zaxxer.hikari.HikariConfig">
    <property name="dataSourceClassName" value="com.sybase.jdbc4.jdbc.SybDataSource" />

    <property name="connectionTestQuery" value="SELECT 1" />
    <property name="maximumPoolSize" value="100" />
    <property name="idleTimeout" value="60000" />
    <property name="username" value="${username}" />
    <property name="password" value="${password}" />
    <property name="dataSourceProperties">
       <props>
          <property name="serverName" value="${hostname}" />
          <property name="port" value="${port}" />
          <property name="databaseName" value="${dbname}" />
       </props>
    </property>
</bean>

-или-

2) Используя jdbcUrl:

<bean id="hikariConfig" class="com.zaxxer.hikari.HikariConfig">
   <property name="jdbcUrl"
             value="jdbc:sybase:Tds:${hostname}:${port}/${dbname}" />
   <property name="connectionTestQuery" value="SELECT 1" />
   <property name="maximumPoolSize" value="100" />
   <property name="idleTimeout" value="60000" />
   <property name="username" value="${username}" />
   <property name="password" value="${password}" />
</bean>

Если ваш драйвер не регистрируется самостоятельно, вам может потребоваться добавить свойство driverClassName к опции № 2.

ОБНОВЛЕНИЕ: Если вы умеете читать по-чешски, вы можете прочитать об этой же ошибке здесь http://blog.prodejna.biz/2013/03/glassfish-jms-broker-perzistujici.html. По сути, не используйте SyDataSource, используйте только подход, основанный на jdbcUrl (и, возможно, driverClassName, если необходимо).

person brettw    schedule 24.06.2015
comment
да, я узнал об этом из readme.md, но почему приходит ошибка или номер порта? - person Amey Jadiye; 24.06.2015

В моем случае я добавил профиль в свой файл конфигурации и начал весну, не указав его.

@Bean
public DataSource dataSource() throws Exception {
    JndiDataSourceLookup dataSourceLookup;
    DataSource dataSource;
    HikariDataSource hikariDataSource;
    HikariConfig hikariConfig;

    try {
        LOGGER.info("BEGIN");
        LOGGER.info("Creating DataSource");

        dataSourceLookup = new JndiDataSourceLookup();
        dataSourceLookup.setResourceRef(true);
        dataSource = dataSourceLookup.getDataSource(CO_PARTICIPACAO_DS);

        //hikariConfig = new HikariConfig(additionalProperties());
        hikariConfig=new HikariConfig();
        hikariConfig.setDataSource(dataSource);
        hikariDataSource = new HikariDataSource(hikariConfig);

        LOGGER.info("END");
        return hikariDataSource;
    } catch (Exception e) {
        LOGGER.error(e.getMessage(), e);
        throw new CoParticipacaoException(e);
    }
}
person LottaLava    schedule 21.08.2018