6736611: [Evt Srv] EventSubscriber.unsubscribe removes other listeners
Reviewed-by: emcmanus
This commit is contained in:
parent
367e60a438
commit
ebdad848b8
@ -149,10 +149,10 @@ public class EventSubscriber implements EventConsumer {
|
|||||||
if (listener == null)
|
if (listener == null)
|
||||||
throw new IllegalArgumentException("Null listener");
|
throw new IllegalArgumentException("Null listener");
|
||||||
|
|
||||||
final ListenerInfo li = new ListenerInfo(listener, filter, handback);
|
final MyListenerInfo li = new MyListenerInfo(listener, filter, handback);
|
||||||
List<ListenerInfo> list;
|
List<MyListenerInfo> list;
|
||||||
|
|
||||||
Map<ObjectName, List<ListenerInfo>> map;
|
Map<ObjectName, List<MyListenerInfo>> map;
|
||||||
Set<ObjectName> names;
|
Set<ObjectName> names;
|
||||||
if (name.isPattern()) {
|
if (name.isPattern()) {
|
||||||
map = patternSubscriptionMap;
|
map = patternSubscriptionMap;
|
||||||
@ -165,7 +165,7 @@ public class EventSubscriber implements EventConsumer {
|
|||||||
synchronized (map) {
|
synchronized (map) {
|
||||||
list = map.get(name);
|
list = map.get(name);
|
||||||
if (list == null) {
|
if (list == null) {
|
||||||
list = new ArrayList<ListenerInfo>();
|
list = new ArrayList<MyListenerInfo>();
|
||||||
map.put(name, list);
|
map.put(name, list);
|
||||||
}
|
}
|
||||||
list.add(li);
|
list.add(li);
|
||||||
@ -186,7 +186,6 @@ public class EventSubscriber implements EventConsumer {
|
|||||||
public void unsubscribe(ObjectName name,
|
public void unsubscribe(ObjectName name,
|
||||||
NotificationListener listener)
|
NotificationListener listener)
|
||||||
throws ListenerNotFoundException, IOException {
|
throws ListenerNotFoundException, IOException {
|
||||||
|
|
||||||
if (logger.traceOn())
|
if (logger.traceOn())
|
||||||
logger.trace("unsubscribe", "" + name);
|
logger.trace("unsubscribe", "" + name);
|
||||||
|
|
||||||
@ -196,7 +195,7 @@ public class EventSubscriber implements EventConsumer {
|
|||||||
if (listener == null)
|
if (listener == null)
|
||||||
throw new ListenerNotFoundException();
|
throw new ListenerNotFoundException();
|
||||||
|
|
||||||
Map<ObjectName, List<ListenerInfo>> map;
|
Map<ObjectName, List<MyListenerInfo>> map;
|
||||||
Set<ObjectName> names;
|
Set<ObjectName> names;
|
||||||
|
|
||||||
if (name.isPattern()) {
|
if (name.isPattern()) {
|
||||||
@ -207,25 +206,42 @@ public class EventSubscriber implements EventConsumer {
|
|||||||
names = Collections.singleton(name);
|
names = Collections.singleton(name);
|
||||||
}
|
}
|
||||||
|
|
||||||
final ListenerInfo li = new ListenerInfo(listener, null, null);
|
List<MyListenerInfo> toRemove = new ArrayList<MyListenerInfo>();
|
||||||
List<ListenerInfo> list;
|
|
||||||
synchronized (map) {
|
synchronized (map) {
|
||||||
list = map.get(name);
|
List<MyListenerInfo> list = map.get(name);
|
||||||
if (list == null || !list.remove(li))
|
if (list == null) {
|
||||||
throw new ListenerNotFoundException();
|
throw new ListenerNotFoundException();
|
||||||
|
}
|
||||||
|
|
||||||
|
for (MyListenerInfo info : list) {
|
||||||
|
if (info.listener == listener) {
|
||||||
|
toRemove.add(info);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (toRemove.isEmpty()) {
|
||||||
|
throw new ListenerNotFoundException();
|
||||||
|
}
|
||||||
|
|
||||||
|
for (MyListenerInfo info : toRemove) {
|
||||||
|
list.remove(info);
|
||||||
|
}
|
||||||
|
|
||||||
if (list.isEmpty())
|
if (list.isEmpty())
|
||||||
map.remove(name);
|
map.remove(name);
|
||||||
}
|
}
|
||||||
|
|
||||||
for (ObjectName mbeanName : names) {
|
for (ObjectName mbeanName : names) {
|
||||||
|
for (MyListenerInfo i : toRemove) {
|
||||||
try {
|
try {
|
||||||
mbeanServer.removeNotificationListener(mbeanName, li.listener);
|
mbeanServer.removeNotificationListener(mbeanName,
|
||||||
|
i.listener, i.filter, i.handback);
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
logger.fine("unsubscribe", "removeNotificationListener", e);
|
logger.fine("unsubscribe", "removeNotificationListener", e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// ---------------------------------
|
// ---------------------------------
|
||||||
// private stuff
|
// private stuff
|
||||||
@ -256,12 +272,12 @@ public class EventSubscriber implements EventConsumer {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
final List<ListenerInfo> listeners = new ArrayList<ListenerInfo>();
|
final List<MyListenerInfo> listeners = new ArrayList<MyListenerInfo>();
|
||||||
|
|
||||||
// If there are subscribers for the exact name that has just arrived
|
// If there are subscribers for the exact name that has just arrived
|
||||||
// then add their listeners to the list.
|
// then add their listeners to the list.
|
||||||
synchronized (exactSubscriptionMap) {
|
synchronized (exactSubscriptionMap) {
|
||||||
List<ListenerInfo> exactListeners = exactSubscriptionMap.get(name);
|
List<MyListenerInfo> exactListeners = exactSubscriptionMap.get(name);
|
||||||
if (exactListeners != null)
|
if (exactListeners != null)
|
||||||
listeners.addAll(exactListeners);
|
listeners.addAll(exactListeners);
|
||||||
}
|
}
|
||||||
@ -277,7 +293,7 @@ public class EventSubscriber implements EventConsumer {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Add all the listeners just found to the new MBean.
|
// Add all the listeners just found to the new MBean.
|
||||||
for (ListenerInfo li : listeners) {
|
for (MyListenerInfo li : listeners) {
|
||||||
try {
|
try {
|
||||||
mbeanServer.addNotificationListener(
|
mbeanServer.addNotificationListener(
|
||||||
name,
|
name,
|
||||||
@ -292,12 +308,12 @@ public class EventSubscriber implements EventConsumer {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
private static class ListenerInfo {
|
private static class MyListenerInfo {
|
||||||
public final NotificationListener listener;
|
public final NotificationListener listener;
|
||||||
public final NotificationFilter filter;
|
public final NotificationFilter filter;
|
||||||
public final Object handback;
|
public final Object handback;
|
||||||
|
|
||||||
public ListenerInfo(NotificationListener listener,
|
public MyListenerInfo(NotificationListener listener,
|
||||||
NotificationFilter filter,
|
NotificationFilter filter,
|
||||||
Object handback) {
|
Object handback) {
|
||||||
|
|
||||||
@ -308,26 +324,6 @@ public class EventSubscriber implements EventConsumer {
|
|||||||
this.filter = filter;
|
this.filter = filter;
|
||||||
this.handback = handback;
|
this.handback = handback;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Two ListenerInfo instances are equal if they have the same
|
|
||||||
* NotificationListener. This means that we can use List.remove
|
|
||||||
* to implement the two-argument removeNotificationListener.
|
|
||||||
*/
|
|
||||||
@Override
|
|
||||||
public boolean equals(Object o) {
|
|
||||||
if (o == this)
|
|
||||||
return true;
|
|
||||||
|
|
||||||
if (!(o instanceof ListenerInfo))
|
|
||||||
return false;
|
|
||||||
|
|
||||||
return listener.equals(((ListenerInfo)o).listener);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public int hashCode() {
|
|
||||||
return listener.hashCode();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// ---------------------------------
|
// ---------------------------------
|
||||||
@ -338,10 +334,10 @@ public class EventSubscriber implements EventConsumer {
|
|||||||
// ---------------------------------
|
// ---------------------------------
|
||||||
private final MBeanServer mbeanServer;
|
private final MBeanServer mbeanServer;
|
||||||
|
|
||||||
private final Map<ObjectName, List<ListenerInfo>> exactSubscriptionMap =
|
private final Map<ObjectName, List<MyListenerInfo>> exactSubscriptionMap =
|
||||||
new HashMap<ObjectName, List<ListenerInfo>>();
|
new HashMap<ObjectName, List<MyListenerInfo>>();
|
||||||
private final Map<ObjectName, List<ListenerInfo>> patternSubscriptionMap =
|
private final Map<ObjectName, List<MyListenerInfo>> patternSubscriptionMap =
|
||||||
new HashMap<ObjectName, List<ListenerInfo>>();
|
new HashMap<ObjectName, List<MyListenerInfo>>();
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
125
jdk/test/javax/management/eventService/SubUnsubTest.java
Normal file
125
jdk/test/javax/management/eventService/SubUnsubTest.java
Normal file
@ -0,0 +1,125 @@
|
|||||||
|
/*
|
||||||
|
* 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 SubUnsubTest
|
||||||
|
* @bug 6736611
|
||||||
|
* @summary Test not to remove other listeners when calling unsubscribe
|
||||||
|
* @author Shanliang JIANG
|
||||||
|
* @run clean SubUnsubTest
|
||||||
|
* @run build SubUnsubTest
|
||||||
|
* @run main SubUnsubTest
|
||||||
|
*/
|
||||||
|
|
||||||
|
import java.lang.management.ManagementFactory;
|
||||||
|
import javax.management.MBeanServer;
|
||||||
|
import javax.management.Notification;
|
||||||
|
import javax.management.NotificationFilter;
|
||||||
|
import javax.management.NotificationBroadcasterSupport;
|
||||||
|
import javax.management.NotificationListener;
|
||||||
|
import javax.management.ObjectName;
|
||||||
|
import javax.management.event.EventSubscriber;
|
||||||
|
import javax.management.event.EventClient;
|
||||||
|
public class SubUnsubTest {
|
||||||
|
private static class CountListener implements NotificationListener {
|
||||||
|
volatile int count;
|
||||||
|
|
||||||
|
public void handleNotification(Notification n, Object h) {
|
||||||
|
count++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static interface SenderMBean {}
|
||||||
|
|
||||||
|
public static class Sender extends NotificationBroadcasterSupport
|
||||||
|
implements SenderMBean {
|
||||||
|
void send() {
|
||||||
|
Notification n = new Notification("type", this, 1L);
|
||||||
|
sendNotification(n);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void main(String[] args) throws Exception {
|
||||||
|
System.out.println("Testing EventSubscriber-unsubscribe method.");
|
||||||
|
|
||||||
|
MBeanServer mbs = ManagementFactory.getPlatformMBeanServer();
|
||||||
|
ObjectName name1 = new ObjectName("d:type=Sender,id=1");
|
||||||
|
ObjectName name2 = new ObjectName("d:type=Sender,id=2");
|
||||||
|
ObjectName pattern = new ObjectName("d:type=Sender,*");
|
||||||
|
Sender sender1 = new Sender();
|
||||||
|
Sender sender2 = new Sender();
|
||||||
|
mbs.registerMBean(sender1, name1);
|
||||||
|
mbs.registerMBean(sender2, name2);
|
||||||
|
|
||||||
|
EventSubscriber sub = EventSubscriber.getEventSubscriber(mbs);
|
||||||
|
|
||||||
|
System.out.println("Single subscribe covering both MBeans");
|
||||||
|
CountListener listener = new CountListener();
|
||||||
|
|
||||||
|
System.out.println("Subscribing and adding listeners ...");
|
||||||
|
sub.subscribe(pattern, listener, null, null);
|
||||||
|
sub.subscribe(name2, listener, null, null);
|
||||||
|
mbs.addNotificationListener(name2, listener, null, null);
|
||||||
|
|
||||||
|
sender1.send();
|
||||||
|
sender2.send();
|
||||||
|
if (listener.count != 4) {
|
||||||
|
throw new RuntimeException("Do not receive all notifications: "+
|
||||||
|
"Expect 4, got "+listener.count);
|
||||||
|
}
|
||||||
|
|
||||||
|
System.out.println("Unsubscribe the listener with the pattern.");
|
||||||
|
sub.unsubscribe(pattern, listener);
|
||||||
|
listener.count = 0;
|
||||||
|
sender1.send();
|
||||||
|
sender2.send();
|
||||||
|
if (listener.count != 2) {
|
||||||
|
throw new RuntimeException("The method unsubscribe removes wrong listeners.");
|
||||||
|
}
|
||||||
|
|
||||||
|
System.out.println("Unsubscribe the listener with the ObjectName.");
|
||||||
|
sub.unsubscribe(name2, listener);
|
||||||
|
listener.count = 0;
|
||||||
|
sender1.send();
|
||||||
|
sender2.send();
|
||||||
|
if (listener.count != 1) {
|
||||||
|
throw new RuntimeException("The method unsubscribe removes wrong listeners.");
|
||||||
|
}
|
||||||
|
|
||||||
|
System.out.println("Subscribe twice to same MBean with same listener " +
|
||||||
|
"but different handback.");
|
||||||
|
sub.subscribe(name1, listener, null, new Object());
|
||||||
|
sub.subscribe(name1, listener, null, new Object());
|
||||||
|
listener.count = 0;
|
||||||
|
|
||||||
|
sub.unsubscribe(name1, listener);
|
||||||
|
sender1.send();
|
||||||
|
if (listener.count > 0) {
|
||||||
|
throw new RuntimeException("EventSubscriber: the method unsubscribe" +
|
||||||
|
" does not remove a listener which was subscribed 2 times.");
|
||||||
|
}
|
||||||
|
|
||||||
|
System.out.println("Bye bye!");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user