/*
* Copyright 2007 Joachim Grueneis
*
* Licensed 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.exolab.castor.xml.util.resolvers;
import java.io.IOException;
import java.io.InputStream;
import java.net.URL;
import java.util.ArrayList;
import java.util.Enumeration;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Properties;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.exolab.castor.mapping.ClassDescriptor;
import org.exolab.castor.xml.ResolverException;
import org.exolab.castor.xml.XMLConstants;
/**
* Resolves a class by reading the package level class-descriptor-resolver file
* and loading all descriptors mentioned inside.
*
* @author <a href="mailto:jgrueneis AT gmail DOT com">Joachim Grueneis</a>
* @author <a href="mailto:stevendolg AT gxm DOT at">Steven Dolg</a>
* @version $Revision$ $Date$
* @since 1.2
*/
public class ByCDR extends AbstractResolverPackageCommand {
private static final Log LOG = LogFactory.getLog(ByCDR.class);
private List<String> _loadedPackages = new ArrayList<String>();
/**
* No specific stuff needed.
*/
public ByCDR() {
super();
}
/**
* Creates a Properties object and initializes it with the contents of
* the given URL. The provided URL must exist and be suitable for loading
* properties from.
*
* @param url The URL to load the properties from. This must not be null!
* @return The loaded properties.
* @throws IOException If loading the properties from the given ULR failed.
*
* @see Properties
* @see Properties#load(InputStream)
*/
private Properties getProperties(URL url) throws java.io.IOException {
Properties cdrList = new Properties();
java.io.InputStream stream = url.openStream();
cdrList.load(stream);
stream.close();
return cdrList;
}
/**
* Tries to load the CDR file for the given package name using the provided
* class loader. If the CDR file is available and could be loaded properly the
* descriptors listed in it are added to this cache.
* <br>
* If a descriptor is listed in the CDR file for the given package but
* could not be loaded (e.g. because the reference class file is not
* available) the descriptor is ignored but no exception is thrown.
* <br>
* If a CDR file is not available for the given package this method will
* not load any descriptors and not throw any exceptions.
* <br>
* Further calls to this method with the same package name will not be
* processed.
* <br>
* {@inheritDoc}
*/
protected Map<String, ClassDescriptor> internalResolve(final String packageName, final ClassLoader classLoader,
final Map properties) throws ResolverException {
Map<String, ClassDescriptor> results = new HashMap<String, ClassDescriptor>();
if (!isEmptyPackageName(packageName) && _loadedPackages.contains(packageName)) {
if (LOG.isDebugEnabled()) {
LOG.debug("Package: " + packageName + " has already been loaded.");
}
return results;
}
if (!isEmptyPackageName(packageName)) {
_loadedPackages.add(packageName);
}
URL url = classLoader.getResource(ResolveHelpers.getQualifiedFileName(
XMLConstants.PKG_CDR_LIST_FILE, packageName));
if (url == null) {
return results;
}
try {
Properties cdrList = this.getProperties(url);
for (Object clazz : cdrList.keySet()) {
String clazzName = (String) clazz;
String descriptorClassName = (String) cdrList.get(clazzName);
try {
Class<?> descriptorClass = classLoader.loadClass(descriptorClassName);
if (LOG.isDebugEnabled()) {
LOG.debug("Found descriptor: " + descriptorClass);
}
if (descriptorClass != null) {
ClassDescriptor instance = (ClassDescriptor) descriptorClass.newInstance();
results.put(clazzName, instance);
} else if (LOG.isDebugEnabled()) {
LOG.debug("Loading of descriptor class: " + descriptorClassName
+ " for class: " + clazzName + " has failed - continue without");
}
} catch (Exception e) {
if (LOG.isDebugEnabled()) {
LOG.debug("Ignored problem at loading of: " + descriptorClassName
+ " with exception: " + e);
}
}
}
} catch (IOException iox) {
String message = "Failed to load package: " + packageName + " with exception: " + iox;
LOG.warn(message);
throw new ResolverException(message);
}
return results;
}
}