blob: d673d2312bd28fc18b0c3fa5fcd9f6f6fc1f8b5d [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.http.base.internal.whiteboard.tracker;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import org.apache.felix.http.base.internal.runtime.WhiteboardServiceInfo;
import org.apache.felix.http.base.internal.whiteboard.WhiteboardManager;
import org.osgi.framework.BundleContext;
import org.osgi.framework.Constants;
import org.osgi.framework.InvalidSyntaxException;
import org.osgi.framework.ServiceReference;
import org.osgi.util.tracker.ServiceTracker;
/**
* Service tracker for all whiteboard services except servlet context helper.
* This tracker does not get/unget the service objects itself, but just forwards the service reference
* by creating an info data object. Each sub class creates a different
* data object.
*/
public abstract class WhiteboardServiceTracker<T> extends ServiceTracker<T, ServiceReference<T>>
{
/** Map containing all info objects reported from the trackers. */
private final Map<Long, WhiteboardServiceInfo<T>> allInfos = new ConcurrentHashMap<Long, WhiteboardServiceInfo<T>>();
private static org.osgi.framework.Filter createFilter(final BundleContext btx, final String expr)
{
try
{
return btx.createFilter(expr);
}
catch ( final InvalidSyntaxException ise)
{
// we can safely ignore it as the filter is a constant
}
return null; // we never get here - and if we get an NPE which is fine
}
/** The manager is called for each added/removed reference. */
private final WhiteboardManager contextManager;
/**
* Create a new tracker
* @param contextManager The context manager
* @param bundleContext The bundle context.
* @param filterExpr The filter expression for the services to track
*/
public WhiteboardServiceTracker(final WhiteboardManager contextManager,
final BundleContext bundleContext, final String filterExpr)
{
super(bundleContext, createFilter(bundleContext, filterExpr), null);
this.contextManager = contextManager;
}
@Override
public void close() {
super.close();
this.allInfos.clear();
}
@Override
public final ServiceReference<T> addingService(final ServiceReference<T> ref)
{
this.added(ref);
return ref;
}
@Override
public final void modifiedService(final ServiceReference<T> ref, final ServiceReference<T> service)
{
this.modified(ref);
}
@Override
public final void removedService(final ServiceReference<T> ref, final ServiceReference<T> service)
{
this.removed(ref);
}
private void modified(final ServiceReference<T> ref)
{
removed(ref);
added(ref);
}
private void added(final ServiceReference<T> ref)
{
final WhiteboardServiceInfo<T> info = this.getServiceInfo(ref);
if ( this.contextManager.addWhiteboardService(info) )
{
this.allInfos.put((Long)ref.getProperty(Constants.SERVICE_ID), info);
}
}
private void removed(final ServiceReference<T> ref)
{
final WhiteboardServiceInfo<T> info = this.allInfos.remove(ref.getProperty(Constants.SERVICE_ID));
if ( info != null )
{
this.contextManager.removeWhiteboardService(info);
}
}
/**
* Implemented by sub classes to create the correct whiteboard service info object.
* @param ref The service reference
* @return A whiteboard service info
*/
protected abstract WhiteboardServiceInfo<T> getServiceInfo(final ServiceReference<T> ref);
}