Browse Source

MBeanExporter detects FactoryBean-exported resources independent from declaration order

git-svn-id: https://src.springframework.org/svn/spring-framework/trunk@2643 50f2f4bb-b051-0410-bef5-90022cba6387
pull/1/head
Juergen Hoeller 16 years ago
parent
commit
197c2d7ca2
  1. 67
      org.springframework.context/src/main/java/org/springframework/jmx/export/MBeanExporter.java
  2. 5
      org.springframework.context/src/test/java/org/springframework/jmx/export/annotation/AnnotationLazyInitMBeanTests.java
  3. 45
      org.springframework.context/src/test/java/org/springframework/jmx/export/annotation/AnnotationTestBeanFactory.java
  4. 25
      org.springframework.context/src/test/java/org/springframework/jmx/export/annotation/FactoryCreatedAnnotationTestBean.java
  5. 12
      org.springframework.context/src/test/java/org/springframework/jmx/export/annotation/lazyAssembling.xml
  6. 4
      org.springframework.context/src/test/java/org/springframework/jmx/export/annotation/lazyNaming.xml

67
org.springframework.context/src/main/java/org/springframework/jmx/export/MBeanExporter.java

@ -21,6 +21,7 @@ import java.util.Arrays;
import java.util.HashMap; import java.util.HashMap;
import java.util.HashSet; import java.util.HashSet;
import java.util.LinkedHashMap; import java.util.LinkedHashMap;
import java.util.LinkedHashSet;
import java.util.List; import java.util.List;
import java.util.Map; import java.util.Map;
import java.util.Set; import java.util.Set;
@ -43,11 +44,11 @@ import org.springframework.aop.target.LazyInitTargetSource;
import org.springframework.beans.factory.BeanClassLoaderAware; import org.springframework.beans.factory.BeanClassLoaderAware;
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.CannotLoadBeanClassException;
import org.springframework.beans.factory.DisposableBean; import org.springframework.beans.factory.DisposableBean;
import org.springframework.beans.factory.InitializingBean; import org.springframework.beans.factory.InitializingBean;
import org.springframework.beans.factory.ListableBeanFactory; import org.springframework.beans.factory.ListableBeanFactory;
import org.springframework.beans.factory.NoSuchBeanDefinitionException; import org.springframework.beans.factory.config.ConfigurableBeanFactory;
import org.springframework.beans.factory.config.BeanDefinition;
import org.springframework.beans.factory.config.ConfigurableListableBeanFactory; import org.springframework.beans.factory.config.ConfigurableListableBeanFactory;
import org.springframework.core.Constants; import org.springframework.core.Constants;
import org.springframework.jmx.export.assembler.AutodetectCapableMBeanInfoAssembler; import org.springframework.jmx.export.assembler.AutodetectCapableMBeanInfoAssembler;
@ -142,7 +143,7 @@ public class MBeanExporter extends MBeanRegistrationSupport
/** The autodetect mode to use for this MBeanExporter */ /** The autodetect mode to use for this MBeanExporter */
private Integer autodetectMode; private Integer autodetectMode;
/** Whether to eagerly init candidate beans when autodetecting MBeans */ /** Whether to eagerly initialize candidate beans when autodetecting MBeans */
private boolean allowEagerInit = false; private boolean allowEagerInit = false;
/** Indicates whether Spring should modify generated ObjectNames */ /** Indicates whether Spring should modify generated ObjectNames */
@ -381,14 +382,13 @@ public class MBeanExporter extends MBeanRegistrationSupport
* <code>ListableBeanFactory</code> is required). * <code>ListableBeanFactory</code> is required).
* @see #setBeans * @see #setBeans
* @see #setAutodetect * @see #setAutodetect
* @throws IllegalArgumentException if the supplied <code>beanFactory</code> is not a {@link ListableBeanFactory}.
*/ */
public void setBeanFactory(BeanFactory beanFactory) { public void setBeanFactory(BeanFactory beanFactory) {
if (beanFactory instanceof ListableBeanFactory) { if (beanFactory instanceof ListableBeanFactory) {
this.beanFactory = (ListableBeanFactory) beanFactory; this.beanFactory = (ListableBeanFactory) beanFactory;
} }
else { else {
logger.info("MBeanExporter not running in a ListableBeanFactory: Autodetection of MBeans not available."); logger.info("MBeanExporter not running in a ListableBeanFactory: autodetection of MBeans not available.");
} }
} }
@ -537,17 +537,8 @@ public class MBeanExporter extends MBeanRegistrationSupport
* @see org.springframework.beans.factory.config.BeanDefinition#isLazyInit * @see org.springframework.beans.factory.config.BeanDefinition#isLazyInit
*/ */
protected boolean isBeanDefinitionLazyInit(ListableBeanFactory beanFactory, String beanName) { protected boolean isBeanDefinitionLazyInit(ListableBeanFactory beanFactory, String beanName) {
if (!(beanFactory instanceof ConfigurableListableBeanFactory)) { return (beanFactory instanceof ConfigurableListableBeanFactory && beanFactory.containsBeanDefinition(beanName) &&
return false; ((ConfigurableListableBeanFactory) beanFactory).getBeanDefinition(beanName).isLazyInit());
}
try {
BeanDefinition bd = ((ConfigurableListableBeanFactory) beanFactory).getBeanDefinition(beanName);
return bd.isLazyInit();
}
catch (NoSuchBeanDefinitionException ex) {
// Probably a directly registered singleton.
return false;
}
} }
/** /**
@ -594,7 +585,7 @@ public class MBeanExporter extends MBeanRegistrationSupport
// Plain bean instance -> register it directly. // Plain bean instance -> register it directly.
if (this.beanFactory != null) { if (this.beanFactory != null) {
Map<String, ?> beansOfSameType = Map<String, ?> beansOfSameType =
this.beanFactory.getBeansOfType(mapValue.getClass(), false, false); this.beanFactory.getBeansOfType(mapValue.getClass(), false, this.allowEagerInit);
for (Map.Entry<String, ?> entry : beansOfSameType.entrySet()) { for (Map.Entry<String, ?> entry : beansOfSameType.entrySet()) {
if (entry.getValue() == mapValue) { if (entry.getValue() == mapValue) {
String beanName = entry.getKey(); String beanName = entry.getKey();
@ -875,27 +866,39 @@ public class MBeanExporter extends MBeanRegistrationSupport
* whether to include a bean or not * whether to include a bean or not
*/ */
private void autodetect(AutodetectCallback callback) { private void autodetect(AutodetectCallback callback) {
String[] beanNames = this.beanFactory.getBeanNamesForType(Object.class, true, this.allowEagerInit); Set<String> beanNames = new LinkedHashSet<String>(this.beanFactory.getBeanDefinitionCount());
beanNames.addAll(Arrays.asList(this.beanFactory.getBeanDefinitionNames()));
if (this.beanFactory instanceof ConfigurableBeanFactory) {
beanNames.addAll(Arrays.asList(((ConfigurableBeanFactory) this.beanFactory).getSingletonNames()));
}
for (String beanName : beanNames) { for (String beanName : beanNames) {
if (!isExcluded(beanName)) { if (!isExcluded(beanName)) {
Class beanClass = this.beanFactory.getType(beanName); try {
if (beanClass != null && callback.include(beanClass, beanName)) { Class beanClass = this.beanFactory.getType(beanName);
boolean lazyInit = isBeanDefinitionLazyInit(this.beanFactory, beanName); if (beanClass != null && callback.include(beanClass, beanName)) {
Object beanInstance = (!lazyInit ? this.beanFactory.getBean(beanName) : null); boolean lazyInit = isBeanDefinitionLazyInit(this.beanFactory, beanName);
if (!this.beans.containsValue(beanName) && (beanInstance == null || Object beanInstance = (!lazyInit ? this.beanFactory.getBean(beanName) : null);
!CollectionUtils.containsInstance(this.beans.values(), beanInstance))) { if (!this.beans.containsValue(beanName) && (beanInstance == null ||
// Not already registered for JMX exposure. !CollectionUtils.containsInstance(this.beans.values(), beanInstance))) {
this.beans.put(beanName, (beanInstance != null ? beanInstance : beanName)); // Not already registered for JMX exposure.
if (logger.isInfoEnabled()) { this.beans.put(beanName, (beanInstance != null ? beanInstance : beanName));
logger.info("Bean with name '" + beanName + "' has been autodetected for JMX exposure"); if (logger.isInfoEnabled()) {
logger.info("Bean with name '" + beanName + "' has been autodetected for JMX exposure");
}
} }
} else {
else { if (logger.isDebugEnabled()) {
if (logger.isDebugEnabled()) { logger.debug("Bean with name '" + beanName + "' is already registered for JMX exposure");
logger.debug("Bean with name '" + beanName + "' is already registered for JMX exposure"); }
} }
} }
} }
catch (CannotLoadBeanClassException ex) {
if (this.allowEagerInit) {
throw ex;
}
// otherwise ignore beans where the class is not resolvable
}
} }
} }
} }

5
org.springframework.context/src/test/java/org/springframework/jmx/export/annotation/AnnotationLazyInitMBeanTests.java

@ -57,6 +57,11 @@ public class AnnotationLazyInitMBeanTests extends TestCase {
String name = (String) server.getAttribute(oname, "Name"); String name = (String) server.getAttribute(oname, "Name");
assertEquals("Invalid name returned", "TEST", name); assertEquals("Invalid name returned", "TEST", name);
oname = ObjectNameManager.getInstance("bean:name=testBean5");
assertNotNull(server.getObjectInstance(oname));
name = (String) server.getAttribute(oname, "Name");
assertEquals("Invalid name returned", "FACTORY", name);
oname = ObjectNameManager.getInstance("spring:mbean=true"); oname = ObjectNameManager.getInstance("spring:mbean=true");
assertNotNull(server.getObjectInstance(oname)); assertNotNull(server.getObjectInstance(oname));
name = (String) server.getAttribute(oname, "Name"); name = (String) server.getAttribute(oname, "Name");

45
org.springframework.context/src/test/java/org/springframework/jmx/export/annotation/AnnotationTestBeanFactory.java

@ -0,0 +1,45 @@
/*
* Copyright 2002-2009 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.jmx.export.annotation;
import org.springframework.beans.factory.FactoryBean;
import org.springframework.jmx.IJmxTestBean;
/**
* @author Juergen Hoeller
*/
public class AnnotationTestBeanFactory implements FactoryBean<IJmxTestBean> {
private final FactoryCreatedAnnotationTestBean instance = new FactoryCreatedAnnotationTestBean();
public AnnotationTestBeanFactory() {
this.instance.setName("FACTORY");
}
public IJmxTestBean getObject() throws Exception {
return this.instance;
}
public Class<? extends IJmxTestBean> getObjectType() {
return FactoryCreatedAnnotationTestBean.class;
}
public boolean isSingleton() {
return true;
}
}

25
org.springframework.context/src/test/java/org/springframework/jmx/export/annotation/FactoryCreatedAnnotationTestBean.java

@ -0,0 +1,25 @@
/*
* Copyright 2002-2009 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.jmx.export.annotation;
/**
* @author Juergen Hoeller
*/
@ManagedResource(objectName = "bean:name=testBean5")
public class FactoryCreatedAnnotationTestBean extends AnnotationTestBean {
}

12
org.springframework.context/src/test/java/org/springframework/jmx/export/annotation/lazyAssembling.xml

@ -5,21 +5,25 @@
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.5.xsd xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.5.xsd
http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-2.5.xsd"> http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-2.5.xsd">
<context:mbean-export server="server" registration="replaceExisting"/> <context:mbean-export server="server" registration="replaceExisting"/>
<context:mbean-export server="server" registration="replaceExisting"/> <context:mbean-export server="server" registration="replaceExisting"/>
<bean id="server" class="org.springframework.jmx.support.MBeanServerFactoryBean"/> <bean id="server" class="org.springframework.jmx.support.MBeanServerFactoryBean"/>
<bean name="bean:name=testBean4" class="org.springframework.jmx.export.annotation.AnnotationTestBean" lazy-init="true"> <bean name="bean:name=testBean4" class="org.springframework.jmx.export.annotation.AnnotationTestBean" lazy-init="true">
<property name="name" value="TEST"/> <property name="name" value="TEST"/>
<property name="age" value="100"/> <property name="age" value="100"/>
</bean> </bean>
<bean name="bean:name=testBean5" class="org.springframework.jmx.export.annotation.AnnotationTestBeanFactory"/>
<bean name="spring:mbean=true" class="org.springframework.jmx.export.TestDynamicMBean" lazy-init="true"/> <bean name="spring:mbean=true" class="org.springframework.jmx.export.TestDynamicMBean" lazy-init="true"/>
<bean name="spring:mbean=another" class="org.springframework.jmx.export.MBeanExporterTests$Person" lazy-init="true"> <bean name="spring:mbean=another" class="org.springframework.jmx.export.MBeanExporterTests$Person" lazy-init="true">
<property name="name" value="Juergen Hoeller"/> <property name="name" value="Juergen Hoeller"/>
</bean> </bean>
<bean id="notLoadable" class="does.not.exist" lazy-init="true"/>
</beans> </beans>

4
org.springframework.context/src/test/java/org/springframework/jmx/export/annotation/lazyNaming.xml

@ -5,9 +5,9 @@
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.5.xsd xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.5.xsd
http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-2.5.xsd"> http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-2.5.xsd">
<context:mbean-export server="server"/> <context:mbean-export server="server"/>
<bean id="server" class="org.springframework.jmx.support.MBeanServerFactoryBean"/> <bean id="server" class="org.springframework.jmx.support.MBeanServerFactoryBean"/>
<bean id="testBean" class="org.springframework.jmx.export.annotation.AnnotationTestBean" lazy-init="true"> <bean id="testBean" class="org.springframework.jmx.export.annotation.AnnotationTestBean" lazy-init="true">
<property name="name" value="TEST"/> <property name="name" value="TEST"/>

Loading…
Cancel
Save