8008982: Adjust JMX for underlying interface changes

Reviewed-by: mchung, dholmes, dfuchs, skoivu
This commit is contained in:
Jaroslav Bachorik 2013-03-28 09:39:26 +01:00
parent 7938121238
commit 9f8bb22b73
3 changed files with 71 additions and 39 deletions

View File

@ -228,6 +228,11 @@ public class Introspector {
MXBeanIntrospector.getInstance().getAnalyzer(interfaceClass); MXBeanIntrospector.getInstance().getAnalyzer(interfaceClass);
} }
public static void testComplianceMBeanInterface(Class<?> interfaceClass)
throws NotCompliantMBeanException{
StandardMBeanIntrospector.getInstance().getAnalyzer(interfaceClass);
}
/** /**
* Basic method for testing if a given class is a JMX compliant * Basic method for testing if a given class is a JMX compliant
* Standard MBean. This method is only called by the legacy code * Standard MBean. This method is only called by the legacy code

View File

@ -27,7 +27,9 @@ package javax.management;
import com.sun.jmx.mbeanserver.Introspector; import com.sun.jmx.mbeanserver.Introspector;
import java.lang.reflect.InvocationHandler; import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Modifier;
import java.lang.reflect.Proxy; import java.lang.reflect.Proxy;
import sun.reflect.misc.ReflectUtil;
/** /**
* Static methods from the JMX API. There are no instances of this class. * Static methods from the JMX API. There are no instances of this class.
@ -203,11 +205,7 @@ public class JMX {
ObjectName objectName, ObjectName objectName,
Class<T> interfaceClass, Class<T> interfaceClass,
boolean notificationEmitter) { boolean notificationEmitter) {
return MBeanServerInvocationHandler.newProxyInstance( return createProxy(connection, objectName, interfaceClass, notificationEmitter, false);
connection,
objectName,
interfaceClass,
notificationEmitter);
} }
/** /**
@ -345,26 +343,7 @@ public class JMX {
ObjectName objectName, ObjectName objectName,
Class<T> interfaceClass, Class<T> interfaceClass,
boolean notificationEmitter) { boolean notificationEmitter) {
// Check interface for MXBean compliance return createProxy(connection, objectName, interfaceClass, notificationEmitter, true);
//
try {
Introspector.testComplianceMXBeanInterface(interfaceClass);
} catch (NotCompliantMBeanException e) {
throw new IllegalArgumentException(e);
}
InvocationHandler handler = new MBeanServerInvocationHandler(
connection, objectName, true);
final Class<?>[] interfaces;
if (notificationEmitter) {
interfaces =
new Class<?>[] {interfaceClass, NotificationEmitter.class};
} else
interfaces = new Class<?>[] {interfaceClass};
Object proxy = Proxy.newProxyInstance(
interfaceClass.getClassLoader(),
interfaces,
handler);
return interfaceClass.cast(proxy);
} }
/** /**
@ -392,4 +371,65 @@ public class JMX {
// exactly the string "MXBean" since that would mean there // exactly the string "MXBean" since that would mean there
// was no package name, which is pretty unlikely in practice. // was no package name, which is pretty unlikely in practice.
} }
/**
* Centralised M(X)Bean proxy creation code
* @param connection {@linkplain MBeanServerConnection} to use
* @param objectName M(X)Bean object name
* @param interfaceClass M(X)Bean interface class
* @param notificationEmitter Is a notification emitter?
* @param isMXBean Is an MXBean?
* @return Returns an M(X)Bean proxy generated for the provided interface class
* @throws SecurityException
* @throws IllegalArgumentException
*/
private static <T> T createProxy(MBeanServerConnection connection,
ObjectName objectName,
Class<T> interfaceClass,
boolean notificationEmitter,
boolean isMXBean) {
if (System.getSecurityManager() != null) {
checkProxyInterface(interfaceClass);
}
try {
if (isMXBean) {
// Check interface for MXBean compliance
Introspector.testComplianceMXBeanInterface(interfaceClass);
} else {
// Check interface for MBean compliance
Introspector.testComplianceMBeanInterface(interfaceClass);
}
} catch (NotCompliantMBeanException e) {
throw new IllegalArgumentException(e);
}
InvocationHandler handler = new MBeanServerInvocationHandler(
connection, objectName, isMXBean);
final Class<?>[] interfaces;
if (notificationEmitter) {
interfaces =
new Class<?>[] {interfaceClass, NotificationEmitter.class};
} else
interfaces = new Class<?>[] {interfaceClass};
Object proxy = Proxy.newProxyInstance(
interfaceClass.getClassLoader(),
interfaces,
handler);
return interfaceClass.cast(proxy);
}
/**
* Checks for the M(X)Bean proxy interface being public and not restricted
* @param interfaceClass MBean proxy interface
* @throws SecurityException when the proxy interface comes from a restricted
* package or is not public
*/
private static void checkProxyInterface(Class<?> interfaceClass) {
if (!Modifier.isPublic(interfaceClass.getModifiers())) {
throw new SecurityException("mbean proxy interface non-public");
}
ReflectUtil.checkPackageAccess(interfaceClass);
}
} }

View File

@ -231,20 +231,7 @@ public class MBeanServerInvocationHandler implements InvocationHandler {
ObjectName objectName, ObjectName objectName,
Class<T> interfaceClass, Class<T> interfaceClass,
boolean notificationBroadcaster) { boolean notificationBroadcaster) {
final InvocationHandler handler = return JMX.newMBeanProxy(connection, objectName, interfaceClass, notificationBroadcaster);
new MBeanServerInvocationHandler(connection, objectName);
final Class<?>[] interfaces;
if (notificationBroadcaster) {
interfaces =
new Class<?>[] {interfaceClass, NotificationEmitter.class};
} else
interfaces = new Class<?>[] {interfaceClass};
Object proxy =
Proxy.newProxyInstance(interfaceClass.getClassLoader(),
interfaces,
handler);
return interfaceClass.cast(proxy);
} }
public Object invoke(Object proxy, Method method, Object[] args) public Object invoke(Object proxy, Method method, Object[] args)