blob: 077f32c4204034f04582e5ba363b76916cb6926b [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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.apache.felix.eventadmin.impl.handler;
import java.util.Collection;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.CopyOnWriteArrayList;
import org.osgi.framework.BundleContext;
import org.osgi.framework.ServiceReference;
import org.osgi.service.event.Event;
import org.osgi.service.event.EventHandler;
import org.osgi.util.tracker.ServiceTracker;
/**
* The event handler tracker keeps track of all event handler services.
*
* @author <a href="mailto:dev@felix.apache.org">Felix Project Team</a>
*/
public class EventHandlerTracker extends ServiceTracker<EventHandler, EventHandlerProxy> {
/** The proxies in this list match all events. */
private final List<EventHandlerProxy> matchingAllEvents;
/** This is a map for exact topic matches. The key is the topic,
* the value is a list of proxies.
*/
private final Map<String, List<EventHandlerProxy>> matchingTopic;
/** This is a map for wildcard topics. The key is the prefix of the topic,
* the value is a list of proxies
*/
private final Map<String, List<EventHandlerProxy>> matchingPrefixTopic;
/** The context for the proxies. */
private HandlerContext handlerContext;
public EventHandlerTracker(final BundleContext context) {
super(context, EventHandler.class.getName(), null);
// we start with empty collections
this.matchingAllEvents = new CopyOnWriteArrayList<EventHandlerProxy>();
this.matchingTopic = new ConcurrentHashMap<String, List<EventHandlerProxy>>();
this.matchingPrefixTopic = new ConcurrentHashMap<String, List<EventHandlerProxy>>();
}
/**
* Update the timeout configuration.
* @param ignoreTimeout
*/
public void update(final String[] ignoreTimeout, final boolean requireTopic) {
final Matcher[] ignoreTimeoutMatcher;
if ( ignoreTimeout == null || ignoreTimeout.length == 0 )
{
ignoreTimeoutMatcher = null;
}
else
{
ignoreTimeoutMatcher = new Matcher[ignoreTimeout.length];
for(int i=0;i<ignoreTimeout.length;i++)
{
String value = ignoreTimeout[i];
if ( value != null )
{
value = value.trim();
}
if ( value != null && value.length() > 0 )
{
if ( value.endsWith(".") )
{
ignoreTimeoutMatcher[i] = new PackageMatcher(value.substring(0, value.length() - 1));
}
else if ( value.endsWith("*") )
{
ignoreTimeoutMatcher[i] = new SubPackageMatcher(value.substring(0, value.length() - 1));
}
else
{
ignoreTimeoutMatcher[i] = new ClassMatcher(value);
}
}
}
}
this.handlerContext = new HandlerContext(this.context, ignoreTimeoutMatcher, requireTopic);
}
/**
* @see org.osgi.util.tracker.ServiceTracker#addingService(org.osgi.framework.ServiceReference)
*/
@Override
public EventHandlerProxy addingService(final ServiceReference<EventHandler> reference) {
final EventHandlerProxy proxy = new EventHandlerProxy(this.handlerContext, reference);
if ( proxy.update() ) {
this.put(proxy);
}
return proxy;
}
/**
* @see org.osgi.util.tracker.ServiceTracker#modifiedService(org.osgi.framework.ServiceReference, java.lang.Object)
*/
@Override
public void modifiedService(final ServiceReference<EventHandler> reference, final EventHandlerProxy proxy) {
this.remove(proxy);
if ( proxy.update() ) {
this.put(proxy);
}
}
/**
* @see org.osgi.util.tracker.ServiceTracker#removedService(org.osgi.framework.ServiceReference, java.lang.Object)
*/
@Override
public void removedService(final ServiceReference<EventHandler> reference, final EventHandlerProxy proxy) {
this.remove(proxy);
proxy.dispose();
}
private void updateMap(final Map<String, List<EventHandlerProxy>> proxyListMap, final String key, final EventHandlerProxy proxy, final boolean add) {
List<EventHandlerProxy> proxies = proxyListMap.get(key);
if (proxies == null)
{
if ( !add )
{
return;
}
proxies = new CopyOnWriteArrayList<EventHandlerProxy>();
proxyListMap.put(key, proxies);
}
if ( add )
{
proxies.add(proxy);
}
else
{
proxies.remove(proxy);
if ( proxies.size() == 0 )
{
proxyListMap.remove(key);
}
}
}
/**
* Check the topics of the event handler and put it into the
* corresponding collections.
*/
private synchronized void put(final EventHandlerProxy proxy) {
final String[] topics = proxy.getTopics();
if ( topics == null )
{
this.matchingAllEvents.add(proxy);
}
else
{
for(int i = 0; i < topics.length; i++) {
final String topic = topics[i];
if ( topic.endsWith("/*") )
{
// prefix topic: we remove the /*
final String prefix = topic.substring(0, topic.length() - 2);
this.updateMap(this.matchingPrefixTopic, prefix, proxy, true);
}
else
{
// exact match
this.updateMap(this.matchingTopic, topic, proxy, true);
}
}
}
}
/**
* Check the topics of the event handler and remove it from the
* corresponding collections.
*/
private synchronized void remove(final EventHandlerProxy proxy) {
final String[] topics = proxy.getTopics();
if ( topics == null )
{
this.matchingAllEvents.remove(proxy);
} else {
for(int i = 0; i < topics.length; i++) {
final String topic = topics[i];
if ( topic.endsWith("/*") )
{
// prefix topic: we remove the /*
final String prefix = topic.substring(0, topic.length() - 2);
this.updateMap(this.matchingPrefixTopic, prefix, proxy, false);
}
else
{
// exact match
this.updateMap(this.matchingTopic, topic, proxy, false);
}
}
}
}
/**
* Get all handlers for this event
*
* @param event The event topic
* @return All handlers for the event
*/
public Collection<EventHandlerProxy> getHandlers(final Event event) {
final String topic = event.getTopic();
final Set<EventHandlerProxy> handlers = new HashSet<EventHandlerProxy>();
// Add all handlers matching everything
this.checkHandlerAndAdd(handlers, this.matchingAllEvents, event);
// Now check for prefix matches
if ( !this.matchingPrefixTopic.isEmpty() )
{
int pos = topic.lastIndexOf('/');
while (pos != -1)
{
final String prefix = topic.substring(0, pos);
this.checkHandlerAndAdd(handlers, this.matchingPrefixTopic.get(prefix), event);
pos = prefix.lastIndexOf('/');
}
}
// Add the handlers for matching topic names
this.checkHandlerAndAdd(handlers, this.matchingTopic.get(topic), event);
return handlers;
}
/**
* Checks each handler from the proxy list if it can deliver the event
* If the event can be delivered, the proxy is added to the handlers.
*/
private void checkHandlerAndAdd( final Set<EventHandlerProxy> handlers,
final List<EventHandlerProxy> proxies,
final Event event)
{
if ( proxies != null )
{
for(final EventHandlerProxy p : proxies)
{
if ( p.canDeliver(event) )
{
handlers.add(p);
}
}
}
}
static Matcher[] createMatchers(final String[] config)
{
final Matcher[] matchers;
if ( config == null || config.length == 0 )
{
matchers = null;
}
else
{
matchers = new Matcher[config.length];
for(int i=0;i<config.length;i++)
{
String value = config[i];
if ( value != null )
{
value = value.trim();
}
if ( value != null && value.length() > 0 )
{
if ( value.endsWith(".") )
{
matchers[i] = new PackageMatcher(value.substring(0, value.length() - 1));
}
else if ( value.endsWith("*") )
{
if ( value.equals("*") )
{
return new Matcher[] {new MatcherAll()};
}
matchers[i] = new SubPackageMatcher(value.substring(0, value.length() - 1));
}
else
{
matchers[i] = new ClassMatcher(value);
}
}
}
}
return matchers;
}
/**
* The matcher interface for checking if timeout handling
* is disabled for the handler.
* Matching is based on the class name of the event handler.
*/
static interface Matcher
{
boolean match(String className);
}
/** Match all. */
private static final class MatcherAll implements Matcher
{
@Override
public boolean match(final String className)
{
return true;
}
}
/** Match a package. */
private static final class PackageMatcher implements Matcher
{
private final String m_packageName;
public PackageMatcher(final String name)
{
m_packageName = name;
}
@Override
public boolean match(final String className)
{
final int pos = className.lastIndexOf('.');
return pos > -1 && className.substring(0, pos).equals(m_packageName);
}
}
/** Match a package or sub package. */
private static final class SubPackageMatcher implements Matcher
{
private final String m_packageName;
public SubPackageMatcher(final String name)
{
m_packageName = name + '.';
}
@Override
public boolean match(final String className)
{
final int pos = className.lastIndexOf('.');
return pos > -1 && className.substring(0, pos + 1).startsWith(m_packageName);
}
}
/** Match a class name. */
private static final class ClassMatcher implements Matcher
{
private final String m_className;
public ClassMatcher(final String name)
{
m_className = name;
}
@Override
public boolean match(final String className)
{
return m_className.equals(className);
}
}
/**
* The context object passed to the proxies.
*/
static final class HandlerContext
{
/** The bundle context. */
public final BundleContext bundleContext;
/** The matchers for ignore timeout handling. */
public final Matcher[] ignoreTimeoutMatcher;
/** Is a topic required. */
public final boolean requireTopic;
public HandlerContext(final BundleContext bundleContext,
final Matcher[] ignoreTimeoutMatcher,
final boolean requireTopic)
{
this.bundleContext = bundleContext;
this.ignoreTimeoutMatcher = ignoreTimeoutMatcher;
this.requireTopic = requireTopic;
}
}
}