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 |