blob: 42f864dd3d553789ed58bcfa5f1d56359ae6b8e7 [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.servicemix.specs.locator;
import java.io.BufferedReader;
import java.io.InputStreamReader;
import java.net.URL;
import java.util.Enumeration;
import java.util.HashMap;
import java.util.Map;
import java.util.concurrent.Callable;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;
import org.osgi.framework.Bundle;
import org.osgi.framework.BundleActivator;
import org.osgi.framework.BundleContext;
import org.osgi.framework.BundleEvent;
import org.osgi.framework.SynchronousBundleListener;
public class Activator implements BundleActivator, SynchronousBundleListener {
private static boolean debug = false;
private ConcurrentMap<Long, Map<String, Callable<Class>>> factories = new ConcurrentHashMap<Long, Map<String, Callable<Class>>>();
private BundleContext bundleContext;
static {
try {
String prop = System.getProperty("org.apache.servicemix.specs.debug");
debug = prop != null && !"false".equals(prop);
} catch (Throwable t) { }
}
/**
* <p>Output debugging messages.</p>
*
* @param msg <code>String</code> to print to <code>stderr</code>.
*/
protected void debugPrintln(String msg) {
if (debug) {
System.err.println("Spec(" + bundleContext.getBundle().getBundleId() + "): " + msg);
}
}
public synchronized void start(BundleContext bundleContext) throws Exception {
this.bundleContext = bundleContext;
debugPrintln("activating");
debugPrintln("adding bundle listener");
bundleContext.addBundleListener(this);
debugPrintln("checking existing bundles");
for (Bundle bundle : bundleContext.getBundles()) {
if (bundle.getState() == Bundle.RESOLVED || bundle.getState() == Bundle.STARTING ||
bundle.getState() == Bundle.ACTIVE || bundle.getState() == Bundle.STOPPING) {
register(bundle);
}
}
debugPrintln("activated");
}
public synchronized void stop(BundleContext bundleContext) throws Exception {
debugPrintln("deactivating");
bundleContext.removeBundleListener(this);
while (!factories.isEmpty()) {
unregister(factories.keySet().iterator().next());
}
debugPrintln("deactivated");
this.bundleContext = null;
}
public void bundleChanged(BundleEvent event) {
synchronized (this) {
if (bundleContext == null) {
return;
}
}
if (event.getType() == BundleEvent.RESOLVED) {
register(event.getBundle());
} else if (event.getType() == BundleEvent.UNRESOLVED || event.getType() == BundleEvent.UNINSTALLED) {
unregister(event.getBundle().getBundleId());
}
}
protected void register(final Bundle bundle) {
debugPrintln("checking bundle " + bundle.getBundleId());
Map<String, Callable<Class>> map = factories.get(bundle.getBundleId());
Enumeration e = bundle.findEntries("META-INF/services/", "*", false);
if (e != null) {
while (e.hasMoreElements()) {
final URL u = (URL) e.nextElement();
final String url = u.toString();
if (url.endsWith("/")) {
continue;
}
final String factoryId = url.substring(url.lastIndexOf("/") + 1);
if (map == null) {
map = new HashMap<String, Callable<Class>>();
factories.put(bundle.getBundleId(), map);
}
map.put(factoryId, new BundleFactoryLoader(factoryId, u, bundle));
}
}
if (map != null) {
for (Map.Entry<String, Callable<Class>> entry : map.entrySet()) {
debugPrintln("registering service for key " + entry.getKey() + " with value " + entry.getValue());
OsgiLocator.register(entry.getKey(), entry.getValue());
}
}
}
protected void unregister(long bundleId) {
Map<String, Callable<Class>> map = factories.remove(bundleId);
if (map != null) {
for (Map.Entry<String, Callable<Class>> entry : map.entrySet()) {
debugPrintln("unregistering service for key " + entry.getKey() + " with value " + entry.getValue());
OsgiLocator.unregister(entry.getKey(), entry.getValue());
}
}
}
private class BundleFactoryLoader implements Callable<Class> {
private final String factoryId;
private final URL u;
private final Bundle bundle;
private volatile Class<?> clazz;
public BundleFactoryLoader(String factoryId, URL u, Bundle bundle) {
this.factoryId = factoryId;
this.u = u;
this.bundle = bundle;
}
public Class call() throws Exception {
try {
debugPrintln("loading factory for key: " + factoryId);
if (clazz == null){
synchronized (this) {
if (clazz == null){
debugPrintln("creating factory for key: " + factoryId);
BufferedReader br = new BufferedReader(new InputStreamReader(u.openStream(), "UTF-8"));
try {
String factoryClassName = br.readLine();
while (factoryClassName != null) {
factoryClassName = factoryClassName.trim();
if (factoryClassName.charAt(0) != '#') {
debugPrintln("factory implementation: " + factoryClassName);
clazz = bundle.loadClass(factoryClassName);
return clazz;
}
factoryClassName = br.readLine();
}
} finally {
br.close();
}
}
}
}
return clazz;
} catch (Exception e) {
debugPrintln("exception caught while creating factory: " + e);
throw e;
} catch (Error e) {
debugPrintln("error caught while creating factory: " + e);
throw e;
}
}
@Override
public String toString() {
return u.toString();
}
@Override
public int hashCode() {
return u.hashCode();
}
@Override
public boolean equals(Object obj) {
if (obj instanceof BundleFactoryLoader) {
return u.equals(((BundleFactoryLoader) obj).u);
} else {
return false;
}
}
}
}