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

jaxx.compiler.JAXXCompiler Maven / Gradle / Ivy

/*
 * #%L
 * JAXX :: Compiler
 * 
 * $Id: JAXXCompiler.java 2379 2012-07-04 16:02:22Z tchemit $
 * $HeadURL: http://svn.nuiton.org/svn/jaxx/tags/jaxx-2.5.19/jaxx-compiler/src/main/java/jaxx/compiler/JAXXCompiler.java $
 * %%
 * Copyright (C) 2008 - 2010 CodeLutin
 * %%
 * This program is free software: you can redistribute it and/or modify
 * it under the terms of the GNU Lesser General Public License as 
 * published by the Free Software Foundation, either version 3 of the 
 * License, or (at your option) any later version.
 * 
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Lesser Public License for more details.
 * 
 * You should have received a copy of the GNU General Lesser Public 
 * License along with this program.  If not, see
 * .
 * #L%
 */

package jaxx.compiler;

import jaxx.compiler.binding.DataBindingHelper;
import jaxx.compiler.css.StylesheetHelper;
import jaxx.compiler.finalizers.JAXXCompilerFinalizer;
import jaxx.compiler.java.JavaElementFactory;
import jaxx.compiler.java.JavaField;
import jaxx.compiler.java.JavaFile;
import jaxx.compiler.java.JavaFileGenerator;
import jaxx.compiler.java.JavaMethod;
import jaxx.compiler.java.parser.ParseException;
import jaxx.compiler.reflect.ClassDescriptor;
import jaxx.compiler.reflect.ClassDescriptorHelper;
import jaxx.compiler.reflect.FieldDescriptor;
import jaxx.compiler.reflect.MethodDescriptor;
import jaxx.compiler.script.ScriptManager;
import jaxx.compiler.tags.DefaultObjectHandler;
import jaxx.compiler.tags.TagHandler;
import jaxx.compiler.tags.TagManager;
import jaxx.runtime.ComponentDescriptor;
import jaxx.runtime.JAXXObjectDescriptor;
import jaxx.runtime.css.Rule;
import jaxx.runtime.css.Stylesheet;
import org.apache.commons.lang3.StringUtils;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.nuiton.eugene.java.JavaGeneratorUtil;
import org.nuiton.eugene.java.extension.ImportsManager;
import org.w3c.dom.Document;
import org.w3c.dom.Element;
import org.xml.sax.Attributes;
import org.xml.sax.InputSource;
import org.xml.sax.Locator;
import org.xml.sax.SAXException;
import org.xml.sax.SAXParseException;
import org.xml.sax.helpers.AttributesImpl;
import org.xml.sax.helpers.XMLFilterImpl;

import javax.xml.parsers.ParserConfigurationException;
import javax.xml.parsers.SAXParser;
import javax.xml.parsers.SAXParserFactory;
import javax.xml.transform.ErrorListener;
import javax.xml.transform.Transformer;
import javax.xml.transform.TransformerConfigurationException;
import javax.xml.transform.TransformerException;
import javax.xml.transform.TransformerFactory;
import javax.xml.transform.dom.DOMResult;
import javax.xml.transform.sax.SAXSource;
import java.awt.BorderLayout;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException;
import java.io.InputStream;
import java.io.PrintWriter;
import java.io.StringWriter;
import java.io.UnsupportedEncodingException;
import java.lang.reflect.Modifier;
import java.net.URL;
import java.net.URLDecoder;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.Stack;

/**
 * Compiles a given {@link #jaxxFile} into a {@link #javaFile}.
 *
 * @author tchemit 
 * @since 1.0.0
 */
public class JAXXCompiler {

    /** Logger */
    protected static final Log log = LogFactory.getLog(JAXXCompiler.class);

    /**
     * True to throw exceptions when we encounter unresolvable classes,
     * false to ignore.
     * 

* This is currently set to false until JAXX has full support for * inner classes (including enumerations), because currently they don't * always resolve (but will generally run without error anyway). */ public static boolean STRICT_CHECKS; public static final String JAXX_NAMESPACE = "http://www.jaxxframework.org/"; public static final String JAXX_INTERNAL_NAMESPACE = "http://www.jaxxframework.org/internal"; /** Maximum length of an inlinable creation method. */ public static final int INLINE_THRESHOLD = 300; /** Line separator cached value */ protected static String lineSeparator = System.getProperty("line.separator", "\n"); public static final String BORDER_LAYOUT_PREFIX = BorderLayout.class.getSimpleName() + "."; /** The unique object handler used in first pass */ protected final DefaultObjectHandler firstPassClassTagHandler; /*------------------------------------------------------------------------*/ /*-- compiler fields -----------------------------------------------------*/ /*------------------------------------------------------------------------*/ /** flag to detect if an error occurs while compiling jaxx file */ protected boolean failed; /** Object corresponding to the root tag in the document. */ protected CompiledObject root; /** Contains strings of the form "javax.swing." */ protected Set importedPackages; /** Contains strings of the form "javax.swing.Timer" */ protected Set importedClasses; /** Keeps track of open components (components still having children added). */ protected Stack openComponents; /** to generate ids */ protected final IDHelper idHelper; /** Binding Util */ protected final DataBindingHelper bindingHelper; /** table of symbols for this compiler */ protected SymbolTable symbolTable; /** * Base directory used for path resolution (normally the directory in * which the .jaxx file resides). */ protected File baseDir; /** jaxx file being compiled. */ protected File src; /** Parsed XML of src file. */ protected Document document; /** Name of class being compiled. */ protected String outputClassName; /** script manager */ protected ScriptManager scriptManager; /** * Combination of all stylesheets registered using * {@link #registerStylesheet(Stylesheet)}. */ protected Stylesheet stylesheet; /** Contains all attributes defined inline on class tags. */ protected List inlineStyles; /** * Maps objects (expressed in Java code) to event listener classes * (e.g. MouseListener) to Lists of EventHandlers. The final list * contains all event handlers of a particular type attached to a * particular object (again, as represented by a Java expression). */ protected Map>> eventHandlers; /** Map of event handler method names used in compiler */ protected Map eventHandlerMethodNames; /** * ClassLoader which searches the user-specified class path in * addition to the normal class path */ protected ClassLoader classLoader; /** * A list of Runnables which will be run after the first compilation pass. *

* This is primarily used to trigger the creation of CompiledObjects, * which cannot be created during the first pass and must be created * in document order. */ protected List initializers; /** * extra interfaces which can by passed to root object via the * 'implements' attribute */ private String[] extraInterfaces; /** a flag to generate a abstract class */ private boolean abstractClass; /** the possible generic type of the class */ private String genericType; /** the possible generic type of the super class */ private String superGenericType; /** Extra code to be added to the instance initializer. */ protected StringBuilder initializer; /** Extra code to be added at the end of the instance initializer. */ protected StringBuilder lateInitializer; /** Extra code to be added to the class body. */ protected StringBuilder bodyCode; /** true if a main() method has been declared in a script */ protected boolean mainDeclared; /** the file to be generated */ protected JavaFile javaFile; /** Used for error reporting purposes, so we can report the right line number. */ protected Stack tagsBeingCompiled; /** Used for error reporting purposes, so we can report the right source file. */ protected Stack sourceFiles; /** * Maps object ID strings to the objects themselves. *

* These are created during the second compilation pass. */ protected Map objects; /** * Maps objects to their ID strings. *

* These are created during the second compilation pass. */ protected Map ids; /** default decodator to use if none specified */ protected CompiledObjectDecorator defaultDecorator; /** * engine which references this compiler * (can be null if compiler is standalone) */ protected final JAXXEngine engine; protected final CompilerConfiguration defaultConfiguration = new DefaultCompilerConfiguration(); protected final JAXXCompilerFile jaxxFile; /** * Flag to know if jaxx file ident css was found, otherwise add it * when {@link CompilerConfiguration#isAutoImportCss()} is sets to {@code true} * at the end of the second compile pass. * * @since 2.0.2 */ protected boolean identCssFound; /** * A flag to know if SwingUtil msut be imported. * * @since 2.4 * @deprecated since 2.4.1, will be removed in version 3.0, it is not a good * idea to do special treatment for a particular class, to use SwingUtil, * do like for other class : import it! */ @Deprecated protected boolean needSwingUtil; public static final String[] EMPTY_STRING_ARRAY = new String[0]; public JAXXCompiler() { this(null, null, null); } /** * Creates a new JAXXCompiler. * * @param engine engine which use the compiler (could be null if not attach to any engine) * @param jaxxFile the file to compile * @param defaultImports list of default imports to add to java files */ public JAXXCompiler(JAXXEngine engine, JAXXCompilerFile jaxxFile, List defaultImports) { this.engine = engine; this.jaxxFile = jaxxFile; ids = new LinkedHashMap(); objects = new LinkedHashMap(); bodyCode = new StringBuilder(); lateInitializer = new StringBuilder(); initializer = new StringBuilder(); tagsBeingCompiled = new Stack(); initializers = new ArrayList(); eventHandlerMethodNames = new HashMap(); eventHandlers = new HashMap>>(); inlineStyles = new ArrayList(); scriptManager = new ScriptManager(this); symbolTable = new SymbolTable(); openComponents = new Stack(); importedPackages = new HashSet(); importedClasses = new HashSet(); sourceFiles = new Stack(); if (jaxxFile == null) { src = null; baseDir = null; outputClassName = null; } else { src = jaxxFile.getJaxxFile(); baseDir = src.getParentFile(); outputClassName = jaxxFile.getClassName(); sourceFiles.push(src); addImport(outputClassName.substring( 0, outputClassName.lastIndexOf(".") + 1) + "*"); } firstPassClassTagHandler = new DefaultObjectHandler( ClassDescriptorHelper.getClassDescriptor(Object.class) ); bindingHelper = new DataBindingHelper(this); if (defaultImports != null) { for (String defaultImport : defaultImports) { addDefaultImport(defaultImport); } } if (engine != null) { CompilerConfiguration configuration = engine.getConfiguration(); // add extra imports from configuration if (configuration.getExtraImports() != null) { for (String extraImport : configuration.getExtraImports()) { addImport(extraImport); } } defaultDecorator = engine.getDecorator( configuration.getDefaultDecoratorClass()); if (defaultDecorator == null) { throw new IllegalArgumentException( "could not find default decorator : " + configuration.getDefaultDecoratorClass() ); } idHelper = new IDHelper(configuration.isOptimize()); } else { idHelper = new IDHelper(false); } } /*------------------------------------------------------------------------*/ /*-- Initializer methods -------------------------------------------------*/ /*------------------------------------------------------------------------*/ public void runInitializers() { for (Runnable runnable : initializers) { if (log.isDebugEnabled()) { log.debug(runnable); } try { runnable.run(); } catch (Exception e) { //TC - 20081018 report error and quit reportError(e.getMessage()); return; } } initializers.clear(); } /** * Registers a Runnable which will be executed after the first * compilation pass is complete. * * @param r runnable to register */ public void registerInitializer(Runnable r) { initializers.add(r); } /*------------------------------------------------------------------------*/ /*-- Compile methods -----------------------------------------------------*/ /*------------------------------------------------------------------------*/ public void compileFirstPass(final Element tag) throws IOException { tagsBeingCompiled.push(tag); String namespace = tag.getNamespaceURI(); String fullClassName; String localName = tag.getLocalName(); boolean namespacePrefix = tag.getPrefix() != null; // resolve class tags into fully-qualified class name if (namespace != null && namespace.endsWith("*")) { String packageName = namespace.substring(0, namespace.length() - 1); if (localName.startsWith(packageName)) { // class name is fully-qualified already fullClassName = TagManager.resolveClassName(localName, this); } else { // namespace not included in class name, probably need the // namespace to resolve fullClassName = TagManager.resolveClassName( packageName + localName, this); if (fullClassName == null && !namespacePrefix) { // it was just a default namespace, try again without // using the namespace fullClassName = TagManager.resolveClassName(localName, this); } } } else { fullClassName = TagManager.resolveClassName(localName, this); } if (fullClassName != null) { // we are definitely dealing with a class tag addDependencyClass(fullClassName); namespace = fullClassName.substring( 0, fullClassName.lastIndexOf(".") + 1) + "*"; if (symbolTable.getSuperclassName() == null) { symbolTable.setSuperclassName(fullClassName); } String id = tag.getAttribute(DefaultObjectHandler.ID_ATTRIBUTE); MethodDescriptor methodDescriptor; if (!id.isEmpty()) { symbolTable.getClassTagIds().put(id, fullClassName); if (tag.getAttributeNode(DefaultObjectHandler.JAVA_BEAN_ATTRIBUTE) != null) { // add java bean support for this property String capitalizeName = StringUtils.capitalize(id); // add method methodDescriptor = new MethodDescriptor( "get" + capitalizeName, Modifier.PUBLIC, fullClassName, EMPTY_STRING_ARRAY, classLoader ); symbolTable.getScriptMethods().add(methodDescriptor); if (Boolean.class.getName().equals(fullClassName)) { methodDescriptor = new MethodDescriptor( "is" + capitalizeName, Modifier.PUBLIC, fullClassName, EMPTY_STRING_ARRAY, classLoader ); symbolTable.getScriptMethods().add(methodDescriptor); } methodDescriptor = new MethodDescriptor( "set" + capitalizeName, Modifier.PUBLIC, JAXXCompilerFinalizer.TYPE_VOID, new String[]{fullClassName}, classLoader ); symbolTable.getScriptMethods().add(methodDescriptor); } else { // add simple get support String capitalizeName = StringUtils.capitalize(id); // add method methodDescriptor = new MethodDescriptor( "get" + capitalizeName, Modifier.PUBLIC, fullClassName, EMPTY_STRING_ARRAY, classLoader ); symbolTable.getScriptMethods().add(methodDescriptor); } } String interfacesStr = tag.getAttribute(DefaultObjectHandler.IMPLEMENTS_ATTRIBUTE); if (!interfacesStr.isEmpty()) { // there is some interfaces to deal with try { String[] interfaces = JavaGeneratorUtil.splitFqnList(interfacesStr, ','); if (log.isDebugEnabled()) { log.debug("detect interfaces : " + Arrays.toString(interfaces)); } symbolTable.setInterfaces(interfaces); } catch (Exception e) { // the syntax is not valid (missed one >) throw new CompilerException( "Syntax error of interfaces " + interfacesStr); } // StringTokenizer stk = new StringTokenizer(interfacesStr, ","); // List tmp = new ArrayList(); // while (stk.hasMoreTokens()) { // String c = stk.nextToken(); // if (c.contains("<") && !c.contains(">")) { // // deal with a generic interface with more than one // // parameter // boolean done = false; // while (stk.hasMoreTokens()) { // // String next = stk.nextToken(); // if (!next.contains(">")) { // // still a parameter of the generic type // continue; // } // // ok find the ending token // done = true; // break; // } // if (!done) { // // the syntax is not valid (missed one >) // throw new CompilerException( // "Syntax error of interfaces " + // interfacesStr); // } // //c = c.substring(0, c.indexOf("<")); // } // tmp.add(c.trim()); // } // String[] interfaces = tmp.toArray(new String[tmp.size()]); // if (log.isDebugEnabled()) { // log.debug("detect interfaces : " + // Arrays.toString(interfaces)); // } // symbolTable.setInterfaces(interfaces); } } // during the first pass, we can't create ClassDescriptors for JAXX // files because they may not have been processed yet (and we can't // wait until they have been processed because of circular // dependencies). // So we don't do any processing during the first pass which requires // having a ClassDescriptor; here we determine whether we have a class // tag or not (class tag namespaces end in "*") and use a generic // handler if so. // The real handler is used during the second pass. TagHandler handler = namespace != null && namespace.endsWith("*") ? firstPassClassTagHandler : TagManager.getTagHandler(tag.getNamespaceURI(), localName, namespacePrefix, this ); if (!firstPassClassTagHandler.equals(handler) && handler instanceof DefaultObjectHandler) { fullClassName = ((DefaultObjectHandler) handler).getBeanClass().getName(); handler = firstPassClassTagHandler; } if (firstPassClassTagHandler.equals(handler)) { final String finalClassName = fullClassName; // register an initializer which will create the // CompiledObject after pass 1 registerInitializer(new Runnable() { @Override public void run() { DefaultObjectHandler handler = (DefaultObjectHandler) TagManager.getTagHandler( null, finalClassName, JAXXCompiler.this ); if (handler == null) { throw new CompilerException( "Internal error: missing TagHandler for '" + finalClassName + "'"); } handler.registerCompiledObject(tag, JAXXCompiler.this); } }); } if (handler == null) { reportError("Could not find a Java class corresponding to: <" + tag.getTagName() + ">"); failed = true; } else { try { handler.compileFirstPass(tag, this); } catch (CompilerException e) { reportError(e); } } Element finished = tagsBeingCompiled.pop(); if (finished != tag) { throw new IllegalStateException( "internal error: just finished compiling " + tag + ", but top of tagsBeingCompiled stack is " + finished); } } public void compileSecondPass(Element tag) throws IOException { tagsBeingCompiled.push(tag); TagHandler handler = TagManager.getTagHandler( tag.getNamespaceURI(), tag.getLocalName(), tag.getPrefix() != null, this ); if (handler == null) { reportError("Could not find a Java class corresponding to: <" + tag.getTagName() + ">"); failed = true; } else { handler.compileSecondPass(tag, this); } Element finished = tagsBeingCompiled.pop(); if (!tag.equals(finished)) { throw new RuntimeException( "internal error: just finished compiling " + tag + ", but top of tagsBeingCompiled stack is " + finished); } } public void compileFirstPass() throws IOException { InputStream in = new FileInputStream(src); try { document = parseDocument(in); compileFirstPass(document.getDocumentElement()); } catch (SAXParseException e) { reportError(e.getLineNumber(), "Invalid XML: " + e.getMessage()); } catch (SAXException e) { reportError(null, "Error parsing XML document: " + e); } finally { in.close(); } } public void compileSecondPass() throws IOException { if (!tagsBeingCompiled.isEmpty()) { throw new RuntimeException( "Internal error: starting pass two, but tagsBeingCompiled" + " is not empty: " + tagsBeingCompiled); } compileSecondPass(document.getDocumentElement()); } /*------------------------------------------------------------------------*/ /*-- CompiledObject methods ----------------------------------------------*/ /*------------------------------------------------------------------------*/ public void openComponent(CompiledObject component) throws CompilerException { openComponent(component, null); } public void openComponent(CompiledObject component, String constraints) throws CompilerException { if (constraints != null) { // try to add the constraints class in imports if (constraints.startsWith(BORDER_LAYOUT_PREFIX)) { addImport(BorderLayout.class.getName()); } } CompiledObject parent = getOpenComponent(); openInvisibleComponent(component); if (parent != null && !component.isOverride()) { parent.addChild(component, constraints, this); } } public void openInvisibleComponent(CompiledObject component) { if (!ids.containsKey(component)) { registerCompiledObject(component); } openComponents.push(component); } public CompiledObject getOpenComponent() { if (openComponents.isEmpty()) { return null; } return openComponents.peek(); } public void closeComponent(CompiledObject component) { if (component == null) { throw new NullPointerException("can not close a null component"); } if (!component.equals(openComponents.pop())) { throw new IllegalArgumentException( "can only close the topmost open object"); } } public void registerCompiledObject(CompiledObject object) { // assert engine.symbolTables.values().contains(symbolTable) : // "attempting to register CompiledObject before pass 1 is complete"; if (root == null) { root = object; } String id = object.getId(); if (ids.containsKey(object)) { reportError("object '" + object + "' is already registered with id '" + ids.get(object) + "', cannot re-register as '" + id + "'" ); } if (objects.containsKey(id) && !(objects.get(id) instanceof Element)) { reportError("id '" + id + "' is already registered to component " + objects.get(id)); } objects.put(id, object); ids.put(object, id); if (object.getDecorator() == null) { // use default decorator object.setDecorator(defaultDecorator); } } public CompiledObject getCompiledObject(String id) { runInitializers(); // assert engine.symbolTables.values().contains(symbolTable) : // "attempting to retrieve CompiledObject before pass 1 is complete"; return objects.get(id); } public boolean inlineCreation(CompiledObject object) { return object.getId().startsWith("$") && object.getInitializationCode(this).length() < INLINE_THRESHOLD; } public void checkOverride(CompiledObject object) throws CompilerException { String fieldName = object.getId(); if (fieldName.startsWith("$")) { return; } ClassDescriptor ancestor = root.getObjectClass(); if (object.getObjectClass().equals(ancestor)) { return; } while (ancestor != null) { try { FieldDescriptor f = ancestor.getDeclaredFieldDescriptor(fieldName); if (!f.getType().isAssignableFrom(object.getObjectClass())) { reportError( "attempting to redefine superclass member '" + fieldName + "' as incompatible type (was " + f.getType() + ", redefined as " + object.getObjectClass() + ")" ); } object.setOverride(true); object.setOverrideType(f.getType()); if (!getCanonicalName(f.getType()).equals(getCanonicalName(object))) { // types are not the same String simpleType = getImportedType(getCanonicalName(object)); if (log.isDebugEnabled()) { log.debug("Simple type for " + object.getId() + " : " + getCanonicalName(object) + " against : " + getCanonicalName(f.getType())); } object.setSimpleType(simpleType); } break; } catch (NoSuchFieldException e) { if (log.isDebugEnabled()) { log.debug(">>>>> could not find declared field [" + fieldName + "] in " + ancestor.getName()); } ancestor = ancestor.getSuperclass(); } } } /*------------------------------------------------------------------------*/ /*-- DataBinding methods -------------------------------------------------*/ /*------------------------------------------------------------------------*/ public DataBindingHelper getBindingHelper() { return bindingHelper; } public void registerEventHandler(EventHandler handler) { String objectCode = handler.getObjectCode(); Map> listeners = eventHandlers.get(objectCode); if (listeners == null) { listeners = new HashMap>(); eventHandlers.put(objectCode, listeners); } ClassDescriptor listenerClass = handler.getListenerClass(); List handlerList = listeners.get(listenerClass); if (handlerList == null) { handlerList = new ArrayList(); listeners.put(listenerClass, handlerList); } handlerList.add(handler); } public String getEventHandlerMethodName(EventHandler handler) { String result = eventHandlerMethodNames.get(handler); if (result == null) { if (getConfiguration().isOptimize()) { result = "$ev" + eventHandlerMethodNames.size(); } else { //TC-20090309 must get the goal property from the event id // to make possible inheritance String id = handler.getEventId().substring( 0, handler.getEventId().indexOf(".")); result = "do" + StringUtils.capitalize(handler.getListenerMethod().getName()) + "__on__" + id; //TC-20091105 : check the method name is available // using css can have multi methods with result name (see priority) if (eventHandlerMethodNames.containsValue(result)) { int index = 0; String result2; do { result2 = result + "_" + index++; } while (eventHandlerMethodNames.containsValue(result2)); result = result2; } } eventHandlerMethodNames.put(handler, result); } return result; } /*------------------------------------------------------------------------*/ /*-- Script methods ------------------------------------------------------*/ /*------------------------------------------------------------------------*/ public void addScriptField(FieldDescriptor field) { symbolTable.getScriptFields().add(field); } public void addScriptMethod(MethodDescriptor method) { if (method.getName().equals("main") && method.getParameterTypes().length == 1 && method.getParameterTypes()[0].getName().equals("[Ljava.lang.String;")) { setMainDeclared(true); } symbolTable.getScriptMethods().add(method); } public void registerScript(String script) throws CompilerException { registerScript(script, null); } public void registerScript(String script, File sourceFile) throws CompilerException { if (sourceFile != null) { sourceFiles.push(sourceFile); } script = script.trim(); if (!"".equals(script) && !script.endsWith("}") && !script.endsWith(";")) { script += ";"; } scriptManager.registerScript(script); if (sourceFile != null) { File pop = sourceFiles.pop(); if (!sourceFile.equals(pop)) { throw new RuntimeException( "leaving registerScript(), but " + sourceFile + " was not the top entry on the stack (found " + pop + " instead)"); } } } public String preprocessScript(String script) throws CompilerException { return scriptManager.preprocessScript(script); } /*------------------------------------------------------------------------*/ /*-- StyleSheet methods --------------------------------------------------*/ /*------------------------------------------------------------------------*/ // public File getIdentCssFile() { // String extension = FileUtil.extension(src); // String jaxxFileName = src.getName(); // String identCssFilename = jaxxFileName.substring( // 0, // jaxxFileName.length() - extension.length()) + "css"; // File identCssFile = new File(src.getParentFile(), identCssFilename); // return identCssFile; // } public boolean isIdentCssFound() { return identCssFound; } public void registerStyleSheetFile(File styleFile) throws IOException { if (!identCssFound && getConfiguration().isAutoImportCss()) { // detects if the given css file is ident to jaxx file File identCssFile = jaxxFile.getCssFile(); if (identCssFile.exists()) { // ok found ident css file identCssFound = true; reportWarning("autoImportCss mode is on, you can remove " + "style declaration with source " + styleFile); } } String content = loadFile(styleFile); // String content = StylesheetHelper.loadCssFile(this, styleFile); getSourceFiles().push(styleFile); try { Stylesheet stylesheet = StylesheetHelper.processStylesheet(content); registerStylesheet(stylesheet); } finally { // whatever could be result, must pop this source file getSourceFiles().pop(); } } public void applyStylesheets() { for (Object o : new ArrayList(objects.values())) { CompiledObject object = (CompiledObject) o; DefaultObjectHandler tagHandler = TagManager.getTagHandler(object.getObjectClass()); if (log.isDebugEnabled()) { log.debug("will apply css on object " + object.getId() + " from handler " + tagHandler); } tagHandler.applyStylesheets(object, this); } } public void registerStylesheet(Stylesheet stylesheet) { if (this.stylesheet == null) { this.stylesheet = stylesheet; } else { this.stylesheet.add(stylesheet.getRules()); } } public void addInlineStyle(CompiledObject object, String propertyName, boolean dataBinding) { Rule style = StylesheetHelper.inlineAttribute( object, propertyName, dataBinding ); inlineStyles.add(style); } /*------------------------------------------------------------------------*/ /*-- Report methods ------------------------------------------------------*/ /*------------------------------------------------------------------------*/ public void reportWarning(String warning) { Element currentTag = null; if (!tagsBeingCompiled.isEmpty()) { currentTag = tagsBeingCompiled.peek(); } reportWarning(currentTag, warning, 0); } public void reportWarning(Element tag, String warning, int lineOffset) { String lineNumber = null; if (tag != null) { String lineAttr = tag.getAttributeNS(JAXX_INTERNAL_NAMESPACE, "line"); if (lineAttr.length() > 0) { lineNumber = lineAttr; } } File srcFile = sourceFiles.peek(); try { srcFile = srcFile.getCanonicalFile(); } catch (IOException e) { // ignore ? } StringBuilder buffer = new StringBuilder(); buffer.append(srcFile); if (lineNumber != null) { buffer.append(":"); buffer.append(sourceFiles.size() == 1 ? Integer.parseInt(lineNumber) + lineOffset : lineOffset + 1 ); } buffer.append(getLineSeparator()).append(warning.trim()); if (engine != null) { engine.addWarning(buffer.toString()); } else { System.err.println(buffer.toString()); } } public void reportError(String error) { Element currentTag = null; if (!tagsBeingCompiled.isEmpty()) { currentTag = tagsBeingCompiled.peek(); } reportError(currentTag, error); } public void reportError(CompilerException ex) { reportError(null, ex); } public void reportError(String extraMessage, CompilerException ex) { String message = ex.getMessage(); if (UnsupportedAttributeException.class.equals(ex.getClass()) || UnsupportedTagException.class.equals(ex.getClass())) { String exceptionName = ex.getClass().getName(); message = exceptionName.substring( exceptionName.lastIndexOf(".") + 1) + ": " + message; } int lineOffset; if (ex instanceof ParseException) { lineOffset = Math.max(0, ((ParseException) ex).getLine() - 1); } else { lineOffset = 0; } Element currentTag = null; if (!tagsBeingCompiled.isEmpty()) { currentTag = tagsBeingCompiled.peek(); } reportError(currentTag, extraMessage != null ? extraMessage + message : message, lineOffset ); } public void reportError(Element tag, String error) { reportError(tag, error, 0); } public void reportError(Element tag, String error, int lineOffset) { int lineNumber = 0; if (tag != null) { String lineAttr = tag.getAttributeNS(JAXX_INTERNAL_NAMESPACE, "line"); if (lineAttr.length() > 0) { lineNumber = Integer.parseInt(lineAttr); } } lineNumber = Math.max(lineNumber, 1) + lineOffset; reportError(lineNumber, error); } public void reportError(int lineNumber, String error) { File errorFile = sourceFiles.isEmpty() ? null : sourceFiles.peek(); try { if (errorFile != null) { errorFile = errorFile.getCanonicalFile(); } } catch (IOException e) { // ignore ? } StringBuilder buffer = new StringBuilder(); buffer.append(errorFile != null ? errorFile.getPath() : ""); if (lineNumber > 0) { buffer.append(":").append(lineNumber); } buffer.append(getLineSeparator()).append(": ").append(error); if (engine != null) { engine.addError(buffer.toString()); } else { System.err.println(buffer.toString()); } failed = true; } /*------------------------------------------------------------------------*/ /*-- Getter methods ------------------------------------------------------*/ /*------------------------------------------------------------------------*/ public Map getObjects() { return objects; } public Map>> getEventHandlers() { return eventHandlers; } public CompilerConfiguration getConfiguration() { JAXXEngine engine = getEngine(); if (engine == null) { return defaultConfiguration; } return engine.getConfiguration(); } public String getOutputClassName() { return outputClassName; } public File getBaseDir() { return baseDir; } public Set getImportedClasses() { return importedClasses; } public Set getImportedPackages() { return importedPackages; } public Iterator getObjectCreationOrder() { return objects.values().iterator(); } public CompiledObject getRootObject() { return root; } public Stack getSourceFiles() { return sourceFiles; } public ScriptManager getScriptManager() { return scriptManager; } public SymbolTable getSymbolTable() { return symbolTable; } public Stylesheet getStylesheet() { Stylesheet merged = new Stylesheet(); if (stylesheet != null) { merged.add(stylesheet.getRules()); } merged.add(inlineStyles.toArray(new Rule[inlineStyles.size()])); return merged; } public FieldDescriptor[] getScriptFields() { List scriptFields = symbolTable.getScriptFields(); return scriptFields.toArray( new FieldDescriptor[scriptFields.size()]); } public FieldDescriptor getScriptField(String fieldName) { for (FieldDescriptor f : symbolTable.getScriptFields()) { if (fieldName.equals(f.getName())) { return f; } } return null; } public MethodDescriptor[] getScriptMethods() { List scriptMethods = symbolTable.getScriptMethods(); return scriptMethods.toArray( new MethodDescriptor[scriptMethods.size()]); } public MethodDescriptor getScriptMethod(String methodName) { for (MethodDescriptor m : symbolTable.getScriptMethods()) { if (methodName.equals(m.getName())) { return m; } } return null; } public boolean isFailed() { return failed; } public IDHelper getIdHelper() { return idHelper; } /** * Returns a ClassLoader which searches the user-specified * class path in addition to the normal system class path. * * @return ClassLoader to use while resolving class references */ public ClassLoader getClassLoader() { if (classLoader == null) { CompilerConfiguration configuration = getConfiguration(); if (configuration.getClassLoader() != null) { classLoader = configuration.getClassLoader(); } else { throw new NullPointerException( "compiler configuration requires a classLoader! :\n" + configuration); } } return classLoader; } /** * Checks if the super class of the mirrored javaFile is aware of the * iven {@code type}. *

* Note: If no super-class exist, then returns * {@code false}. * * @param type the type to check against super class * @return {@code true} if super class exists and is assignable against the * given type, {@code false} otherwise * @throws ClassNotFoundException if could not find class descriptor for * super-class */ public boolean isSuperClassAware(Class type) throws ClassNotFoundException { ClassDescriptor superClass = root.getObjectClass(); if (superClass == null) { // no super class, not awre of anything return false; } boolean aware = ClassDescriptorHelper.isAssignableFrom(superClass, type); return aware; } public JAXXObjectDescriptor getJAXXObjectDescriptor() { runInitializers(); CompiledObject[] components = new ArrayList( objects.values()).toArray(new CompiledObject[objects.size()]); assert initializers.isEmpty() : "there are pending initializers remaining"; assert root != null : "root object has not been defined"; assert Arrays.asList(components).contains(root) : "root object is not registered"; ComponentDescriptor[] descriptors = new ComponentDescriptor[components.length]; // as we print, sort the array so that component's parents are always // before the components themselves for (int i = 0; i < components.length; i++) { CompiledObject parent = components[i].getParent(); while (parent != null) { boolean found = false; for (int j = i + 1; j < components.length; j++) { // found parent after component, swap them if (components[j] == parent) { components[j] = components[i]; components[i] = parent; found = true; break; } } if (!found) { break; } parent = components[i].getParent(); } int parentIndex = -1; if (parent != null) { for (int j = 0; j < i; j++) { if (components[j].equals(parent)) { parentIndex = j; break; } } } descriptors[i] = new ComponentDescriptor( components[i].getId(), components[i] == root ? outputClassName : components[i].getObjectClass().getName(), components[i].getStyleClass(), parentIndex != -1 ? descriptors[parentIndex] : null); } Stylesheet css = getStylesheet(); if (css == null) { css = new Stylesheet(); } return new JAXXObjectDescriptor(descriptors, css); } /*------------------------------------------------------------------------*/ /*-- Buffer --------------------------------------------------------------*/ /*------------------------------------------------------------------------*/ public StringBuilder getInitializer() { return initializer; } public StringBuilder getLateInitializer() { return lateInitializer; } public StringBuilder getBodyCode() { return bodyCode; } public boolean isMainDeclared() { return mainDeclared; } public void setMainDeclared(boolean mainDeclared) { this.mainDeclared = mainDeclared; } public void appendInitializerCode(String code) { initializer.append(code); } public void appendBodyCode(String code) { bodyCode.append(code); } public void appendLateInitializer(String code) { lateInitializer.append(code); } public boolean hasMethod(String methodName) { for (JavaMethod method : getJavaFile().getMethods()) { if (methodName.equals(method.getName())) { return true; } } return false; } /*------------------------------------------------------------------------*/ /*-- Other methods -------------------------------------------------------*/ /*------------------------------------------------------------------------*/ public void addImport(Class clazz) { addImport(clazz.getName()); } public void addImport(String text) { if (text.endsWith("*")) { importedPackages.add(text.substring(0, text.length() - 1)); } else { importedClasses.add(text); } if (!text.equals("*")) { getJavaFile().addImport(text); } } protected void addDefaultImport(String text) { if (text.endsWith("*")) { importedPackages.add(text.substring(0, text.length() - 1)); } else { importedClasses.add(text); } } public void addDependencyClass(String className) { if (engine.containsJaxxFileClassName(className)) { // already registred in engine return; } // register the jaxx file into the engine URL jaxxURL = ClassDescriptorHelper.getURL(getClassLoader(), className, "jaxx"); if (jaxxURL == null) { // not a jaxx file, can quit now return; } if (!engine.isCompileFirstPassTask()) { throw new IllegalStateException( "Internal error: adding dependency class " + className + " during second compilation pass"); } //FIXME tchemit 20100518 Must unify this with ClassDescriptorHelper to use the less costless resource if (jaxxURL.toString().startsWith("file:")) { File jaxxFile = URLtoFile(jaxxURL); try { jaxxFile = jaxxFile.getCanonicalFile(); } catch (IOException ex) { // ignore ? (at least log it...) log.error(ex); } String jaxxFileName = className.substring(className.lastIndexOf(".") + 1) + ".jaxx"; assert jaxxFile.getName().equalsIgnoreCase(jaxxFileName) : "expecting file name to match " + className + ", but found " + jaxxFile.getName(); if (jaxxFile.getName().equals(jaxxFileName)) { engine.addFileToCompile(jaxxFile, className); } } } /** * Loads the given file and returns its content. *

* Note: If any errors encounters, then report an error in the * compiler and returns an empty string. * * @param file the file to load * @return the content of the file or empty string if something was wrong */ public String loadFile(File file) { if (!file.exists()) { reportError("Could not found file " + file); return ""; } try { FileReader in = new FileReader(file); try { StringWriter styleBuffer = new StringWriter(); char[] readBuffer = new char[2048]; int c; while ((c = in.read(readBuffer)) > 0) { styleBuffer.write(readBuffer, 0, c); } return styleBuffer.toString(); } finally { in.close(); } } catch (IOException e) { reportError("Could not read file " + file + " for reason " + e.getMessage()); return ""; } } /** * Verifies that a snippet of Java code parses correctly. *

* A warning is generated if the string has enclosing curly braces. * * @param javaCode the Java code snippet to test * @return a "cooked" version of the string which has enclosing curly * braces removed. * @throws CompilerException if the code cannot be parsed */ public String checkJavaCode(String javaCode) { javaCode = scriptManager.trimScript(javaCode); scriptManager.checkParse(javaCode); return javaCode; } /** * Check that a reference exists in symbol table on second compil pass * * @param tag the current tag * @param reference the required reference * @param strict flag to report an error if reference was not found * @param attribute (if not null reference the attribute where is defined * the reference) * @return true if reference was found, false * otherwise and add an error in compiler */ public boolean checkReference(Element tag, String reference, boolean strict, String attribute) { String component = getSymbolTable().getClassTagIds().get(reference); if (component == null) { if (strict) { String msg; if (attribute != null) { msg = "tag '" + tag.getLocalName() + "' could not find the reference '" + reference + "' on attribute [" + attribute + "]"; } else { msg = "tag '" + tag.getLocalName() + "' could not find the reference '" + reference + "'"; } reportError(msg); } return false; } return true; } public String getAutoId(String name) { return idHelper.nextId(name); } public String getUniqueId(Object object) { return idHelper.getUniqueId(object); } public void setExtraInterfaces(String[] extraInterfaces) { this.extraInterfaces = extraInterfaces; } public String[] getExtraInterfaces() { return extraInterfaces; } public boolean isAbstractClass() { return abstractClass; } public void setAbstractClass(boolean abstractClass) { this.abstractClass = abstractClass; } public String getGenericType() { return genericType; } public void setGenericType(String genericType) { this.genericType = genericType; } public String getSuperGenericType() { return superGenericType; } public void setSuperGenericType(String superGenericType) { this.superGenericType = superGenericType; } public void addSimpleField(JavaField javaField) { getJavaFile().addSimpleField(javaField); } public JavaFile getJavaFile() { if (javaFile == null) { String outputClassName = getOutputClassName(); if (outputClassName == null) { javaFile = JavaElementFactory.newFile(0, ""); } else { int dotPos = outputClassName.lastIndexOf("."); String packageName = dotPos != -1 ? outputClassName.substring(0, dotPos) : null; String simpleClassName = outputClassName.substring(dotPos + 1); javaFile = JavaElementFactory.newFile( Modifier.PUBLIC, packageName + "." + simpleClassName); } } return javaFile; } public void preFinalizeCompiler() throws Exception { } public void finalizeCompiler() throws Exception { int dotPos = getOutputClassName().lastIndexOf("."); String packageName = dotPos != -1 ? getOutputClassName().substring(0, dotPos) : null; String simpleClassName = getOutputClassName().substring(dotPos + 1); CompiledObject compiledObject = getRootObject(); String genericType = getGenericType(); if (StringUtils.isNotEmpty(genericType)) { // add the generic type to the root object // generictype can be on form E extends XXX //keep only the first thing... StringBuilder sb = new StringBuilder(); String[] allTypes = JavaGeneratorUtil.splitFqnList(genericType, ','); for (String type : allTypes) { int anExtends = type.indexOf("extends"); if (anExtends > -1) { type = type.substring(0, anExtends - 1).trim(); } sb.append(", ").append(type); } String finalType; if (allTypes.length > 0) { finalType = sb.substring(2); } else { finalType = sb.toString(); } compiledObject.setGenericTypes(finalType); } // finalize all objects via their decorator for (CompiledObject object : getObjects().values()) { CompiledObjectDecorator decorator = object.getDecorator(); decorator.finalizeCompiler(this, root, object, javaFile, packageName, simpleClassName, getOutputClassName()); } // obtain list of finalizers to apply List realFinalizers = new ArrayList(); for (JAXXCompilerFinalizer finalizer : getConfiguration().getFinalizers().values()) { if (finalizer.accept(this)) { realFinalizers.add(finalizer); } } // call the finalizers finalizeCompiler method for (JAXXCompilerFinalizer finalizer : realFinalizers) { // check if finalizer can be apply of this compiler if (finalizer.accept(this)) { finalizer.finalizeCompiler(compiledObject, this, javaFile, packageName, simpleClassName ); } } // call the compiled objects finalizeCompiler method for (CompiledObject object : getObjects().values()) { object.finalizeCompiler(this); } // compile bindings getBindingHelper().finalizeBindings(); // call the finalizers prepareJavaFile method for (JAXXCompilerFinalizer finalizer : realFinalizers) { finalizer.prepareJavaFile(compiledObject, this, javaFile, packageName, simpleClassName ); } } public void generate(JavaFileGenerator generator) throws IOException { File dest; String fqn = getOutputClassName(); if (getConfiguration().getTargetDirectory() != null) { dest = new File(getConfiguration().getTargetDirectory(), fqn.replace('.', File.separatorChar) + ".java" ); } else { dest = new File(getBaseDir(), fqn.substring(fqn.lastIndexOf(".") + 1) + ".java" ); } // make sure directory exists File parentFile = dest.getParentFile(); if (parentFile == null) { throw new IOException("No parent file for " + dest); } if (!parentFile.exists() && !parentFile.mkdirs()) { throw new IOException("Could not create directory " + parentFile); } if (dest.exists() && !dest.setLastModified(System.currentTimeMillis())) { log.warn("could not touch file " + dest); } PrintWriter out = new PrintWriter(new FileWriter(dest)); try { generator.generateFile(javaFile, out); } finally { out.close(); } } /** * Returns the system line separator string. * * @return the string used to separate lines */ public static String getLineSeparator() { return lineSeparator; } // 1.5 adds getCanonicalName; unfortunately we can't depend on 1.5 features yet public static String getCanonicalName(Class clazz) { if (clazz.isArray()) { String canonicalName = getCanonicalName(clazz.getComponentType()); if (canonicalName != null) { return canonicalName + "[]"; } return null; } return clazz.getName().replace('$', '.'); } public static String getCanonicalName(ClassDescriptor clazz) { if (clazz.isArray()) { String canonicalName = getCanonicalName(clazz.getComponentType()); if (canonicalName != null) { return canonicalName + "[]"; } return null; } return clazz.getName().replace('$', '.'); } public static String getCanonicalName(CompiledObject compiled) { ClassDescriptor clazz = compiled.getObjectClass(); if (clazz.isArray()) { String canonicalName = getCanonicalName(clazz.getComponentType()); if (canonicalName != null) { if (compiled.getGenericTypesLength() > 0) { canonicalName += compiled.getGenericTypes(); } return canonicalName + "[]"; } return null; } String canonicalName = clazz.getName().replace('$', '.'); if (compiled.getGenericTypesLength() > 0) { canonicalName += compiled.getGenericTypes(); } return canonicalName; } /** * Escapes a string using standard Java escape sequences, generally in * preparation to including it in a string literal in a compiled Java file. * * @param raw the raw string to be escape * @return a string in which all 'dangerous' characters have been replaced * by equivalent Java escape sequences */ public static String escapeJavaString(String raw) { StringBuilder out = new StringBuilder(raw); for (int i = 0; i < out.length(); i++) { char c = out.charAt(i); if (c == '\\' || c == '"') { out.insert(i, '\\'); i++; } else if (c == '\n') { out.replace(i, i + 1, "\\n"); i++; } else if (c == '\r') { out.replace(i, i + 1, "\\r"); i++; } else if (c < 32 || c > 127) { String value = Integer.toString((int) c, 16); while (value.length() < 4) { value = "0" + value; } out.replace(i, i + 1, "\\u" + value); i += 5; } } return out.toString(); } public static File URLtoFile(URL url) { return URLtoFile(url.toString()); } public static File URLtoFile(String urlString) { if (!urlString.startsWith("file:")) { throw new IllegalArgumentException("url must start with 'file:'"); } urlString = urlString.substring("file:".length()); if (urlString.startsWith("/") && System.getProperty("os.name").startsWith("Windows")) { urlString = urlString.substring(1); } try { return new File(URLDecoder.decode( urlString.replace('/', File.separatorChar), "utf-8")); } catch (UnsupportedEncodingException e) { throw new RuntimeException(e); } } public static SAXParser getSAXParser() { try { SAXParserFactory factory = SAXParserFactory.newInstance(); factory.setNamespaceAware(true); SAXParser parser; parser = factory.newSAXParser(); return parser; } catch (SAXException e) { throw new RuntimeException(e); } catch (ParserConfigurationException e) { throw new RuntimeException(e); } } public static Document parseDocument(InputStream in) throws IOException, SAXException { try { TransformerFactory factory = TransformerFactory.newInstance(); Transformer transformer = factory.newTransformer(); transformer.setErrorListener(new ErrorListener() { @Override public void warning( TransformerException ex) throws TransformerException { throw ex; } @Override public void error( TransformerException ex) throws TransformerException { throw ex; } @Override public void fatalError( TransformerException ex) throws TransformerException { throw ex; } }); DOMResult result = new DOMResult(); transformer.transform( new SAXSource(new XMLFilterImpl(getSAXParser().getXMLReader()) { Locator locator; @Override public void setDocumentLocator(Locator locator) { this.locator = locator; } @Override public void startElement(String uri, String localName, String qName, Attributes atts) throws SAXException { AttributesImpl resultAtts = new AttributesImpl(atts); resultAtts.addAttribute( JAXX_INTERNAL_NAMESPACE, "line", "internal:line", "CDATA", String.valueOf(locator.getLineNumber()) ); getContentHandler().startElement(uri, localName, qName, resultAtts ); } }, new InputSource(in)), result); return (Document) result.getNode(); } catch (TransformerConfigurationException e) { throw new RuntimeException(e); } catch (TransformerException e) { Throwable ex = e; while (ex.getCause() != null) { ex = ex.getCause(); } if (ex instanceof IOException) { throw (IOException) ex; } if (ex instanceof SAXException) { throw (SAXException) ex; } if (ex instanceof RuntimeException) { throw (RuntimeException) ex; } throw new RuntimeException(ex); } } public JAXXEngine getEngine() { return engine; } public void clear() { idHelper.clear(); bindingHelper.clear(); objects.clear(); ids.clear(); if (symbolTable != null) { symbolTable.clear(); } } public void setIdentCssFound(boolean identCssFound) { this.identCssFound = identCssFound; } public void setClassLoader(ClassLoader classLoader) { this.classLoader = classLoader; } /** * Try to import the given type into the underlying java file of this compiler. *

* If import can not be done, will then returns the fully qualified name of * the type. * * @param type the type to simplify * @return the simplify type or the fqn if type could not be imported for the underlying java file. * @since 2.4.1 */ public String getImportedType(Class type) { return getJavaFile().getImportedType(type); } /** * Try to import the given type into the underlying java file of this compiler. *

* If import can not be done, will then returns the fully qualified name of * the type. * * @param type the fqn of the type to simplify * @return the simplify type or the fqn if type could not be imported for the underlying java file. * @since 2.4.1 */ public String getImportedType(String type) { return getJavaFile().getImportedType(type); } /** * @return the javafile import manager * @since 2.4 * @deprecated since 2.4.1, will be removed in version 3.0 : do not want * to expose eugene export manager in rest of api, this is purpose of * JavaFile only */ @Deprecated public ImportsManager getImportManager() { return getJavaFile().getImportManager(); } /** * @return * @since 2.4 * @deprecated since 2.4.1, will be removed in version 3.0, has no effect do not use it... */ @Deprecated public boolean isNeedSwingUtil() { return needSwingUtil; } /** * @param needSwingUtil * @since 2.4 * @deprecated since 2.4.1, will be removed in version 3.0, has no effect do not use it... */ @Deprecated public void setNeedSwingUtil(boolean needSwingUtil) { this.needSwingUtil = needSwingUtil; } public boolean containsScriptField(String fieldName) { FieldDescriptor[] scriptFields = getScriptFields(); for (FieldDescriptor f : scriptFields) { if (fieldName.equals(f.getName())) { return true; } } return false; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy