OsgiRegistry.java
/*
* Copyright (c) 2012, 2017 Oracle and/or its affiliates. All rights reserved.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License v. 2.0, which is available at
* http://www.eclipse.org/legal/epl-2.0.
*
* This Source Code may also be made available under the following Secondary
* Licenses when the conditions for such availability set forth in the
* Eclipse Public License v. 2.0 are satisfied: GNU General Public License,
* version 2 with the GNU Classpath Exception, which is available at
* https://www.gnu.org/software/classpath/license.html.
*
* SPDX-License-Identifier: EPL-2.0 OR GPL-2.0 WITH Classpath-exception-2.0
*/
package org.glassfish.tyrus.core;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.net.URI;
import java.net.URISyntaxException;
import java.net.URL;
import java.util.ArrayList;
import java.util.Collections;
import java.util.Enumeration;
import java.util.HashMap;
import java.util.Iterator;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
import java.util.PropertyResourceBundle;
import java.util.ResourceBundle;
import java.util.concurrent.Callable;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.locks.ReadWriteLock;
import java.util.concurrent.locks.ReentrantReadWriteLock;
import java.util.jar.JarEntry;
import java.util.jar.JarInputStream;
import java.util.logging.Level;
import java.util.logging.Logger;
import org.glassfish.tyrus.core.l10n.LocalizationMessages;
import org.osgi.framework.Bundle;
import org.osgi.framework.BundleContext;
import org.osgi.framework.BundleEvent;
import org.osgi.framework.BundleReference;
import org.osgi.framework.FrameworkUtil;
import org.osgi.framework.SynchronousBundleListener;
/**
* Taken from Jersey 2. Utility class to deal with OSGi runtime specific behavior.
* This is mainly to handle META-INF/services lookup
* and generic/application class lookup issue in OSGi.
* <p>
* When OSGi runtime is detected by the {@link ServiceFinder} class,
* an instance of OsgiRegistry is created and associated with given
* OSGi BundleContext. META-INF/services entries are then being accessed
* via the OSGi Bundle API as direct ClassLoader#getResource() method invocation
* does not work in this case within OSGi.
*
* @author Jakub Podlesak (jakub.podlesak at oracle.com)
*/
public final class OsgiRegistry implements SynchronousBundleListener {
private static final String CoreBundleSymbolicNAME = "org.glassfish.jersey.core.jersey-common";
private static final Logger LOGGER = Logger.getLogger(OsgiRegistry.class.getName());
private final BundleContext bundleContext;
private final Map<Long, Map<String, Callable<List<Class<?>>>>> factories =
new HashMap<Long, Map<String, Callable<List<Class<?>>>>>();
private final ReadWriteLock lock = new ReentrantReadWriteLock();
private static OsgiRegistry instance;
private Map<String, Bundle> classToBundleMapping = new HashMap<String, Bundle>();
/**
* Returns an {@code OsgiRegistry} instance. Call this method only if sure that the application is running in OSGi
* environment, otherwise a call to this method can lead to an {@link ClassNotFoundException}.
*
* @return an {@code OsgiRegistry} instance.
*/
public static synchronized OsgiRegistry getInstance() {
if (instance == null) {
final ClassLoader classLoader = ReflectionHelper.class.getClassLoader();
if (classLoader instanceof BundleReference) {
BundleContext context = FrameworkUtil.getBundle(OsgiRegistry.class).getBundleContext();
if (context != null) { // context could be still null in GlassFish
instance = new OsgiRegistry(context);
}
}
}
return instance;
}
private final class OsgiServiceFinder extends ServiceFinder.ServiceIteratorProvider {
final ServiceFinder.ServiceIteratorProvider defaultIterator =
new ServiceFinder.DefaultServiceIteratorProvider();
@Override
public <T> Iterator<T> createIterator(final Class<T> serviceClass, final String serviceName, ClassLoader loader,
boolean ignoreOnClassNotFound) {
final List<Class<?>> providerClasses = locateAllProviders(serviceName);
if (!providerClasses.isEmpty()) {
return new Iterator<T>() {
Iterator<Class<?>> it = providerClasses.iterator();
@Override
public boolean hasNext() {
return it.hasNext();
}
@Override
public T next() {
Class<T> nextClass = (Class<T>) it.next();
try {
return nextClass.newInstance();
} catch (Exception ex) {
ServiceConfigurationError sce = new ServiceConfigurationError(
serviceName + ": " + LocalizationMessages
.PROVIDER_COULD_NOT_BE_CREATED(nextClass.getName(), serviceClass,
ex.getLocalizedMessage()));
sce.initCause(ex);
throw sce;
}
}
@Override
public void remove() {
throw new UnsupportedOperationException();
}
};
}
return defaultIterator.createIterator(serviceClass, serviceName, loader, ignoreOnClassNotFound);
}
@Override
public <T> Iterator<Class<T>> createClassIterator(Class<T> service, String serviceName, ClassLoader loader,
boolean ignoreOnClassNotFound) {
final List<Class<?>> providerClasses = locateAllProviders(serviceName);
if (!providerClasses.isEmpty()) {
return new Iterator<Class<T>>() {
Iterator<Class<?>> it = providerClasses.iterator();
@Override
public boolean hasNext() {
return it.hasNext();
}
@SuppressWarnings("unchecked")
@Override
public Class<T> next() {
return (Class<T>) it.next();
}
@Override
public void remove() {
throw new UnsupportedOperationException();
}
};
}
return defaultIterator.createClassIterator(service, serviceName, loader, ignoreOnClassNotFound);
}
}
private static class BundleSpiProvidersLoader implements Callable<List<Class<?>>> {
private final String spi;
private final URI spiRegistryUri;
private final Bundle bundle;
BundleSpiProvidersLoader(final String spi, final URI spiRegistryUri, final Bundle bundle) {
this.spi = spi;
this.spiRegistryUri = spiRegistryUri;
this.bundle = bundle;
}
@Override
public List<Class<?>> call() throws Exception {
try {
if (LOGGER.isLoggable(Level.FINEST)) {
LOGGER.log(Level.FINEST, "Loading providers for SPI: {0}", spi);
}
final BufferedReader br =
new BufferedReader(new InputStreamReader(spiRegistryUri.toURL().openStream(), "UTF-8"));
String providerClassName;
final List<Class<?>> providerClasses = new ArrayList<Class<?>>();
while ((providerClassName = br.readLine()) != null) {
if (providerClassName.trim().length() == 0) {
continue;
}
if (LOGGER.isLoggable(Level.FINEST)) {
LOGGER.log(Level.FINEST, "SPI provider: {0}", providerClassName);
}
providerClasses.add(bundle.loadClass(providerClassName));
}
br.close();
return providerClasses;
} catch (Exception e) {
LOGGER.log(Level.WARNING, LocalizationMessages.EXCEPTION_CAUGHT_WHILE_LOADING_SPI_PROVIDERS(), e);
throw e;
} catch (Error e) {
LOGGER.log(Level.WARNING, LocalizationMessages.ERROR_CAUGHT_WHILE_LOADING_SPI_PROVIDERS(), e);
throw e;
}
}
@Override
public String toString() {
return spiRegistryUri.toString();
}
@Override
public int hashCode() {
return spiRegistryUri.hashCode();
}
@Override
public boolean equals(Object obj) {
if (obj instanceof BundleSpiProvidersLoader) {
return spiRegistryUri.equals(((BundleSpiProvidersLoader) obj).spiRegistryUri);
} else {
return false;
}
}
}
@Override
public void bundleChanged(BundleEvent event) {
if (event.getType() == BundleEvent.RESOLVED) {
register(event.getBundle());
} else if (event.getType() == BundleEvent.UNRESOLVED || event.getType() == BundleEvent.UNINSTALLED) {
final Bundle unregisteredBundle = event.getBundle();
lock.writeLock().lock();
try {
factories.remove(unregisteredBundle.getBundleId());
if (unregisteredBundle.getSymbolicName().equals(CoreBundleSymbolicNAME)) {
bundleContext.removeBundleListener(this);
factories.clear();
}
} finally {
lock.writeLock().unlock();
}
}
}
@SuppressWarnings("unchecked")
public Enumeration<URL> getPackageResources(final String packagePath, final ClassLoader classLoader) {
List<URL> result = new LinkedList<URL>();
classToBundleMapping.clear();
for (Bundle bundle : bundleContext.getBundles()) {
// Look for resources at the given <packagePath> and at WEB-INF/classes/<packagePath> in case a WAR is
// being examined.
for (String bundlePackagePath : new String[]{packagePath, "WEB-INF/classes/" + packagePath}) {
final Enumeration<URL> enumeration =
(Enumeration<URL>) bundle.findEntries(bundlePackagePath, "*", false);
if (enumeration != null) {
while (enumeration.hasMoreElements()) {
final URL url = enumeration.nextElement();
final String path = url.getPath();
final String className = (packagePath + path.substring(path.lastIndexOf('/')))
.replace('/', '.').replace(".class", "");
classToBundleMapping.put(className, bundle);
result.add(url);
}
}
}
// Now interested only in .jar provided by current bundle.
final Enumeration<URL> jars = bundle.findEntries("/", "*.jar", true);
if (jars != null) {
while (jars.hasMoreElements()) {
JarInputStream jarInputStream = null;
try {
final InputStream inputStream = classLoader.getResourceAsStream(jars.nextElement().getPath());
jarInputStream = new JarInputStream(inputStream);
JarEntry jarEntry;
while ((jarEntry = jarInputStream.getNextJarEntry()) != null) {
final String jarEntryName = jarEntry.getName();
if (jarEntryName.endsWith(".class") && jarEntryName.contains(packagePath)) {
classToBundleMapping.put(jarEntryName.replace(".class", "").replace('/', '.'), bundle);
result.add(bundle.getResource(jarEntryName));
}
}
} catch (Exception e) {
// Ignore.
} finally {
if (jarInputStream != null) {
try {
jarInputStream.close();
} catch (IOException ioe) {
// ignore
}
}
}
}
}
}
return Collections.enumeration(result);
}
/**
* Get the Class from the class name.
* <p>
* The context class loader will be utilized if accessible and non-null.
* Otherwise the defining class loader of this class will
* be utilized.
*
* @param className the class name.
* @return the Class, otherwise null if the class cannot be found.
* @throws ClassNotFoundException if the class cannot be found.
*/
public Class<?> classForNameWithException(final String className) throws ClassNotFoundException {
final Bundle bundle = classToBundleMapping.get(className);
if (bundle == null) {
throw new ClassNotFoundException(className);
}
return bundle.loadClass(className);
}
/**
* Tries to load resource bundle via OSGi means. No caching involved here,
* as localization properties are being cached in Localizer class already.
*
* @param bundleName name of the resource bundle to load
* @return resource bundle instance if found, null otherwise
*/
public ResourceBundle getResourceBundle(final String bundleName) {
final int lastDotIndex = bundleName.lastIndexOf('.');
final String path = bundleName.substring(0, lastDotIndex).replace('.', '/');
final String propertiesName = bundleName.substring(lastDotIndex + 1, bundleName.length()) + ".properties";
for (Bundle bundle : bundleContext.getBundles()) {
final Enumeration entries = bundle.findEntries(path, propertiesName, false);
if (entries != null && entries.hasMoreElements()) {
final URL entryUrl = (URL) entries.nextElement();
try {
return new PropertyResourceBundle(entryUrl.openStream());
} catch (IOException ex) {
if (LOGGER.isLoggable(Level.FINE)) {
// does not make sense to localize this
LOGGER.fine("Exception caught when tried to load resource bundle in OSGi");
}
return null;
}
}
}
return null;
}
/**
* Creates a new OsgiRegistry instance bound to a particular OSGi runtime.
* The only parameter must be an instance of a {@link BundleContext}.
*
* @param bundleContext must be a non-null instance of a BundleContext
*/
private OsgiRegistry(BundleContext bundleContext) {
this.bundleContext = bundleContext;
}
/**
* Will hook up this instance with the OSGi runtime.
* This is to actually update SPI provider lookup and class loading mechanisms in Jersey
* to utilize OSGi features.
*/
public void hookUp() {
setOSGiServiceFinderIteratorProvider();
bundleContext.addBundleListener(this);
registerExistingBundles();
}
private void registerExistingBundles() {
for (Bundle bundle : bundleContext.getBundles()) {
if (bundle.getState() == Bundle.RESOLVED || bundle.getState() == Bundle.STARTING
|| bundle.getState() == Bundle.ACTIVE || bundle.getState() == Bundle.STOPPING) {
register(bundle);
}
}
}
private void setOSGiServiceFinderIteratorProvider() {
ServiceFinder.setIteratorProvider(new OsgiServiceFinder());
}
private void register(final Bundle bundle) {
if (LOGGER.isLoggable(Level.FINEST)) {
LOGGER.log(Level.FINEST, "checking bundle {0}", bundle.getBundleId());
}
Map<String, Callable<List<Class<?>>>> map;
lock.writeLock().lock();
try {
map = factories.get(bundle.getBundleId());
if (map == null) {
map = new ConcurrentHashMap<String, Callable<List<Class<?>>>>();
factories.put(bundle.getBundleId(), map);
}
} finally {
lock.writeLock().unlock();
}
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);
try {
map.put(factoryId, new BundleSpiProvidersLoader(factoryId, u.toURI(), bundle));
} catch (URISyntaxException e1) {
// ignore.
}
}
}
}
private List<Class<?>> locateAllProviders(String serviceName) {
lock.readLock().lock();
try {
final List<Class<?>> result = new LinkedList<Class<?>>();
for (Map<String, Callable<List<Class<?>>>> value : factories.values()) {
if (value.containsKey(serviceName)) {
try {
result.addAll(value.get(serviceName).call());
} catch (Exception ex) {
// ignore
}
}
}
return result;
} finally {
lock.readLock().unlock();
}
}
}