org.apache.axis2.jaxws.message.databinding.JAXBUtils Maven / Gradle / Ivy
/*
* 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.axis2.jaxws.message.databinding;
import org.apache.axis2.java.security.AccessController;
import org.apache.axis2.jaxws.ExceptionFactory;
import org.apache.axis2.jaxws.i18n.Messages;
import org.apache.axis2.jaxws.message.databinding.JAXBUtilsMonitor;
import org.apache.axis2.jaxws.message.factory.ClassFinderFactory;
import org.apache.axis2.jaxws.registry.FactoryRegistry;
import org.apache.axis2.jaxws.utility.ClassUtils;
import org.apache.axis2.jaxws.utility.JavaUtils;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import javax.xml.bind.JAXBContext;
import javax.xml.bind.JAXBException;
import javax.xml.bind.JAXBIntrospector;
import javax.xml.bind.Marshaller;
import javax.xml.bind.PropertyException;
import javax.xml.bind.Unmarshaller;
import javax.xml.bind.annotation.XmlType;
import javax.xml.ws.Holder;
import javax.xml.ws.wsaddressing.W3CEndpointReference;
import java.io.File;
import java.io.IOException;
import java.io.UnsupportedEncodingException;
import java.lang.annotation.Annotation;
import java.lang.ref.SoftReference;
import java.lang.reflect.AnnotatedElement;
import java.lang.reflect.Array;
import java.net.URL;
import java.net.URLDecoder;
import java.security.PrivilegedActionException;
import java.security.PrivilegedExceptionAction;
import java.security.PrivilegedAction;
import java.util.ArrayList;
import java.util.Enumeration;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.TreeSet;
import java.util.concurrent.ConcurrentHashMap;
/**
* JAXB Utilites to pool JAXBContext and related objects.
*/
public class JAXBUtils {
private static final Log log = LogFactory.getLog(JAXBUtils.class);
// Create a concurrent map to get the JAXBObject:
// key is the String (sorted packages)
// value is a SoftReference to a ConcurrentHashMap of Classloader keys and JAXBContextValue objects
// It is a soft map to encourage GC in low memory situations
private static Map<
String,
SoftReference>> jaxbMap =
new ConcurrentHashMap>>();
private static Pool mpool = new Pool();
private static Pool upool = new Pool();
private static Pool ipool = new Pool();
// From Lizet Ernand:
// If you really care about the performance,
// and/or your application is going to read a lot of small documents,
// then creating Unmarshaller could be relatively an expensive operation.
// In that case, consider pooling Unmarshaller objects.
// Different threads may reuse one Unmarshaller instance,
// as long as you don't use one instance from two threads at the same time.
// ENABLE_ADV_POOLING is false...which means they are obtained from the JAXBContext instead of
// from the pool.
private static boolean ENABLE_MARSHALL_POOLING = true;
private static boolean ENABLE_UNMARSHALL_POOLING = true;
private static boolean ENABLE_INTROSPECTION_POOLING = false;
private static int MAX_LOAD_FACTOR = 32; // Maximum number of JAXBContext to store
// Construction Type
public enum CONSTRUCTION_TYPE {
BY_CLASS_ARRAY, // New Instance with Class[]
BY_CONTEXT_PATH, // New Instance with context path string (preferred)
BY_CLASS_ARRAY_PLUS_ARRAYS, // New Instance with Class[] plus arrays of each class are added
UNKNOWN}
;
// Some packages have a known set of classes.
// This map is immutable after its static creation.
private static final Map> specialMap = new HashMap>();
static {
// The javax.xml.ws.wsaddressing package has a single class (W3CEndpointReference)
List classes = new ArrayList();
classes.add(W3CEndpointReference.class);
specialMap.put("javax.xml.ws.wsaddressing", classes);
}
public static final String DEFAULT_NAMESPACE_REMAP = getDefaultNamespaceRemapProperty();
/**
* Get a JAXBContext for the class
*
* @param contextPackage Set
* @return JAXBContext
* @throws JAXBException
* @deprecated
*/
public static JAXBContext getJAXBContext(TreeSet contextPackages) throws JAXBException {
return getJAXBContext(contextPackages, new Holder(),
contextPackages.toString(), null, null);
}
/**
* Get a JAXBContext for the class
*
* Note: The contextPackage object is used by multiple threads. It should be considered immutable
* and not altered by this method.
*
* @param contextPackage Set
* @param cacheKey ClassLoader
* @return JAXBContext
* @throws JAXBException
* @deprecated
*/
public static JAXBContext getJAXBContext(TreeSet contextPackages, ClassLoader
cacheKey) throws JAXBException {
return getJAXBContext(contextPackages, new Holder(),
contextPackages.toString(), cacheKey, null);
}
public static JAXBContext getJAXBContext(TreeSet contextPackages,
Holder constructionType,
String key)
throws JAXBException {
return getJAXBContext(contextPackages, constructionType, key, null, null);
}
/**
* Get a JAXBContext for the class
*
* Note: The contextPackage object is used by multiple threads. It should be considered immutable
* and not altered by this method.
*
* @param contextPackage Set
* @param contructionType (output value that indicates how the context was constructed)
* @param cacheKey ClassLoader
* @return JAXBContext
* @throws JAXBException
*/
public static JAXBContext getJAXBContext(TreeSet contextPackages,
Holder constructionType,
String key,
ClassLoader cacheKey,
Map properties)
throws JAXBException {
return getJAXBContext(contextPackages,
constructionType,
false,
key,
cacheKey,
properties);
}
/**
* Get a JAXBContext for the class
*
* Note: The contextPackage object is used by multiple threads. It should be considered immutable
* and not altered by this method.
*
* @param contextPackage Set
* @param contructionType (output value that indicates how the context was constructed)
* @param forceArrays (forces the returned JAXBContext to include the array types)
* @param cacheKey ClassLoader
* @return JAXBContext
* @throws JAXBException
*/
public static JAXBContext getJAXBContext(TreeSet contextPackages,
Holder constructionType,
boolean forceArrays,
String key,
ClassLoader cacheKey,
Map properties)
throws JAXBException {
// JAXBContexts for the same class can be reused and are supposed to be thread-safe
if (log.isDebugEnabled()) {
log.debug("Following packages are in this batch of getJAXBContext() :");
for (String pkg : contextPackages) {
log.debug(pkg);
}
}
if (JAXBUtilsMonitor.isMonitoring()) {
JAXBUtilsMonitor.addPackageKey(contextPackages.toString());
}
// Get or Create The InnerMap using the package key
ConcurrentHashMap innerMap = null;
SoftReference>
softRef = jaxbMap.get(key);
if (softRef != null) {
innerMap = softRef.get();
}
if (innerMap == null) {
synchronized(jaxbMap) {
softRef = jaxbMap.get(key);
if (softRef != null) {
innerMap = softRef.get();
}
if (innerMap == null) {
innerMap = new ConcurrentHashMap();
softRef =
new SoftReference>(innerMap);
jaxbMap.put(key, softRef);
}
}
}
// Now get the contextValue using either the classloader key or
// the current Classloader
ClassLoader cl = getContextClassLoader();
JAXBContextValue contextValue = null;
if(cacheKey != null) {
if(log.isDebugEnabled()) {
log.debug("Using supplied classloader to retrieve JAXBContext: " +
cacheKey);
}
contextValue = innerMap.get(cacheKey);
} else {
if(log.isDebugEnabled()) {
log.debug("Using classloader from Thread to retrieve JAXBContext: " +
cl);
}
contextValue = innerMap.get(cl);
}
// If the context value is found, but the caller requested that the JAXBContext
// contain arrays, then rebuild the JAXBContext value
if (forceArrays &&
contextValue != null &&
contextValue.constructionType != JAXBUtils.CONSTRUCTION_TYPE.BY_CLASS_ARRAY_PLUS_ARRAYS) {
if(log.isDebugEnabled()) {
log.debug("Found a JAXBContextValue with constructionType=" +
contextValue.constructionType + " but the caller requested a JAXBContext " +
" that includes arrays. A new JAXBContext will be built");
}
contextValue = null;
}
if (contextPackages == null) {
contextPackages = new TreeSet();
}
if (contextValue == null) {
synchronized (innerMap) {
// Try to get the contextValue once more since sync was temporarily exited.
ClassLoader clKey = (cacheKey != null) ? cacheKey:cl;
contextValue = innerMap.get(clKey);
adjustPoolSize(innerMap);
if (forceArrays &&
contextValue != null &&
contextValue.constructionType != JAXBUtils.CONSTRUCTION_TYPE.BY_CLASS_ARRAY_PLUS_ARRAYS) {
contextValue = null;
}
if (contextValue==null) {
// Create a copy of the contextPackages. This new TreeSet will
// contain only the valid contextPackages.
// Note: The original contextPackage set is accessed by multiple
// threads and should not be altered.
TreeSet validContextPackages = new TreeSet(contextPackages);
List classRefs = pruneDirectives(validContextPackages);
int numPackages = validContextPackages.size();
contextValue = createJAXBContextValue(validContextPackages,
clKey,
forceArrays,
properties,
classRefs);
synchronized (jaxbMap) {
// Add the context value with the original package set
ConcurrentHashMap map1 = null;
SoftReference>
softRef1 = jaxbMap.get(key);
if (softRef1 != null) {
map1 = softRef1.get();
}
if (map1 == null) {
map1 = new ConcurrentHashMap();
softRef1 =
new SoftReference>(map1);
jaxbMap.put(key, softRef1);
}
map1.put(clKey, contextValue);
String validPackagesKey = validContextPackages.toString();
// Add the context value with the new package set
ConcurrentHashMap map2 = null;
SoftReference>
softRef2 = jaxbMap.get(validPackagesKey);
if (softRef2 != null) {
map2 = softRef2.get();
}
if (map2 == null) {
map2 = new ConcurrentHashMap();
softRef2 =
new SoftReference>(map2);
jaxbMap.put(validPackagesKey, softRef2);
}
map2.put(clKey, contextValue);
if (log.isDebugEnabled()) {
log.debug("JAXBContext [created] for " + key);
log.debug("JAXBContext also stored by the list of valid packages:" + validPackagesKey);
}
}
}
}
} else {
if (log.isDebugEnabled()) {
log.debug("JAXBContext [from pool] for " + key);
}
}
if (log.isDebugEnabled()) {
log.debug("JAXBContext constructionType= " + contextValue.constructionType);
log.debug("JAXBContextValue = " + JavaUtils.getObjectIdentity(contextValue));
log.debug("JAXBContext = " + JavaUtils.getObjectIdentity(contextValue.jaxbContext));
}
constructionType.value = contextValue.constructionType;
return contextValue.jaxbContext;
}
/**
* The contextPackages may declare overrides.
* Example:
* "com.A"
* "com.B"
* "com.C"
* "@com.A" <-- Indicates a reference to a class (versus ns 2 pkg conversion)
* "com.A > com.B" <-- This says com.A overrides com.B
*
* This method prunes the overrides and overriden packages.
* Example return:
* "com.A"
* "com.C"
* @param contextPackages
* @return List class references
*/
protected static List pruneDirectives(TreeSet contextPackages) {
List removePkgsList = new ArrayList();
List strongPkgsList = new ArrayList();
List classRefs = new ArrayList();
// Walk the contextPackages looking for entries representing directives
Iterator it = contextPackages.iterator();
while (it.hasNext()) {
String entry = it.next();
// If the entry contains an override character
if (entry.contains(">")) {
if (log.isDebugEnabled()) {
log.debug("Override found:" + entry);
}
// Remove the entry using an iterator remove()
it.remove();
// Store the overridden package
String removePkg = entry.substring(entry.indexOf(">") + 1);
removePkg = removePkg.trim();
removePkgsList.add(removePkg);
}
if (entry.startsWith("@")) {
if (log.isDebugEnabled()) {
log.debug("Strong (class) reference found:" + entry);
}
// Remove the entry using an iterator remove()
it.remove();
// Store the overridden package
String strongPkg = entry.substring(1);
strongPkg = strongPkg.trim();
strongPkgsList.add(strongPkg);
}
if (entry.startsWith("[")) {
if (log.isDebugEnabled()) {
log.debug("Class Reference found:" + entry);
}
// Remove the entry using an iterator remove()
it.remove();
// Store the class
String cls = entry.substring(1, entry.length()-1);
classRefs.add(cls);
}
}
// Now walk the contextPackages and remove the overriden packages
it = contextPackages.iterator();
while (it.hasNext()) {
String entry = it.next();
// If the entry contains an override character
if (removePkgsList.contains(entry)) {
if (log.isDebugEnabled()) {
log.debug("Removing override package:" + entry);
}
// Remove the overridden package using an iterator remove()
it.remove();
}
}
// Now add back all of the strong packages
contextPackages.addAll(strongPkgsList);
return classRefs;
}
/**
* Create a JAXBContext using the contextPackages
*
* @param contextPackages Set
* @param cl ClassLoader
* @param forceArrays boolean (true if JAXBContext must include all arrays)
* @param properties Map of properties for the JAXBContext.newInstance creation method
* @param classRefs List of class references
* @return JAXBContextValue (JAXBContext + constructionType)
* @throws JAXBException
*/
private static JAXBContextValue createJAXBContextValue(TreeSet contextPackages,
ClassLoader cl,
boolean forceArrays,
Map properties,
List classRefs) throws JAXBException {
JAXBContextValue contextValue = null;
if (log.isDebugEnabled()) {
log.debug("Following packages are in this batch of getJAXBContext() :");
for (String pkg : contextPackages) {
log.debug(pkg);
}
log.debug("This classloader will be used to construct the JAXBContext" + cl);
}
// The contextPackages is a set of package names that are constructed using PackageSetBuilder.
// PackageSetBuilder gets the packages names from various sources.
// a) It walks the various annotations on the WebService collecting package names.
// b) It walks the wsdl/schemas and builds package names for each target namespace.
//
// The combination of these two sources should produce all of the package names.
// -------------
// Note that (b) is necessary for the following case:
// An operation has a parameter named BASE.
// Object DERIVED is an extension of BASE and is defined in a different package/schema.
// In this case, there will not be any annotations on the WebService that reference DERIVED.
// The only way to find the package for DERIVED is to walk the schemas.
// -------------
Iterator it = contextPackages.iterator();
while (it.hasNext()) {
String p = it.next();
// Don't consider java and javax packages
// REVIEW: We might have to refine this
if (p.startsWith("javax.xml.ws.wsaddressing")) {
continue;
}
if (p.startsWith("java.") ||
p.startsWith("javax.")) {
it.remove();
}
}
// There are two ways to construct the context.
// 1) USE A CONTEXTPATH, which is a string containing
// all of the packages separated by colons.
// 2) USE A CLASS[], which is an array of all of the classes
// involved in the marshal/unmarshal.
//
// There are pros/cons with both approaches.
// USE A CONTEXTPATH:
// Pros: preferred way of doing this.
// performant
// most dynamic
// Cons: Each package in context path must have an ObjectFactory
//
//
// USE CLASS[]:
// Pros: Doesn't require ObjectFactory in each package
// Cons: Hard to set up, must account for JAX-WS classes, etc.
// Does not work if arrays of classes are needed
// slower
//
// The following code attempts to build a context path. It then
// choose one of the two constructions above (prefer USE A CONTEXT_PATH)
//
// The packages are examined to see if they have ObjectFactory/package-info classes.
// Invalid packages are removed from the list
it = contextPackages.iterator();
boolean contextConstruction = (!forceArrays);
boolean isJAXBFound = false;
while (it.hasNext()) {
String p = it.next();
// See if this package has an ObjectFactory or package-info
if (checkPackage(p, cl)) {
// Flow to here indicates package can be used for CONTEXT construction
isJAXBFound = true;
if (log.isDebugEnabled()) {
log.debug("Package " + p + " contains an ObjectFactory or package-info class.");
}
} else {
// Flow to here indicates that the package is not valid for context construction.
// Perhaps the package is invalid.
if (log.isDebugEnabled()) {
log.debug("Package " + p +
" does not contain an ObjectFactory or package-info class. Searching for JAXB classes");
}
List classes = null;
classes = getAllClassesFromPackage(p, cl);
if (classes == null || classes.size() == 0) {
if (log.isDebugEnabled()) {
log.debug("Package " + p +
" does not have any JAXB classes. It is removed from the JAXB context path.");
}
it.remove();
} else {
// Classes are found in the package. We cannot use the CONTEXT construction
contextConstruction = false;
if (log.isDebugEnabled()) {
log.debug("Package " + p +
" does not contain ObjectFactory, but it does contain other JAXB classes.");
}
}
}
}
if (!isJAXBFound) {
if (log.isDebugEnabled()) {
log.debug("ObjectFactory & package-info are not found in package hierachy");
}
}
// The code above may have removed some packages from the list.
// Retry our lookup with the updated list
if (contextConstruction) {
if (log.isDebugEnabled()) {
log.debug("Recheck Cache Start: Some packages have been removed from the list. Rechecking cache.");
}
String key = contextPackages.toString();
ConcurrentHashMap innerMap = null;
SoftReference> softRef = jaxbMap.get(key);
if (softRef != null) {
innerMap = softRef.get();
}
if (innerMap != null) {
contextValue = innerMap.get(cl);
if (forceArrays &&
contextValue != null &&
contextValue.constructionType != JAXBUtils.CONSTRUCTION_TYPE.BY_CLASS_ARRAY_PLUS_ARRAYS) {
if(log.isDebugEnabled()) {
log.debug("Found a JAXBContextValue with constructionType=" +
contextValue.constructionType + " but the caller requested a JAXBContext " +
" that includes arrays. A new JAXBContext will be built");
}
contextValue = null;
}
if (contextValue != null) {
if (log.isDebugEnabled()) {
log.debug("Successfully found JAXBContext with updated context list:" +
contextValue.jaxbContext.toString());
}
return contextValue;
}
}
if (log.isDebugEnabled()) {
log.debug("Recheck Cache End: Did not find a JAXBContext. Will build a new JAXBContext.");
}
}
// CONTEXT construction
if (contextConstruction) {
if (log.isDebugEnabled()) {
log.debug("Try building a JAXBContext using the packages only.");
}
JAXBContext context = createJAXBContextUsingContextPath(contextPackages, cl, classRefs);
if (context != null) {
contextValue = new JAXBContextValue(context, CONSTRUCTION_TYPE.BY_CONTEXT_PATH);
}
if (log.isDebugEnabled()) {
log.debug("Building a JAXBContext with packages only success=" + (contextValue != null));
}
}
// CLASS construction
if (contextValue == null) {
if (log.isDebugEnabled()) {
log.debug("Try building a JAXBContext using a list of classes.");
log.debug("Start finding classes");
}
it = contextPackages.iterator();
List fullList = new ArrayList();
while (it.hasNext()) {
String pkg = it.next();
fullList.addAll(getAllClassesFromPackage(pkg, cl));
}
//Lets add all common array classes
addCommonArrayClasses(fullList);
Class[] classArray = fullList.toArray(new Class[0]);
if (log.isDebugEnabled()) {
log.debug("End finding classes");
}
JAXBContext context = JAXBContext_newInstance(classArray, cl, properties, classRefs);
if (context != null) {
if (forceArrays) {
contextValue = new JAXBContextValue(context, CONSTRUCTION_TYPE.BY_CLASS_ARRAY_PLUS_ARRAYS);
} else {
contextValue = new JAXBContextValue(context, CONSTRUCTION_TYPE.BY_CLASS_ARRAY);
}
}
}
if (log.isDebugEnabled()) {
log.debug("Successfully created JAXBContext " + contextValue.jaxbContext.toString());
}
return contextValue;
}
/**
* Get the unmarshaller. You must call releaseUnmarshaller to put it back into the pool
*
* @param context JAXBContext
* @return Unmarshaller
* @throws JAXBException
*/
public static Unmarshaller getJAXBUnmarshaller(JAXBContext context) throws JAXBException {
if (!ENABLE_UNMARSHALL_POOLING) {
if (log.isDebugEnabled()) {
log.debug("Unmarshaller created [no pooling]");
}
return internalCreateUnmarshaller(context);
}
Unmarshaller unm = upool.get(context);
if (unm == null) {
if (log.isDebugEnabled()) {
log.debug("Unmarshaller created [not in pool]");
}
unm = internalCreateUnmarshaller(context);
} else {
if (log.isDebugEnabled()) {
log.debug("Unmarshaller obtained [from pool]");
}
}
return unm;
}
private static Unmarshaller internalCreateUnmarshaller(final JAXBContext context) throws JAXBException {
Unmarshaller unm;
try {
unm = (Unmarshaller) AccessController.doPrivileged(
new PrivilegedExceptionAction() {
public Object run() throws JAXBException {
return context.createUnmarshaller();
}
}
);
} catch (PrivilegedActionException e) {
throw (JAXBException) e.getCause();
}
return unm;
}
private static Marshaller internalCreateMarshaller(final JAXBContext context) throws JAXBException {
Marshaller marshaller;
try {
marshaller = (Marshaller) AccessController.doPrivileged(
new PrivilegedExceptionAction() {
public Object run() throws JAXBException {
return context.createMarshaller();
}
}
);
} catch (PrivilegedActionException e) {
throw (JAXBException) e.getCause();
}
return marshaller;
}
/**
* Release Unmarshaller Do not call this method if an exception occurred while using the
* Unmarshaller. We object my be in an invalid state.
*
* @param context JAXBContext
* @param unmarshaller Unmarshaller
*/
public static void releaseJAXBUnmarshaller(JAXBContext context, Unmarshaller unmarshaller) {
if (log.isDebugEnabled()) {
log.debug("Unmarshaller placed back into pool");
}
if (ENABLE_UNMARSHALL_POOLING) {
unmarshaller.setAttachmentUnmarshaller(null);
upool.put(context, unmarshaller);
}
}
/**
* Get JAXBMarshaller
*
* @param context JAXBContext
* @return Marshaller
* @throws JAXBException
*/
public static Marshaller getJAXBMarshaller(JAXBContext context) throws JAXBException {
Marshaller m = null;
if (!ENABLE_MARSHALL_POOLING) {
if (log.isDebugEnabled()) {
log.debug("Marshaller created [no pooling]");
}
m = internalCreateMarshaller(context);
} else {
m = mpool.get(context);
if (m == null) {
if (log.isDebugEnabled()) {
log.debug("Marshaller created [not in pool]");
}
m = internalCreateMarshaller(context);
} else {
if (log.isDebugEnabled()) {
log.debug("Marshaller obtained [from pool]");
}
}
}
m.setProperty(Marshaller.JAXB_FRAGMENT, Boolean.TRUE); // No PIs
return m;
}
/**
* releaseJAXBMarshalller
* Do not call this method if an exception occurred while using the
* Marshaller. We don't want an object in an invalid state.
*
* @param context JAXBContext
* @param marshaller Marshaller
*/
public static void releaseJAXBMarshaller(JAXBContext context, Marshaller marshaller) {
if (log.isDebugEnabled()) {
log.debug("Marshaller placed back into pool");
log.debug(" Marshaller = " + JavaUtils.getObjectIdentity(marshaller));
log.debug(" JAXBContext = " + JavaUtils.getObjectIdentity(context));
}
if (ENABLE_MARSHALL_POOLING) {
// Make sure to clear any state or properties
try {
marshaller.setAttachmentMarshaller(null);
// Set the JAXB_ENCODING back to the default value UTF-8
marshaller.setProperty(Marshaller.JAXB_ENCODING, "UTF-8");
mpool.put(context, marshaller);
} catch (Throwable t) {
// Log the problem, and continue without pooling
if (log.isDebugEnabled()) {
log.debug("The following exception is ignored. Processing continues " + t);
}
}
}
}
/**
* get JAXB Introspector
*
* @param context JAXBContext
* @return JAXBIntrospector
* @throws JAXBException
*/
public static JAXBIntrospector getJAXBIntrospector(final JAXBContext context) throws JAXBException {
JAXBIntrospector i = null;
if (!ENABLE_INTROSPECTION_POOLING) {
if (log.isDebugEnabled()) {
log.debug("JAXBIntrospector created [no pooling]");
}
i = internalCreateIntrospector(context);
} else {
i = ipool.get(context);
if (i == null) {
if (log.isDebugEnabled()) {
log.debug("JAXBIntrospector created [not in pool]");
}
i = internalCreateIntrospector(context);
} else {
if (log.isDebugEnabled()) {
log.debug("JAXBIntrospector obtained [from pool]");
}
}
}
return i;
}
private static JAXBIntrospector internalCreateIntrospector(final JAXBContext context) {
JAXBIntrospector i;
i = (JAXBIntrospector) AccessController.doPrivileged(
new PrivilegedAction() {
public Object run() {
return context.createJAXBIntrospector();
}
}
);
return i;
}
/**
* Release JAXBIntrospector Do not call this method if an exception occurred while using the
* JAXBIntrospector. We object my be in an invalid state.
*
* @param context JAXBContext
* @param introspector JAXBIntrospector
*/
public static void releaseJAXBIntrospector(JAXBContext context, JAXBIntrospector introspector) {
if (log.isDebugEnabled()) {
log.debug("JAXBIntrospector placed back into pool");
}
if (ENABLE_INTROSPECTION_POOLING) {
ipool.put(context, introspector);
}
}
/**
* @param p Package
* @param cl
* @return true if each package has a ObjectFactory class or package-info
*/
private static boolean checkPackage(String p, ClassLoader cl) {
// Each package must have an ObjectFactory
if (log.isDebugEnabled()) {
log.debug("checking package :" + p);
}
try {
Class cls = forName(p + ".ObjectFactory", false, cl);
if (cls != null) {
return true;
}
} catch (Throwable e) {
//Catch Throwable as ClassLoader can throw an NoClassDefFoundError that
//does not extend Exception. So we will absorb any Throwable exception here.
if (log.isDebugEnabled()) {
log.debug("ObjectFactory Class Not Found " + e);
}
}
try {
Class cls = forName(p + ".package-info", false, cl);
if (cls != null) {
return true;
}
} catch (Throwable e) {
//Catch Throwable as ClassLoader can throw an NoClassDefFoundError that
//does not extend Exception. So we will absorb any Throwable exception here.
if (log.isDebugEnabled()) {
log.debug("package-info Class Not Found " + e);
}
}
return false;
}
/**
* Create a JAXBContext using the contextpath approach
*
* @param packages
* @param cl ClassLoader
* @param List classRefs
* @return JAXBContext or null if unsuccessful
*/
private static JAXBContext createJAXBContextUsingContextPath(TreeSet packages,
ClassLoader cl,
List classRefs) {
JAXBContext context = null;
String contextpath = "";
// Iterate through the classes and build the contextpath
Iterator it = packages.iterator();
while (it.hasNext()) {
String p = it.next();
if (contextpath.length() != 0) {
contextpath += ":";
}
contextpath += p;
}
try {
if (log.isDebugEnabled()) {
log.debug("Attempting to create JAXBContext with contextPath=" + contextpath);
}
context = JAXBContext_newInstance(contextpath, cl);
if (!containsClasses(context, classRefs)) {
if (log.isDebugEnabled()) {
log.debug(" Unsuccessful: Will now use an alterative JAXBConstruct construction");
}
return null;
}
if (log.isDebugEnabled()) {
log.debug(" Successfully created JAXBContext:" + context);
}
} catch (Throwable e) {
if (log.isDebugEnabled()) {
log.debug(
" Unsuccessful: We will now use an alterative JAXBConstruct construction");
log.debug(" Reason " + e.toString());
}
}
return context;
}
/**
* containsClasses
* @param JAXBContext
* @param List classRefs
*/
private static boolean containsClasses(JAXBContext context, List classRefs) {
String text = context.toString();
text = text.replace('\n', ' ');
text = text.replace('\t', ' ');
text = text.replace('\r', ' ');
text = text.replace('<', ' ');
text = text.replace('[', ' ');
text = text.replace(']', ' ');
for (String classRef: classRefs) {
// Strip off generic and array chars
int index = classRef.indexOf('<');
if (index > 0) {
classRef = classRef.substring(0, index);
}
index = classRef.indexOf('[');
if (index > 0) {
classRef = classRef.substring(0, index);
}
if (classRef.length() == 0 ||
classRef.endsWith(".ObjectFactory") ||
classRef.startsWith("java.util.") ||
classRef.startsWith("java.lang.")) {
// skip these
} else {
String search = " " + classRef + " ";
if (!text.contains(search)) {
if (log.isDebugEnabled()) {
log.debug("The context does not contain " + classRef + " " + context);
}
return false;
}
}
}
return true;
}
/**
* This method will return all the Class names needed to construct a JAXBContext
*
* @param pkg Package
* @param ClassLoader cl
* @return
* @throws ClassNotFoundException if error occurs getting package
*/
private static List getAllClassesFromPackage(String pkg, ClassLoader cl) {
if (log.isDebugEnabled()) {
log.debug("Start: getAllClassesFromPackage for " + pkg);
}
if (pkg == null) {
if (log.isDebugEnabled()) {
log.debug("End: getAllClassesFromPackage (package is null)");
}
return new ArrayList();
}
// See if this is a special package that has a set of known classes.
List knownClasses = specialMap.get(pkg);
if (knownClasses != null) {
if (log.isDebugEnabled()) {
try {
log.debug("End: getAllClassesFromPackage (package is special) returning: " + knownClasses);
} catch(Throwable t) {
// In case classes cannot be toString'd
log.debug("End: getAllClassesFromPackage (package is special)");
}
}
return knownClasses;
}
/*
* This method is a best effort method. We should always return an object.
*/
ArrayList classes = new ArrayList();
if (log.isDebugEnabled()) {
log.debug("Start: Obtain packages from similar directory");
}
try {
//
List classesFromDir = getClassesFromDirectory(pkg, cl);
checkClasses(classesFromDir, pkg);
classes.addAll(classesFromDir);
} catch (ClassNotFoundException e) {
if (log.isDebugEnabled()) {
log.debug("getClassesFromDirectory failed to get Classes");
}
}
if (log.isDebugEnabled()) {
log.debug("End: Obtain packages from similar directory");
log.debug("Start: Obtain packages from ClassFinder plugin");
}
try {
//This will load classes from jar file.
ClassFinderFactory cff =
(ClassFinderFactory)FactoryRegistry.getFactory(ClassFinderFactory.class);
ClassFinder cf = cff.getClassFinder();
List classesFromJar = cf.getClassesFromJarFile(pkg, cl);
checkClasses(classesFromJar, pkg);
classes.addAll(classesFromJar);
} catch (ClassNotFoundException e) {
if (log.isDebugEnabled()) {
log.debug("getClassesFromJarFile failed to get Classes");
}
}
if (log.isDebugEnabled()) {
log.debug("End: Obtain packages from ClassFinder plugin");
}
if (log.isDebugEnabled()) {
log.debug("End: Obtain packages from ClassFinder plugin");
}
if (log.isDebugEnabled()) {
try {
log.debug("End: getAllClassesFromPackage for " + pkg + "with classes " + classes);
} catch (Throwable e) {
// In case classes cannot be toString'd
log.debug("End: getAllClassesFromPackage for " + pkg );
}
}
return classes;
}
/**
* @param list
* @param pkg
*/
private static void checkClasses(List list, String pkg) {
// The installed classfinder or directory search may inadvertently add too many
// classes. This rountine is a 'double check' to make sure that the classes
// are acceptable.
for (int i=0; i getClassesFromDirectory(String pkg, ClassLoader cl)
throws ClassNotFoundException {
// This will hold a list of directories matching the pckgname. There may be more than one if a package is split over multiple jars/paths
String pckgname = pkg;
ArrayList directories = new ArrayList();
try {
String path = pckgname.replace('.', '/');
// Ask for all resources for the path
Enumeration resources = cl.getResources(path);
while (resources.hasMoreElements()) {
directories.add(new File(
URLDecoder.decode(resources.nextElement().getPath(), "UTF-8")));
}
} catch (UnsupportedEncodingException e) {
if (log.isDebugEnabled()) {
log.debug(
pckgname + " does not appear to be a valid package (Unsupported encoding)");
}
throw new ClassNotFoundException(Messages.getMessage("ClassUtilsErr2", pckgname));
} catch (IOException e) {
if (log.isDebugEnabled()) {
log.debug(
"IOException was thrown when trying to get all resources for " + pckgname);
}
throw new ClassNotFoundException(Messages.getMessage("ClassUtilsErr3", pckgname));
}
ArrayList classes = new ArrayList();
// For every directory identified capture all the .class files
for (File directory : directories) {
if (log.isDebugEnabled()) {
log.debug(" Adding JAXB classes from directory: " + directory.getName());
}
if (directory.exists()) {
// Get the list of the files contained in the package
String[] files = directory.list();
for (String file : files) {
// we are only interested in .class files
if (file.endsWith(".class")) {
// removes the .class extension
// TODO Java2 Sec
String className = pckgname + '.' + file.substring(0, file.length() - 6);
try {
Class clazz = forName(className,
false, getContextClassLoader());
// Don't add any interfaces or JAXWS specific classes.
// Only classes that represent data and can be marshalled
// by JAXB should be added.
if (!clazz.isInterface()
&& (clazz.isEnum() ||
getAnnotation(clazz, XmlType.class) != null ||
ClassUtils.getDefaultPublicConstructor(clazz) != null)
&& !ClassUtils.isJAXWSClass(clazz)
&& !isSkipClass(clazz)
&& !java.lang.Exception.class.isAssignableFrom(clazz)) {
// Ensure that all the referenced classes are loadable too
clazz.getDeclaredMethods();
clazz.getDeclaredFields();
if (log.isDebugEnabled()) {
log.debug("Adding class: " + file);
}
classes.add(clazz);
// REVIEW:
// Support of RPC list (and possibly other scenarios) requires that the array classes should also be present.
// This is a hack until we can determine how to get this information.
// The arrayName and loadable name are different. Get the loadable
// name, load the array class, and add it to our list
//className += "[]";
//String loadableName = ClassUtils.getLoadableClassName(className);
//Class aClazz = Class.forName(loadableName, false, Thread.currentThread().getContextClassLoader());
}
//Catch Throwable as ClassLoader can throw an NoClassDefFoundError that
//does not extend Exception
} catch (Throwable e) {
if (log.isDebugEnabled()) {
log.debug("Tried to load class " + className +
" while constructing a JAXBContext. This class will be skipped. Processing Continues.");
log.debug(" The reason that class could not be loaded:" +
e.toString());
log.trace(JavaUtils.stackToString(e));
}
}
}
}
}
}
return classes;
}
private static String[] commonArrayClasses = new String[] {
// primitives
"boolean[]",
"byte[]",
"char[]",
"double[]",
"float[]",
"int[]",
"long[]",
"short[]",
"java.lang.String[]",
// Others
"java.lang.Object[]",
"java.awt.Image[]",
"java.math.BigDecimal[]",
"java.math.BigInteger[]",
"java.util.Calendar[]",
"javax.xml.namespace.QName[]" };
private static void addCommonArrayClasses(List list) {
// Add common primitives arrays (necessary for RPC list type support)
ClassLoader cl = getContextClassLoader();
for (int i = 0; i < commonArrayClasses.length; i++) {
String className = commonArrayClasses[i];
try {
// Load and add the class
Class cls = forName(ClassUtils.getLoadableClassName(className), false, cl);
list.add(cls);
//Catch Throwable as ClassLoader can throw an NoClassDefFoundError that
//does not extend Exception
} catch (Throwable e) {
if (log.isDebugEnabled()) {
log.debug("Tried to load class " + className +
" while constructing a JAXBContext. This class will be skipped. Processing Continues.");
log.debug(" The reason that class could not be loaded:" + e.toString());
log.trace(JavaUtils.stackToString(e));
}
}
}
}
/** @return ClassLoader */
private static ClassLoader getContextClassLoader() {
// NOTE: This method must remain private because it uses AccessController
ClassLoader cl = null;
try {
cl = (ClassLoader)AccessController.doPrivileged(
new PrivilegedExceptionAction() {
public Object run() throws ClassNotFoundException {
return Thread.currentThread().getContextClassLoader();
}
}
);
} catch (PrivilegedActionException e) {
if (log.isDebugEnabled()) {
log.debug("Exception thrown from AccessController: " + e);
}
throw ExceptionFactory.makeWebServiceException(e.getException());
}
return cl;
}
/**
* @return true if clazz is a class that should be skipped (should not
* be a considered for JAXB)
*/
private static boolean isSkipClass(Class clazz) {
Class cls = clazz;
while (cls != null) {
// Check the name of the class to see if it should be excluded.
String clsName = cls.getCanonicalName();
if (clsName != null &&
(clsName.equals("javax.ejb.SessionBean") ||
clsName.equals("org.apache.axis2.jaxws.spi.JAXBExclude"))) {
if (log.isDebugEnabled()) {
log.debug("isSkipClass returns true for : " + clazz);
log.debug(" (It is skipped because the class extends " + clsName);
}
return true;
}
// Check the interfaces of the class to see if it should be excluded
Class[] intferfaces = getInterfaces_priv(cls);
if (intferfaces != null) {
for (int i=0; i
* @param ClassRefs List
* @return
* @throws Exception
*/
private static JAXBContext JAXBContext_newInstance(final Class[] classArray,
final ClassLoader cl,
Map properties,
List classRefs)
throws JAXBException {
// NOTE: This method must remain private because it uses AccessController
JAXBContext jaxbContext = null;
if (log.isDebugEnabled()) {
if (classArray == null || classArray.length == 0) {
log.debug("JAXBContext is constructed with 0 input classes.");
} else {
log.debug("JAXBContext is constructed with " + classArray.length +
" input classes.");
}
}
// Get JAXBContext from classes
jaxbContext = JAXBContextFromClasses.newInstance(classArray,
cl,
properties,
classRefs);
return jaxbContext;
}
/** Holds the JAXBContext and the manner by which it was constructed */
static class JAXBContextValue {
public JAXBContext jaxbContext;
public CONSTRUCTION_TYPE constructionType;
public JAXBContextValue(JAXBContext jaxbContext, CONSTRUCTION_TYPE constructionType) {
this.jaxbContext = jaxbContext;
this.constructionType = constructionType;
}
}
static private void adjustPoolSize(Map map) {
if (map.size() > MAX_LOAD_FACTOR) {
// Remove every other Entry in the map.
Iterator it = map.entrySet().iterator();
boolean removeIt = false;
while (it.hasNext()) {
it.next();
if (removeIt) {
it.remove();
}
removeIt = !removeIt;
}
}
}
/**
* Pool a list of items for a specific key
*
* @param Key
* @param Pooled object
*/
private static class Pool {
private SoftReference
© 2015 - 2024 Weber Informatics LLC | Privacy Policy