Browse Source

Make R2DBC auto-config back off without a connection provider

Closes gh-26439
pull/28398/head
Andy Wilkinson 5 years ago
parent
commit
a27dfcb9b0
  1. 63
      spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/r2dbc/NoConnectionFactoryBeanFailureAnalyzer.java
  2. 2
      spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/r2dbc/R2dbcAutoConfiguration.java
  3. 1
      spring-boot-project/spring-boot-autoconfigure/src/main/resources/META-INF/spring.factories
  4. 60
      spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/r2dbc/NoConnectionFactoryBeanFailureAnalyzerTests.java
  5. 9
      spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/r2dbc/R2dbcAutoConfigurationTests.java
  6. 2
      spring-boot-project/spring-boot-starters/spring-boot-starter-jooq/build.gradle

63
spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/r2dbc/NoConnectionFactoryBeanFailureAnalyzer.java

@ -0,0 +1,63 @@ @@ -0,0 +1,63 @@
/*
* Copyright 2012-2021 the original author or authors.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* https://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.springframework.boot.autoconfigure.r2dbc;
import io.r2dbc.spi.ConnectionFactory;
import org.springframework.beans.factory.NoSuchBeanDefinitionException;
import org.springframework.boot.diagnostics.AbstractFailureAnalyzer;
import org.springframework.boot.diagnostics.FailureAnalysis;
import org.springframework.core.Ordered;
/**
* An {@link AbstractFailureAnalyzer} that produces failure analysis when a
* {@link NoSuchBeanDefinitionException} for a {@link ConnectionFactory} bean is thrown
* and there is no {@code META-INF/services/io.r2dbc.spi.ConnectionFactoryProvider}
* resource on the classpath.
*
* @author Andy Wilkinson
*/
class NoConnectionFactoryBeanFailureAnalyzer extends AbstractFailureAnalyzer<NoSuchBeanDefinitionException>
implements Ordered {
private final ClassLoader classLoader;
NoConnectionFactoryBeanFailureAnalyzer() {
this(NoConnectionFactoryBeanFailureAnalyzer.class.getClassLoader());
}
NoConnectionFactoryBeanFailureAnalyzer(ClassLoader classLoader) {
this.classLoader = classLoader;
}
@Override
protected FailureAnalysis analyze(Throwable rootFailure, NoSuchBeanDefinitionException cause) {
if (ConnectionFactory.class.equals(cause.getBeanType())
&& this.classLoader.getResource("META-INF/services/io.r2dbc.spi.ConnectionFactoryProvider") == null) {
return new FailureAnalysis("No R2DBC ConnectionFactory bean is available "
+ "and no /META-INF/services/io.r2dbc.spi.ConnectionFactoryProvider resource could be found.",
"Check that the R2DBC driver for your database is on the classpath.", cause);
}
return null;
}
@Override
public int getOrder() {
return 0;
}
}

2
spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/r2dbc/R2dbcAutoConfiguration.java

@ -21,6 +21,7 @@ import io.r2dbc.spi.ConnectionFactory; @@ -21,6 +21,7 @@ import io.r2dbc.spi.ConnectionFactory;
import org.springframework.boot.autoconfigure.AutoConfigureBefore;
import org.springframework.boot.autoconfigure.EnableAutoConfiguration;
import org.springframework.boot.autoconfigure.condition.ConditionalOnClass;
import org.springframework.boot.autoconfigure.condition.ConditionalOnResource;
import org.springframework.boot.autoconfigure.jdbc.DataSourceAutoConfiguration;
import org.springframework.boot.autoconfigure.sql.init.SqlInitializationAutoConfiguration;
import org.springframework.boot.context.properties.EnableConfigurationProperties;
@ -36,6 +37,7 @@ import org.springframework.context.annotation.Import; @@ -36,6 +37,7 @@ import org.springframework.context.annotation.Import;
*/
@Configuration(proxyBeanMethods = false)
@ConditionalOnClass(ConnectionFactory.class)
@ConditionalOnResource(resources = "classpath:META-INF/services/io.r2dbc.spi.ConnectionFactoryProvider")
@AutoConfigureBefore({ DataSourceAutoConfiguration.class, SqlInitializationAutoConfiguration.class })
@EnableConfigurationProperties(R2dbcProperties.class)
@Import({ ConnectionFactoryConfigurations.Pool.class, ConnectionFactoryConfigurations.Generic.class,

1
spring-boot-project/spring-boot-autoconfigure/src/main/resources/META-INF/spring.factories

@ -165,6 +165,7 @@ org.springframework.boot.autoconfigure.jdbc.DataSourceBeanCreationFailureAnalyze @@ -165,6 +165,7 @@ org.springframework.boot.autoconfigure.jdbc.DataSourceBeanCreationFailureAnalyze
org.springframework.boot.autoconfigure.jdbc.HikariDriverConfigurationFailureAnalyzer,\
org.springframework.boot.autoconfigure.jooq.NoDslContextBeanFailureAnalyzer,\
org.springframework.boot.autoconfigure.r2dbc.ConnectionFactoryBeanCreationFailureAnalyzer,\
org.springframework.boot.autoconfigure.r2dbc.NoConnectionFactoryBeanFailureAnalyzer,\
org.springframework.boot.autoconfigure.session.NonUniqueSessionRepositoryFailureAnalyzer
# Template availability providers

60
spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/r2dbc/NoConnectionFactoryBeanFailureAnalyzerTests.java

@ -0,0 +1,60 @@ @@ -0,0 +1,60 @@
/*
* Copyright 2012-2021 the original author or authors.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* https://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.springframework.boot.autoconfigure.r2dbc;
import io.r2dbc.spi.ConnectionFactory;
import io.r2dbc.spi.ConnectionFactoryProvider;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.NoSuchBeanDefinitionException;
import org.springframework.boot.test.context.FilteredClassLoader;
import static org.assertj.core.api.Assertions.assertThat;
/**
* Tests for {@link NoConnectionFactoryBeanFailureAnalyzer}.
*
* @author Andy Wilkinson
*/
class NoConnectionFactoryBeanFailureAnalyzerTests {
@Test
void analyzeWhenNotNoSuchBeanDefinitionExceptionShouldReturnNull() {
assertThat(new NoConnectionFactoryBeanFailureAnalyzer().analyze(new Exception())).isNull();
}
@Test
void analyzeWhenNoSuchBeanDefinitionExceptionForDifferentTypeShouldReturnNull() {
assertThat(
new NoConnectionFactoryBeanFailureAnalyzer().analyze(new NoSuchBeanDefinitionException(String.class)))
.isNull();
}
@Test
void analyzeWhenNoSuchBeanDefinitionExceptionButProviderIsAvailableShouldReturnNull() {
assertThat(new NoConnectionFactoryBeanFailureAnalyzer()
.analyze(new NoSuchBeanDefinitionException(ConnectionFactory.class))).isNull();
}
@Test
void analyzeWhenNoSuchBeanDefinitionExceptionAndNoProviderShouldAnalyze() {
assertThat(new NoConnectionFactoryBeanFailureAnalyzer(
new FilteredClassLoader(("META-INF/services/" + ConnectionFactoryProvider.class.getName())::equals))
.analyze(new NoSuchBeanDefinitionException(ConnectionFactory.class))).isNotNull();
}
}

9
spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/r2dbc/R2dbcAutoConfigurationTests.java

@ -28,6 +28,7 @@ import io.r2dbc.h2.H2ConnectionFactory; @@ -28,6 +28,7 @@ import io.r2dbc.h2.H2ConnectionFactory;
import io.r2dbc.pool.ConnectionPool;
import io.r2dbc.pool.PoolMetrics;
import io.r2dbc.spi.ConnectionFactory;
import io.r2dbc.spi.ConnectionFactoryProvider;
import io.r2dbc.spi.Option;
import io.r2dbc.spi.Wrapped;
import org.assertj.core.api.InstanceOfAssertFactories;
@ -255,6 +256,14 @@ class R2dbcAutoConfigurationTests { @@ -255,6 +256,14 @@ class R2dbcAutoConfigurationTests {
});
}
@Test
void configureWithoutUrlAndNoConnectionFactoryProviderBacksOff() {
this.contextRunner
.withClassLoader(new FilteredClassLoader(
("META-INF/services/" + ConnectionFactoryProvider.class.getName())::equals))
.run((context) -> assertThat(context).doesNotHaveBean(R2dbcAutoConfiguration.class));
}
@Test
void configureWithDataSourceAutoConfigurationDoesNotCreateDataSource() {
this.contextRunner.withConfiguration(AutoConfigurations.of(DataSourceAutoConfiguration.class))

2
spring-boot-project/spring-boot-starters/spring-boot-starter-jooq/build.gradle

@ -2,7 +2,7 @@ plugins { @@ -2,7 +2,7 @@ plugins {
id "org.springframework.boot.starter"
}
description = "Starter for using jOOQ to access SQL databases. An alternative to spring-boot-starter-data-jpa or spring-boot-starter-jdbc"
description = "Starter for using jOOQ to access SQL databases with JDBC. An alternative to spring-boot-starter-data-jpa or spring-boot-starter-jdbc"
dependencies {
api(project(":spring-boot-project:spring-boot-starters:spring-boot-starter-jdbc"))

Loading…
Cancel
Save