All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.jace.proxy.ClassPath Maven / Gradle / Ivy

There is a newer version: 1.2.22
Show newest version
package org.jace.proxy;

import com.google.common.collect.Lists;
import org.jace.metaclass.ArrayMetaClass;
import org.jace.metaclass.ClassMetaClass;
import org.jace.metaclass.MetaClass;
import org.jace.metaclass.MetaClassFactory;
import org.jace.metaclass.TypeName;
import org.jace.metaclass.TypeNameFactory;
import org.jace.parser.ClassFile;
import org.jace.util.WildcardFileFilter;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.InputStream;
import java.util.Arrays;
import java.util.List;
import java.util.zip.ZipEntry;
import java.util.zip.ZipFile;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/**
 * A source for class files.
 *
 * @author Toby Reyelts
 */
public class ClassPath
{
	private final List elements;
	private final Logger log = LoggerFactory.getLogger(ClassPath.class);

	/**
	 * Creates a new ClassPath.
	 *
	 * @param elements the paths to be used to search for classes
	 * @throws IllegalArgumentException if elements is null
	 */
	public ClassPath(List elements)
	{
		if (elements == null)
			throw new IllegalArgumentException("classPath may not be null");
		this.elements = elements;
	}

	/**
	 * Creates a new ClassPath.
	 *
	 * @param text the String representation of the classpath
	 * @throws IllegalArgumentException if text is null
	 */
	public ClassPath(String text)
	{
		if (text == null)
			throw new IllegalArgumentException("text may not be null");
		this.elements = Lists.newArrayList();
		List classPathArray = Arrays.asList(text.split(File.pathSeparator));
		for (String path: classPathArray)
		{
			if (path.contains("*") || path.contains("?"))
			{
				String normalizedPath = path.replace(File.separator, "/");
				int index = normalizedPath.lastIndexOf('/');
				String directory;
				String filename;
				if (index == -1)
				{
					directory = ".";
					filename = normalizedPath;
				}
				else
				{
					directory = normalizedPath.substring(0, index);
					filename = normalizedPath.substring(index + "/".length());
				}
				if (directory.contains("*") || directory.contains("?"))
				{
					throw new IllegalArgumentException("classpath directories may not contain wildcards");
				}
				WildcardFileFilter filter = new WildcardFileFilter(filename);
				File[] files = new File(directory).listFiles(filter);
				elements.addAll(Arrays.asList(files));
			}
			else
				elements.add(new File(path));
		}
	}

	/**
	 * Returns the first match for name in the class path. The class may be packed within
	 * a zip or jar file.
	 *
	 * @param name the class name
	 * @return the directory, zip or jar file containing the class file. Null if the class was not found.
	 * @throws IOException if an I/O error occurs while reading a zip or jar file
	 */
	public File getFirstMatch(TypeName name) throws IOException
	{
		if (log.isTraceEnabled())
			log.trace("getFirstMatch(" + name + ")");
		for (File path: elements)
		{
			if (!path.exists())
				continue;
			if (log.isTraceEnabled())
				log.trace("Checking " + path);

			// if the file is a directory, search for the .class file in the appropriate subfolder
			if (path.isDirectory())
			{
				MetaClass metaClass = MetaClassFactory.getMetaClass(name);
				String directory = metaClass.getPackage().toName("/", false);
				File subDirectory = new File(path.getAbsolutePath(), directory);

				if (log.isTraceEnabled())
					log.trace("Looking for directory " + subDirectory);
				if (subDirectory.exists())
				{
					String fileName = ((ClassMetaClass) metaClass).getTrueName() + ".class";
					File classFile = new File(subDirectory, fileName);

					if (log.isTraceEnabled())
						log.trace("Looking for file " + classFile);
					if (classFile.exists())
						return classFile.getParentFile();
				}
				continue;
			}

			// Search inside zip or jar files
			String fileName = path.getName();
			if (!path.getName().endsWith(".zip") && !path.getName().endsWith(".jar"))
				continue;
			if (log.isTraceEnabled())
				log.trace("Checking compressed file " + fileName);
			ZipFile zipFile = new ZipFile(path);
			MetaClass metaClass = MetaClassFactory.getMetaClass(name);
			if (metaClass instanceof ArrayMetaClass)
				metaClass = ((ArrayMetaClass) metaClass).getInnermostElementType();
			String entryName = ((ClassMetaClass) metaClass).getFullyQualifiedTrueName("/") + ".class";
			if (log.isTraceEnabled())
				log.trace("Looking for entry " + entryName);
			ZipEntry entry = zipFile.getEntry(entryName);
			if (entry != null)
				return path;
		}
		return null;
	}

	/**
	 * Reads the supplied class path looking for the class name.
	 * Returns the InputStream to the class. (The class may be packed within
	 * a Zip or Jar file).
	 *
	 * @param name this may be any of the canonical styles of class naming
	 * @return the InputStream to the class
	 * @throws ClassNotFoundException if no matching class can be found.
	 * @throws IOException if an error occurs while locating the class file
	 */
	public InputStream openClass(TypeName name) throws ClassNotFoundException, IOException
	{
		if (log.isTraceEnabled())
			log.trace("openClass(" + name + ")");
		// TODO: reuse result from getFirstMatch() to pick up from where we left off
		for (File path: elements)
		{
			if (!path.exists())
				continue;
			if (log.isTraceEnabled())
				log.trace("checking " + path);

			// if the file is a directory, search for the .class file in the appropriate subfolder
			if (path.isDirectory())
			{
				MetaClass metaClass = MetaClassFactory.getMetaClass(name);
				String packagePath = metaClass.getPackage().toName("/", false);
				File subDirectory = new File(path.getAbsolutePath(), packagePath);

				if (log.isTraceEnabled())
					log.trace("Looking for directory " + subDirectory);

				if (subDirectory.exists())
				{
					String fileName = ((ClassMetaClass) metaClass).getTrueName() + ".class";
					File classFile = new File(subDirectory, fileName);

					if (log.isTraceEnabled())
						log.trace("Looking for file " + classFile);
					if (classFile.exists())
					{
						try
						{
							return new FileInputStream(classFile);
						}
						catch (FileNotFoundException e)
						{
							// this shouldn't happen, because we just checked
							throw new RuntimeException("The file, " + fileName + " could not be found.");
						}
					}
				}
				continue;
			}

			// the file is a Jar or Zip file, so we need to search for the entry within the file
			String fileName = path.getName();

			if (!path.getName().endsWith(".zip") && !path.getName().endsWith(".jar"))
			{
				// not a zip file
				continue;
			}
			if (log.isTraceEnabled())
				log.trace("Checking compressed file " + fileName);
			ZipFile zipFile = new ZipFile(path);
			MetaClass metaClass = MetaClassFactory.getMetaClass(name);
			if (metaClass instanceof ArrayMetaClass)
				metaClass = ((ArrayMetaClass) metaClass).getInnermostElementType();
			String entryName = ((ClassMetaClass) metaClass).getFullyQualifiedTrueName("/") + ".class";
			if (log.isTraceEnabled())
				log.trace("Looking for entry " + entryName);
			ZipEntry entry = zipFile.getEntry(entryName);

			if (entry != null)
			{
				// This is kinda nasty, right here. We should really be closing the ZipFile and we're not.
				// There are a couple of different approaches we could take.
				//
				// 1) Read the ZipEntry into a byte[] and return that as a ByteArrayInputStream
				// so we can close the file immediately - Disadvantage - more memory, slower.
				// Advantage - pretty simple.
				//
				// 2) Create a wrapper InputStream that upon close, would close the ZipFile.
				// Advantage - no copying required. Disadvantage - Wrappering takes a minor performance hit.
				return zipFile.getInputStream(entry);
			}
		}
		throw new ClassNotFoundException("The class, " + name
																		 + " could not be found on the class path, "
																		 + elements);
	}

	@Override
	public String toString()
	{
		return getClass().getName() + "[" + elements + "]";
	}

	/**
	 * Tests ClassPath.
	 *
	 * @param args the command-line argument
	 */
	@SuppressWarnings("UseOfSystemOutOrSystemErr")
	public static void main(String[] args)
	{
		String classPath = args[0];
		ClassPath source = new ClassPath(classPath);
		try
		{
			InputStream input = source.openClass(TypeNameFactory.fromPath(args[1]));
			ClassFile classFile = new ClassFile(input);
			System.out.println(classFile.toString());
			input.close();
		}
		catch (ClassNotFoundException e)
		{
			Logger log = LoggerFactory.getLogger(ClassPath.class);
			log.error("", e);
		}
		catch (IOException e)
		{
			Logger log = LoggerFactory.getLogger(ClassPath.class);
			log.error("", e);
		}
	}
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy