blob: 8f12ad215a03893cd9891c9a9d5f647f378e4b3d [file] [log] [blame]
/**
*
* Copyright 2005 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.gbean.kernel.simple;
import java.util.Collections;
import java.util.Iterator;
import java.util.LinkedHashMap;
import java.util.Map;
import java.util.List;
import java.util.LinkedList;
import java.util.Set;
import javax.management.ObjectName;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.gbean.kernel.LifecycleListener;
import org.gbean.kernel.runtime.LifecycleBroadcaster;
/**
* @version $Rev: 71492 $ $Date: 2004-11-14 21:31:50 -0800 (Sun, 14 Nov 2004) $
*/
public class SimpleLifecycleMonitor {
private static final Log log = LogFactory.getLog(SimpleLifecycleMonitor.class);
private final Map boundListeners = new LinkedHashMap();
private final Map listenerPatterns = new LinkedHashMap();
public void start() {
}
/**
* Frees all resources associated with the monitor. No futher events will be processed.
*/
public synchronized void stop() {
boundListeners.clear();
listenerPatterns.clear();
}
/**
* Create a lifecycle broadcaster for the specified bean. This is typically given to the
* instance manager to broadcast lifecycle changes
* @param objectName the name of the object for which a broadcaster should be created
* @return the lifecycle broadcaster
*/
public LifecycleBroadcaster createLifecycleBroadcaster(ObjectName objectName) {
return new SimpleLifecycleBroadcaster(objectName);
}
/**
* Registers a listener to revieve life cycle events for a set of object name patterns.
* @param listener the listener that will receive life cycle events
* @param patterns a set of ObjectName patterns
*/
public synchronized void addLifecycleListener(LifecycleListener listener, Set patterns) {
for (Iterator patternIterator = patterns.iterator(); patternIterator.hasNext();) {
ObjectName pattern = (ObjectName) patternIterator.next();
for (Iterator iterator = boundListeners.entrySet().iterator(); iterator.hasNext();) {
Map.Entry entry = (Map.Entry) iterator.next();
ObjectName source = (ObjectName) entry.getKey();
if (pattern.apply(source)) {
List listeners = (List) entry.getValue();
listeners.add(listener);
}
}
}
listenerPatterns.put(listener, patterns);
}
/**
* Removes the listener from all notifications.
* @param listener the listener to unregister
*/
public synchronized void removeLifecycleListener(LifecycleListener listener) {
for (Iterator iterator = boundListeners.values().iterator(); iterator.hasNext();) {
List listeners = (List) iterator.next();
listeners.remove(listener);
}
listenerPatterns.remove(listener);
}
private synchronized void addSource(ObjectName source) {
if (boundListeners.containsKey(source)) {
// alreayd registered
return;
}
// find all listeners interested in events from this source
List listeners = new LinkedList();
for (Iterator listenerIterator = listenerPatterns.entrySet().iterator(); listenerIterator.hasNext();) {
Map.Entry entry = (Map.Entry) listenerIterator.next();
Set patterns = (Set) entry.getValue();
for (Iterator patternIterator = patterns.iterator(); patternIterator.hasNext();) {
ObjectName pattern = (ObjectName) patternIterator.next();
if (pattern.apply(source)) {
LifecycleListener listener = (LifecycleListener) entry.getKey();
listeners.add(listener);
}
}
}
boundListeners.put(source, listeners);
}
private synchronized void removeSource(ObjectName source) {
boundListeners.remove(source);
}
private synchronized List getTargets(ObjectName source) {
List targets = (List) boundListeners.get(source);
if (targets == null) {
// no one is interested in this event
return Collections.EMPTY_LIST;
} else {
return new LinkedList(targets);
}
}
private void fireLoadedEvent(ObjectName objectName) {
List targets = getTargets(objectName);
for (Iterator iterator = targets.iterator(); iterator.hasNext();) {
LifecycleListener listener = (LifecycleListener) iterator.next();
try {
listener.loaded(objectName);
} catch (Throwable e) {
log.warn("Exception occured while notifying listener", e);
}
}
}
private void fireStartingEvent(ObjectName source) {
List targets = getTargets(source);
for (Iterator iterator = targets.iterator(); iterator.hasNext();) {
LifecycleListener listener = (LifecycleListener) iterator.next();
try {
listener.starting(source);
} catch (Throwable e) {
log.warn("Exception occured while notifying listener", e);
}
}
}
private void fireRunningEvent(ObjectName source) {
List targets = getTargets(source);
for (Iterator iterator = targets.iterator(); iterator.hasNext();) {
LifecycleListener listener = (LifecycleListener) iterator.next();
try {
listener.running(source);
} catch (Throwable e) {
log.warn("Exception occured while notifying listener", e);
}
}
}
private void fireStoppingEvent(ObjectName source) {
List targets = getTargets(source);
Collections.reverse(targets);
for (Iterator iterator = targets.iterator(); iterator.hasNext();) {
LifecycleListener listener = (LifecycleListener) iterator.next();
try {
listener.stopping(source);
} catch (Throwable e) {
log.warn("Exception occured while notifying listener", e);
}
}
}
private void fireStoppedEvent(ObjectName source) {
List targets = getTargets(source);
Collections.reverse(targets);
for (Iterator iterator = targets.iterator(); iterator.hasNext();) {
LifecycleListener listener = (LifecycleListener) iterator.next();
try {
listener.stopped(source);
} catch (Throwable e) {
log.warn("Exception occured while notifying listener", e);
}
}
}
private void fireUnloadedEvent(ObjectName source) {
List targets = getTargets(source);
Collections.reverse(targets);
for (Iterator iterator = targets.iterator(); iterator.hasNext();) {
LifecycleListener listener = (LifecycleListener) iterator.next();
try {
listener.unloaded(source);
} catch (Throwable e) {
log.warn("Exception occured while notifying listener", e);
}
}
}
private class SimpleLifecycleBroadcaster implements LifecycleBroadcaster {
private final ObjectName objectName;
public SimpleLifecycleBroadcaster(ObjectName objectName) {
this.objectName = objectName;
}
public void fireLoadedEvent() {
addSource(objectName);
SimpleLifecycleMonitor.this.fireLoadedEvent(objectName);
}
public void fireStartingEvent() {
SimpleLifecycleMonitor.this.fireStartingEvent(objectName);
}
public void fireRunningEvent() {
SimpleLifecycleMonitor.this.fireRunningEvent(objectName);
}
public void fireStoppingEvent() {
SimpleLifecycleMonitor.this.fireStoppingEvent(objectName);
}
public void fireStoppedEvent() {
SimpleLifecycleMonitor.this.fireStoppedEvent(objectName);
}
public void fireUnloadedEvent() {
SimpleLifecycleMonitor.this.fireUnloadedEvent(objectName);
removeSource(objectName);
}
}
}