AbstractClassPathRepository.java
/*
* 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.bcel.util;
import java.io.IOException;
import java.io.InputStream;
import org.apache.bcel.classfile.ClassParser;
import org.apache.bcel.classfile.JavaClass;
import org.apache.bcel.classfile.Utility;
/**
* This abstract class provides a logic of a loading {@link JavaClass} objects class names via {@link ClassPath}.
*
* <p>
* Subclasses can choose caching strategy of the objects by implementing the abstract methods (e.g.,
* {@link #storeClass(JavaClass)} and {@link #findClass(String)}).
* </p>
*
* @since 6.4.0
*/
abstract class AbstractClassPathRepository implements Repository {
private final ClassPath classPath;
AbstractClassPathRepository(final ClassPath classPath) {
this.classPath = classPath;
}
@Override
public abstract void clear();
@Override
public abstract JavaClass findClass(final String className);
@Override
public ClassPath getClassPath() {
return classPath;
}
/**
* Finds the JavaClass object for a runtime Class object. If a class with the same name is already in this Repository,
* the Repository version is returned. Otherwise, getResourceAsStream() is called on the Class object to find the
* class's representation. If the representation is found, it is added to the Repository.
*
* @see Class
* @param clazz the runtime Class object
* @return JavaClass object for given runtime class
* @throws ClassNotFoundException if the class is not in the Repository, and its representation could not be found
*/
@Override
public JavaClass loadClass(final Class<?> clazz) throws ClassNotFoundException {
final String className = clazz.getName();
final JavaClass repositoryClass = findClass(className);
if (repositoryClass != null) {
return repositoryClass;
}
String name = className;
final int i = name.lastIndexOf('.');
if (i > 0) {
name = name.substring(i + 1);
}
try (InputStream clsStream = clazz.getResourceAsStream(name + JavaClass.EXTENSION)) {
return loadClass(clsStream, className);
} catch (final IOException e) {
return null;
}
}
private JavaClass loadClass(final InputStream inputStream, final String className) throws ClassNotFoundException {
try {
if (inputStream != null) {
final ClassParser parser = new ClassParser(inputStream, className);
final JavaClass clazz = parser.parse();
storeClass(clazz);
return clazz;
}
} catch (final IOException e) {
throw new ClassNotFoundException("Exception while looking for class " + className + ": " + e, e);
}
throw new ClassNotFoundException("ClassRepository could not load " + className);
}
/**
* Finds a JavaClass object by name. If it is already in this Repository, the Repository version is returned. Otherwise,
* the Repository's classpath is searched for the class (and it is added to the Repository if found).
*
* @param className the name of the class
* @return the JavaClass object
* @throws ClassNotFoundException if the class is not in the Repository, and could not be found on the classpath
*/
@Override
public JavaClass loadClass(String className) throws ClassNotFoundException {
if (className == null || className.isEmpty()) {
throw new IllegalArgumentException("Invalid class name " + className);
}
className = Utility.pathToPackage(className); // Just in case, canonical form
final JavaClass clazz = findClass(className);
if (clazz != null) {
return clazz;
}
try (InputStream inputStream = classPath.getInputStream(className)) {
return loadClass(inputStream, className);
} catch (final IOException e) {
throw new ClassNotFoundException("Exception while looking for class " + className + ": " + e, e);
}
}
@Override
public abstract void removeClass(final JavaClass javaClass);
@Override
public abstract void storeClass(final JavaClass javaClass);
}