org.glassfish.common.util.OSGiObjectInputOutputStreamFactoryImpl Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of payara-micro Show documentation
Show all versions of payara-micro Show documentation
Micro Distribution of the Payara Project
/*
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
*
* Copyright (c) 2009-2013 Oracle and/or its affiliates. All rights reserved.
*
* The contents of this file are subject to the terms of either the GNU
* General Public License Version 2 only ("GPL") or the Common Development
* and Distribution License("CDDL") (collectively, the "License"). You
* may not use this file except in compliance with the License. You can
* obtain a copy of the License at
* https://glassfish.dev.java.net/public/CDDL+GPL_1_1.html
* or packager/legal/LICENSE.txt. See the License for the specific
* language governing permissions and limitations under the License.
*
* When distributing the software, include this License Header Notice in each
* file and include the License file at packager/legal/LICENSE.txt.
*
* GPL Classpath Exception:
* Oracle designates this particular file as subject to the "Classpath"
* exception as provided by Oracle in the GPL Version 2 section of the License
* file that accompanied this code.
*
* Modifications:
* If applicable, add the following below the License Header, with the fields
* enclosed by brackets [] replaced by your own identifying information:
* "Portions Copyright [year] [name of copyright owner]"
*
* Contributor(s):
* If you wish your version of this file to be governed by only the CDDL or
* only the GPL Version 2, indicate your decision by adding "[Contributor]
* elects to include this software in this distribution under the [CDDL or GPL
* Version 2] license." If you don't indicate a single choice of license, a
* recipient has the option to distribute your version of this file under
* either the CDDL, the GPL Version 2 or to extend the choice of license to
* its licensees as provided above. However, if you add GPL Version 2 code
* and therefore, elected the GPL Version 2 license, then the option applies
* only if the new code is made subject to such option by the copyright
* holder.
*/
// Portions Copyright [2022] Payara Foundation and/or affiliates
package org.glassfish.common.util;
import org.osgi.framework.Bundle;
import org.osgi.framework.BundleContext;
import org.osgi.framework.BundleEvent;
import org.osgi.framework.ServiceReference;
import org.osgi.service.packageadmin.PackageAdmin;
import org.osgi.util.tracker.BundleTracker;
import org.osgi.util.tracker.BundleTrackerCustomizer;
import com.sun.enterprise.util.CULoggerInfo;
import java.io.*;
import java.lang.reflect.Array;
import java.util.concurrent.ConcurrentHashMap;
import java.util.logging.Level;
import java.util.logging.Logger;
import static com.sun.enterprise.util.Utility.getClassLoader;
/**
* @author [email protected]
*/
public class OSGiObjectInputOutputStreamFactoryImpl
implements ObjectInputOutputStreamFactory {
private static final Logger logger = CULoggerInfo.getLogger();
private BundleContext ctx;
PackageAdmin pkgAdm;
private ConcurrentHashMap name2Id = new ConcurrentHashMap();
// Since bundle id starts with 0, we use -1 to indicate a non-bundle
// private static final long NOT_A_BUNDLE_ID = -1;
private static final String NOT_A_BUNDLE_KEY = ":";
public OSGiObjectInputOutputStreamFactoryImpl(BundleContext ctx)
{
this.ctx = ctx;
ServiceReference ref = ctx.getServiceReference(PackageAdmin.class.getName());
pkgAdm = PackageAdmin.class.cast(ctx.getService(ref));
BundleTracker bt = new BundleTracker(ctx, Bundle.INSTALLED | Bundle.RESOLVED | Bundle.STARTING | Bundle.ACTIVE, new BundleTrackerCustomizer() {
@Override
public void modifiedBundle(Bundle bundle, BundleEvent bundleEvent, Object o) {
}
@Override
public Object addingBundle(Bundle bundle, BundleEvent bundleEvent) {
String key = makeKey(bundle);
name2Id.put(key, bundle.getBundleId());
if (logger != null && logger.isLoggable(Level.FINER)) {
logger.log(Level.FINER, "BundleTracker.addingBundle BUNDLE {0} ==> {1} for {2}", new Object[]{key, bundle.getBundleId(), bundle.getSymbolicName()});
}
return null;
}
@Override
public void removedBundle(Bundle bundle, BundleEvent bundleEvent, Object o) {
String key = makeKey(bundle);
Long bundleID = name2Id.remove(key);
if (logger.isLoggable(Level.FINER)) {
logger.log(Level.FINER, "BundleTracker.removedBundle BUNDLE {0} ==> {1}", new Object[]{key, bundle.getSymbolicName()});
}
if (bundleID == null) {
logger.log(Level.WARNING, CULoggerInfo.NULL_BUNDLE, key);
}
}
});
bt.open();
}
private String makeKey(Bundle bundle) {
return bundle.getSymbolicName() + ":" + bundle.getVersion();
}
@Override
public ObjectInputStream createObjectInputStream(InputStream in)
throws IOException
{
ClassLoader loader = Thread.currentThread().getContextClassLoader();
if (loader == null) {
loader = getClassLoader();
}
return new OSGiObjectInputStream(in, loader);
}
@Override
public ObjectOutputStream createObjectOutputStream(OutputStream out)
throws IOException
{
return new OSGiObjectOutputStream(out);
}
private class OSGiObjectInputStream extends ObjectInputStreamWithLoader
{
public OSGiObjectInputStream(InputStream in, ClassLoader loader) throws IOException
{
super(in, loader);
}
@Override
protected Class> resolveClass(ObjectStreamClass desc)
throws IOException, ClassNotFoundException
{
Class clazz =
OSGiObjectInputOutputStreamFactoryImpl.this.resolveClass(this, desc);
if (clazz == null) {
clazz = super.resolveClass(desc);
}
return clazz;
}
}
private class OSGiObjectOutputStream extends ObjectOutputStream {
private OSGiObjectOutputStream(OutputStream out) throws IOException
{
super(out);
}
@Override
protected void annotateClass(Class> cl) throws IOException
{
OSGiObjectInputOutputStreamFactoryImpl.this.annotateClass(this, cl);
}
}
@Override
public Class> resolveClass(ObjectInputStream in, final ObjectStreamClass desc)
throws IOException, ClassNotFoundException
{
try {
String key = in.readUTF();
if (! NOT_A_BUNDLE_KEY.equals(key)) {
Long bundleId = name2Id.get(key);
if (bundleId != null) {
final Bundle b = ctx.getBundle(bundleId);
String cname = desc.getName();
if (cname.startsWith("[")) {
return loadArrayClass(b, cname);
} else {
return loadClassFromBundle(b, cname);
}
}
}
}catch(IOException e) {
logger.fine("Missing Bundle Key continuing...");
}
return null;
}
@Override
public void annotateClass(ObjectOutputStream out, Class> cl) throws IOException
{
String key = NOT_A_BUNDLE_KEY;
Bundle b = pkgAdm.getBundle(cl);
if (b != null) {
key = makeKey(b);
}
out.writeUTF(key);
}
private Class loadArrayClass(Bundle b, String cname) throws ClassNotFoundException {
// We are never called with primitive types, so we don't have to check for primitive types.
assert(cname.charAt(0) == '['); // An array
Class component; // component class
int dcount; // dimension
for (dcount = 1; cname.charAt(dcount) == '['; dcount++){
}
assert(cname.charAt(dcount) == 'L');
component = loadClassFromBundle(b, cname.substring(dcount + 1, cname.length() - 1));
int[] dim = new int[dcount];
for (int i = 0; i < dcount; i++) {
dim[i] = 0;
}
return Array.newInstance(component, dim).getClass();
}
private Class loadClassFromBundle(final Bundle b, final String cname) throws ClassNotFoundException {
if (System.getSecurityManager() == null) {
return b.loadClass(cname);
} else {
try {
return (Class) java.security.AccessController.doPrivileged(
new java.security.PrivilegedExceptionAction() {
@Override
public java.lang.Object run() throws ClassNotFoundException {
return b.loadClass(cname);
}
});
} catch (java.security.PrivilegedActionException pae) {
throw (ClassNotFoundException) pae.getException();
}
}
}
}