/* * The Apache Software License, Version 1.1 * * Copyright (c) 1999 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 * . */ package org.apache.tools.ant.taskdefs; import org.apache.tools.ant.*; import org.apache.tools.ant.types.Reference; import java.io.*; import java.util.*; /** * Will set a Project property. Used to be a hack in ProjectHelper * Will not override values set by the command line or parent projects. * * @author costin@dnt.ro * @author Sam Ruby * @author Glenn McAllister */ public class Property extends Task { protected String name; protected String value; protected File file; protected String resource; protected Reference ref = null; protected boolean userProperty=false; // set read-only properties public void setName(String name) { this.name = name; } public String getName() { return name; } public void setLocation(File location) { setValue(location.getAbsolutePath()); } public void setValue(String value) { this.value = value; } public String getValue() { return value; } public void setFile(File file) { this.file = file; } public File getFile() { return file; } public void setRefid(Reference ref) { this.ref = ref; } public Reference getRefid() { return ref; } public void setResource(String resource) { this.resource = resource; } public String getResource() { return resource; } public void setUserProperty(boolean userProperty) { this.userProperty = userProperty; } public String toString() { return value == null ? "" : value; } public void execute() throws BuildException { try { if ((name != null) && (value != null)) { addProperty(name, value); } if (file != null) loadFile(file); if (resource != null) loadResource(resource); if ((name != null) && (ref != null)) { Object obj = ref.getReferencedObject(getProject()); if (obj != null) { addProperty(name, obj.toString()); } } } catch (Exception e) { throw new BuildException(e, location); } } protected void loadFile (File file) throws BuildException { Properties props = new Properties(); log("Loading " + file.getAbsolutePath(), Project.MSG_VERBOSE); try { if (file.exists()) { FileInputStream fis = new FileInputStream(file); try { props.load(fis); } finally { if (fis != null) { fis.close(); } } addProperties(props); } else { log("Unable to find " + file.getAbsolutePath(), Project.MSG_VERBOSE); } } catch(Exception ex) { throw new BuildException(ex.getMessage(), ex, location); } } protected void loadResource( String name ) { Properties props = new Properties(); log("Resource Loading " + name, Project.MSG_VERBOSE); try { ClassLoader cL = this.getClass().getClassLoader(); InputStream is = null; if (cL == null) { is = ClassLoader.getSystemResourceAsStream(name); } else { is = cL.getResourceAsStream(name); } if (is != null) { props.load(is); addProperties(props); } else { log("Unable to find resource " + name, Project.MSG_WARN); } } catch (Exception ex) { throw new BuildException(ex, location); } } protected void addProperties(Properties props) { resolveAllProperties(props); Enumeration e = props.keys(); while (e.hasMoreElements()) { String name = (String) e.nextElement(); String value = (String) props.getProperty(name); String v = ProjectHelper.replaceProperties(project, value, project.getProperties()); addProperty(name, value); } } protected void addProperty(String n, String v) { if( userProperty ) { if (project.getUserProperty(n) == null) { project.setUserProperty(n, v); } else { log("Override ignored for " + n, Project.MSG_VERBOSE); } } else { if (project.getProperty(n) == null) { project.setProperty(n, v); } else { log("Override ignored for " + n, Project.MSG_VERBOSE); } } } private void resolveAllProperties(Hashtable props) { Hashtable unresolvableProperties = new Hashtable(); for (Enumeration e = props.keys(); e.hasMoreElements(); ) { String name = (String) e.nextElement(); String value = (String) props.get(name); boolean resolved = false; while (!resolved) { Vector propsInValue = new Vector(); // assume it will be resolved resolved = true; boolean unresolvable = false; if (extractProperties(value, propsInValue)) { for (int i=0; i < propsInValue.size(); i++) { String elem = (String) propsInValue.elementAt(i); if (elem.equals(name) || unresolvableProperties.containsKey(elem)) { // we won't try further resolving elements with circular // property dependencies or dependencies on unresolvable elements unresolvable = true; break; } if (project.getProperties().containsKey(elem) || props.containsKey(elem)) { resolved = false; } } } if (unresolvable) { unresolvableProperties.put(name, value); resolved = true; } if (!resolved) { value = ProjectHelper.replaceProperties(project, value, project.getProperties()); value = ProjectHelper.replaceProperties(project, value, props); props.put(name, value); } } } } private boolean extractProperties(String source, Vector properties) { // This is an abreviated version of // ProjectHelper.replaceProperties method int i=0; int prev=0; int pos; while( (pos=source.indexOf( "$", prev )) >= 0 ) { if( pos == (source.length() - 1)) { prev = pos + 1; } else if (source.charAt( pos + 1 ) != '{' ) { prev=pos+2; } else { int endName=source.indexOf( '}', pos ); String n=source.substring( pos+2, endName ); properties.addElement(n); prev=endName+1; } } return (properties.size() > 0); } }