Exclude @ManagedResources from Endpoint MBeans
If an Endpoint is already @ManagedResource then it doesn't need an additional (probably wrong) MBEan registration based on the invoke() method.
This commit is contained in:
parent
c0305ecbbb
commit
123ffd736c
|
|
@ -38,9 +38,11 @@ import org.springframework.context.ApplicationContext;
|
||||||
import org.springframework.context.ApplicationContextAware;
|
import org.springframework.context.ApplicationContextAware;
|
||||||
import org.springframework.context.ApplicationListener;
|
import org.springframework.context.ApplicationListener;
|
||||||
import org.springframework.context.SmartLifecycle;
|
import org.springframework.context.SmartLifecycle;
|
||||||
|
import org.springframework.core.annotation.AnnotationUtils;
|
||||||
import org.springframework.jmx.export.MBeanExportException;
|
import org.springframework.jmx.export.MBeanExportException;
|
||||||
import org.springframework.jmx.export.MBeanExporter;
|
import org.springframework.jmx.export.MBeanExporter;
|
||||||
import org.springframework.jmx.export.annotation.AnnotationJmxAttributeSource;
|
import org.springframework.jmx.export.annotation.AnnotationJmxAttributeSource;
|
||||||
|
import org.springframework.jmx.export.annotation.ManagedResource;
|
||||||
import org.springframework.jmx.export.assembler.MetadataMBeanInfoAssembler;
|
import org.springframework.jmx.export.assembler.MetadataMBeanInfoAssembler;
|
||||||
import org.springframework.jmx.export.naming.MetadataNamingStrategy;
|
import org.springframework.jmx.export.naming.MetadataNamingStrategy;
|
||||||
import org.springframework.jmx.export.naming.SelfNaming;
|
import org.springframework.jmx.export.naming.SelfNaming;
|
||||||
|
|
@ -144,6 +146,18 @@ public class EndpointMBeanExporter extends MBeanExporter implements SmartLifecyc
|
||||||
}
|
}
|
||||||
|
|
||||||
protected void registerEndpoint(String beanName, Endpoint<?> endpoint) {
|
protected void registerEndpoint(String beanName, Endpoint<?> endpoint) {
|
||||||
|
@SuppressWarnings("rawtypes")
|
||||||
|
Class<? extends Endpoint> type = endpoint.getClass();
|
||||||
|
if (AnnotationUtils.findAnnotation(type, ManagedResource.class) != null) {
|
||||||
|
// Already managed
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (type.isMemberClass()
|
||||||
|
&& AnnotationUtils.findAnnotation(type.getEnclosingClass(),
|
||||||
|
ManagedResource.class) != null) {
|
||||||
|
// Nested class with @ManagedResource in parent
|
||||||
|
return;
|
||||||
|
}
|
||||||
try {
|
try {
|
||||||
registerBeanNameOrInstance(getEndpointMBean(beanName, endpoint), beanName);
|
registerBeanNameOrInstance(getEndpointMBean(beanName, endpoint), beanName);
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -19,12 +19,14 @@ package org.springframework.boot.actuate.endpoint.jmx;
|
||||||
import org.springframework.boot.actuate.endpoint.Endpoint;
|
import org.springframework.boot.actuate.endpoint.Endpoint;
|
||||||
import org.springframework.boot.actuate.endpoint.ShutdownEndpoint;
|
import org.springframework.boot.actuate.endpoint.ShutdownEndpoint;
|
||||||
import org.springframework.jmx.export.annotation.ManagedOperation;
|
import org.springframework.jmx.export.annotation.ManagedOperation;
|
||||||
|
import org.springframework.jmx.export.annotation.ManagedResource;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Special endpoint wrapper for {@link ShutdownEndpoint}.
|
* Special endpoint wrapper for {@link ShutdownEndpoint}.
|
||||||
*
|
*
|
||||||
* @author Christian Dupuis
|
* @author Christian Dupuis
|
||||||
*/
|
*/
|
||||||
|
@ManagedResource
|
||||||
public class ShutdownEndpointMBean extends EndpointMBean {
|
public class ShutdownEndpointMBean extends EndpointMBean {
|
||||||
|
|
||||||
public ShutdownEndpointMBean(String beanName, Endpoint<?> endpoint) {
|
public ShutdownEndpointMBean(String beanName, Endpoint<?> endpoint) {
|
||||||
|
|
|
||||||
|
|
@ -25,18 +25,26 @@ import javax.management.ReflectionException;
|
||||||
import org.junit.After;
|
import org.junit.After;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
import org.springframework.beans.factory.NoSuchBeanDefinitionException;
|
import org.springframework.beans.factory.NoSuchBeanDefinitionException;
|
||||||
|
import org.springframework.boot.actuate.endpoint.AbstractEndpoint;
|
||||||
|
import org.springframework.boot.actuate.endpoint.Endpoint;
|
||||||
import org.springframework.boot.actuate.endpoint.jmx.EndpointMBeanExporter;
|
import org.springframework.boot.actuate.endpoint.jmx.EndpointMBeanExporter;
|
||||||
|
import org.springframework.boot.autoconfigure.PropertyPlaceholderAutoConfiguration;
|
||||||
import org.springframework.boot.autoconfigure.jmx.JmxAutoConfiguration;
|
import org.springframework.boot.autoconfigure.jmx.JmxAutoConfiguration;
|
||||||
import org.springframework.context.ApplicationContext;
|
import org.springframework.context.ApplicationContext;
|
||||||
import org.springframework.context.annotation.AnnotationConfigApplicationContext;
|
import org.springframework.context.annotation.AnnotationConfigApplicationContext;
|
||||||
|
import org.springframework.context.annotation.Bean;
|
||||||
import org.springframework.context.annotation.Configuration;
|
import org.springframework.context.annotation.Configuration;
|
||||||
import org.springframework.context.annotation.EnableMBeanExport;
|
import org.springframework.context.annotation.EnableMBeanExport;
|
||||||
import org.springframework.jmx.export.MBeanExporter;
|
import org.springframework.jmx.export.MBeanExporter;
|
||||||
|
import org.springframework.jmx.export.annotation.ManagedResource;
|
||||||
import org.springframework.jmx.support.ObjectNameManager;
|
import org.springframework.jmx.support.ObjectNameManager;
|
||||||
import org.springframework.mock.env.MockEnvironment;
|
import org.springframework.mock.env.MockEnvironment;
|
||||||
|
import org.springframework.stereotype.Component;
|
||||||
import org.springframework.util.ObjectUtils;
|
import org.springframework.util.ObjectUtils;
|
||||||
|
|
||||||
|
import static org.junit.Assert.assertFalse;
|
||||||
import static org.junit.Assert.assertNotNull;
|
import static org.junit.Assert.assertNotNull;
|
||||||
|
import static org.junit.Assert.assertTrue;
|
||||||
import static org.junit.Assert.fail;
|
import static org.junit.Assert.fail;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
@ -55,13 +63,51 @@ public class EndpointMBeanExportAutoConfigurationTests {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testEndpointMBeanExporterIsInstalled() {
|
public void testEndpointMBeanExporterIsInstalled() throws Exception {
|
||||||
this.context = new AnnotationConfigApplicationContext();
|
this.context = new AnnotationConfigApplicationContext();
|
||||||
this.context.register(TestConfiguration.class, JmxAutoConfiguration.class,
|
this.context.register(TestConfiguration.class, JmxAutoConfiguration.class,
|
||||||
EndpointAutoConfiguration.class,
|
EndpointAutoConfiguration.class,
|
||||||
EndpointMBeanExportAutoConfiguration.class);
|
EndpointMBeanExportAutoConfiguration.class,
|
||||||
|
PropertyPlaceholderAutoConfiguration.class);
|
||||||
this.context.refresh();
|
this.context.refresh();
|
||||||
assertNotNull(this.context.getBean(EndpointMBeanExporter.class));
|
assertNotNull(this.context.getBean(EndpointMBeanExporter.class));
|
||||||
|
MBeanExporter mbeanExporter = this.context.getBean(EndpointMBeanExporter.class);
|
||||||
|
|
||||||
|
assertFalse(mbeanExporter.getServer()
|
||||||
|
.queryNames(getObjectName("*", "*,*", this.context), null).isEmpty());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testEndpointMBeanExporterIsNotInstalledIfManagedResource()
|
||||||
|
throws Exception {
|
||||||
|
this.context = new AnnotationConfigApplicationContext();
|
||||||
|
this.context.register(TestConfiguration.class, JmxAutoConfiguration.class,
|
||||||
|
ManagedEndpoint.class, EndpointMBeanExportAutoConfiguration.class,
|
||||||
|
PropertyPlaceholderAutoConfiguration.class);
|
||||||
|
this.context.refresh();
|
||||||
|
assertNotNull(this.context.getBean(EndpointMBeanExporter.class));
|
||||||
|
|
||||||
|
MBeanExporter mbeanExporter = this.context.getBean(EndpointMBeanExporter.class);
|
||||||
|
|
||||||
|
assertTrue(mbeanExporter.getServer()
|
||||||
|
.queryNames(getObjectName("*", "*,*", this.context), null).isEmpty());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testEndpointMBeanExporterIsNotInstalledIfNestedInManagedResource()
|
||||||
|
throws Exception {
|
||||||
|
this.context = new AnnotationConfigApplicationContext();
|
||||||
|
this.context.register(TestConfiguration.class, JmxAutoConfiguration.class,
|
||||||
|
NestedInManagedEndpoint.class,
|
||||||
|
EndpointMBeanExportAutoConfiguration.class,
|
||||||
|
PropertyPlaceholderAutoConfiguration.class);
|
||||||
|
this.context.refresh();
|
||||||
|
assertNotNull(this.context.getBean(EndpointMBeanExporter.class));
|
||||||
|
|
||||||
|
MBeanExporter mbeanExporter = this.context.getBean(EndpointMBeanExporter.class);
|
||||||
|
|
||||||
|
assertTrue(mbeanExporter.getServer()
|
||||||
|
.queryNames(getObjectName("*", "*,*", this.context), null).isEmpty());
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test(expected = NoSuchBeanDefinitionException.class)
|
@Test(expected = NoSuchBeanDefinitionException.class)
|
||||||
|
|
@ -125,21 +171,23 @@ public class EndpointMBeanExportAutoConfigurationTests {
|
||||||
|
|
||||||
private ObjectName getObjectName(String domain, String beanKey,
|
private ObjectName getObjectName(String domain, String beanKey,
|
||||||
ApplicationContext applicationContext) throws MalformedObjectNameException {
|
ApplicationContext applicationContext) throws MalformedObjectNameException {
|
||||||
|
String name = "%s:type=Endpoint,name=%s";
|
||||||
if (applicationContext.getParent() != null) {
|
if (applicationContext.getParent() != null) {
|
||||||
return ObjectNameManager
|
name = name + ",context=%s";
|
||||||
.getInstance(String.format(
|
}
|
||||||
"%s:type=Endpoint,name=%s,context=%s,identity=%s", domain,
|
if (applicationContext.getEnvironment().getProperty("endpoints.jmx.unique_names",
|
||||||
beanKey,
|
Boolean.class, false)) {
|
||||||
ObjectUtils.getIdentityHexString(applicationContext),
|
name = name
|
||||||
ObjectUtils.getIdentityHexString(applicationContext
|
+ ",identity="
|
||||||
.getBean(beanKey))));
|
+ ObjectUtils.getIdentityHexString(applicationContext
|
||||||
|
.getBean(beanKey));
|
||||||
|
}
|
||||||
|
if (applicationContext.getParent() != null) {
|
||||||
|
return ObjectNameManager.getInstance(String.format(name, domain, beanKey,
|
||||||
|
ObjectUtils.getIdentityHexString(applicationContext)));
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
return ObjectNameManager
|
return ObjectNameManager.getInstance(String.format(name, domain, beanKey));
|
||||||
.getInstance(String.format("%s:type=Endpoint,name=%s,identity=%s",
|
|
||||||
domain, beanKey, ObjectUtils
|
|
||||||
.getIdentityHexString(applicationContext
|
|
||||||
.getBean(beanKey))));
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -148,4 +196,43 @@ public class EndpointMBeanExportAutoConfigurationTests {
|
||||||
public static class TestConfiguration {
|
public static class TestConfiguration {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Component
|
||||||
|
@ManagedResource
|
||||||
|
protected static class ManagedEndpoint extends AbstractEndpoint<Boolean> {
|
||||||
|
|
||||||
|
public ManagedEndpoint() {
|
||||||
|
super("managed", true, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Boolean invoke() {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Configuration
|
||||||
|
@ManagedResource
|
||||||
|
protected static class NestedInManagedEndpoint {
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
public Endpoint<Boolean> nested() {
|
||||||
|
return new Nested();
|
||||||
|
}
|
||||||
|
|
||||||
|
class Nested extends AbstractEndpoint<Boolean> {
|
||||||
|
|
||||||
|
public Nested() {
|
||||||
|
super("managed", true, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Boolean invoke() {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
||||||
Loading…
Reference in New Issue