Add support for reactive Elasticsearch healthcheck
Prior to this commit, configuring a reactive Elasticsearch client would auto-configure an Actuator Health check using a synchronous client, with the default configuration properties (so tarting localhost:9200). This would lead to false reports of unhealthy Elasticsearch clusters when using reactive clients. This commit reproduces the logic for MongoDB repositories: if a reactive variant is available, it is selected for the health check infrastructure. See gh-21042
This commit is contained in:
parent
79770b9615
commit
203878a16f
|
|
@ -97,6 +97,7 @@ dependencies {
|
||||||
optional("org.springframework.data:spring-data-mongodb")
|
optional("org.springframework.data:spring-data-mongodb")
|
||||||
optional("org.springframework.data:spring-data-neo4j")
|
optional("org.springframework.data:spring-data-neo4j")
|
||||||
optional("org.springframework.data:spring-data-redis")
|
optional("org.springframework.data:spring-data-redis")
|
||||||
|
optional("org.springframework.data:spring-data-elasticsearch")
|
||||||
optional("org.springframework.data:spring-data-solr")
|
optional("org.springframework.data:spring-data-solr")
|
||||||
optional("org.springframework.integration:spring-integration-core")
|
optional("org.springframework.integration:spring-integration-core")
|
||||||
optional("org.springframework.kafka:spring-kafka")
|
optional("org.springframework.kafka:spring-kafka")
|
||||||
|
|
|
||||||
|
|
@ -0,0 +1,59 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2012-2020 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.actuate.autoconfigure.elasticsearch;
|
||||||
|
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
import reactor.core.publisher.Flux;
|
||||||
|
|
||||||
|
import org.springframework.boot.actuate.autoconfigure.health.CompositeReactiveHealthContributorConfiguration;
|
||||||
|
import org.springframework.boot.actuate.autoconfigure.health.ConditionalOnEnabledHealthIndicator;
|
||||||
|
import org.springframework.boot.actuate.elasticsearch.ElasticsearchReactiveHealthIndicator;
|
||||||
|
import org.springframework.boot.actuate.health.ReactiveHealthContributor;
|
||||||
|
import org.springframework.boot.autoconfigure.AutoConfigureAfter;
|
||||||
|
import org.springframework.boot.autoconfigure.EnableAutoConfiguration;
|
||||||
|
import org.springframework.boot.autoconfigure.condition.ConditionalOnBean;
|
||||||
|
import org.springframework.boot.autoconfigure.condition.ConditionalOnClass;
|
||||||
|
import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean;
|
||||||
|
import org.springframework.boot.autoconfigure.data.elasticsearch.ReactiveElasticsearchRestClientAutoConfiguration;
|
||||||
|
import org.springframework.context.annotation.Bean;
|
||||||
|
import org.springframework.context.annotation.Configuration;
|
||||||
|
import org.springframework.data.elasticsearch.client.reactive.ReactiveElasticsearchClient;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@link EnableAutoConfiguration Auto-configuration} for
|
||||||
|
* {@link ElasticsearchReactiveHealthIndicator} using the
|
||||||
|
* {@link ReactiveElasticsearchClient}.
|
||||||
|
*
|
||||||
|
* @author Aleksander Lech
|
||||||
|
* @since 2.3
|
||||||
|
*/
|
||||||
|
@Configuration(proxyBeanMethods = false)
|
||||||
|
@ConditionalOnClass({ ReactiveElasticsearchClient.class, Flux.class })
|
||||||
|
@ConditionalOnBean(ReactiveElasticsearchClient.class)
|
||||||
|
@ConditionalOnEnabledHealthIndicator("elasticsearch")
|
||||||
|
@AutoConfigureAfter(ReactiveElasticsearchRestClientAutoConfiguration.class)
|
||||||
|
public class ElasticSearchReactiveHealthContributorAutoConfiguration extends
|
||||||
|
CompositeReactiveHealthContributorConfiguration<ElasticsearchReactiveHealthIndicator, ReactiveElasticsearchClient> {
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
@ConditionalOnMissingBean(name = { "elasticsearchHealthIndicator", "elasticsearchHealthContributor" })
|
||||||
|
public ReactiveHealthContributor elasticsearchHealthContributor(Map<String, ReactiveElasticsearchClient> clients) {
|
||||||
|
return createContributor(clients);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
@ -49,8 +49,8 @@ public class ElasticSearchRestHealthContributorAutoConfiguration
|
||||||
extends CompositeHealthContributorConfiguration<ElasticsearchRestHealthIndicator, RestClient> {
|
extends CompositeHealthContributorConfiguration<ElasticsearchRestHealthIndicator, RestClient> {
|
||||||
|
|
||||||
@Bean
|
@Bean
|
||||||
@ConditionalOnMissingBean(name = { "elasticsearchRestHealthIndicator", "elasticsearchRestHealthContributor" })
|
@ConditionalOnMissingBean(name = { "elasticsearchHealthIndicator", "elasticsearchHealthContributor" })
|
||||||
public HealthContributor elasticsearchRestHealthContributor(Map<String, RestClient> clients) {
|
public HealthContributor elasticsearchHealthContributor(Map<String, RestClient> clients) {
|
||||||
return createContributor(clients);
|
return createContributor(clients);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -15,6 +15,7 @@ org.springframework.boot.actuate.autoconfigure.context.ShutdownEndpointAutoConfi
|
||||||
org.springframework.boot.actuate.autoconfigure.couchbase.CouchbaseHealthContributorAutoConfiguration,\
|
org.springframework.boot.actuate.autoconfigure.couchbase.CouchbaseHealthContributorAutoConfiguration,\
|
||||||
org.springframework.boot.actuate.autoconfigure.couchbase.CouchbaseReactiveHealthContributorAutoConfiguration,\
|
org.springframework.boot.actuate.autoconfigure.couchbase.CouchbaseReactiveHealthContributorAutoConfiguration,\
|
||||||
org.springframework.boot.actuate.autoconfigure.elasticsearch.ElasticSearchRestHealthContributorAutoConfiguration,\
|
org.springframework.boot.actuate.autoconfigure.elasticsearch.ElasticSearchRestHealthContributorAutoConfiguration,\
|
||||||
|
org.springframework.boot.actuate.autoconfigure.elasticsearch.ElasticSearchReactiveHealthContributorAutoConfiguration,\
|
||||||
org.springframework.boot.actuate.autoconfigure.endpoint.EndpointAutoConfiguration,\
|
org.springframework.boot.actuate.autoconfigure.endpoint.EndpointAutoConfiguration,\
|
||||||
org.springframework.boot.actuate.autoconfigure.endpoint.jmx.JmxEndpointAutoConfiguration,\
|
org.springframework.boot.actuate.autoconfigure.endpoint.jmx.JmxEndpointAutoConfiguration,\
|
||||||
org.springframework.boot.actuate.autoconfigure.endpoint.web.WebEndpointAutoConfiguration,\
|
org.springframework.boot.actuate.autoconfigure.endpoint.web.WebEndpointAutoConfiguration,\
|
||||||
|
|
|
||||||
|
|
@ -0,0 +1,67 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2012-2020 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.actuate.autoconfigure.elasticsearch;
|
||||||
|
|
||||||
|
import org.junit.jupiter.api.Test;
|
||||||
|
|
||||||
|
import org.springframework.boot.actuate.autoconfigure.health.HealthContributorAutoConfiguration;
|
||||||
|
import org.springframework.boot.actuate.elasticsearch.ElasticsearchReactiveHealthIndicator;
|
||||||
|
import org.springframework.boot.actuate.elasticsearch.ElasticsearchRestHealthIndicator;
|
||||||
|
import org.springframework.boot.autoconfigure.AutoConfigurations;
|
||||||
|
import org.springframework.boot.autoconfigure.data.elasticsearch.ElasticsearchDataAutoConfiguration;
|
||||||
|
import org.springframework.boot.autoconfigure.data.elasticsearch.ReactiveElasticsearchRestClientAutoConfiguration;
|
||||||
|
import org.springframework.boot.autoconfigure.elasticsearch.ElasticsearchRestClientAutoConfiguration;
|
||||||
|
import org.springframework.boot.test.context.runner.ApplicationContextRunner;
|
||||||
|
|
||||||
|
import static org.assertj.core.api.Assertions.assertThat;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests for {@link ElasticSearchReactiveHealthContributorAutoConfiguration}.
|
||||||
|
*
|
||||||
|
* @author Aleksander Lech
|
||||||
|
*/
|
||||||
|
class ElasticsearchReactiveHealthContributorAutoConfigurationTests {
|
||||||
|
|
||||||
|
private ApplicationContextRunner contextRunner = new ApplicationContextRunner().withConfiguration(AutoConfigurations
|
||||||
|
.of(ElasticsearchDataAutoConfiguration.class, ReactiveElasticsearchRestClientAutoConfiguration.class,
|
||||||
|
ElasticsearchRestClientAutoConfiguration.class,
|
||||||
|
ElasticSearchReactiveHealthContributorAutoConfiguration.class,
|
||||||
|
HealthContributorAutoConfiguration.class));
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void runShouldCreateIndicator() {
|
||||||
|
this.contextRunner.run((context) -> assertThat(context)
|
||||||
|
.hasSingleBean(ElasticsearchReactiveHealthIndicator.class).hasBean("elasticsearchHealthContributor"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void runWithRegularIndicatorShouldOnlyCreateReactiveIndicator() {
|
||||||
|
this.contextRunner
|
||||||
|
.withConfiguration(AutoConfigurations.of(ElasticSearchRestHealthContributorAutoConfiguration.class))
|
||||||
|
.run((context) -> assertThat(context).hasSingleBean(ElasticsearchReactiveHealthIndicator.class)
|
||||||
|
.hasBean("elasticsearchHealthContributor")
|
||||||
|
.doesNotHaveBean(ElasticsearchRestHealthIndicator.class));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void runWhenDisabledShouldNotCreateIndicator() {
|
||||||
|
this.contextRunner.withPropertyValues("management.health.elasticsearch.enabled:false")
|
||||||
|
.run((context) -> assertThat(context).doesNotHaveBean(ElasticsearchReactiveHealthIndicator.class)
|
||||||
|
.doesNotHaveBean("elasticsearchHealthContributor"));
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
@ -57,6 +57,7 @@ dependencies {
|
||||||
optional("org.springframework.data:spring-data-neo4j")
|
optional("org.springframework.data:spring-data-neo4j")
|
||||||
optional("org.springframework.data:spring-data-redis")
|
optional("org.springframework.data:spring-data-redis")
|
||||||
optional("org.springframework.data:spring-data-rest-webmvc")
|
optional("org.springframework.data:spring-data-rest-webmvc")
|
||||||
|
optional("org.springframework.data:spring-data-elasticsearch")
|
||||||
optional("org.springframework.data:spring-data-solr")
|
optional("org.springframework.data:spring-data-solr")
|
||||||
optional("org.springframework.integration:spring-integration-core")
|
optional("org.springframework.integration:spring-integration-core")
|
||||||
optional("org.springframework.security:spring-security-core")
|
optional("org.springframework.security:spring-security-core")
|
||||||
|
|
|
||||||
|
|
@ -0,0 +1,61 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2012-2020 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.actuate.elasticsearch;
|
||||||
|
|
||||||
|
import java.util.stream.Collectors;
|
||||||
|
|
||||||
|
import reactor.core.publisher.Mono;
|
||||||
|
|
||||||
|
import org.springframework.boot.actuate.health.AbstractReactiveHealthIndicator;
|
||||||
|
import org.springframework.boot.actuate.health.Health;
|
||||||
|
import org.springframework.boot.actuate.health.HealthIndicator;
|
||||||
|
import org.springframework.data.elasticsearch.client.reactive.ReactiveElasticsearchClient;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@link HealthIndicator} for an Elasticsearch cluster using a
|
||||||
|
* {@link ReactiveElasticsearchClient}.
|
||||||
|
*
|
||||||
|
* @author Aleksander Lech
|
||||||
|
* @since 2.3
|
||||||
|
*/
|
||||||
|
public class ElasticsearchReactiveHealthIndicator extends AbstractReactiveHealthIndicator {
|
||||||
|
|
||||||
|
private final ReactiveElasticsearchClient client;
|
||||||
|
|
||||||
|
public ElasticsearchReactiveHealthIndicator(ReactiveElasticsearchClient client) {
|
||||||
|
super("Elasticsearch health check failed");
|
||||||
|
this.client = client;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected Mono<Health> doHealthCheck(Health.Builder builder) {
|
||||||
|
return this.client.status().map((status) -> {
|
||||||
|
if (status.isOk()) {
|
||||||
|
builder.up();
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
builder.down();
|
||||||
|
}
|
||||||
|
|
||||||
|
builder.withDetails(status.hosts().stream().collect(Collectors
|
||||||
|
.toMap((host) -> host.getEndpoint().getHostString(), (host) -> host.getState().toString())));
|
||||||
|
|
||||||
|
return builder.build();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
Loading…
Reference in New Issue