org.apache.aries.util.internal.DefaultWorker Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of aem-sdk-api Show documentation
Show all versions of aem-sdk-api Show documentation
The Adobe Experience Manager SDK
/*
* 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.aries.util.internal;
import java.security.AccessController;
import java.security.PrivilegedAction;
import java.util.IdentityHashMap;
import java.util.Map;
import org.osgi.framework.Bundle;
import org.osgi.framework.BundleContext;
import org.osgi.framework.BundleEvent;
import org.osgi.framework.BundleListener;
import org.osgi.framework.Constants;
import org.osgi.framework.FrameworkEvent;
import org.osgi.framework.FrameworkListener;
import org.osgi.framework.FrameworkUtil;
public class DefaultWorker implements FrameworkUtilWorker, BundleListener, FrameworkListener
{
private Map classLoaders = new IdentityHashMap();
private static final Bundle myFrameworkBundle;
static
{
Bundle bundle = FrameworkUtil.getBundle(DefaultWorker.class);
BundleContext myContext = bundle == null? null: bundle.getBundleContext();
// This may be created during framework shutdown when the bundle context is null.
// So we need to cope and not NPE during construction.
if (myContext != null) {
myFrameworkBundle = myContext.getBundle(0);
} else {
myFrameworkBundle = null;
}
}
public ClassLoader getClassLoader(final Bundle b)
{
ClassLoader cl = get(b);
if (cl != null) return cl;
// so first off try to get the real classloader. We can do this by loading a known class
// such as the bundle activator. There is no guarantee this will work, so we have a back door too.
String activator = (String) b.getHeaders().get(Constants.BUNDLE_ACTIVATOR);
if (activator != null) {
try {
Class> clazz = b.loadClass(activator);
// so we have the class, but it could have been imported, so we make sure the two bundles
// are the same. A reference check should work here because there will be one.
Bundle activatorBundle = FrameworkUtil.getBundle(clazz);
if (activatorBundle == b) {
cl = clazz.getClassLoader();
}
} catch (ClassNotFoundException e) {
}
}
if (cl == null) {
// ok so we haven't found a class loader yet, so we need to create a wapper class loader
cl = AccessController.doPrivileged(new PrivilegedAction() {
public ClassLoader run() {
return new BundleToClassLoaderAdapter(b);
}
});
}
if (cl != null) {
setupListener(b);
cl = put(b, cl);
}
return cl;
}
private void setupListener(Bundle b)
{
// So we need to cope with multiple equinox frameworks, so we can't just listen to our
// BundleContext. Instead we add a listener to Bundle 0 of the framework bundle associated
// with the bundle passed in.
BundleContext ctx = b.getBundleContext().getBundle(0).getBundleContext();
ctx.addBundleListener(this);
ctx.addFrameworkListener(this);
}
private synchronized ClassLoader put(Bundle b, ClassLoader cl)
{
// If the bundle is uninstalled or installed then there is no classloader so we should
// just return null. This is a last second sanity check to avoid memory leaks that could
// occur if a bundle is uninstalled or unresolved while someone is calling getClassLoader
if (b.getState() == Bundle.UNINSTALLED || b.getState() == Bundle.INSTALLED) return null;
ClassLoader previous = classLoaders.put(b, cl);
// OK, so we could cause a replace to occur here, so we want to check to
// see if previous is not null. If it is not null we need to do a replace
// and return the previous classloader. This ensures we have one classloader
// in use for a bundle.
if (previous != null) {
cl = previous;
classLoaders.put(b, cl);
}
return cl;
}
private synchronized ClassLoader get(Bundle b)
{
return classLoaders.get(b);
}
private synchronized void remove(Bundle bundle)
{
classLoaders.remove(bundle);
}
public boolean isValid()
{
return true;
}
public void bundleChanged(BundleEvent event)
{
if (event.getType() == BundleEvent.UNINSTALLED || event.getType() == BundleEvent.UNRESOLVED) {
Bundle b = event.getBundle();
remove(b);
if (b.getBundleId() == 0) {
clearBundles(b);
}
}
}
private void clearBundles(Bundle b) {
// we have been told about the system bundle, so we need to clear up any state for this framework.
BundleContext ctx = b.getBundleContext();
ctx.removeBundleListener(this);
Bundle[] bundles = ctx.getBundles();
for (Bundle bundle : bundles) {
remove(bundle);
}
}
public void frameworkEvent(FrameworkEvent event)
{
if (event.getType() == FrameworkEvent.STOPPED) {
Bundle b = event.getBundle();
if (b == myFrameworkBundle) {
classLoaders.clear();
} else if (b != null) {
clearBundles(b);
}
b.getBundleContext().removeFrameworkListener(this);
}
}
}
© 2015 - 2024 Weber Informatics LLC | Privacy Policy