Polish Elasticsearch health indicator
- Nest the configuration class in HealthIndicatorAutoConfiguration, bringing it into line with the other health indicator configuration classes - Include the statistics from the response in the health’s details - Map YELLOW to UP rather than UNKNOWN as it indicates that the cluster is running but that “the primary shard is allocated but replicas are not” [1]. The details can be used to determine the precise state of the cluster. - Add a property to configure the time that the health indicator will wait to receive a response from the cluster - Document the configuration properties - Update the tests to cover the updated functionality See gh-2399 [1] http://www.elastic.co/guide/en/elasticsearch/reference/1.x/cluster-health.html
This commit is contained in:
parent
58c8c2ccdc
commit
03b109a2c8
|
@ -0,0 +1 @@
|
||||||
|
/data
|
|
@ -97,8 +97,8 @@
|
||||||
<optional>true</optional>
|
<optional>true</optional>
|
||||||
</dependency>
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>org.springframework.data</groupId>
|
<groupId>org.elasticsearch</groupId>
|
||||||
<artifactId>spring-data-elasticsearch</artifactId>
|
<artifactId>elasticsearch</artifactId>
|
||||||
<optional>true</optional>
|
<optional>true</optional>
|
||||||
</dependency>
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
|
@ -173,35 +173,40 @@
|
||||||
<optional>true</optional>
|
<optional>true</optional>
|
||||||
</dependency>
|
</dependency>
|
||||||
<!-- Test -->
|
<!-- Test -->
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.springframework.boot</groupId>
|
||||||
|
<artifactId>spring-boot</artifactId>
|
||||||
|
<type>test-jar</type>
|
||||||
|
<scope>test</scope>
|
||||||
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>ch.qos.logback</groupId>
|
<groupId>ch.qos.logback</groupId>
|
||||||
<artifactId>logback-classic</artifactId>
|
<artifactId>logback-classic</artifactId>
|
||||||
<scope>test</scope>
|
<scope>test</scope>
|
||||||
</dependency>
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.apache.tomcat.embed</groupId>
|
||||||
|
<artifactId>tomcat-embed-logging-juli</artifactId>
|
||||||
|
<scope>test</scope>
|
||||||
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>org.crashub</groupId>
|
<groupId>org.crashub</groupId>
|
||||||
<artifactId>crash.connectors.telnet</artifactId>
|
<artifactId>crash.connectors.telnet</artifactId>
|
||||||
<scope>test</scope>
|
<scope>test</scope>
|
||||||
</dependency>
|
</dependency>
|
||||||
<dependency>
|
|
||||||
<groupId>org.springframework</groupId>
|
|
||||||
<artifactId>spring-test</artifactId>
|
|
||||||
<scope>test</scope>
|
|
||||||
</dependency>
|
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>org.hsqldb</groupId>
|
<groupId>org.hsqldb</groupId>
|
||||||
<artifactId>hsqldb</artifactId>
|
<artifactId>hsqldb</artifactId>
|
||||||
<scope>test</scope>
|
<scope>test</scope>
|
||||||
</dependency>
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>org.springframework.boot</groupId>
|
<groupId>org.springframework</groupId>
|
||||||
<artifactId>spring-boot</artifactId>
|
<artifactId>spring-test</artifactId>
|
||||||
<type>test-jar</type>
|
|
||||||
<scope>test</scope>
|
<scope>test</scope>
|
||||||
</dependency>
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>org.apache.tomcat.embed</groupId>
|
<groupId>org.springframework.data</groupId>
|
||||||
<artifactId>tomcat-embed-logging-juli</artifactId>
|
<artifactId>spring-data-elasticsearch</artifactId>
|
||||||
<scope>test</scope>
|
<scope>test</scope>
|
||||||
</dependency>
|
</dependency>
|
||||||
</dependencies>
|
</dependencies>
|
||||||
|
|
|
@ -1,55 +0,0 @@
|
||||||
/*
|
|
||||||
* Copyright 2014 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
|
|
||||||
*
|
|
||||||
* http://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;
|
|
||||||
|
|
||||||
import org.elasticsearch.client.Client;
|
|
||||||
import org.springframework.boot.actuate.health.ElasticsearchHealthIndicator;
|
|
||||||
import org.springframework.boot.actuate.health.ElasticsearchHealthIndicatorProperties;
|
|
||||||
import org.springframework.boot.actuate.health.HealthIndicator;
|
|
||||||
import org.springframework.boot.autoconfigure.AutoConfigureAfter;
|
|
||||||
import org.springframework.boot.autoconfigure.AutoConfigureBefore;
|
|
||||||
import org.springframework.boot.autoconfigure.condition.ConditionalOnBean;
|
|
||||||
import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean;
|
|
||||||
import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty;
|
|
||||||
import org.springframework.context.annotation.Bean;
|
|
||||||
import org.springframework.context.annotation.Configuration;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* {@link org.springframework.boot.autoconfigure.EnableAutoConfiguration Auto-configuration} for
|
|
||||||
* {@link org.springframework.boot.actuate.health.ElasticsearchHealthIndicator}.
|
|
||||||
*
|
|
||||||
* @author Binwei Yang
|
|
||||||
* @since 1.2.2
|
|
||||||
*/
|
|
||||||
@Configuration
|
|
||||||
@AutoConfigureBefore({EndpointAutoConfiguration.class})
|
|
||||||
@AutoConfigureAfter({HealthIndicatorAutoConfiguration.class})
|
|
||||||
@ConditionalOnProperty(prefix = "management.health.elasticsearch", name = "enabled", matchIfMissing = true)
|
|
||||||
public class ElasticsearchHealthIndicatorConfiguration {
|
|
||||||
|
|
||||||
@Bean
|
|
||||||
@ConditionalOnBean(Client.class)
|
|
||||||
@ConditionalOnMissingBean(name = "elasticsearchHealthIndicator")
|
|
||||||
public HealthIndicator elasticsearchHealthIndicator() {
|
|
||||||
return new ElasticsearchHealthIndicator();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Bean
|
|
||||||
public ElasticsearchHealthIndicatorProperties elasticsearchHealthIndicatorProperties() {
|
|
||||||
return new ElasticsearchHealthIndicatorProperties();
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -23,6 +23,7 @@ import javax.jms.ConnectionFactory;
|
||||||
import javax.sql.DataSource;
|
import javax.sql.DataSource;
|
||||||
|
|
||||||
import org.apache.solr.client.solrj.SolrServer;
|
import org.apache.solr.client.solrj.SolrServer;
|
||||||
|
import org.elasticsearch.client.Client;
|
||||||
import org.springframework.amqp.rabbit.core.RabbitTemplate;
|
import org.springframework.amqp.rabbit.core.RabbitTemplate;
|
||||||
import org.springframework.beans.factory.InitializingBean;
|
import org.springframework.beans.factory.InitializingBean;
|
||||||
import org.springframework.beans.factory.annotation.Autowired;
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
|
@ -31,6 +32,8 @@ import org.springframework.boot.actuate.health.CompositeHealthIndicator;
|
||||||
import org.springframework.boot.actuate.health.DataSourceHealthIndicator;
|
import org.springframework.boot.actuate.health.DataSourceHealthIndicator;
|
||||||
import org.springframework.boot.actuate.health.DiskSpaceHealthIndicator;
|
import org.springframework.boot.actuate.health.DiskSpaceHealthIndicator;
|
||||||
import org.springframework.boot.actuate.health.DiskSpaceHealthIndicatorProperties;
|
import org.springframework.boot.actuate.health.DiskSpaceHealthIndicatorProperties;
|
||||||
|
import org.springframework.boot.actuate.health.ElasticsearchHealthIndicator;
|
||||||
|
import org.springframework.boot.actuate.health.ElasticsearchHealthIndicatorProperties;
|
||||||
import org.springframework.boot.actuate.health.HealthAggregator;
|
import org.springframework.boot.actuate.health.HealthAggregator;
|
||||||
import org.springframework.boot.actuate.health.HealthIndicator;
|
import org.springframework.boot.actuate.health.HealthIndicator;
|
||||||
import org.springframework.boot.actuate.health.JmsHealthIndicator;
|
import org.springframework.boot.actuate.health.JmsHealthIndicator;
|
||||||
|
@ -47,6 +50,7 @@ import org.springframework.boot.autoconfigure.amqp.RabbitAutoConfiguration;
|
||||||
import org.springframework.boot.autoconfigure.condition.ConditionalOnBean;
|
import org.springframework.boot.autoconfigure.condition.ConditionalOnBean;
|
||||||
import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean;
|
import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean;
|
||||||
import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty;
|
import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty;
|
||||||
|
import org.springframework.boot.autoconfigure.elasticsearch.ElasticsearchAutoConfiguration;
|
||||||
import org.springframework.boot.autoconfigure.jdbc.DataSourceAutoConfiguration;
|
import org.springframework.boot.autoconfigure.jdbc.DataSourceAutoConfiguration;
|
||||||
import org.springframework.boot.autoconfigure.jdbc.metadata.DataSourcePoolMetadata;
|
import org.springframework.boot.autoconfigure.jdbc.metadata.DataSourcePoolMetadata;
|
||||||
import org.springframework.boot.autoconfigure.jdbc.metadata.DataSourcePoolMetadataProvider;
|
import org.springframework.boot.autoconfigure.jdbc.metadata.DataSourcePoolMetadataProvider;
|
||||||
|
@ -79,7 +83,8 @@ import org.springframework.mail.javamail.JavaMailSenderImpl;
|
||||||
@AutoConfigureAfter({ DataSourceAutoConfiguration.class, MongoAutoConfiguration.class,
|
@AutoConfigureAfter({ DataSourceAutoConfiguration.class, MongoAutoConfiguration.class,
|
||||||
MongoDataAutoConfiguration.class, RedisAutoConfiguration.class,
|
MongoDataAutoConfiguration.class, RedisAutoConfiguration.class,
|
||||||
RabbitAutoConfiguration.class, SolrAutoConfiguration.class,
|
RabbitAutoConfiguration.class, SolrAutoConfiguration.class,
|
||||||
MailSenderAutoConfiguration.class, JmsAutoConfiguration.class })
|
MailSenderAutoConfiguration.class, JmsAutoConfiguration.class,
|
||||||
|
ElasticsearchAutoConfiguration.class })
|
||||||
@EnableConfigurationProperties({ HealthIndicatorAutoConfigurationProperties.class })
|
@EnableConfigurationProperties({ HealthIndicatorAutoConfigurationProperties.class })
|
||||||
public class HealthIndicatorAutoConfiguration {
|
public class HealthIndicatorAutoConfiguration {
|
||||||
|
|
||||||
|
@ -306,4 +311,30 @@ public class HealthIndicatorAutoConfiguration {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Configuration
|
||||||
|
@ConditionalOnBean(Client.class)
|
||||||
|
@ConditionalOnProperty(prefix = "management.health.elasticsearch", name = "enabled", matchIfMissing = true)
|
||||||
|
@EnableConfigurationProperties(ElasticsearchHealthIndicatorProperties.class)
|
||||||
|
public static class ElasticsearchHealthIndicatorConfiguration extends
|
||||||
|
CompositeHealthIndicatorConfiguration<ElasticsearchHealthIndicator, Client> {
|
||||||
|
|
||||||
|
@Autowired
|
||||||
|
private Map<String, Client> clients;
|
||||||
|
|
||||||
|
@Autowired
|
||||||
|
private ElasticsearchHealthIndicatorProperties properties;
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
@ConditionalOnMissingBean(name = "elasticsearchHealthIndicator")
|
||||||
|
public HealthIndicator elasticsearchHealthIndicator() {
|
||||||
|
return createHealthIndicator(this.clients);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected ElasticsearchHealthIndicator createHealthIndicator(Client client) {
|
||||||
|
return new ElasticsearchHealthIndicator(client, this.properties);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright 2014 the original author or authors.
|
* Copyright 2012-2015 the original author or authors.
|
||||||
*
|
*
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
* you may not use this file except in compliance with the License.
|
* you may not use this file except in compliance with the License.
|
||||||
|
@ -16,47 +16,58 @@
|
||||||
|
|
||||||
package org.springframework.boot.actuate.health;
|
package org.springframework.boot.actuate.health;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
import org.elasticsearch.action.admin.cluster.health.ClusterHealthResponse;
|
import org.elasticsearch.action.admin.cluster.health.ClusterHealthResponse;
|
||||||
import org.elasticsearch.client.Client;
|
import org.elasticsearch.client.Client;
|
||||||
import org.elasticsearch.client.Requests;
|
import org.elasticsearch.client.Requests;
|
||||||
import org.springframework.beans.factory.annotation.Autowired;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Simple implementation of a {@link HealthIndicator} returning health status information for
|
* {@link HealthIndicator} for an Elasticsearch cluster.
|
||||||
* ElasticSearch cluster.
|
|
||||||
*
|
*
|
||||||
* @author Binwei Yang
|
* @author Binwei Yang
|
||||||
* @since 1.2.2
|
* @author Andy Wilkinson
|
||||||
|
* @since 1.3.0
|
||||||
*/
|
*/
|
||||||
public class ElasticsearchHealthIndicator extends ApplicationHealthIndicator {
|
public class ElasticsearchHealthIndicator extends AbstractHealthIndicator {
|
||||||
@Autowired
|
|
||||||
private Client client;
|
|
||||||
|
|
||||||
@Autowired
|
private final Client client;
|
||||||
private ElasticsearchHealthIndicatorProperties properties;
|
|
||||||
|
|
||||||
@Override
|
private final ElasticsearchHealthIndicatorProperties properties;
|
||||||
protected void doHealthCheck(Health.Builder builder) throws Exception {
|
|
||||||
try {
|
|
||||||
ClusterHealthResponse response = client.admin().cluster().health(Requests.clusterHealthRequest(
|
|
||||||
properties.getIndexNamesAsArray()
|
|
||||||
)).actionGet(100);
|
|
||||||
|
|
||||||
switch (response.getStatus()) {
|
public ElasticsearchHealthIndicator(Client client,
|
||||||
case GREEN:
|
ElasticsearchHealthIndicatorProperties properties) {
|
||||||
builder.up();
|
this.client = client;
|
||||||
break;
|
this.properties = properties;
|
||||||
case RED:
|
}
|
||||||
builder.down();
|
|
||||||
break;
|
@Override
|
||||||
case YELLOW:
|
protected void doHealthCheck(Health.Builder builder) throws Exception {
|
||||||
default:
|
List<String> indices = this.properties.getIndices();
|
||||||
builder.unknown();
|
ClusterHealthResponse response = this.client
|
||||||
break;
|
.admin()
|
||||||
}
|
.cluster()
|
||||||
builder.withDetail("clusterHealth", response);
|
.health(Requests.clusterHealthRequest(indices.isEmpty() ? null : indices
|
||||||
} catch (Exception handled) {
|
.toArray(new String[indices.size()])))
|
||||||
builder.unknown().withDetail("exception", handled);
|
.actionGet(this.properties.getResponseTimeout());
|
||||||
}
|
|
||||||
}
|
switch (response.getStatus()) {
|
||||||
|
case GREEN:
|
||||||
|
case YELLOW:
|
||||||
|
builder.up();
|
||||||
|
break;
|
||||||
|
case RED:
|
||||||
|
default:
|
||||||
|
builder.down();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
builder.withDetail("clusterName", response.getClusterName());
|
||||||
|
builder.withDetail("numberOfNodes", response.getNumberOfNodes());
|
||||||
|
builder.withDetail("numberOfDataNodes", response.getNumberOfDataNodes());
|
||||||
|
builder.withDetail("activePrimaryShards", response.getActivePrimaryShards());
|
||||||
|
builder.withDetail("activeShards", response.getActiveShards());
|
||||||
|
builder.withDetail("relocatingShards", response.getRelocatingShards());
|
||||||
|
builder.withDetail("initializingShards", response.getInitializingShards());
|
||||||
|
builder.withDetail("unassignedShards", response.getUnassignedShards());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright 2014 the original author or authors.
|
* Copyright 2012-2015 the original author or authors.
|
||||||
*
|
*
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
* you may not use this file except in compliance with the License.
|
* you may not use this file except in compliance with the License.
|
||||||
|
@ -16,37 +16,41 @@
|
||||||
|
|
||||||
package org.springframework.boot.actuate.health;
|
package org.springframework.boot.actuate.health;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
import org.springframework.boot.context.properties.ConfigurationProperties;
|
import org.springframework.boot.context.properties.ConfigurationProperties;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* External configuration properties for {@link ElasticsearchHealthIndicator}
|
* External configuration properties for {@link ElasticsearchHealthIndicator}
|
||||||
*
|
*
|
||||||
* @author Binwei Yang
|
* @author Binwei Yang
|
||||||
* @since 1.2.2
|
* @author Andy Wilkinson
|
||||||
|
* @since 1.3.0
|
||||||
*/
|
*/
|
||||||
@ConfigurationProperties("management.health.elasticsearch")
|
@ConfigurationProperties("management.health.elasticsearch")
|
||||||
public class ElasticsearchHealthIndicatorProperties {
|
public class ElasticsearchHealthIndicatorProperties {
|
||||||
|
|
||||||
public static final String ALL = "_all";
|
/**
|
||||||
|
* Comma-separated index names
|
||||||
|
*/
|
||||||
|
private List<String> indices = new ArrayList<String>();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* comma separated index names. the default includes all indices.
|
* The time, in milliseconds, to wait for a response from the cluster
|
||||||
*/
|
*/
|
||||||
private String indices = ALL;
|
private long responseTimeout = 100L;
|
||||||
|
|
||||||
public String getIndices() {
|
public List<String> getIndices() {
|
||||||
return indices;
|
return this.indices;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void setIndices(String indices) {
|
public long getResponseTimeout() {
|
||||||
this.indices = indices;
|
return this.responseTimeout;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void setResponseTimeout(long responseTimeout) {
|
||||||
|
this.responseTimeout = responseTimeout;
|
||||||
|
}
|
||||||
|
|
||||||
String[] getIndexNamesAsArray() {
|
|
||||||
if (null == indices) {
|
|
||||||
return new String[]{ALL};
|
|
||||||
} else {
|
|
||||||
return indices.split(",");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -12,5 +12,4 @@ org.springframework.boot.actuate.autoconfigure.MetricFilterAutoConfiguration,\
|
||||||
org.springframework.boot.actuate.autoconfigure.MetricRepositoryAutoConfiguration,\
|
org.springframework.boot.actuate.autoconfigure.MetricRepositoryAutoConfiguration,\
|
||||||
org.springframework.boot.actuate.autoconfigure.PublicMetricsAutoConfiguration,\
|
org.springframework.boot.actuate.autoconfigure.PublicMetricsAutoConfiguration,\
|
||||||
org.springframework.boot.actuate.autoconfigure.TraceRepositoryAutoConfiguration,\
|
org.springframework.boot.actuate.autoconfigure.TraceRepositoryAutoConfiguration,\
|
||||||
org.springframework.boot.actuate.autoconfigure.TraceWebFilterAutoConfiguration,\
|
org.springframework.boot.actuate.autoconfigure.TraceWebFilterAutoConfiguration
|
||||||
org.springframework.boot.actuate.autoconfigure.ElasticsearchHealthIndicatorConfiguration
|
|
|
@ -26,6 +26,7 @@ import org.junit.Test;
|
||||||
import org.springframework.boot.actuate.health.ApplicationHealthIndicator;
|
import org.springframework.boot.actuate.health.ApplicationHealthIndicator;
|
||||||
import org.springframework.boot.actuate.health.DataSourceHealthIndicator;
|
import org.springframework.boot.actuate.health.DataSourceHealthIndicator;
|
||||||
import org.springframework.boot.actuate.health.DiskSpaceHealthIndicator;
|
import org.springframework.boot.actuate.health.DiskSpaceHealthIndicator;
|
||||||
|
import org.springframework.boot.actuate.health.ElasticsearchHealthIndicator;
|
||||||
import org.springframework.boot.actuate.health.HealthIndicator;
|
import org.springframework.boot.actuate.health.HealthIndicator;
|
||||||
import org.springframework.boot.actuate.health.JmsHealthIndicator;
|
import org.springframework.boot.actuate.health.JmsHealthIndicator;
|
||||||
import org.springframework.boot.actuate.health.MailHealthIndicator;
|
import org.springframework.boot.actuate.health.MailHealthIndicator;
|
||||||
|
@ -35,6 +36,7 @@ import org.springframework.boot.actuate.health.RedisHealthIndicator;
|
||||||
import org.springframework.boot.actuate.health.SolrHealthIndicator;
|
import org.springframework.boot.actuate.health.SolrHealthIndicator;
|
||||||
import org.springframework.boot.autoconfigure.PropertyPlaceholderAutoConfiguration;
|
import org.springframework.boot.autoconfigure.PropertyPlaceholderAutoConfiguration;
|
||||||
import org.springframework.boot.autoconfigure.amqp.RabbitAutoConfiguration;
|
import org.springframework.boot.autoconfigure.amqp.RabbitAutoConfiguration;
|
||||||
|
import org.springframework.boot.autoconfigure.elasticsearch.ElasticsearchAutoConfiguration;
|
||||||
import org.springframework.boot.autoconfigure.jdbc.DataSourceBuilder;
|
import org.springframework.boot.autoconfigure.jdbc.DataSourceBuilder;
|
||||||
import org.springframework.boot.autoconfigure.jdbc.DataSourceProperties;
|
import org.springframework.boot.autoconfigure.jdbc.DataSourceProperties;
|
||||||
import org.springframework.boot.autoconfigure.jdbc.EmbeddedDataSourceConfiguration;
|
import org.springframework.boot.autoconfigure.jdbc.EmbeddedDataSourceConfiguration;
|
||||||
|
@ -59,6 +61,7 @@ import static org.junit.Assert.assertEquals;
|
||||||
*
|
*
|
||||||
* @author Christian Dupuis
|
* @author Christian Dupuis
|
||||||
* @author Stephane Nicoll
|
* @author Stephane Nicoll
|
||||||
|
* @author Andy Wilkinson
|
||||||
*/
|
*/
|
||||||
public class HealthIndicatorAutoConfigurationTests {
|
public class HealthIndicatorAutoConfigurationTests {
|
||||||
|
|
||||||
|
@ -362,6 +365,39 @@ public class HealthIndicatorAutoConfigurationTests {
|
||||||
.getClass());
|
.getClass());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void elasticSearchHealthIndicator() {
|
||||||
|
this.context = new AnnotationConfigApplicationContext();
|
||||||
|
EnvironmentTestUtils.addEnvironment(this.context,
|
||||||
|
"management.health.diskspace.enabled:false");
|
||||||
|
this.context.register(ElasticsearchAutoConfiguration.class,
|
||||||
|
ManagementServerProperties.class, HealthIndicatorAutoConfiguration.class);
|
||||||
|
this.context.refresh();
|
||||||
|
|
||||||
|
Map<String, HealthIndicator> beans = this.context
|
||||||
|
.getBeansOfType(HealthIndicator.class);
|
||||||
|
assertEquals(1, beans.size());
|
||||||
|
assertEquals(ElasticsearchHealthIndicator.class, beans.values().iterator().next()
|
||||||
|
.getClass());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void notElasticSearchHealthIndicator() {
|
||||||
|
this.context = new AnnotationConfigApplicationContext();
|
||||||
|
EnvironmentTestUtils.addEnvironment(this.context,
|
||||||
|
"management.health.elasticsearch.enabled:false",
|
||||||
|
"management.health.diskspace.enabled:false");
|
||||||
|
this.context.register(ElasticsearchAutoConfiguration.class,
|
||||||
|
ManagementServerProperties.class, HealthIndicatorAutoConfiguration.class);
|
||||||
|
this.context.refresh();
|
||||||
|
|
||||||
|
Map<String, HealthIndicator> beans = this.context
|
||||||
|
.getBeansOfType(HealthIndicator.class);
|
||||||
|
assertEquals(1, beans.size());
|
||||||
|
assertEquals(ApplicationHealthIndicator.class, beans.values().iterator().next()
|
||||||
|
.getClass());
|
||||||
|
}
|
||||||
|
|
||||||
@Configuration
|
@Configuration
|
||||||
@EnableConfigurationProperties
|
@EnableConfigurationProperties
|
||||||
protected static class DataSourceConfig {
|
protected static class DataSourceConfig {
|
||||||
|
|
|
@ -1,84 +0,0 @@
|
||||||
/*
|
|
||||||
* Copyright 2014 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
|
|
||||||
*
|
|
||||||
* http://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.health;
|
|
||||||
|
|
||||||
import org.elasticsearch.client.Client;
|
|
||||||
import org.junit.After;
|
|
||||||
import org.junit.Before;
|
|
||||||
import org.junit.Test;
|
|
||||||
import org.springframework.boot.actuate.autoconfigure.ElasticsearchHealthIndicatorConfiguration;
|
|
||||||
import org.springframework.boot.actuate.autoconfigure.EndpointAutoConfiguration;
|
|
||||||
import org.springframework.boot.autoconfigure.PropertyPlaceholderAutoConfiguration;
|
|
||||||
import org.springframework.boot.autoconfigure.elasticsearch.ElasticsearchAutoConfiguration;
|
|
||||||
import org.springframework.boot.autoconfigure.elasticsearch.ElasticsearchDataAutoConfiguration;
|
|
||||||
import org.springframework.context.annotation.AnnotationConfigApplicationContext;
|
|
||||||
|
|
||||||
import static org.junit.Assert.assertEquals;
|
|
||||||
import static org.junit.Assert.assertNotNull;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Test for {@link ElasticsearchHealthIndicator} for ElasticSearch cluster.
|
|
||||||
*
|
|
||||||
* @author Binwei Yang
|
|
||||||
* @since 1.2.2
|
|
||||||
*/
|
|
||||||
public class ElasticsearchHealthIndicatorTest {
|
|
||||||
private AnnotationConfigApplicationContext context;
|
|
||||||
|
|
||||||
@Before
|
|
||||||
public void setUp() throws Exception {
|
|
||||||
this.context = new AnnotationConfigApplicationContext(
|
|
||||||
PropertyPlaceholderAutoConfiguration.class,
|
|
||||||
ElasticsearchAutoConfiguration.class,
|
|
||||||
ElasticsearchDataAutoConfiguration.class,
|
|
||||||
EndpointAutoConfiguration.class,
|
|
||||||
ElasticsearchHealthIndicatorConfiguration.class
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
@After
|
|
||||||
public void close() {
|
|
||||||
if (null != context) {
|
|
||||||
context.close();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Test
|
|
||||||
public void indicatorExists() {
|
|
||||||
assertEquals(1, this.context.getBeanNamesForType(Client.class).length);
|
|
||||||
|
|
||||||
ElasticsearchHealthIndicator healthIndicator = this.context.getBean(ElasticsearchHealthIndicator.class);
|
|
||||||
assertNotNull(healthIndicator);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Test
|
|
||||||
public void configPropertiesUsed() {
|
|
||||||
ElasticsearchHealthIndicatorProperties properties = this.context.getBean(ElasticsearchHealthIndicatorProperties.class);
|
|
||||||
|
|
||||||
// test default index
|
|
||||||
assertEquals(ElasticsearchHealthIndicatorProperties.ALL, properties.getIndices());
|
|
||||||
assertEquals(1, properties.getIndexNamesAsArray().length);
|
|
||||||
assertEquals(ElasticsearchHealthIndicatorProperties.ALL, properties.getIndexNamesAsArray()[0]);
|
|
||||||
|
|
||||||
// test specific indices
|
|
||||||
properties.setIndices("no-such-index");
|
|
||||||
|
|
||||||
ElasticsearchHealthIndicator healthIndicator = this.context.getBean(ElasticsearchHealthIndicator.class);
|
|
||||||
Health health = healthIndicator.health();
|
|
||||||
assertEquals(Status.UNKNOWN, health.getStatus());
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -0,0 +1,238 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2012-2015 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
|
||||||
|
*
|
||||||
|
* http://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.health;
|
||||||
|
|
||||||
|
import java.util.Arrays;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
import org.elasticsearch.ElasticsearchException;
|
||||||
|
import org.elasticsearch.ElasticsearchTimeoutException;
|
||||||
|
import org.elasticsearch.action.admin.cluster.health.ClusterHealthRequest;
|
||||||
|
import org.elasticsearch.action.admin.cluster.health.ClusterHealthResponse;
|
||||||
|
import org.elasticsearch.action.admin.cluster.health.ClusterHealthStatus;
|
||||||
|
import org.elasticsearch.action.support.PlainActionFuture;
|
||||||
|
import org.elasticsearch.client.AdminClient;
|
||||||
|
import org.elasticsearch.client.Client;
|
||||||
|
import org.elasticsearch.client.ClusterAdminClient;
|
||||||
|
import org.junit.Before;
|
||||||
|
import org.junit.Test;
|
||||||
|
import org.junit.runner.RunWith;
|
||||||
|
import org.mockito.ArgumentCaptor;
|
||||||
|
import org.mockito.Mock;
|
||||||
|
import org.mockito.runners.MockitoJUnitRunner;
|
||||||
|
|
||||||
|
import static org.hamcrest.Matchers.arrayContaining;
|
||||||
|
import static org.hamcrest.Matchers.containsString;
|
||||||
|
import static org.hamcrest.Matchers.equalTo;
|
||||||
|
import static org.hamcrest.Matchers.is;
|
||||||
|
import static org.hamcrest.Matchers.nullValue;
|
||||||
|
import static org.junit.Assert.assertThat;
|
||||||
|
import static org.mockito.BDDMockito.given;
|
||||||
|
import static org.mockito.Matchers.any;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test for {@link ElasticsearchHealthIndicator}.
|
||||||
|
*
|
||||||
|
* @author Andy Wilkinson
|
||||||
|
*/
|
||||||
|
@RunWith(MockitoJUnitRunner.class)
|
||||||
|
public class ElasticsearchHealthIndicatorTests {
|
||||||
|
|
||||||
|
@Mock
|
||||||
|
private Client client;
|
||||||
|
|
||||||
|
@Mock
|
||||||
|
private AdminClient admin;
|
||||||
|
|
||||||
|
@Mock
|
||||||
|
private ClusterAdminClient cluster;
|
||||||
|
|
||||||
|
private ElasticsearchHealthIndicator indicator;
|
||||||
|
|
||||||
|
private ElasticsearchHealthIndicatorProperties properties = new ElasticsearchHealthIndicatorProperties();
|
||||||
|
|
||||||
|
@Before
|
||||||
|
public void setUp() throws Exception {
|
||||||
|
given(this.client.admin()).willReturn(this.admin);
|
||||||
|
given(this.admin.cluster()).willReturn(this.cluster);
|
||||||
|
|
||||||
|
this.indicator = new ElasticsearchHealthIndicator(this.client, this.properties);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void defaultConfigurationQueriesAllIndicesWith100msTimeout() {
|
||||||
|
TestActionFuture responseFuture = new TestActionFuture();
|
||||||
|
responseFuture.onResponse(new StubClusterHealthResponse());
|
||||||
|
ArgumentCaptor<ClusterHealthRequest> requestCaptor = ArgumentCaptor
|
||||||
|
.forClass(ClusterHealthRequest.class);
|
||||||
|
given(this.cluster.health(requestCaptor.capture())).willReturn(responseFuture);
|
||||||
|
Health health = this.indicator.health();
|
||||||
|
assertThat(responseFuture.getTimeout, is(100L));
|
||||||
|
assertThat(requestCaptor.getValue().indices(), is(nullValue()));
|
||||||
|
assertThat(health.getStatus(), is(Status.UP));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void certainIndices() {
|
||||||
|
PlainActionFuture<ClusterHealthResponse> responseFuture = new PlainActionFuture<ClusterHealthResponse>();
|
||||||
|
responseFuture.onResponse(new StubClusterHealthResponse());
|
||||||
|
ArgumentCaptor<ClusterHealthRequest> requestCaptor = ArgumentCaptor
|
||||||
|
.forClass(ClusterHealthRequest.class);
|
||||||
|
given(this.cluster.health(requestCaptor.capture())).willReturn(responseFuture);
|
||||||
|
this.properties.getIndices()
|
||||||
|
.addAll(Arrays.asList("test-index-1", "test-index-2"));
|
||||||
|
Health health = this.indicator.health();
|
||||||
|
assertThat(requestCaptor.getValue().indices(),
|
||||||
|
is(arrayContaining("test-index-1", "test-index-2")));
|
||||||
|
assertThat(health.getStatus(), is(Status.UP));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void customTimeout() {
|
||||||
|
TestActionFuture responseFuture = new TestActionFuture();
|
||||||
|
responseFuture.onResponse(new StubClusterHealthResponse());
|
||||||
|
ArgumentCaptor<ClusterHealthRequest> requestCaptor = ArgumentCaptor
|
||||||
|
.forClass(ClusterHealthRequest.class);
|
||||||
|
given(this.cluster.health(requestCaptor.capture())).willReturn(responseFuture);
|
||||||
|
this.properties.setResponseTimeout(1000L);
|
||||||
|
this.indicator.health();
|
||||||
|
assertThat(responseFuture.getTimeout, is(1000L));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void healthDetails() {
|
||||||
|
PlainActionFuture<ClusterHealthResponse> responseFuture = new PlainActionFuture<ClusterHealthResponse>();
|
||||||
|
responseFuture.onResponse(new StubClusterHealthResponse());
|
||||||
|
given(this.cluster.health(any(ClusterHealthRequest.class))).willReturn(
|
||||||
|
responseFuture);
|
||||||
|
Health health = this.indicator.health();
|
||||||
|
assertThat(health.getStatus(), is(Status.UP));
|
||||||
|
Map<String, Object> details = health.getDetails();
|
||||||
|
assertDetail(details, "clusterName", "test-cluster");
|
||||||
|
assertDetail(details, "activeShards", 1);
|
||||||
|
assertDetail(details, "relocatingShards", 2);
|
||||||
|
assertDetail(details, "activePrimaryShards", 3);
|
||||||
|
assertDetail(details, "initializingShards", 4);
|
||||||
|
assertDetail(details, "unassignedShards", 5);
|
||||||
|
assertDetail(details, "numberOfNodes", 6);
|
||||||
|
assertDetail(details, "numberOfDataNodes", 7);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void redResponseMapsToDown() {
|
||||||
|
PlainActionFuture<ClusterHealthResponse> responseFuture = new PlainActionFuture<ClusterHealthResponse>();
|
||||||
|
responseFuture.onResponse(new StubClusterHealthResponse(ClusterHealthStatus.RED));
|
||||||
|
given(this.cluster.health(any(ClusterHealthRequest.class))).willReturn(
|
||||||
|
responseFuture);
|
||||||
|
assertThat(this.indicator.health().getStatus(), is(Status.DOWN));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void yellowResponseMapsToUp() {
|
||||||
|
PlainActionFuture<ClusterHealthResponse> responseFuture = new PlainActionFuture<ClusterHealthResponse>();
|
||||||
|
responseFuture.onResponse(new StubClusterHealthResponse(
|
||||||
|
ClusterHealthStatus.YELLOW));
|
||||||
|
given(this.cluster.health(any(ClusterHealthRequest.class))).willReturn(
|
||||||
|
responseFuture);
|
||||||
|
assertThat(this.indicator.health().getStatus(), is(Status.UP));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void responseTimeout() {
|
||||||
|
PlainActionFuture<ClusterHealthResponse> responseFuture = new PlainActionFuture<ClusterHealthResponse>();
|
||||||
|
given(this.cluster.health(any(ClusterHealthRequest.class))).willReturn(
|
||||||
|
responseFuture);
|
||||||
|
Health health = this.indicator.health();
|
||||||
|
assertThat(health.getStatus(), is(Status.DOWN));
|
||||||
|
assertThat((String) health.getDetails().get("error"),
|
||||||
|
containsString(ElasticsearchTimeoutException.class.getName()));
|
||||||
|
}
|
||||||
|
|
||||||
|
@SuppressWarnings("unchecked")
|
||||||
|
private <T> void assertDetail(Map<String, Object> details, String detail, T value) {
|
||||||
|
assertThat((T) details.get(detail), is(equalTo(value)));
|
||||||
|
}
|
||||||
|
|
||||||
|
private static class StubClusterHealthResponse extends ClusterHealthResponse {
|
||||||
|
|
||||||
|
private final ClusterHealthStatus status;
|
||||||
|
|
||||||
|
private StubClusterHealthResponse() {
|
||||||
|
this(ClusterHealthStatus.GREEN);
|
||||||
|
}
|
||||||
|
|
||||||
|
private StubClusterHealthResponse(ClusterHealthStatus status) {
|
||||||
|
super("test-cluster", null);
|
||||||
|
this.status = status;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int getActiveShards() {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int getRelocatingShards() {
|
||||||
|
return 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int getActivePrimaryShards() {
|
||||||
|
return 3;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int getInitializingShards() {
|
||||||
|
return 4;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int getUnassignedShards() {
|
||||||
|
return 5;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int getNumberOfNodes() {
|
||||||
|
return 6;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int getNumberOfDataNodes() {
|
||||||
|
return 7;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public ClusterHealthStatus getStatus() {
|
||||||
|
return this.status;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
private static class TestActionFuture extends
|
||||||
|
PlainActionFuture<ClusterHealthResponse> {
|
||||||
|
|
||||||
|
private long getTimeout = -1L;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public ClusterHealthResponse actionGet(long timeoutMillis)
|
||||||
|
throws ElasticsearchException {
|
||||||
|
this.getTimeout = timeoutMillis;
|
||||||
|
return super.actionGet(timeoutMillis);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
|
@ -63,6 +63,7 @@
|
||||||
<ehcache.version>2.9.1</ehcache.version>
|
<ehcache.version>2.9.1</ehcache.version>
|
||||||
<flyway.version>3.2</flyway.version>
|
<flyway.version>3.2</flyway.version>
|
||||||
<freemarker.version>2.3.22</freemarker.version>
|
<freemarker.version>2.3.22</freemarker.version>
|
||||||
|
<elasticsearch.version>1.4.4</elasticsearch.version>
|
||||||
<gemfire.version>7.0.2</gemfire.version>
|
<gemfire.version>7.0.2</gemfire.version>
|
||||||
<glassfish-el.version>3.0.0</glassfish-el.version>
|
<glassfish-el.version>3.0.0</glassfish-el.version>
|
||||||
<gradle.version>1.12</gradle.version>
|
<gradle.version>1.12</gradle.version>
|
||||||
|
@ -1084,6 +1085,11 @@
|
||||||
<artifactId>websocket-server</artifactId>
|
<artifactId>websocket-server</artifactId>
|
||||||
<version>${jetty.version}</version>
|
<version>${jetty.version}</version>
|
||||||
</dependency>
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.elasticsearch</groupId>
|
||||||
|
<artifactId>elasticsearch</artifactId>
|
||||||
|
<version>${elasticsearch.version}</version>
|
||||||
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>org.flywaydb</groupId>
|
<groupId>org.flywaydb</groupId>
|
||||||
<artifactId>flyway-core</artifactId>
|
<artifactId>flyway-core</artifactId>
|
||||||
|
|
|
@ -542,13 +542,15 @@ content into your application; rather pick only the properties that you need.
|
||||||
|
|
||||||
# HEALTH INDICATORS (previously health.*)
|
# HEALTH INDICATORS (previously health.*)
|
||||||
management.health.db.enabled=true
|
management.health.db.enabled=true
|
||||||
|
management.health.elasticsearch.enabled=true
|
||||||
|
management.health.elasticsearch.response-timeout=100 # the time, in milliseconds, to wait for a response from the cluster
|
||||||
management.health.diskspace.enabled=true
|
management.health.diskspace.enabled=true
|
||||||
|
management.health.diskspace.path=.
|
||||||
|
management.health.diskspace.threshold=10485760
|
||||||
management.health.mongo.enabled=true
|
management.health.mongo.enabled=true
|
||||||
management.health.rabbit.enabled=true
|
management.health.rabbit.enabled=true
|
||||||
management.health.redis.enabled=true
|
management.health.redis.enabled=true
|
||||||
management.health.solr.enabled=true
|
management.health.solr.enabled=true
|
||||||
management.health.diskspace.path=.
|
|
||||||
management.health.diskspace.threshold=10485760
|
|
||||||
management.health.status.order=DOWN, OUT_OF_SERVICE, UNKNOWN, UP
|
management.health.status.order=DOWN, OUT_OF_SERVICE, UNKNOWN, UP
|
||||||
|
|
||||||
# MVC ONLY ENDPOINTS
|
# MVC ONLY ENDPOINTS
|
||||||
|
|
Loading…
Reference in New Issue