Commit 02917ff0 authored by Phillip Webb's avatar Phillip Webb

Merge pull request #7722 from artembilan/IntegrationManagement

* pr/7722:
  Polish Spring Integration metrics support
  Refactor Spring Integration metrics support
parents 2e0f87e7 b87e02dd
...@@ -193,11 +193,6 @@ ...@@ -193,11 +193,6 @@
</exclusion> </exclusion>
</exclusions> </exclusions>
</dependency> </dependency>
<dependency>
<groupId>org.springframework.integration</groupId>
<artifactId>spring-integration-jmx</artifactId>
<optional>true</optional>
</dependency>
<dependency> <dependency>
<groupId>org.springframework.integration</groupId> <groupId>org.springframework.integration</groupId>
<artifactId>spring-integration-core</artifactId> <artifactId>spring-integration-core</artifactId>
...@@ -363,6 +358,11 @@ ...@@ -363,6 +358,11 @@
<artifactId>spring-data-rest-webmvc</artifactId> <artifactId>spring-data-rest-webmvc</artifactId>
<scope>test</scope> <scope>test</scope>
</dependency> </dependency>
<dependency>
<groupId>org.springframework.integration</groupId>
<artifactId>spring-integration-jmx</artifactId>
<scope>test</scope>
</dependency>
<dependency> <dependency>
<groupId>org.springframework.security</groupId> <groupId>org.springframework.security</groupId>
<artifactId>spring-security-test</artifactId> <artifactId>spring-security-test</artifactId>
......
...@@ -46,13 +46,15 @@ import org.springframework.boot.autoconfigure.condition.ConditionalOnJava; ...@@ -46,13 +46,15 @@ import org.springframework.boot.autoconfigure.condition.ConditionalOnJava;
import org.springframework.boot.autoconfigure.condition.ConditionalOnJava.JavaVersion; import org.springframework.boot.autoconfigure.condition.ConditionalOnJava.JavaVersion;
import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean;
import org.springframework.boot.autoconfigure.condition.ConditionalOnWebApplication; import org.springframework.boot.autoconfigure.condition.ConditionalOnWebApplication;
import org.springframework.boot.autoconfigure.condition.SearchStrategy;
import org.springframework.boot.autoconfigure.integration.IntegrationAutoConfiguration; import org.springframework.boot.autoconfigure.integration.IntegrationAutoConfiguration;
import org.springframework.boot.autoconfigure.jdbc.DataSourceAutoConfiguration; import org.springframework.boot.autoconfigure.jdbc.DataSourceAutoConfiguration;
import org.springframework.boot.autoconfigure.jdbc.metadata.DataSourcePoolMetadataProvider; import org.springframework.boot.autoconfigure.jdbc.metadata.DataSourcePoolMetadataProvider;
import org.springframework.cache.CacheManager; import org.springframework.cache.CacheManager;
import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration; import org.springframework.context.annotation.Configuration;
import org.springframework.integration.monitor.IntegrationMBeanExporter; import org.springframework.integration.config.EnableIntegrationManagement;
import org.springframework.integration.support.management.IntegrationManagementConfigurer;
import org.springframework.lang.UsesJava7; import org.springframework.lang.UsesJava7;
/** /**
...@@ -61,6 +63,7 @@ import org.springframework.lang.UsesJava7; ...@@ -61,6 +63,7 @@ import org.springframework.lang.UsesJava7;
* @author Stephane Nicoll * @author Stephane Nicoll
* @author Phillip Webb * @author Phillip Webb
* @author Johannes Edmeier * @author Johannes Edmeier
* @author Artem Bilan
* @since 1.2.0 * @since 1.2.0
*/ */
@Configuration @Configuration
...@@ -139,18 +142,26 @@ public class PublicMetricsAutoConfiguration { ...@@ -139,18 +142,26 @@ public class PublicMetricsAutoConfiguration {
} }
@Configuration @Configuration
@ConditionalOnClass(IntegrationMBeanExporter.class) @ConditionalOnClass(EnableIntegrationManagement.class)
@ConditionalOnBean(IntegrationMBeanExporter.class)
@ConditionalOnJava(JavaVersion.SEVEN) @ConditionalOnJava(JavaVersion.SEVEN)
@UsesJava7 @UsesJava7
static class IntegrationMetricsConfiguration { static class IntegrationMetricsConfiguration {
@Bean(name = IntegrationManagementConfigurer.MANAGEMENT_CONFIGURER_NAME)
@ConditionalOnMissingBean(value = IntegrationManagementConfigurer.class, name = IntegrationManagementConfigurer.MANAGEMENT_CONFIGURER_NAME, search = SearchStrategy.CURRENT)
public IntegrationManagementConfigurer managementConfigurer() {
IntegrationManagementConfigurer configurer = new IntegrationManagementConfigurer();
configurer.setDefaultCountsEnabled(true);
configurer.setDefaultStatsEnabled(true);
return configurer;
}
@Bean @Bean
@ConditionalOnMissingBean(name = "springIntegrationPublicMetrics") @ConditionalOnMissingBean(name = "springIntegrationPublicMetrics")
public MetricReaderPublicMetrics springIntegrationPublicMetrics( public MetricReaderPublicMetrics springIntegrationPublicMetrics(
IntegrationMBeanExporter exporter) { IntegrationManagementConfigurer managementConfigurer) {
return new MetricReaderPublicMetrics( return new MetricReaderPublicMetrics(
new SpringIntegrationMetricReader(exporter)); new SpringIntegrationMetricReader(managementConfigurer));
} }
} }
......
...@@ -22,24 +22,29 @@ import java.util.List; ...@@ -22,24 +22,29 @@ import java.util.List;
import org.springframework.boot.actuate.metrics.Metric; import org.springframework.boot.actuate.metrics.Metric;
import org.springframework.boot.actuate.metrics.reader.MetricReader; import org.springframework.boot.actuate.metrics.reader.MetricReader;
import org.springframework.integration.monitor.IntegrationMBeanExporter; import org.springframework.integration.support.management.IntegrationManagementConfigurer;
import org.springframework.integration.support.management.MessageChannelMetrics;
import org.springframework.integration.support.management.MessageHandlerMetrics;
import org.springframework.integration.support.management.MessageSourceMetrics;
import org.springframework.integration.support.management.PollableChannelManagement;
import org.springframework.integration.support.management.Statistics; import org.springframework.integration.support.management.Statistics;
import org.springframework.lang.UsesJava7; import org.springframework.lang.UsesJava7;
/** /**
* A {@link MetricReader} for Spring Integration metrics (as provided by * A {@link MetricReader} for Spring Integration metrics (as provided by
* spring-integration-jmx). * {@link IntegrationManagementConfigurer}).
* *
* @author Dave Syer * @author Dave Syer
* @author Artem Bilan
* @since 1.3.0 * @since 1.3.0
*/ */
@UsesJava7 @UsesJava7
public class SpringIntegrationMetricReader implements MetricReader { public class SpringIntegrationMetricReader implements MetricReader {
private final IntegrationMBeanExporter exporter; private final IntegrationManagementConfigurer configurer;
public SpringIntegrationMetricReader(IntegrationMBeanExporter exporter) { public SpringIntegrationMetricReader(IntegrationManagementConfigurer configurer) {
this.exporter = exporter; this.configurer = configurer;
} }
@Override @Override
...@@ -49,51 +54,80 @@ public class SpringIntegrationMetricReader implements MetricReader { ...@@ -49,51 +54,80 @@ public class SpringIntegrationMetricReader implements MetricReader {
@Override @Override
public Iterable<Metric<?>> findAll() { public Iterable<Metric<?>> findAll() {
IntegrationMBeanExporter exporter = this.exporter; List<Metric<?>> result = new ArrayList<Metric<?>>();
List<Metric<?>> metrics = new ArrayList<Metric<?>>(); String[] channelNames = this.configurer.getChannelNames();
for (String name : exporter.getChannelNames()) { String[] handlerNames = this.configurer.getHandlerNames();
String[] sourceNames = this.configurer.getSourceNames();
addChannelMetrics(result, channelNames);
addHandlerMetrics(result, handlerNames);
addSourceMetrics(result, sourceNames);
result.add(new Metric<Integer>("integration.handlerCount", handlerNames.length));
result.add(new Metric<Integer>("integration.channelCount", channelNames.length));
result.add(new Metric<Integer>("integration.sourceCount", sourceNames.length));
return result;
}
private void addChannelMetrics(List<Metric<?>> result, String[] names) {
for (String name : names) {
addChannelMetrics(result, name, this.configurer.getChannelMetrics(name));
}
}
private void addChannelMetrics(List<Metric<?>> result, String name,
MessageChannelMetrics metrics) {
String prefix = "integration.channel." + name; String prefix = "integration.channel." + name;
metrics.addAll(getStatistics(prefix + ".errorRate", result.addAll(getStatistics(prefix + ".errorRate", metrics.getErrorRate()));
exporter.getChannelErrorRate(name))); result.add(new Metric<Long>(prefix + ".sendCount", metrics.getSendCountLong()));
metrics.add(new Metric<Long>(prefix + ".sendCount", result.addAll(getStatistics(prefix + ".sendRate", metrics.getSendRate()));
exporter.getChannelSendCountLong(name))); if (metrics instanceof PollableChannelManagement) {
metrics.addAll(getStatistics(prefix + ".sendRate", result.add(new Metric<Long>(prefix + ".receiveCount",
exporter.getChannelSendRate(name))); ((PollableChannelManagement) metrics).getReceiveCountLong()));
metrics.add(new Metric<Long>(prefix + ".receiveCount",
exporter.getChannelReceiveCountLong(name)));
} }
for (String name : exporter.getHandlerNames()) {
metrics.addAll(getStatistics("integration.handler." + name + ".duration",
exporter.getHandlerDuration(name)));
} }
metrics.add(new Metric<Integer>("integration.activeHandlerCount",
exporter.getActiveHandlerCount())); private void addHandlerMetrics(List<Metric<?>> result, String[] names) {
metrics.add(new Metric<Integer>("integration.handlerCount", for (String name : names) {
exporter.getHandlerCount())); addHandlerMetrics(result, name, this.configurer.getHandlerMetrics(name));
metrics.add(new Metric<Integer>("integration.channelCount", }
exporter.getChannelCount())); }
metrics.add(new Metric<Integer>("integration.queuedMessageCount",
exporter.getQueuedMessageCount())); private void addHandlerMetrics(List<Metric<?>> result, String name,
return metrics; MessageHandlerMetrics metrics) {
String prefix = "integration.handler." + name;
result.addAll(getStatistics(prefix + ".duration", metrics.getDuration()));
long activeCount = metrics.getActiveCountLong();
result.add(new Metric<Long>(prefix + ".activeCount", activeCount));
}
private void addSourceMetrics(List<Metric<?>> result, String[] names) {
for (String name : names) {
addSourceMetrics(result, name, this.configurer.getSourceMetrics(name));
}
}
private void addSourceMetrics(List<Metric<?>> result, String name,
MessageSourceMetrics sourceMetrics) {
String prefix = "integration.source." + name;
result.add(new Metric<Long>(prefix + ".messageCount",
sourceMetrics.getMessageCountLong()));
} }
private Collection<? extends Metric<?>> getStatistics(String name, private Collection<? extends Metric<?>> getStatistics(String name, Statistics stats) {
Statistics statistic) {
List<Metric<?>> metrics = new ArrayList<Metric<?>>(); List<Metric<?>> metrics = new ArrayList<Metric<?>>();
metrics.add(new Metric<Double>(name + ".mean", statistic.getMean())); metrics.add(new Metric<Double>(name + ".mean", stats.getMean()));
metrics.add(new Metric<Double>(name + ".max", statistic.getMax())); metrics.add(new Metric<Double>(name + ".max", stats.getMax()));
metrics.add(new Metric<Double>(name + ".min", statistic.getMin())); metrics.add(new Metric<Double>(name + ".min", stats.getMin()));
metrics.add( metrics.add(new Metric<Double>(name + ".stdev", stats.getStandardDeviation()));
new Metric<Double>(name + ".stdev", statistic.getStandardDeviation())); metrics.add(new Metric<Long>(name + ".count", stats.getCountLong()));
metrics.add(new Metric<Long>(name + ".count", statistic.getCountLong()));
return metrics; return metrics;
} }
@Override @Override
public long count() { public long count() {
int totalChannelCount = this.exporter.getChannelCount() * 11; int totalChannelCount = this.configurer.getChannelNames().length;
int totalHandlerCount = this.exporter.getHandlerCount() * 5; int totalHandlerCount = this.configurer.getHandlerNames().length;
return totalChannelCount + totalHandlerCount + 4; int totalSourceCount = this.configurer.getSourceNames().length;
return totalChannelCount + totalHandlerCount + totalSourceCount;
} }
} }
...@@ -92,7 +92,7 @@ public class PublicMetricsAutoConfigurationTests { ...@@ -92,7 +92,7 @@ public class PublicMetricsAutoConfigurationTests {
public void metricReaderPublicMetrics() throws Exception { public void metricReaderPublicMetrics() throws Exception {
load(); load();
assertThat(this.context.getBeansOfType(MetricReaderPublicMetrics.class)) assertThat(this.context.getBeansOfType(MetricReaderPublicMetrics.class))
.hasSize(1); .hasSize(2);
} }
@Test @Test
......
/*
* Copyright 2012-2016 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.metrics.integration;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.boot.actuate.autoconfigure.PublicMetricsAutoConfiguration;
import org.springframework.boot.actuate.endpoint.MetricReaderPublicMetrics;
import org.springframework.boot.autoconfigure.integration.IntegrationAutoConfiguration;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.Import;
import org.springframework.test.annotation.DirtiesContext;
import org.springframework.test.context.junit4.SpringRunner;
import static org.assertj.core.api.Assertions.assertThat;
/**
* Tests for {@link SpringIntegrationMetricReader}.
*
* @author Artem Bilan
*/
@RunWith(SpringRunner.class)
@SpringBootTest("spring.jmx.enabled=false")
@DirtiesContext
public class SpringIntegrationMetricReaderNoJmxTests {
@Autowired
@Qualifier("springIntegrationPublicMetrics")
private MetricReaderPublicMetrics integrationMetricReader;
@Test
public void test() {
assertThat(this.integrationMetricReader.metrics().size() > 0).isTrue();
}
@Configuration
@Import({ IntegrationAutoConfiguration.class, PublicMetricsAutoConfiguration.class })
protected static class TestConfiguration {
}
}
...@@ -26,7 +26,7 @@ import org.springframework.boot.test.context.SpringBootTest; ...@@ -26,7 +26,7 @@ import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration; import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.Import; import org.springframework.context.annotation.Import;
import org.springframework.integration.monitor.IntegrationMBeanExporter; import org.springframework.integration.support.management.IntegrationManagementConfigurer;
import org.springframework.test.annotation.DirtiesContext; import org.springframework.test.annotation.DirtiesContext;
import org.springframework.test.context.junit4.SpringRunner; import org.springframework.test.context.junit4.SpringRunner;
...@@ -36,6 +36,7 @@ import static org.assertj.core.api.Assertions.assertThat; ...@@ -36,6 +36,7 @@ import static org.assertj.core.api.Assertions.assertThat;
* Tests for {@link SpringIntegrationMetricReader}. * Tests for {@link SpringIntegrationMetricReader}.
* *
* @author Dave Syer * @author Dave Syer
* @author Artem Bilan
*/ */
@RunWith(SpringRunner.class) @RunWith(SpringRunner.class)
@SpringBootTest("spring.jmx.enabled=true") @SpringBootTest("spring.jmx.enabled=true")
...@@ -55,8 +56,9 @@ public class SpringIntegrationMetricReaderTests { ...@@ -55,8 +56,9 @@ public class SpringIntegrationMetricReaderTests {
protected static class TestConfiguration { protected static class TestConfiguration {
@Bean @Bean
public SpringIntegrationMetricReader reader(IntegrationMBeanExporter exporter) { public SpringIntegrationMetricReader reader(
return new SpringIntegrationMetricReader(exporter); IntegrationManagementConfigurer managementConfigurer) {
return new SpringIntegrationMetricReader(managementConfigurer);
} }
} }
......
...@@ -21,8 +21,6 @@ import javax.management.MBeanServer; ...@@ -21,8 +21,6 @@ import javax.management.MBeanServer;
import org.springframework.beans.BeansException; import org.springframework.beans.BeansException;
import org.springframework.beans.factory.BeanFactory; import org.springframework.beans.factory.BeanFactory;
import org.springframework.beans.factory.BeanFactoryAware; import org.springframework.beans.factory.BeanFactoryAware;
import org.springframework.beans.factory.ObjectProvider;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.boot.autoconfigure.AutoConfigureAfter; import org.springframework.boot.autoconfigure.AutoConfigureAfter;
import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass;
import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean;
...@@ -35,6 +33,7 @@ import org.springframework.context.annotation.Bean; ...@@ -35,6 +33,7 @@ import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration; import org.springframework.context.annotation.Configuration;
import org.springframework.core.env.Environment; import org.springframework.core.env.Environment;
import org.springframework.integration.config.EnableIntegration; import org.springframework.integration.config.EnableIntegration;
import org.springframework.integration.config.EnableIntegrationManagement;
import org.springframework.integration.jmx.config.EnableIntegrationMBeanExport; import org.springframework.integration.jmx.config.EnableIntegrationMBeanExport;
import org.springframework.integration.monitor.IntegrationMBeanExporter; import org.springframework.integration.monitor.IntegrationMBeanExporter;
import org.springframework.integration.support.management.IntegrationManagementConfigurer; import org.springframework.integration.support.management.IntegrationManagementConfigurer;
...@@ -67,17 +66,10 @@ public class IntegrationAutoConfiguration { ...@@ -67,17 +66,10 @@ public class IntegrationAutoConfiguration {
protected static class IntegrationJmxConfiguration protected static class IntegrationJmxConfiguration
implements EnvironmentAware, BeanFactoryAware { implements EnvironmentAware, BeanFactoryAware {
private final IntegrationManagementConfigurer configurer;
private BeanFactory beanFactory; private BeanFactory beanFactory;
private RelaxedPropertyResolver propertyResolver; private RelaxedPropertyResolver propertyResolver;
protected IntegrationJmxConfiguration(
@Qualifier(IntegrationManagementConfigurer.MANAGEMENT_CONFIGURER_NAME) ObjectProvider<IntegrationManagementConfigurer> configurerProvider) {
this.configurer = configurerProvider.getIfAvailable();
}
@Override @Override
public void setBeanFactory(BeanFactory beanFactory) throws BeansException { public void setBeanFactory(BeanFactory beanFactory) throws BeansException {
this.beanFactory = beanFactory; this.beanFactory = beanFactory;
...@@ -100,15 +92,22 @@ public class IntegrationAutoConfiguration { ...@@ -100,15 +92,22 @@ public class IntegrationAutoConfiguration {
if (StringUtils.hasLength(server)) { if (StringUtils.hasLength(server)) {
exporter.setServer(this.beanFactory.getBean(server, MBeanServer.class)); exporter.setServer(this.beanFactory.getBean(server, MBeanServer.class));
} }
if (this.configurer != null) { return exporter;
if (this.configurer.getDefaultCountsEnabled() == null) {
this.configurer.setDefaultCountsEnabled(true);
}
if (this.configurer.getDefaultStatsEnabled() == null) {
this.configurer.setDefaultStatsEnabled(true);
} }
} }
return exporter;
@Configuration
@ConditionalOnClass({ EnableIntegrationManagement.class,
EnableIntegrationMBeanExport.class })
@ConditionalOnMissingBean(value = IntegrationManagementConfigurer.class, name = IntegrationManagementConfigurer.MANAGEMENT_CONFIGURER_NAME, search = SearchStrategy.CURRENT)
@ConditionalOnProperty(prefix = "spring.jmx", name = "enabled", havingValue = "true", matchIfMissing = true)
protected static class IntegrationManagementConfiguration {
@Configuration
@EnableIntegrationManagement(defaultCountsEnabled = "true", defaultStatsEnabled = "true")
protected static class EnableIntegrationManagementConfiguration {
} }
} }
......
...@@ -31,6 +31,7 @@ import org.springframework.context.annotation.Bean; ...@@ -31,6 +31,7 @@ import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration; import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.Primary; import org.springframework.context.annotation.Primary;
import org.springframework.integration.support.channel.HeaderChannelRegistry; import org.springframework.integration.support.channel.HeaderChannelRegistry;
import org.springframework.integration.support.management.IntegrationManagementConfigurer;
import org.springframework.jmx.export.MBeanExporter; import org.springframework.jmx.export.MBeanExporter;
import org.springframework.test.context.support.TestPropertySourceUtils; import org.springframework.test.context.support.TestPropertySourceUtils;
...@@ -86,12 +87,17 @@ public class IntegrationAutoConfigurationTests { ...@@ -86,12 +87,17 @@ public class IntegrationAutoConfigurationTests {
MBeanServer mBeanServer = this.context.getBean(MBeanServer.class); MBeanServer mBeanServer = this.context.getBean(MBeanServer.class);
assertDomains(mBeanServer, true, "org.springframework.integration", assertDomains(mBeanServer, true, "org.springframework.integration",
"org.springframework.integration.monitor"); "org.springframework.integration.monitor");
Object bean = this.context
.getBean(IntegrationManagementConfigurer.MANAGEMENT_CONFIGURER_NAME);
assertThat(bean).isNotNull();
} }
@Test @Test
public void disableJmxIntegration() { public void disableJmxIntegration() {
load("spring.jmx.enabled=false"); load("spring.jmx.enabled=false");
assertThat(this.context.getBeansOfType(MBeanServer.class)).hasSize(0); assertThat(this.context.getBeansOfType(MBeanServer.class)).hasSize(0);
assertThat(this.context.getBeansOfType(IntegrationManagementConfigurer.class))
.isEmpty();
} }
@Test @Test
......
...@@ -34,9 +34,5 @@ ...@@ -34,9 +34,5 @@
<groupId>org.springframework.integration</groupId> <groupId>org.springframework.integration</groupId>
<artifactId>spring-integration-java-dsl</artifactId> <artifactId>spring-integration-java-dsl</artifactId>
</dependency> </dependency>
<dependency>
<groupId>org.springframework.integration</groupId>
<artifactId>spring-integration-jmx</artifactId>
</dependency>
</dependencies> </dependencies>
</project> </project>
provides: spring-integration-core,spring-integration-java-dsl,spring-integration-jmx provides: spring-integration-core,spring-integration-java-dsl
\ No newline at end of file
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment