org.jace.proxy.BatchGenerator Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of jace-core-java Show documentation
Show all versions of jace-core-java Show documentation
Jace core module, Java source-code
package org.jace.proxy;
import org.jace.metaclass.ClassMetaClass;
import org.jace.metaclass.MetaClassFactory;
import org.jace.metaclass.TypeNameFactory;
import org.jace.parser.ClassFile;
import org.jace.proxy.ProxyGenerator.AcceptAll;
import org.jace.proxy.ProxyGenerator.AccessibilityType;
import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.util.Collections;
import java.util.jar.JarEntry;
import java.util.jar.JarInputStream;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* Batch generates C++ proxy classes. The BatchGenerator reads a Jar file,
* and generates a Jace proxy for each and every Java class in the file.
*
* We should allow for all sorts of options, but unfortunately time is pressing,
* so we'll only provide the basics for now.
*
* @author Toby Reyelts
* @author Gili Tzabari
*/
public class BatchGenerator
{
private static final String newLine = System.getProperty("line.separator");
private final Logger log = LoggerFactory.getLogger(BatchGenerator.class);
private final ClassPath classPath;
private final File outputHeaders;
private final File outputSources;
private final AccessibilityType accessibility;
/**
* Creates a new BatchGenerator
*
* @param classPath the path to search for class files when resolving class dependencies
* @param outputHeaders the directory to which the proxy header files should be written
* @param outputSources the directory to which the proxy source files should be written
* @param accessibility the class accessibility to expose
*/
public BatchGenerator(ClassPath classPath, File outputHeaders, File outputSources,
AccessibilityType accessibility)
{
if (classPath == null)
throw new IllegalArgumentException("classPath may not be null");
if (outputHeaders == null)
throw new IllegalArgumentException("outputHeaders may not be null");
if (outputSources == null)
throw new IllegalArgumentException("outputSources may not be null");
if (accessibility == null)
throw new IllegalArgumentException("accessibility may not be null");
if (!outputHeaders.isDirectory())
throw new IllegalArgumentException("outputHeaders must be a directory");
if (!outputSources.isDirectory())
throw new IllegalArgumentException("outputSources must be a directory");
this.classPath = classPath;
this.outputHeaders = outputHeaders;
this.outputSources = outputSources;
this.accessibility = accessibility;
}
/**
* Generates a proxy for every class in a jar file.
*
* @param file the jar file
* @throws IOException if an I/O error occurs while generating proxies
*/
private void generateFromJar(File jarFile) throws IOException
{
JarInputStream in = new JarInputStream(new FileInputStream(jarFile));
while (true)
{
JarEntry entry = in.getNextJarEntry();
if (entry == null)
break;
if (entry.isDirectory())
{
String dirName = entry.getName();
dirName = "jace" + File.separator + "proxy" + File.separator + dirName;
File headerDir = new File(outputHeaders + File.separator + dirName);
if (!headerDir.exists() && !headerDir.mkdirs())
throw new IOException("Cannot create " + headerDir.getAbsolutePath());
File sourceDir = new File(outputSources + File.separator + dirName);
if (!sourceDir.exists() && !sourceDir.mkdirs())
throw new IOException("Cannot create " + sourceDir.getAbsolutePath());
continue;
}
StringBuilder fullyQualifiedPath = new StringBuilder(entry.getName());
if (!fullyQualifiedPath.toString().endsWith(".class"))
continue;
// Remove the ".class" extension
fullyQualifiedPath.setLength(fullyQualifiedPath.length() - ".class".length());
// Fix for bug 607754. Stealing code from AutoProxy to do this right now
ClassMetaClass metaClass = (ClassMetaClass) MetaClassFactory.getMetaClass(
TypeNameFactory.fromPath(fullyQualifiedPath.toString())).proxy();
String classFileName = metaClass.getFileName();
File targetHeaderFile = new File(outputHeaders, classFileName + ".h");
File targetSourceFile = new File(outputSources, classFileName + ".cpp");
if (!(jarFile.lastModified() > targetSourceFile.lastModified() || jarFile.lastModified() > targetHeaderFile.
lastModified()))
{
// The source-file has not been modified since we last generated the
// target source/header files.
log.debug(jarFile + " has not been modified, skipping...");
continue;
}
ClassFile classFile = new ClassFile(in);
new ProxyGenerator.Builder(classPath, classFile, new AcceptAll()).accessibility(accessibility).
build().
writeProxy(outputHeaders, outputSources);
}
in.close();
}
/**
* Returns a String describing the usage of this tool.
*
* @return String describing the usage of this tool
*/
private static String getUsage()
{
return "Usage: BatchGenerator " + newLine
+ " " + newLine
+ " " + newLine
+ " [ options ]" + newLine + "Where options can be:" + newLine
+ " -public : Generate public fields and methods." + newLine
+ " -protected : Generate public, protected fields and methods." + newLine
+ " -package : Generate public, protected, package-private fields and methods."
+ newLine
+ " -private : Generate public, protected, package-private, private fields and methods."
+ newLine;
}
/**
* Runs the BatchGenerator using the specified options.
*
* @param args the command-line arguments
* @throws IOException if an I/O error occurs reading from the JAR file
*/
@SuppressWarnings("UseOfSystemOutOrSystemErr")
public static void main(String[] args) throws IOException
{
if (args.length < 3)
{
System.out.println(getUsage());
return;
}
AccessibilityType accessibility = AccessibilityType.PUBLIC;
for (int i = 3; i < args.length; ++i)
{
String option = args[i];
if (option.equals("-public"))
accessibility = AccessibilityType.PUBLIC;
else if (option.equals("-protected"))
accessibility = AccessibilityType.PROTECTED;
else if (option.equals("-package"))
accessibility = AccessibilityType.PACKAGE;
else if (option.equals("-private"))
accessibility = AccessibilityType.PRIVATE;
else
{
System.out.println("Not an understood option: [" + option + "]");
System.out.println();
System.out.println(getUsage());
return;
}
}
File jarFile = new File(args[0]);
new BatchGenerator(new ClassPath(Collections.singletonList(jarFile)), new File(args[1]),
new File(args[2]), accessibility).generateFromJar(jarFile);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy