wait for my mail. git-svn-id: https://svn.apache.org/repos/asf/ant/core/trunk@271546 13f79535-47bb-0310-9956-ffa450edef68master
| @@ -0,0 +1 @@ | |||
| build | |||
| @@ -0,0 +1,56 @@ | |||
| Ant's current CVS version contains an <input> task that gathers user | |||
| input by reading from System.in - this is not too nice for people | |||
| embedding Ant in IDEs. 8-) | |||
| <input> also supports an undocumented testinput attribute that is used | |||
| by Ant's test cases to allow them to run without user interaction, but | |||
| could also be used to provide predefined answers to unattended builds. | |||
| This proposal tries to define a very basic input framework for Ant | |||
| that would allow Ant to be easily embedded into IDEs via | |||
| implementations of the org.apache.tools.ant.input.InputHandler | |||
| interface. At the same time an implementation of the interface is | |||
| provided that allows the input to be specified via an external | |||
| property file. | |||
| There are three implementations of the InputHandler interface, | |||
| DefaultInputHandler which reads form System.in just like the <input> | |||
| task originally did, PropertyFileInputHandler for non-interactive | |||
| builds and SwingInputHandler which is nothing more than a proof of | |||
| concept. | |||
| Input requests get encapsulated in instances of the | |||
| org.apache.tools.ant.input.InputRequest class - or subclasses thereof | |||
| - which provide a method to also validate the input, moving this | |||
| responsibility from the <input> task to the InputRequest itself. | |||
| There are two types of InputRequests ATM, InputRequest encapsulates a | |||
| request for a simple unrestricted text input, | |||
| MultipleChoiceInputRequest is a request where valid inputs are | |||
| restricted to a given set of values. | |||
| If you run ant on the build file in this directory, a version of | |||
| ant.jar will be created in the build subdirectory that is identical to | |||
| the main trunk of Ant except for the input task itself and two minor | |||
| changes to Project and Main, that allow InputHandlers to be plugged in | |||
| programmatically or via a commandline switch -inputhandler. | |||
| With this version of Ant, run the build file in proposals/testcases, | |||
| Ant should behave the same way the input task for the main branch does | |||
| - except that it won't allow you to enter invalid input in the multi | |||
| target. | |||
| If you invoke Ant like this: | |||
| ant -f proposal/sandbox/input/src/testcases/input.xml -inputhandler org.apache.tools.ant.input.SwingInputHandler | |||
| You'll get the ugliest dialog you've ever seen, but it works ;-) | |||
| Finally, use | |||
| ANT_OPTS=-Dinput.properties=proposal/sandbox/input/src/testcases/works.properties | |||
| ant -f proposal/sandbox/input/src/testcases/input.xml -inputhandler org.apache.tools.ant.input.PropertyFileInputHandler | |||
| to see the non-interactive build process in action. fails.properties | |||
| provides a sample of possible input failures. | |||
| @@ -0,0 +1,26 @@ | |||
| <project name="input-sandbox" default="main" basedir="."> | |||
| <target name="setup"> | |||
| <property name="main.ant" location="../../.." /> | |||
| <property name="classes.dir" value="build/classes" /> | |||
| <mkdir dir="${classes.dir}" /> | |||
| <ant dir="${main.ant}" inheritall="false" target="build" /> | |||
| <copy toDir="${classes.dir}" preservelastmodified="true" > | |||
| <fileset dir="${main.ant}/${classes.dir}"> | |||
| <include name='**' /> | |||
| <exclude name='org/apache/tools/ant/Project.class' /> | |||
| <exclude name='org/apache/tools/ant/Main.class' /> | |||
| <exclude name='org/apache/tools/ant/taskdefs/Input.class' /> | |||
| </fileset> | |||
| </copy> | |||
| </target> | |||
| <target name="main" depends="setup"> | |||
| <javac srcdir="src/main" destdir="${classes.dir}" /> | |||
| <jar destfile="build/ant.jar"> | |||
| <fileset dir="${classes.dir}"> | |||
| <exclude name="**/optional/*" /> | |||
| </fileset> | |||
| </jar> | |||
| </target> | |||
| </project> | |||
| @@ -0,0 +1,897 @@ | |||
| /* | |||
| * The Apache Software License, Version 1.1 | |||
| * | |||
| * Copyright (c) 2000-2002 The Apache Software Foundation. All rights | |||
| * reserved. | |||
| * | |||
| * Redistribution and use in source and binary forms, with or without | |||
| * modification, are permitted provided that the following conditions | |||
| * are met: | |||
| * | |||
| * 1. Redistributions of source code must retain the above copyright | |||
| * notice, this list of conditions and the following disclaimer. | |||
| * | |||
| * 2. Redistributions in binary form must reproduce the above copyright | |||
| * notice, this list of conditions and the following disclaimer in | |||
| * the documentation and/or other materials provided with the | |||
| * distribution. | |||
| * | |||
| * 3. The end-user documentation included with the redistribution, if | |||
| * any, must include the following acknowlegement: | |||
| * "This product includes software developed by the | |||
| * Apache Software Foundation (http://www.apache.org/)." | |||
| * Alternately, this acknowlegement may appear in the software itself, | |||
| * if and wherever such third-party acknowlegements normally appear. | |||
| * | |||
| * 4. The names "The Jakarta Project", "Ant", and "Apache Software | |||
| * Foundation" must not be used to endorse or promote products derived | |||
| * from this software without prior written permission. For written | |||
| * permission, please contact apache@apache.org. | |||
| * | |||
| * 5. Products derived from this software may not be called "Apache" | |||
| * nor may "Apache" appear in their names without prior written | |||
| * permission of the Apache Group. | |||
| * | |||
| * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED | |||
| * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES | |||
| * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | |||
| * DISCLAIMED. IN NO EVENT SHALL THE APACHE SOFTWARE FOUNDATION OR | |||
| * ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, | |||
| * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT | |||
| * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF | |||
| * USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | |||
| * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, | |||
| * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT | |||
| * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF | |||
| * SUCH DAMAGE. | |||
| * ==================================================================== | |||
| * | |||
| * This software consists of voluntary contributions made by many | |||
| * individuals on behalf of the Apache Software Foundation. For more | |||
| * information on the Apache Software Foundation, please see | |||
| * <http://www.apache.org/>. | |||
| */ | |||
| package org.apache.tools.ant; | |||
| import org.apache.tools.ant.input.DefaultInputHandler; | |||
| import org.apache.tools.ant.input.InputHandler; | |||
| import java.io.File; | |||
| import java.io.FileInputStream; | |||
| import java.io.PrintStream; | |||
| import java.io.FileOutputStream; | |||
| import java.io.IOException; | |||
| import java.io.InputStream; | |||
| import java.util.Vector; | |||
| import java.util.Properties; | |||
| import java.util.Enumeration; | |||
| /** | |||
| * Command line entry point into Ant. This class is entered via the | |||
| * cannonical `public static void main` entry point and reads the | |||
| * command line arguments. It then assembles and executes an Ant | |||
| * project. | |||
| * <p> | |||
| * If you integrating Ant into some other tool, this is not the class | |||
| * to use as an entry point. Please see the source code of this | |||
| * class to see how it manipulates the Ant project classes. | |||
| * | |||
| * @author duncan@x180.com | |||
| */ | |||
| public class Main { | |||
| /** The default build file name. */ | |||
| public final static String DEFAULT_BUILD_FILENAME = "build.xml"; | |||
| /** Our current message output status. Follows Project.MSG_XXX. */ | |||
| private int msgOutputLevel = Project.MSG_INFO; | |||
| /** File that we are using for configuration. */ | |||
| private File buildFile; /* null */ | |||
| /** Stream to use for logging. */ | |||
| private PrintStream out = System.out; | |||
| /** Stream that we are using for logging error messages. */ | |||
| private PrintStream err = System.err; | |||
| /** The build targets. */ | |||
| private Vector targets = new Vector(5); | |||
| /** Set of properties that can be used by tasks. */ | |||
| private Properties definedProps = new Properties(); | |||
| /** Names of classes to add as listeners to project. */ | |||
| private Vector listeners = new Vector(5); | |||
| /** File names of property files to load on startup. */ | |||
| private Vector propertyFiles = new Vector(5); | |||
| /** | |||
| * The Ant logger class. There may be only one logger. It will have | |||
| * the right to use the 'out' PrintStream. The class must implements the | |||
| * BuildLogger interface. | |||
| */ | |||
| private String loggerClassname = null; | |||
| /** | |||
| * The Ant InputHandler class. There may be only one input | |||
| * handler. | |||
| * | |||
| * XXX input proposal. | |||
| */ | |||
| private String inputHandlerClassname = null; | |||
| /** | |||
| * Whether or not output to the log is to be unadorned. | |||
| */ | |||
| private boolean emacsMode = false; | |||
| /** | |||
| * Whether or not this instance has successfully been | |||
| * constructed and is ready to run. | |||
| */ | |||
| private boolean readyToRun = false; | |||
| /** | |||
| * Whether or not we should only parse and display the project help | |||
| * information. | |||
| */ | |||
| private boolean projectHelp = false; | |||
| /** | |||
| * Prints the message of the Throwable if it (the message) is not | |||
| * <code>null</code>. | |||
| * | |||
| * @param t Throwable to print the message of. | |||
| * Must not be <code>null</code>. | |||
| */ | |||
| private static void printMessage(Throwable t) { | |||
| String message = t.getMessage(); | |||
| if (message != null) { | |||
| System.err.println(message); | |||
| } | |||
| } | |||
| /** | |||
| * Creates a new instance of this class using the | |||
| * arguments specified, gives it any extra user properties which have been | |||
| * specified, and then runs the build using the classloader provided. | |||
| * | |||
| * @param args Command line arguments. Must not be <code>null</code>. | |||
| * @param additionalUserProperties Any extra properties to use in this | |||
| * build. May be <code>null</code>, which is the equivalent to | |||
| * passing in an empty set of properties. | |||
| * @param coreLoader Classloader used for core classes. May be | |||
| * <code>null</code> in which case the system classloader is used. | |||
| */ | |||
| public static void start(String[] args, Properties additionalUserProperties, | |||
| ClassLoader coreLoader) { | |||
| Main m = null; | |||
| try { | |||
| m = new Main(args); | |||
| } catch(Throwable exc) { | |||
| printMessage(exc); | |||
| System.exit(1); | |||
| } | |||
| if (additionalUserProperties != null) { | |||
| for (Enumeration e = additionalUserProperties.keys(); e.hasMoreElements(); ) { | |||
| String key = (String) e.nextElement(); | |||
| String property = additionalUserProperties.getProperty(key); | |||
| m.definedProps.put(key, property); | |||
| } | |||
| } | |||
| try { | |||
| m.runBuild(coreLoader); | |||
| System.exit(0); | |||
| } catch (BuildException be) { | |||
| if (m.err != System.err) { | |||
| printMessage(be); | |||
| } | |||
| System.exit(1); | |||
| } catch(Throwable exc) { | |||
| exc.printStackTrace(); | |||
| printMessage(exc); | |||
| System.exit(1); | |||
| } | |||
| } | |||
| /** | |||
| * Command line entry point. This method kicks off the building | |||
| * of a project object and executes a build using either a given | |||
| * target or the default target. | |||
| * | |||
| * @param args Command line arguments. Must not be <code>null</code>. | |||
| */ | |||
| public static void main(String[] args) { | |||
| start(args, null, null); | |||
| } | |||
| // XXX: (Jon Skeet) Error handling appears to be inconsistent here. | |||
| // Sometimes there's just a return statement, and sometimes a | |||
| // BuildException is thrown. What's the rationale for when to do | |||
| // what? | |||
| /** | |||
| * Sole constructor, which parses and deals with command line | |||
| * arguments. | |||
| * | |||
| * @param args Command line arguments. Must not be <code>null</code>. | |||
| * | |||
| * @exception BuildException if the specified build file doesn't exist | |||
| * or is a directory. | |||
| */ | |||
| protected Main(String[] args) throws BuildException { | |||
| String searchForThis = null; | |||
| // cycle through given args | |||
| for (int i = 0; i < args.length; i++) { | |||
| String arg = args[i]; | |||
| if (arg.equals("-help")) { | |||
| printUsage(); | |||
| return; | |||
| } else if (arg.equals("-version")) { | |||
| printVersion(); | |||
| return; | |||
| } else if (arg.equals("-quiet") || arg.equals("-q")) { | |||
| msgOutputLevel = Project.MSG_WARN; | |||
| } else if (arg.equals("-verbose") || arg.equals("-v")) { | |||
| printVersion(); | |||
| msgOutputLevel = Project.MSG_VERBOSE; | |||
| } else if (arg.equals("-debug")) { | |||
| printVersion(); | |||
| msgOutputLevel = Project.MSG_DEBUG; | |||
| } else if (arg.equals("-logfile") || arg.equals("-l")) { | |||
| try { | |||
| File logFile = new File(args[i+1]); | |||
| i++; | |||
| out = new PrintStream(new FileOutputStream(logFile)); | |||
| err = out; | |||
| System.setOut(out); | |||
| System.setErr(out); | |||
| } catch (IOException ioe) { | |||
| String msg = "Cannot write on the specified log file. " + | |||
| "Make sure the path exists and you have write permissions."; | |||
| System.out.println(msg); | |||
| return; | |||
| } catch (ArrayIndexOutOfBoundsException aioobe) { | |||
| String msg = "You must specify a log file when " + | |||
| "using the -log argument"; | |||
| System.out.println(msg); | |||
| return; | |||
| } | |||
| } else if (arg.equals("-buildfile") || arg.equals("-file") || arg.equals("-f")) { | |||
| try { | |||
| buildFile = new File(args[i+1]); | |||
| i++; | |||
| } catch (ArrayIndexOutOfBoundsException aioobe) { | |||
| String msg = "You must specify a buildfile when " + | |||
| "using the -buildfile argument"; | |||
| System.out.println(msg); | |||
| return; | |||
| } | |||
| } else if (arg.equals("-listener")) { | |||
| try { | |||
| listeners.addElement(args[i+1]); | |||
| i++; | |||
| } catch (ArrayIndexOutOfBoundsException aioobe) { | |||
| String msg = "You must specify a classname when " + | |||
| "using the -listener argument"; | |||
| System.out.println(msg); | |||
| return; | |||
| } | |||
| } else if (arg.startsWith("-D")) { | |||
| /* Interestingly enough, we get to here when a user | |||
| * uses -Dname=value. However, in some cases, the JDK | |||
| * goes ahead and parses this out to args | |||
| * {"-Dname", "value"} | |||
| * so instead of parsing on "=", we just make the "-D" | |||
| * characters go away and skip one argument forward. | |||
| * | |||
| * I don't know how to predict when the JDK is going | |||
| * to help or not, so we simply look for the equals sign. | |||
| */ | |||
| String name = arg.substring(2, arg.length()); | |||
| String value = null; | |||
| int posEq = name.indexOf("="); | |||
| if (posEq > 0) { | |||
| value = name.substring(posEq+1); | |||
| name = name.substring(0, posEq); | |||
| } else if (i < args.length-1) { | |||
| value = args[++i]; | |||
| } | |||
| definedProps.put(name, value); | |||
| } else if (arg.equals("-logger")) { | |||
| if (loggerClassname != null) { | |||
| System.out.println("Only one logger class may be specified."); | |||
| return; | |||
| } | |||
| try { | |||
| loggerClassname = args[++i]; | |||
| } | |||
| catch (ArrayIndexOutOfBoundsException aioobe) { | |||
| System.out.println("You must specify a classname when " + | |||
| "using the -logger argument"); | |||
| return; | |||
| } | |||
| } else if (arg.equals("-inputhandler")) { | |||
| if (inputHandlerClassname != null) { | |||
| System.out.println("Only one input handler class may be specified."); | |||
| return; | |||
| } | |||
| try { | |||
| inputHandlerClassname = args[++i]; | |||
| } | |||
| catch (ArrayIndexOutOfBoundsException aioobe) { | |||
| System.out.println("You must specify a classname when " + | |||
| "using the -inputhandler argument"); | |||
| return; | |||
| } | |||
| } else if (arg.equals("-emacs")) { | |||
| emacsMode = true; | |||
| } else if (arg.equals("-projecthelp")) { | |||
| // set the flag to display the targets and quit | |||
| projectHelp = true; | |||
| } else if (arg.equals("-find")) { | |||
| // eat up next arg if present, default to build.xml | |||
| if (i < args.length-1) { | |||
| searchForThis = args[++i]; | |||
| } else { | |||
| searchForThis = DEFAULT_BUILD_FILENAME; | |||
| } | |||
| } else if (arg.startsWith("-propertyfile")) { | |||
| try { | |||
| propertyFiles.addElement(args[i+1]); | |||
| i++; | |||
| } catch (ArrayIndexOutOfBoundsException aioobe) { | |||
| String msg = "You must specify a property filename when " + | |||
| "using the -propertyfile argument"; | |||
| System.out.println(msg); | |||
| return; | |||
| } | |||
| } else if (arg.startsWith("-")) { | |||
| // we don't have any more args to recognize! | |||
| String msg = "Unknown argument: " + arg; | |||
| System.out.println(msg); | |||
| printUsage(); | |||
| return; | |||
| } else { | |||
| // if it's no other arg, it may be the target | |||
| targets.addElement(arg); | |||
| } | |||
| } | |||
| // if buildFile was not specified on the command line, | |||
| if (buildFile == null) { | |||
| // but -find then search for it | |||
| if (searchForThis != null) { | |||
| buildFile = findBuildFile(System.getProperty("user.dir"), | |||
| searchForThis); | |||
| } else { | |||
| buildFile = new File(DEFAULT_BUILD_FILENAME); | |||
| } | |||
| } | |||
| // make sure buildfile exists | |||
| if (!buildFile.exists()) { | |||
| System.out.println("Buildfile: " + buildFile + " does not exist!"); | |||
| throw new BuildException("Build failed"); | |||
| } | |||
| // make sure it's not a directory (this falls into the ultra | |||
| // paranoid lets check everything catagory | |||
| if (buildFile.isDirectory()) { | |||
| System.out.println("What? Buildfile: " + buildFile + " is a dir!"); | |||
| throw new BuildException("Build failed"); | |||
| } | |||
| // Load the property files specified by -propertyfile | |||
| for (int propertyFileIndex=0; | |||
| propertyFileIndex < propertyFiles.size(); | |||
| propertyFileIndex++) { | |||
| String filename = (String) propertyFiles.elementAt(propertyFileIndex); | |||
| Properties props = new Properties(); | |||
| FileInputStream fis = null; | |||
| try { | |||
| fis = new FileInputStream(filename); | |||
| props.load(fis); | |||
| } | |||
| catch (IOException e) { | |||
| System.out.println("Could not load property file " | |||
| + filename + ": " + e.getMessage()); | |||
| } finally { | |||
| if (fis != null){ | |||
| try { | |||
| fis.close(); | |||
| } catch (IOException e){ | |||
| } | |||
| } | |||
| } | |||
| // ensure that -D properties take precedence | |||
| Enumeration propertyNames = props.propertyNames(); | |||
| while (propertyNames.hasMoreElements()) { | |||
| String name = (String) propertyNames.nextElement(); | |||
| if (definedProps.getProperty(name) == null) { | |||
| definedProps.put(name, props.getProperty(name)); | |||
| } | |||
| } | |||
| } | |||
| readyToRun = true; | |||
| } | |||
| /** | |||
| * Helper to get the parent file for a given file. | |||
| * <p> | |||
| * Added to simulate File.getParentFile() from JDK 1.2. | |||
| * | |||
| * @param file File to find parent of. Must not be <code>null</code>. | |||
| * @return Parent file or null if none | |||
| */ | |||
| private File getParentFile(File file) { | |||
| String filename = file.getAbsolutePath(); | |||
| file = new File(filename); | |||
| filename = file.getParent(); | |||
| if (filename != null && msgOutputLevel >= Project.MSG_VERBOSE) { | |||
| System.out.println("Searching in "+filename); | |||
| } | |||
| return (filename == null) ? null : new File(filename); | |||
| } | |||
| /** | |||
| * Search parent directories for the build file. | |||
| * <p> | |||
| * Takes the given target as a suffix to append to each | |||
| * parent directory in seach of a build file. Once the | |||
| * root of the file-system has been reached an exception | |||
| * is thrown. | |||
| * | |||
| * @param start Leaf directory of search. | |||
| * Must not be <code>null</code>. | |||
| * @param suffix Suffix filename to look for in parents. | |||
| * Must not be <code>null</code>. | |||
| * | |||
| * @return A handle to the build file if one is found | |||
| * | |||
| * @exception BuildException if no build file is found | |||
| */ | |||
| private File findBuildFile(String start, String suffix) throws BuildException { | |||
| if (msgOutputLevel >= Project.MSG_INFO) { | |||
| System.out.println("Searching for " + suffix + " ..."); | |||
| } | |||
| File parent = new File(new File(start).getAbsolutePath()); | |||
| File file = new File(parent, suffix); | |||
| // check if the target file exists in the current directory | |||
| while (!file.exists()) { | |||
| // change to parent directory | |||
| parent = getParentFile(parent); | |||
| // if parent is null, then we are at the root of the fs, | |||
| // complain that we can't find the build file. | |||
| if (parent == null) { | |||
| throw new BuildException("Could not locate a build file!"); | |||
| } | |||
| // refresh our file handle | |||
| file = new File(parent, suffix); | |||
| } | |||
| return file; | |||
| } | |||
| /** | |||
| * Executes the build. If the constructor for this instance failed | |||
| * (e.g. returned after issuing a warning), this method returns | |||
| * immediately. | |||
| * | |||
| * @param coreLoader The classloader to use to find core classes. | |||
| * May be <code>null</code>, in which case the | |||
| * system classloader is used. | |||
| * | |||
| * @exception BuildException if the build fails | |||
| */ | |||
| private void runBuild(ClassLoader coreLoader) throws BuildException { | |||
| if (!readyToRun) { | |||
| return; | |||
| } | |||
| // track when we started | |||
| if (msgOutputLevel >= Project.MSG_INFO) { | |||
| System.out.println("Buildfile: " + buildFile); | |||
| } | |||
| final Project project = new Project(); | |||
| project.setCoreLoader(coreLoader); | |||
| Throwable error = null; | |||
| try { | |||
| addBuildListeners(project); | |||
| addInputHandler(project); | |||
| PrintStream err = System.err; | |||
| PrintStream out = System.out; | |||
| // use a system manager that prevents from System.exit() | |||
| // only in JDK > 1.1 | |||
| SecurityManager oldsm = null; | |||
| if ( !Project.JAVA_1_0.equals(Project.getJavaVersion()) && | |||
| !Project.JAVA_1_1.equals(Project.getJavaVersion()) ){ | |||
| oldsm = System.getSecurityManager(); | |||
| //SecurityManager can not be installed here for backwards | |||
| //compatability reasons (PD). Needs to be loaded prior to | |||
| //ant class if we are going to implement it. | |||
| //System.setSecurityManager(new NoExitSecurityManager()); | |||
| } | |||
| try { | |||
| System.setOut(new PrintStream(new DemuxOutputStream(project, false))); | |||
| System.setErr(new PrintStream(new DemuxOutputStream(project, true))); | |||
| if (!projectHelp) { | |||
| project.fireBuildStarted(); | |||
| } | |||
| project.init(); | |||
| project.setUserProperty("ant.version", getAntVersion()); | |||
| // set user-define properties | |||
| Enumeration e = definedProps.keys(); | |||
| while (e.hasMoreElements()) { | |||
| String arg = (String)e.nextElement(); | |||
| String value = (String)definedProps.get(arg); | |||
| project.setUserProperty(arg, value); | |||
| } | |||
| project.setUserProperty("ant.file" , buildFile.getAbsolutePath() ); | |||
| // first use the ProjectHelper to create the project object | |||
| // from the given build file. | |||
| String noParserMessage = | |||
| "No JAXP compliant XML parser found. Please visit http://xml.apache.org for a suitable parser"; | |||
| try { | |||
| Class.forName("javax.xml.parsers.SAXParserFactory"); | |||
| ProjectHelper.configureProject(project, buildFile); | |||
| } catch (NoClassDefFoundError ncdfe) { | |||
| throw new BuildException(noParserMessage, ncdfe); | |||
| } catch (ClassNotFoundException cnfe) { | |||
| throw new BuildException(noParserMessage, cnfe); | |||
| } catch (NullPointerException npe) { | |||
| throw new BuildException(noParserMessage, npe); | |||
| } | |||
| if (projectHelp) { | |||
| printDescription(project); | |||
| printTargets(project, msgOutputLevel > Project.MSG_INFO ); | |||
| return; | |||
| } | |||
| // make sure that we have a target to execute | |||
| if (targets.size() == 0) { | |||
| targets.addElement(project.getDefaultTarget()); | |||
| } | |||
| project.executeTargets(targets); | |||
| } | |||
| finally { | |||
| // put back the original security manager | |||
| //The following will never eval to true. (PD) | |||
| if (oldsm != null){ | |||
| System.setSecurityManager(oldsm); | |||
| } | |||
| System.setOut(out); | |||
| System.setErr(err); | |||
| } | |||
| } | |||
| catch(RuntimeException exc) { | |||
| error = exc; | |||
| throw exc; | |||
| } | |||
| catch(Error err) { | |||
| error = err; | |||
| throw err; | |||
| } | |||
| finally { | |||
| if (!projectHelp) { | |||
| project.fireBuildFinished(error); | |||
| } | |||
| } | |||
| } | |||
| /** | |||
| * Adds the listeners specified in the command line arguments, | |||
| * along with the default listener, to the specified project. | |||
| * | |||
| * @param project The project to add listeners to. | |||
| * Must not be <code>null</code>. | |||
| */ | |||
| protected void addBuildListeners(Project project) { | |||
| // Add the default listener | |||
| project.addBuildListener(createLogger()); | |||
| for (int i = 0; i < listeners.size(); i++) { | |||
| String className = (String) listeners.elementAt(i); | |||
| try { | |||
| BuildListener listener = | |||
| (BuildListener) Class.forName(className).newInstance(); | |||
| project.addBuildListener(listener); | |||
| } | |||
| catch(Throwable exc) { | |||
| throw new BuildException("Unable to instantiate listener " + className, exc); | |||
| } | |||
| } | |||
| } | |||
| /** | |||
| * Creates the InputHandler and adds it to the project. | |||
| * | |||
| * XXX input proposal | |||
| * | |||
| * @exception BuildException if a specified InputHandler | |||
| * implementation could not be loaded. | |||
| */ | |||
| private void addInputHandler(Project project) { | |||
| InputHandler handler = null; | |||
| if (inputHandlerClassname == null) { | |||
| handler = new DefaultInputHandler(); | |||
| } else { | |||
| try { | |||
| handler = (InputHandler)(Class.forName(inputHandlerClassname).newInstance()); | |||
| } | |||
| catch (ClassCastException e) { | |||
| String msg = "The specified input handler class " | |||
| + inputHandlerClassname | |||
| + " does not implement the InputHandler interface"; | |||
| throw new BuildException(msg); | |||
| } | |||
| catch (Exception e) { | |||
| String msg = "Unable to instantiate specified input handler class " | |||
| + inputHandlerClassname + " : " + e.getClass().getName(); | |||
| throw new BuildException(msg); | |||
| } | |||
| } | |||
| project.setInputHandler(handler); | |||
| } | |||
| // XXX: (Jon Skeet) Any reason for writing a message and then using a bare | |||
| // RuntimeException rather than just using a BuildException here? Is it | |||
| // in case the message could end up being written to no loggers (as the loggers | |||
| // could have failed to be created due to this failure)? | |||
| /** | |||
| * Creates the default build logger for sending build events to the ant log. | |||
| */ | |||
| private BuildLogger createLogger() { | |||
| BuildLogger logger = null; | |||
| if (loggerClassname != null) { | |||
| try { | |||
| logger = (BuildLogger)(Class.forName(loggerClassname).newInstance()); | |||
| } | |||
| catch (ClassCastException e) { | |||
| System.err.println("The specified logger class " + loggerClassname + | |||
| " does not implement the BuildLogger interface"); | |||
| throw new RuntimeException(); | |||
| } | |||
| catch (Exception e) { | |||
| System.err.println("Unable to instantiate specified logger class " + | |||
| loggerClassname + " : " + e.getClass().getName()); | |||
| throw new RuntimeException(); | |||
| } | |||
| } | |||
| else { | |||
| logger = new DefaultLogger(); | |||
| } | |||
| logger.setMessageOutputLevel(msgOutputLevel); | |||
| logger.setOutputPrintStream(out); | |||
| logger.setErrorPrintStream(err); | |||
| logger.setEmacsMode(emacsMode); | |||
| return logger; | |||
| } | |||
| /** | |||
| * Prints the usage information for this class to <code>System.out</code>. | |||
| */ | |||
| private static void printUsage() { | |||
| String lSep = System.getProperty("line.separator"); | |||
| StringBuffer msg = new StringBuffer(); | |||
| msg.append("ant [options] [target [target2 [target3] ...]]" + lSep); | |||
| msg.append("Options: " + lSep); | |||
| msg.append(" -help print this message" + lSep); | |||
| msg.append(" -projecthelp print project help information" + lSep); | |||
| msg.append(" -version print the version information and exit" + lSep); | |||
| msg.append(" -quiet be extra quiet" + lSep); | |||
| msg.append(" -verbose be extra verbose" + lSep); | |||
| msg.append(" -debug print debugging information" + lSep); | |||
| msg.append(" -emacs produce logging information without adornments" + lSep); | |||
| msg.append(" -logfile <file> use given file for log" + lSep); | |||
| msg.append(" -logger <classname> the class which is to perform logging" + lSep); | |||
| msg.append(" -listener <classname> add an instance of class as a project listener" + lSep); | |||
| msg.append(" -buildfile <file> use given buildfile" + lSep); | |||
| msg.append(" -D<property>=<value> use value for given property" + lSep); | |||
| msg.append(" -propertyfile <name> load all properties from file with -D" + lSep); | |||
| msg.append(" properties taking precedence" + lSep); | |||
| msg.append(" -find <file> search for buildfile towards the root of the" + lSep); | |||
| msg.append(" filesystem and use it" + lSep); | |||
| System.out.println(msg.toString()); | |||
| } | |||
| /** | |||
| * Prints the Ant version information to <code>System.out</code>. | |||
| * | |||
| * @exception BuildException if the version information is unavailable | |||
| */ | |||
| private static void printVersion() throws BuildException { | |||
| System.out.println(getAntVersion()); | |||
| } | |||
| /** | |||
| * Cache of the Ant version information when it has been loaded. | |||
| */ | |||
| private static String antVersion = null; | |||
| /** | |||
| * Returns the Ant version information, if available. Once the information | |||
| * has been loaded once, it's cached and returned from the cache on future | |||
| * calls. | |||
| * | |||
| * @return the Ant version information as a String | |||
| * (always non-<code>null</code>) | |||
| * | |||
| * @exception BuildException if the version information is unavailable | |||
| */ | |||
| public static synchronized String getAntVersion() throws BuildException { | |||
| if (antVersion == null) { | |||
| try { | |||
| Properties props = new Properties(); | |||
| InputStream in = | |||
| Main.class.getResourceAsStream("/org/apache/tools/ant/version.txt"); | |||
| props.load(in); | |||
| in.close(); | |||
| String lSep = System.getProperty("line.separator"); | |||
| StringBuffer msg = new StringBuffer(); | |||
| msg.append("Apache Ant version "); | |||
| msg.append(props.getProperty("VERSION")); | |||
| msg.append(" compiled on "); | |||
| msg.append(props.getProperty("DATE")); | |||
| antVersion = msg.toString(); | |||
| } catch (IOException ioe) { | |||
| throw new BuildException("Could not load the version information:" | |||
| + ioe.getMessage()); | |||
| } catch (NullPointerException npe) { | |||
| throw new BuildException("Could not load the version information."); | |||
| } | |||
| } | |||
| return antVersion; | |||
| } | |||
| /** | |||
| * Prints the description of a project (if there is one) to | |||
| * <code>System.out</code>. | |||
| * | |||
| * @param project The project to display a description of. | |||
| * Must not be <code>null</code>. | |||
| */ | |||
| private static void printDescription(Project project) { | |||
| if (project.getDescription() != null) { | |||
| System.out.println(project.getDescription()); | |||
| } | |||
| } | |||
| /** | |||
| * Prints a list of all targets in the specified project to | |||
| * <code>System.out</code>, optionally including subtargets. | |||
| * | |||
| * @param project The project to display a description of. | |||
| * Must not be <code>null</code>. | |||
| * @param printSubTargets Whether or not subtarget names should also be | |||
| * printed. | |||
| */ | |||
| private static void printTargets(Project project, boolean printSubTargets) { | |||
| // find the target with the longest name | |||
| int maxLength = 0; | |||
| Enumeration ptargets = project.getTargets().elements(); | |||
| String targetName; | |||
| String targetDescription; | |||
| Target currentTarget; | |||
| // split the targets in top-level and sub-targets depending | |||
| // on the presence of a description | |||
| Vector topNames = new Vector(); | |||
| Vector topDescriptions = new Vector(); | |||
| Vector subNames = new Vector(); | |||
| while (ptargets.hasMoreElements()) { | |||
| currentTarget = (Target)ptargets.nextElement(); | |||
| targetName = currentTarget.getName(); | |||
| targetDescription = currentTarget.getDescription(); | |||
| // maintain a sorted list of targets | |||
| if (targetDescription == null) { | |||
| int pos = findTargetPosition(subNames, targetName); | |||
| subNames.insertElementAt(targetName, pos); | |||
| } else { | |||
| int pos = findTargetPosition(topNames, targetName); | |||
| topNames.insertElementAt(targetName, pos); | |||
| topDescriptions.insertElementAt(targetDescription, pos); | |||
| if (targetName.length() > maxLength) { | |||
| maxLength = targetName.length(); | |||
| } | |||
| } | |||
| } | |||
| printTargets(topNames, topDescriptions, "Main targets:", maxLength); | |||
| if( printSubTargets ) { | |||
| printTargets(subNames, null, "Subtargets:", 0); | |||
| } | |||
| String defaultTarget = project.getDefaultTarget(); | |||
| if (defaultTarget != null && !"".equals(defaultTarget)) { // shouldn't need to check but... | |||
| System.out.println( "Default target: " + defaultTarget ); | |||
| } | |||
| } | |||
| /** | |||
| * Searches for the correct place to insert a name into a list so as | |||
| * to keep the list sorted alphabetically. | |||
| * | |||
| * @param names The current list of names. Must not be <code>null</code>. | |||
| * @param name The name to find a place for. | |||
| * Must not be <code>null</code>. | |||
| * | |||
| * @return the correct place in the list for the given name | |||
| */ | |||
| private static int findTargetPosition(Vector names, String name) { | |||
| int res = names.size(); | |||
| for (int i=0; i<names.size() && res == names.size(); i++) { | |||
| if (name.compareTo((String)names.elementAt(i)) < 0) { | |||
| res = i; | |||
| } | |||
| } | |||
| return res; | |||
| } | |||
| /** | |||
| * Writes a formatted list of target names to <code>System.out</code> | |||
| * with an optional description | |||
| */ | |||
| private static void printTargets(Vector names, Vector descriptions, String heading, int maxlen) { | |||
| // now, start printing the targets and their descriptions | |||
| String lSep = System.getProperty("line.separator"); | |||
| // got a bit annoyed that I couldn't find a pad function | |||
| String spaces = " "; | |||
| while (spaces.length()<maxlen) { | |||
| spaces += spaces; | |||
| } | |||
| StringBuffer msg = new StringBuffer(); | |||
| msg.append(heading + lSep + lSep); | |||
| for (int i=0; i<names.size(); i++) { | |||
| msg.append(" "); | |||
| msg.append(names.elementAt(i)); | |||
| if (descriptions != null) { | |||
| msg.append(spaces.substring(0, maxlen - ((String)names.elementAt(i)).length() + 2)); | |||
| msg.append(descriptions.elementAt(i)); | |||
| } | |||
| msg.append(lSep); | |||
| } | |||
| System.out.println(msg.toString()); | |||
| } | |||
| } | |||
| @@ -0,0 +1,90 @@ | |||
| /* | |||
| * The Apache Software License, Version 1.1 | |||
| * | |||
| * Copyright (c) 2002 The Apache Software Foundation. All rights | |||
| * reserved. | |||
| * | |||
| * Redistribution and use in source and binary forms, with or without | |||
| * modification, are permitted provided that the following conditions | |||
| * are met: | |||
| * | |||
| * 1. Redistributions of source code must retain the above copyright | |||
| * notice, this list of conditions and the following disclaimer. | |||
| * | |||
| * 2. Redistributions in binary form must reproduce the above copyright | |||
| * notice, this list of conditions and the following disclaimer in | |||
| * the documentation and/or other materials provided with the | |||
| * distribution. | |||
| * | |||
| * 3. The end-user documentation included with the redistribution, if | |||
| * any, must include the following acknowlegement: | |||
| * "This product includes software developed by the | |||
| * Apache Software Foundation (http://www.apache.org/)." | |||
| * Alternately, this acknowlegement may appear in the software itself, | |||
| * if and wherever such third-party acknowlegements normally appear. | |||
| * | |||
| * 4. The names "The Jakarta Project", "Ant", and "Apache Software | |||
| * Foundation" must not be used to endorse or promote products derived | |||
| * from this software without prior written permission. For written | |||
| * permission, please contact apache@apache.org. | |||
| * | |||
| * 5. Products derived from this software may not be called "Apache" | |||
| * nor may "Apache" appear in their names without prior written | |||
| * permission of the Apache Group. | |||
| * | |||
| * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED | |||
| * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES | |||
| * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | |||
| * DISCLAIMED. IN NO EVENT SHALL THE APACHE SOFTWARE FOUNDATION OR | |||
| * ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, | |||
| * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT | |||
| * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF | |||
| * USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | |||
| * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, | |||
| * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT | |||
| * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF | |||
| * SUCH DAMAGE. | |||
| * ==================================================================== | |||
| * | |||
| * This software consists of voluntary contributions made by many | |||
| * individuals on behalf of the Apache Software Foundation. For more | |||
| * information on the Apache Software Foundation, please see | |||
| * <http://www.apache.org/>. | |||
| */ | |||
| package org.apache.tools.ant.input; | |||
| import java.io.BufferedReader; | |||
| import java.io.IOException; | |||
| import java.io.InputStreamReader; | |||
| import org.apache.tools.ant.BuildException; | |||
| /** | |||
| * Prompts on System.out, reads input from System.in | |||
| * | |||
| * @author <a href="mailto:stefan.bodewig@epost.de">Stefan Bodewig</a> | |||
| * @version $Revision$ | |||
| */ | |||
| public class DefaultInputHandler implements InputHandler { | |||
| /** | |||
| * Empty no-arg constructor | |||
| */ | |||
| public DefaultInputHandler() { | |||
| } | |||
| public void handleInput(InputRequest request) throws BuildException { | |||
| do { | |||
| System.out.println(request.getPrompt()); | |||
| try { | |||
| BufferedReader in = | |||
| new BufferedReader(new InputStreamReader(System.in)); | |||
| String input = in.readLine(); | |||
| request.setInput(input); | |||
| } catch (IOException e) { | |||
| throw new BuildException("Failed to read input from Console.", e); | |||
| } | |||
| } while (!request.isInputValid()); | |||
| } | |||
| } | |||
| @@ -0,0 +1,66 @@ | |||
| /* | |||
| * The Apache Software License, Version 1.1 | |||
| * | |||
| * Copyright (c) 2002 The Apache Software Foundation. All rights | |||
| * reserved. | |||
| * | |||
| * Redistribution and use in source and binary forms, with or without | |||
| * modification, are permitted provided that the following conditions | |||
| * are met: | |||
| * | |||
| * 1. Redistributions of source code must retain the above copyright | |||
| * notice, this list of conditions and the following disclaimer. | |||
| * | |||
| * 2. Redistributions in binary form must reproduce the above copyright | |||
| * notice, this list of conditions and the following disclaimer in | |||
| * the documentation and/or other materials provided with the | |||
| * distribution. | |||
| * | |||
| * 3. The end-user documentation included with the redistribution, if | |||
| * any, must include the following acknowlegement: | |||
| * "This product includes software developed by the | |||
| * Apache Software Foundation (http://www.apache.org/)." | |||
| * Alternately, this acknowlegement may appear in the software itself, | |||
| * if and wherever such third-party acknowlegements normally appear. | |||
| * | |||
| * 4. The names "The Jakarta Project", "Ant", and "Apache Software | |||
| * Foundation" must not be used to endorse or promote products derived | |||
| * from this software without prior written permission. For written | |||
| * permission, please contact apache@apache.org. | |||
| * | |||
| * 5. Products derived from this software may not be called "Apache" | |||
| * nor may "Apache" appear in their names without prior written | |||
| * permission of the Apache Group. | |||
| * | |||
| * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED | |||
| * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES | |||
| * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | |||
| * DISCLAIMED. IN NO EVENT SHALL THE APACHE SOFTWARE FOUNDATION OR | |||
| * ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, | |||
| * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT | |||
| * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF | |||
| * USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | |||
| * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, | |||
| * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT | |||
| * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF | |||
| * SUCH DAMAGE. | |||
| * ==================================================================== | |||
| * | |||
| * This software consists of voluntary contributions made by many | |||
| * individuals on behalf of the Apache Software Foundation. For more | |||
| * information on the Apache Software Foundation, please see | |||
| * <http://www.apache.org/>. | |||
| */ | |||
| package org.apache.tools.ant.input; | |||
| /** | |||
| * Plugin to Ant to handle requests for user input. | |||
| * | |||
| * @author <a href="mailto:stefan.bodewig@epost.de">Stefan Bodewig</a> | |||
| * @version $Revision$ | |||
| */ | |||
| public interface InputHandler { | |||
| void handleInput(InputRequest request) | |||
| throws org.apache.tools.ant.BuildException; | |||
| } | |||
| @@ -0,0 +1,99 @@ | |||
| /* | |||
| * The Apache Software License, Version 1.1 | |||
| * | |||
| * Copyright (c) 2002 The Apache Software Foundation. All rights | |||
| * reserved. | |||
| * | |||
| * Redistribution and use in source and binary forms, with or without | |||
| * modification, are permitted provided that the following conditions | |||
| * are met: | |||
| * | |||
| * 1. Redistributions of source code must retain the above copyright | |||
| * notice, this list of conditions and the following disclaimer. | |||
| * | |||
| * 2. Redistributions in binary form must reproduce the above copyright | |||
| * notice, this list of conditions and the following disclaimer in | |||
| * the documentation and/or other materials provided with the | |||
| * distribution. | |||
| * | |||
| * 3. The end-user documentation included with the redistribution, if | |||
| * any, must include the following acknowlegement: | |||
| * "This product includes software developed by the | |||
| * Apache Software Foundation (http://www.apache.org/)." | |||
| * Alternately, this acknowlegement may appear in the software itself, | |||
| * if and wherever such third-party acknowlegements normally appear. | |||
| * | |||
| * 4. The names "The Jakarta Project", "Ant", and "Apache Software | |||
| * Foundation" must not be used to endorse or promote products derived | |||
| * from this software without prior written permission. For written | |||
| * permission, please contact apache@apache.org. | |||
| * | |||
| * 5. Products derived from this software may not be called "Apache" | |||
| * nor may "Apache" appear in their names without prior written | |||
| * permission of the Apache Group. | |||
| * | |||
| * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED | |||
| * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES | |||
| * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | |||
| * DISCLAIMED. IN NO EVENT SHALL THE APACHE SOFTWARE FOUNDATION OR | |||
| * ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, | |||
| * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT | |||
| * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF | |||
| * USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | |||
| * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, | |||
| * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT | |||
| * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF | |||
| * SUCH DAMAGE. | |||
| * ==================================================================== | |||
| * | |||
| * This software consists of voluntary contributions made by many | |||
| * individuals on behalf of the Apache Software Foundation. For more | |||
| * information on the Apache Software Foundation, please see | |||
| * <http://www.apache.org/>. | |||
| */ | |||
| package org.apache.tools.ant.input; | |||
| /** | |||
| * Encapsulates an input request. | |||
| * | |||
| * @author <a href="mailto:stefan.bodewig@epost.de">Stefan Bodewig</a> | |||
| * @version $Revision$ | |||
| */ | |||
| public class InputRequest { | |||
| private String prompt; | |||
| private String input; | |||
| public InputRequest(String prompt) { | |||
| this.prompt = prompt; | |||
| } | |||
| /** | |||
| * Retrieves the prompt text. | |||
| */ | |||
| public String getPrompt() { | |||
| return prompt; | |||
| } | |||
| /** | |||
| * Sets the user provided input. | |||
| */ | |||
| public void setInput(String input) { | |||
| this.input = input; | |||
| } | |||
| /** | |||
| * Is the user input valid? | |||
| */ | |||
| public boolean isInputValid() { | |||
| return true; | |||
| } | |||
| /** | |||
| * Retrieves the user input. | |||
| */ | |||
| public String getInput() { | |||
| return input; | |||
| } | |||
| } | |||
| @@ -0,0 +1,83 @@ | |||
| /* | |||
| * The Apache Software License, Version 1.1 | |||
| * | |||
| * Copyright (c) 2002 The Apache Software Foundation. All rights | |||
| * reserved. | |||
| * | |||
| * Redistribution and use in source and binary forms, with or without | |||
| * modification, are permitted provided that the following conditions | |||
| * are met: | |||
| * | |||
| * 1. Redistributions of source code must retain the above copyright | |||
| * notice, this list of conditions and the following disclaimer. | |||
| * | |||
| * 2. Redistributions in binary form must reproduce the above copyright | |||
| * notice, this list of conditions and the following disclaimer in | |||
| * the documentation and/or other materials provided with the | |||
| * distribution. | |||
| * | |||
| * 3. The end-user documentation included with the redistribution, if | |||
| * any, must include the following acknowlegement: | |||
| * "This product includes software developed by the | |||
| * Apache Software Foundation (http://www.apache.org/)." | |||
| * Alternately, this acknowlegement may appear in the software itself, | |||
| * if and wherever such third-party acknowlegements normally appear. | |||
| * | |||
| * 4. The names "The Jakarta Project", "Ant", and "Apache Software | |||
| * Foundation" must not be used to endorse or promote products derived | |||
| * from this software without prior written permission. For written | |||
| * permission, please contact apache@apache.org. | |||
| * | |||
| * 5. Products derived from this software may not be called "Apache" | |||
| * nor may "Apache" appear in their names without prior written | |||
| * permission of the Apache Group. | |||
| * | |||
| * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED | |||
| * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES | |||
| * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | |||
| * DISCLAIMED. IN NO EVENT SHALL THE APACHE SOFTWARE FOUNDATION OR | |||
| * ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, | |||
| * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT | |||
| * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF | |||
| * USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | |||
| * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, | |||
| * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT | |||
| * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF | |||
| * SUCH DAMAGE. | |||
| * ==================================================================== | |||
| * | |||
| * This software consists of voluntary contributions made by many | |||
| * individuals on behalf of the Apache Software Foundation. For more | |||
| * information on the Apache Software Foundation, please see | |||
| * <http://www.apache.org/>. | |||
| */ | |||
| package org.apache.tools.ant.input; | |||
| import java.util.Vector; | |||
| /** | |||
| * Encapsulates an input request. | |||
| * | |||
| * @author <a href="mailto:stefan.bodewig@epost.de">Stefan Bodewig</a> | |||
| * @version $Revision$ | |||
| */ | |||
| public class MultipleChoiceInputRequest extends InputRequest { | |||
| private Vector choices = new Vector(); | |||
| public MultipleChoiceInputRequest(String prompt, Vector choices) { | |||
| super(prompt); | |||
| this.choices = choices; | |||
| } | |||
| /** | |||
| * The possible values. | |||
| */ | |||
| public Vector getChoices() { | |||
| return choices; | |||
| } | |||
| public boolean isInputValid() { | |||
| return choices.contains(getInput()); | |||
| } | |||
| } | |||
| @@ -0,0 +1,111 @@ | |||
| /* | |||
| * The Apache Software License, Version 1.1 | |||
| * | |||
| * Copyright (c) 2002 The Apache Software Foundation. All rights | |||
| * reserved. | |||
| * | |||
| * Redistribution and use in source and binary forms, with or without | |||
| * modification, are permitted provided that the following conditions | |||
| * are met: | |||
| * | |||
| * 1. Redistributions of source code must retain the above copyright | |||
| * notice, this list of conditions and the following disclaimer. | |||
| * | |||
| * 2. Redistributions in binary form must reproduce the above copyright | |||
| * notice, this list of conditions and the following disclaimer in | |||
| * the documentation and/or other materials provided with the | |||
| * distribution. | |||
| * | |||
| * 3. The end-user documentation included with the redistribution, if | |||
| * any, must include the following acknowlegement: | |||
| * "This product includes software developed by the | |||
| * Apache Software Foundation (http://www.apache.org/)." | |||
| * Alternately, this acknowlegement may appear in the software itself, | |||
| * if and wherever such third-party acknowlegements normally appear. | |||
| * | |||
| * 4. The names "The Jakarta Project", "Ant", and "Apache Software | |||
| * Foundation" must not be used to endorse or promote products derived | |||
| * from this software without prior written permission. For written | |||
| * permission, please contact apache@apache.org. | |||
| * | |||
| * 5. Products derived from this software may not be called "Apache" | |||
| * nor may "Apache" appear in their names without prior written | |||
| * permission of the Apache Group. | |||
| * | |||
| * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED | |||
| * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES | |||
| * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | |||
| * DISCLAIMED. IN NO EVENT SHALL THE APACHE SOFTWARE FOUNDATION OR | |||
| * ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, | |||
| * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT | |||
| * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF | |||
| * USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | |||
| * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, | |||
| * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT | |||
| * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF | |||
| * SUCH DAMAGE. | |||
| * ==================================================================== | |||
| * | |||
| * This software consists of voluntary contributions made by many | |||
| * individuals on behalf of the Apache Software Foundation. For more | |||
| * information on the Apache Software Foundation, please see | |||
| * <http://www.apache.org/>. | |||
| */ | |||
| package org.apache.tools.ant.input; | |||
| import org.apache.tools.ant.BuildException; | |||
| import java.io.FileInputStream; | |||
| import java.io.IOException; | |||
| import java.util.Properties; | |||
| /** | |||
| * Reads input from a property file, the file name is read from the | |||
| * system property input.properties, the prompt is the key for input. | |||
| * | |||
| * @author <a href="mailto:stefan.bodewig@epost.de">Stefan Bodewig</a> | |||
| * @version $Revision$ | |||
| */ | |||
| public class PropertyFileInputHandler implements InputHandler { | |||
| private Properties props = null; | |||
| /** | |||
| * Empty no-arg constructor. | |||
| */ | |||
| public PropertyFileInputHandler() { | |||
| } | |||
| public synchronized void handleInput(InputRequest request) | |||
| throws BuildException { | |||
| if (props == null) { | |||
| readProps(); | |||
| } | |||
| Object o = props.get(request.getPrompt()); | |||
| if (o == null) { | |||
| throw new BuildException("Unable to find input for " | |||
| + request.getPrompt()); | |||
| } | |||
| request.setInput(o.toString()); | |||
| if (!request.isInputValid()) { | |||
| throw new BuildException("Found invalid input " + o | |||
| + " for " + request.getPrompt()); | |||
| } | |||
| } | |||
| private void readProps() throws BuildException { | |||
| String propsFile = System.getProperty("input.properties"); | |||
| if (propsFile == null) { | |||
| throw new BuildException("System property input.properties for PropertyFileInputHandler not set"); | |||
| } | |||
| props = new Properties(); | |||
| try { | |||
| props.load(new FileInputStream(propsFile)); | |||
| } catch (IOException e) { | |||
| throw new BuildException("Couldn't load "+propsFile, e); | |||
| } | |||
| } | |||
| } | |||
| @@ -0,0 +1,135 @@ | |||
| /* | |||
| * The Apache Software License, Version 1.1 | |||
| * | |||
| * Copyright (c) 2002 The Apache Software Foundation. All rights | |||
| * reserved. | |||
| * | |||
| * Redistribution and use in source and binary forms, with or without | |||
| * modification, are permitted provided that the following conditions | |||
| * are met: | |||
| * | |||
| * 1. Redistributions of source code must retain the above copyright | |||
| * notice, this list of conditions and the following disclaimer. | |||
| * | |||
| * 2. Redistributions in binary form must reproduce the above copyright | |||
| * notice, this list of conditions and the following disclaimer in | |||
| * the documentation and/or other materials provided with the | |||
| * distribution. | |||
| * | |||
| * 3. The end-user documentation included with the redistribution, if | |||
| * any, must include the following acknowlegement: | |||
| * "This product includes software developed by the | |||
| * Apache Software Foundation (http://www.apache.org/)." | |||
| * Alternately, this acknowlegement may appear in the software itself, | |||
| * if and wherever such third-party acknowlegements normally appear. | |||
| * | |||
| * 4. The names "The Jakarta Project", "Ant", and "Apache Software | |||
| * Foundation" must not be used to endorse or promote products derived | |||
| * from this software without prior written permission. For written | |||
| * permission, please contact apache@apache.org. | |||
| * | |||
| * 5. Products derived from this software may not be called "Apache" | |||
| * nor may "Apache" appear in their names without prior written | |||
| * permission of the Apache Group. | |||
| * | |||
| * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED | |||
| * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES | |||
| * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | |||
| * DISCLAIMED. IN NO EVENT SHALL THE APACHE SOFTWARE FOUNDATION OR | |||
| * ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, | |||
| * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT | |||
| * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF | |||
| * USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | |||
| * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, | |||
| * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT | |||
| * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF | |||
| * SUCH DAMAGE. | |||
| * ==================================================================== | |||
| * | |||
| * This software consists of voluntary contributions made by many | |||
| * individuals on behalf of the Apache Software Foundation. For more | |||
| * information on the Apache Software Foundation, please see | |||
| * <http://www.apache.org/>. | |||
| */ | |||
| package org.apache.tools.ant.input; | |||
| import org.apache.tools.ant.BuildException; | |||
| import javax.swing.*; | |||
| import java.awt.*; | |||
| import java.awt.event.*; | |||
| import java.util.Vector; | |||
| /** | |||
| * Very, very, very simplistic GUI input handler, nothing more than a | |||
| * proof of concept. | |||
| * | |||
| * <p>I don't intend to commit this to the main branch if my proposal | |||
| * should get accepted.</p> | |||
| * | |||
| * <p>I guess I can use this code to demonstrate why nobody should | |||
| * hire me to do GUI stuff 8-)</p> | |||
| * | |||
| * @author <a href="mailto:stefan.bodewig@epost.de">Stefan Bodewig</a> | |||
| * @version $Revision$ | |||
| */ | |||
| public class SwingInputHandler implements InputHandler { | |||
| private Component inputComponent; | |||
| private boolean done = false; | |||
| public SwingInputHandler() { | |||
| } | |||
| public synchronized void handleInput(final InputRequest request) | |||
| throws BuildException { | |||
| final JFrame frame = new JFrame(); | |||
| frame.setDefaultCloseOperation(JFrame.DO_NOTHING_ON_CLOSE); | |||
| Container cp = frame.getContentPane(); | |||
| cp.setLayout(new BorderLayout()); | |||
| JPanel p = new JPanel(); | |||
| p.setLayout(new BoxLayout(p, BoxLayout.X_AXIS)); | |||
| p.add(new JLabel(request.getPrompt())); | |||
| if (request instanceof MultipleChoiceInputRequest) { | |||
| JComboBox c = new JComboBox(((MultipleChoiceInputRequest) request).getChoices()); | |||
| c.setEditable(false); | |||
| p.add(c); | |||
| inputComponent = c; | |||
| } else { | |||
| p.add((inputComponent = new JTextField(20))); | |||
| } | |||
| cp.add(p, BorderLayout.CENTER); | |||
| JButton button = new JButton("OK"); | |||
| button.addActionListener(new ActionListener() { | |||
| public void actionPerformed(ActionEvent e) { | |||
| request.setInput(getInput()); | |||
| frame.dispose(); | |||
| synchronized (SwingInputHandler.this) { | |||
| SwingInputHandler.this.done = true; | |||
| SwingInputHandler.this.notifyAll(); | |||
| } | |||
| } | |||
| }); | |||
| cp.add(button, BorderLayout.SOUTH); | |||
| frame.pack(); | |||
| frame.show(); | |||
| while (!done) { | |||
| try { | |||
| wait(); | |||
| } catch (InterruptedException e) {} | |||
| } | |||
| done = false; | |||
| } | |||
| private String getInput() { | |||
| if (inputComponent instanceof JTextField) { | |||
| return ((JTextField) inputComponent).getText(); | |||
| } else { | |||
| return ((JComboBox) inputComponent).getSelectedItem().toString(); | |||
| } | |||
| } | |||
| } | |||
| @@ -0,0 +1,149 @@ | |||
| /* | |||
| * The Apache Software License, Version 1.1 | |||
| * | |||
| * Copyright (c) 2001-2002 The Apache Software Foundation. All rights | |||
| * reserved. | |||
| * | |||
| * Redistribution and use in source and binary forms, with or without | |||
| * modification, are permitted provided that the following conditions | |||
| * are met: | |||
| * | |||
| * 1. Redistributions of source code must retain the above copyright | |||
| * notice, this list of conditions and the following disclaimer. | |||
| * | |||
| * 2. Redistributions in binary form must reproduce the above copyright | |||
| * notice, this list of conditions and the following disclaimer in | |||
| * the documentation and/or other materials provided with the | |||
| * distribution. | |||
| * | |||
| * 3. The end-user documentation included with the redistribution, if | |||
| * any, must include the following acknowlegement: | |||
| * "This product includes software developed by the | |||
| * Apache Software Foundation (http://www.apache.org/)." | |||
| * Alternately, this acknowlegement may appear in the software itself, | |||
| * if and wherever such third-party acknowlegements normally appear. | |||
| * | |||
| * 4. The names "The Jakarta Project", "Ant", and "Apache Software | |||
| * Foundation" must not be used to endorse or promote products derived | |||
| * from this software without prior written permission. For written | |||
| * permission, please contact apache@apache.org. | |||
| * | |||
| * 5. Products derived from this software may not be called "Apache" | |||
| * nor may "Apache" appear in their names without prior written | |||
| * permission of the Apache Group. | |||
| * | |||
| * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED | |||
| * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES | |||
| * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | |||
| * DISCLAIMED. IN NO EVENT SHALL THE APACHE SOFTWARE FOUNDATION OR | |||
| * ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, | |||
| * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT | |||
| * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF | |||
| * USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | |||
| * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, | |||
| * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT | |||
| * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF | |||
| * SUCH DAMAGE. | |||
| * ==================================================================== | |||
| * | |||
| * This software consists of voluntary contributions made by many | |||
| * individuals on behalf of the Apache Software Foundation. For more | |||
| * information on the Apache Software Foundation, please see | |||
| * <http://www.apache.org/>. | |||
| */ | |||
| package org.apache.tools.ant.taskdefs; | |||
| import java.util.StringTokenizer; | |||
| import java.util.Vector; | |||
| import org.apache.tools.ant.BuildException; | |||
| import org.apache.tools.ant.Project; | |||
| import org.apache.tools.ant.Task; | |||
| import org.apache.tools.ant.input.InputRequest; | |||
| import org.apache.tools.ant.input.MultipleChoiceInputRequest; | |||
| /** | |||
| * Ant task to read input line from console. | |||
| * | |||
| * @author <a href="mailto:usch@usch.net">Ulrich Schmidt</a> | |||
| */ | |||
| public class Input extends Task { | |||
| private String validargs = null; | |||
| private String message = ""; | |||
| private String addproperty = null; | |||
| /** | |||
| * Defines valid input parameters as comma separated String. If set, input | |||
| * task will reject any input not defined as accepted and requires the user | |||
| * to reenter it. Validargs are case sensitive. If you want 'a' and 'A' to | |||
| * be accepted you need to define both values as accepted arguments. | |||
| * | |||
| * @param validargs A comma separated String defining valid input args. | |||
| */ | |||
| public void setValidargs (String validargs) { | |||
| this.validargs = validargs; | |||
| } | |||
| /** | |||
| * Defines the name of a property to be created from input. Behaviour is | |||
| * according to property task which means that existing properties | |||
| * cannot be overriden. | |||
| * | |||
| * @param addproperty Name for the property to be created from input | |||
| */ | |||
| public void setAddproperty (String addproperty) { | |||
| this.addproperty = addproperty; | |||
| } | |||
| /** | |||
| * Sets the Message which gets displayed to the user during the build run. | |||
| * @param message The message to be displayed. | |||
| */ | |||
| public void setMessage (String message) { | |||
| this.message = message; | |||
| } | |||
| /** | |||
| * Set a multiline message. | |||
| */ | |||
| public void addText(String msg) { | |||
| message += getProject().replaceProperties(msg); | |||
| } | |||
| /** | |||
| * No arg constructor. | |||
| */ | |||
| public Input () { | |||
| } | |||
| /** | |||
| * Actual test method executed by jakarta-ant. | |||
| * @exception BuildException | |||
| */ | |||
| public void execute () throws BuildException { | |||
| InputRequest request = null; | |||
| if (validargs != null) { | |||
| Vector accept = new Vector(); | |||
| StringTokenizer stok = new StringTokenizer(validargs, ",", false); | |||
| while (stok.hasMoreTokens()) { | |||
| accept.addElement(stok.nextToken()); | |||
| } | |||
| request = new MultipleChoiceInputRequest(message, accept); | |||
| } else { | |||
| request = new InputRequest(message); | |||
| } | |||
| getProject().getInputHandler().handleInput(request); | |||
| if (addproperty != null) { | |||
| if (project.getProperty(addproperty) == null) { | |||
| project.setProperty(addproperty, request.getInput()); | |||
| } else { | |||
| log("Override ignored for " + addproperty, | |||
| Project.MSG_VERBOSE); | |||
| } | |||
| } | |||
| } | |||
| } | |||
| @@ -0,0 +1 @@ | |||
| This\ is\ a\ prompt\ (1\ and\ 2\ are\ valid)=3 | |||
| @@ -0,0 +1,15 @@ | |||
| <project name="manual-test-for-input" default="both"> | |||
| <target name="single"> | |||
| <input addproperty="single" message="This is a prompt" /> | |||
| <echo>$${single} is ${single}</echo> | |||
| </target> | |||
| <target name="multi"> | |||
| <input addproperty="multi" | |||
| message="This is a prompt (1 and 2 are valid)" | |||
| validargs="1,2" /> | |||
| <echo>$${multi} is ${multi}</echo> | |||
| </target> | |||
| <target name="both" depends="single,multi" /> | |||
| </project> | |||
| @@ -0,0 +1,2 @@ | |||
| This\ is\ a\ prompt\ (1\ and\ 2\ are\ valid)=2 | |||
| This\ is\ a\ prompt=Hi there | |||