jdk-24/jdk/test/javax/management/eventService/EventClientExecutorTest.java
Shanliang Jiang cf105cf085 5108776: Add reliable event handling to the JMX API
6218920: API bug - impossible to delete last MBeanServerForwarder on a connector

Reviewed-by: emcmanus
2008-07-31 15:31:13 +02:00

192 lines
7.0 KiB
Java

/*
* Copyright 2008 Sun Microsystems, Inc. All Rights Reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
* CA 95054 USA or visit www.sun.com if you need additional information or
* have any questions.
*/
/*
* @test
* @bug 5108776
* @summary Test that the various Executor parameters in an EventClient do
* what they are supposed to.
* @author Eamonn McManus
*/
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.util.HashSet;
import java.util.Set;
import java.util.concurrent.Executor;
import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.ThreadFactory;
import javax.management.MBeanServer;
import javax.management.MBeanServerFactory;
import javax.management.Notification;
import javax.management.NotificationBroadcasterSupport;
import javax.management.NotificationListener;
import javax.management.ObjectName;
import javax.management.event.EventClient;
import javax.management.event.EventClientDelegate;
import javax.management.event.EventClientDelegateMBean;
import javax.management.event.FetchingEventRelay;
import javax.management.remote.MBeanServerForwarder;
public class EventClientExecutorTest {
private static volatile String failure;
private static final Set testedPrefixes = new HashSet();
public static void main(String[] args) throws Exception {
Executor fetchExecutor = Executors.newSingleThreadExecutor(
new NamedThreadFactory("FETCH"));
Executor listenerExecutor = Executors.newSingleThreadExecutor(
new NamedThreadFactory("LISTENER"));
ScheduledExecutorService leaseScheduler =
Executors.newSingleThreadScheduledExecutor(
new NamedThreadFactory("LEASE"));
MBeanServer mbs = MBeanServerFactory.newMBeanServer();
MBeanServerForwarder mbsf = EventClientDelegate.newForwarder();
mbsf.setMBeanServer(mbs);
mbs = mbsf;
EventClientDelegateMBean ecd = EventClientDelegate.getProxy(mbs);
ecd = (EventClientDelegateMBean) Proxy.newProxyInstance(
EventClientDelegateMBean.class.getClassLoader(),
new Class<?>[] {EventClientDelegateMBean.class},
new DelegateCheckIH(ecd));
ObjectName mbeanName = new ObjectName("d:type=Notifier");
Notifier notifier = new Notifier();
mbs.registerMBean(notifier, mbeanName);
FetchingEventRelay eventRelay = new FetchingEventRelay(
ecd, fetchExecutor);
EventClient ec = new EventClient(
ecd, eventRelay, listenerExecutor, leaseScheduler, 1000L);
NotificationListener checkListener = new NotificationListener() {
public void handleNotification(Notification notification,
Object handback) {
assertThreadName("listener dispatch", "LISTENER");
}
};
ec.addNotificationListener(mbeanName, checkListener, null, null);
mbs.invoke(mbeanName, "send", null, null);
// Now wait until we have seen all three thread types.
long deadline = System.currentTimeMillis() + 5000;
synchronized (testedPrefixes) {
while (testedPrefixes.size() < 3 && failure == null) {
long remain = deadline - System.currentTimeMillis();
if (remain <= 0) {
fail("Timed out waiting for all three thread types to show, " +
"saw only " + testedPrefixes);
break;
}
try {
testedPrefixes.wait(remain);
} catch (InterruptedException e) {
fail("Unexpected InterruptedException");
break;
}
}
}
// We deliberately don't close the EventClient to check that it has
// not created any non-daemon threads.
if (failure != null)
throw new Exception("TEST FAILED: " + failure);
else
System.out.println("TEST PASSED");
}
public static interface NotifierMBean {
public void send();
}
public static class Notifier extends NotificationBroadcasterSupport
implements NotifierMBean {
public void send() {
Notification n = new Notification("a.b.c", this, 0L);
sendNotification(n);
}
}
static void fail(String why) {
System.out.println("FAIL: " + why);
failure = why;
}
static void assertThreadName(String what, String prefix) {
String name = Thread.currentThread().getName();
if (!name.startsWith(prefix)) {
fail("Wrong thread for " + what + ": " + name);
return;
}
synchronized (testedPrefixes) {
if (testedPrefixes.add(prefix))
testedPrefixes.notify();
}
}
private static class DelegateCheckIH implements InvocationHandler {
private final EventClientDelegateMBean ecd;
public DelegateCheckIH(EventClientDelegateMBean ecd) {
this.ecd = ecd;
}
public Object invoke(Object proxy, Method method, Object[] args)
throws Throwable {
String methodName = method.getName();
if (methodName.equals("fetchNotifications"))
assertThreadName("fetchNotifications", "FETCH");
else if (methodName.equals("lease"))
assertThreadName("lease renewal", "LEASE");
try {
return method.invoke(ecd, args);
} catch (InvocationTargetException e) {
throw e.getCause();
}
}
}
private static class NamedThreadFactory implements ThreadFactory {
private final String namePrefix;
private int count;
NamedThreadFactory(String namePrefix) {
this.namePrefix = namePrefix;
}
public synchronized Thread newThread(Runnable r) {
Thread t = new Thread(r);
t.setName(namePrefix + " " + ++count);
t.setDaemon(true);
return t;
}
}
}