Java tutorial
/* * The contents of this file are subject to the Mozilla Public License Version * 1.1 (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.mozilla.org/MPL/ * * Software distributed under the License is distributed on an "AS IS" basis, * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License * for the specific language governing rights and limitations under the * License. * * The Original Code is JSeance code, released * May 6, 2009. * * The Initial Developer of the Original Code is * Andres Murillo. * Portions created by the Initial Developer are Copyright (C) 2009 * the Initial Developer. All Rights Reserved. * * Alternatively, the contents of this file may be used under the terms of * the GNU General Public License Version 2 or later (the "GPL"), in which * case the provisions of the GPL are applicable instead of those above. If * you wish to allow use of your version of this file only under the terms of * the GPL and not to allow others to use your version of this file under the * MPL, indicate your decision by deleting the provisions above and replacing * them with the notice and other provisions required by the GPL. If you do * not delete the provisions above, a recipient may use your version of this * file under either the MPL or the GPL. */ package com.CodeSeance.JSeance2.CodeGenXML; import com.CodeSeance.JSeance2.CodeGenXML.DependencyTracking.DependencyManager; import com.CodeSeance.JSeance2.CodeGenXML.DependencyTracking.TemplateDependencies; import com.CodeSeance.JSeance2.CodeGenXML.EntryPoints.Logger; import com.CodeSeance.JSeance2.CodeGenXML.TemplateElements.Template; import org.apache.commons.logging.Log; import org.apache.commons.logging.LogFactory; import org.apache.log4j.PropertyConfigurator; import java.io.File; import java.io.IOException; import java.util.List; import java.util.Properties; /** This is the args4j class to define command line options for the executable Jar execution method also is the entry point for functional tests @author Andres Murillo @version 1.0 */ public class Runtime { public Runtime(String errorLogFileName, String infoLogFileName, String debugLogFileName, boolean ignoreReadOnlyOuputFiles, boolean forceRebuild) { this.errorLogFileName = errorLogFileName; this.infoLogFileName = infoLogFileName; this.debugLogFileName = debugLogFileName; this.ignoreReadOnlyOuputFiles = ignoreReadOnlyOuputFiles; this.forceRebuild = forceRebuild; ConfigureLogger(); } //Uses the specified filename for error logging private final String errorLogFileName; //Uses the specified filename for info logging private final String infoLogFileName; //Uses the specified filename for debugging private final String debugLogFileName; //Skips production of ouput files with readonly flag private final boolean ignoreReadOnlyOuputFiles; //Skips dependency checks and forces a rebuild private final boolean forceRebuild; // The dependency manager used to avoid unnecessary builds //private final DependencyManager dependencyManager; private void ConfigureLogger(Properties properties, String logName, String threshold, String fileName) { properties.setProperty(String.format("log4j.appender.%s", logName), "org.apache.log4j.FileAppender"); properties.setProperty(String.format("log4j.appender.%s.File", logName), fileName); properties.setProperty(String.format("log4j.appender.%s.Append", logName), "false"); properties.setProperty(String.format("log4j.appender.%s.layout", logName), "org.apache.log4j.PatternLayout"); properties.setProperty(String.format("log4j.appender.%s.layout.ConversionPattern", logName), "%-5p %30c - %m%n"); properties.setProperty(String.format("log4j.appender.%s.threshold", logName), threshold); } private void ConfigureLogger() { if (!logConfigured) { Properties log4Jproperties = new Properties(); // override the logger config log4Jproperties.setProperty("log4j.rootLogger", "DEBUG" + (errorLogFileName != null ? ", ErrorLog" : "") + (infoLogFileName != null ? ", InfoLog" : "") + (debugLogFileName != null ? ", DebugLog" : "")); // override the log filenames and append mode if (errorLogFileName != null) { ConfigureLogger(log4Jproperties, "ErrorLog", "WARN", errorLogFileName); } if (infoLogFileName != null) { ConfigureLogger(log4Jproperties, "InfoLog", "INFO", infoLogFileName); } if (debugLogFileName != null) { ConfigureLogger(log4Jproperties, "DebugLog", "DEBUG", debugLogFileName); } // Configure log4j PropertyConfigurator.configure(log4Jproperties); logConfigured = true; } } private static boolean logConfigured = false; public String run(File sourcesDir, File targetDir, List<File> templateFiles, Logger externalLog) { // Directory from where to load include files (relative to) File includesDir = new File(sourcesDir, "/includes"); //Directory from where to load model(xml) files (relative to) File modelsDir = new File(sourcesDir, "/models"); return run(includesDir, modelsDir, targetDir, templateFiles, externalLog); } private boolean errors = false; public boolean hasErrors() { return errors; } public String run(File includesDir, File modelsDir, File targetDir, List<File> templateFiles, Logger externalLog) { errors = false; Log log = LogFactory.getLog("Runtime"); StringBuffer buffer = new StringBuffer(); if (!((targetDir.exists() && targetDir.isDirectory()) || targetDir.mkdirs())) { String message = ExecutionError.INVALID_TARGET_DIR.getMessage(targetDir); externalLog.errorMessage(message); log.error(message); return null; } DependencyManager dependencyManager = new DependencyManager(targetDir); // access non-option arguments and generate the templates for (File templateFile : templateFiles) { TemplateDependencies templateDependencies = dependencyManager.getTemplateDependencies(templateFile); // Track the processing time externalLog.infoMessage(String.format("Processing template file:[%s]", templateFile.toString())); long startMillis = System.currentTimeMillis(); if (!dependencyManager.getTemplateDependencies(templateFile).isUpToDate() || forceRebuild) { dependencyManager.clearTemplateDependencies(templateFile); try { String result = run(templateFile, includesDir, modelsDir, targetDir, templateDependencies); buffer.append(result); dependencyManager.commit(); } catch (Exception ex) { errors = true; externalLog.errorMessage(ex.getMessage()); log.error(ex.getMessage()); } } else { String message = String.format( "File dependencies are up to date, skipping template generation:[%s]", templateFile); externalLog.infoMessage(message); log.info(message); } long elapsedMillis = System.currentTimeMillis() - startMillis; externalLog.infoMessage(String.format("Completed in :[%s] ms", elapsedMillis)); } return buffer.toString(); } private String run(File templateFile, File includesDir, File modelsDir, File targetDir, TemplateDependencies templateDependencies) { // Create a local logger for the static context Log log = com.CodeSeance.JSeance2.CodeGenXML.Runtime.CreateLogger(Runtime.class); if (log.isInfoEnabled()) { log.info(String.format("Loading Template:[%s]", templateFile.toString())); } Template template; try { template = new Template(templateFile); template.loadChildren(template, null); } catch (IOException ex) { throw new RuntimeException(ExecutionError.INVALID_TEMPLATE_FILE.getMessage(templateFile.toString()), ex); } if (log.isInfoEnabled()) { log.info("Template validated"); } // Create a new Context Context context = new Context(includesDir, modelsDir, targetDir, ignoreReadOnlyOuputFiles, templateDependencies); try { // Enter and leave the context on the new template element template.loadAttributes(context); template.onExecutionStart(context); template.onExecutionEnd(context); } finally { // dispose the context manager to release used resources context.dispose(); } return template.getText(); } public static Log CreateLogger(Class classObj) { return LogFactory.getLog(classObj.getName().replace("com.CodeSeance.JSeance2.CodeGenXML.", "")); } }