diff --git a/src/main/org/apache/tools/ant/taskdefs/defaults.properties b/src/main/org/apache/tools/ant/taskdefs/defaults.properties index d2b3a4b1c..308dc0d58 100644 --- a/src/main/org/apache/tools/ant/taskdefs/defaults.properties +++ b/src/main/org/apache/tools/ant/taskdefs/defaults.properties @@ -160,6 +160,7 @@ echoproperties=org.apache.tools.ant.taskdefs.optional.EchoProperties splash=org.apache.tools.ant.taskdefs.optional.splash.SplashTask serverdeploy=org.apache.tools.ant.taskdefs.optional.j2ee.ServerDeploy jarlib-display=org.apache.tools.ant.taskdefs.optional.extension.JarLibDisplayTask +jarlib-manifest=org.apache.tools.ant.taskdefs.optional.extension.JarLibManifestTask # deprecated ant tasks (kept for back compatibility) starteam=org.apache.tools.ant.taskdefs.optional.scm.AntStarTeamCheckOut diff --git a/src/main/org/apache/tools/ant/taskdefs/optional/extension/JarLibManifestTask.java b/src/main/org/apache/tools/ant/taskdefs/optional/extension/JarLibManifestTask.java new file mode 100644 index 000000000..7f485a98c --- /dev/null +++ b/src/main/org/apache/tools/ant/taskdefs/optional/extension/JarLibManifestTask.java @@ -0,0 +1,378 @@ +/* + * 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 + * . + */ +package org.apache.tools.ant.taskdefs.optional.extension; + +import java.io.File; +import java.io.FileOutputStream; +import java.io.IOException; +import java.util.ArrayList; +import java.util.Iterator; +import java.util.jar.Attributes; +import java.util.jar.Manifest; +import org.apache.tools.ant.BuildException; +import org.apache.tools.ant.Task; +import org.apache.tools.ant.Project; + +/** + * Task to generate a manifest that declares all the dependencies + * in manifest. The dependencies are determined by looking in the + * specified path and searching for Extension / "Optional Package" + * specifications in the manifests of the jars. + * + *

Prior to JDK1.3, an "Optional Package" was known as an Extension. + * The specification for this mechanism is available in the JDK1.3 + * documentation in the directory + * $JDK_HOME/docs/guide/extensions/versioning.html. Alternatively it is + * available online at + * http://java.sun.com/j2se/1.3/docs/guide/extensions/versioning.html.

+ * + * @author Peter Donald + * @ant.task name="jarlib-manifest" + */ +public final class JarLibManifestTask + extends Task +{ + /** + * Version of manifest spec that task generates. + */ + private static final String MANIFEST_VERSION = "1.0"; + + /** + * "Created-By" string used when creating manifest. + */ + private static final String CREATED_BY = "Created-By"; + + /** + * The library to display information about. + */ + private File m_destfile; + + /** + * The extension supported by this library (if any). + */ + private Extension m_extension; + + /** + * ExtensionAdapter objects representing + * dependencies required by library. + */ + private final ArrayList m_dependencies = new ArrayList(); + + /** + * ExtensionAdapter objects representing optional + * dependencies required by library. + */ + private final ArrayList m_optionals = new ArrayList(); + + /** + * Extra attributes the user specifies for main section + * in manifest. + */ + private final ArrayList m_extraAttributes = new ArrayList(); + + /** + * The location where generated manifest is placed. + * + * @param destfile The location where generated manifest is placed. + */ + public void setDestfile( final File destfile ) + { + m_destfile = destfile; + } + + /** + * Adds an extension that this library implements. + * + * @param extensionAdapter an extension that this library implements. + */ + public void addConfiguredExtension( final ExtensionAdapter extensionAdapter ) + throws BuildException + { + if( null != m_extension ) + { + final String message = + "Can not have multiple extensions defined in one library."; + throw new BuildException( message ); + } + else + { + m_extension = extensionAdapter.toExtension(); + } + } + + /** + * Adds a set of extensions that this library requires. + * + * @param extensionSet a set of extensions that this library requires. + */ + public void addConfiguredDepends( final ExtensionSet extensionSet ) + { + m_dependencies.add( extensionSet ); + } + + /** + * Adds a set of extensions that this library optionally requires. + * + * @param extensionSet a set of extensions that this library optionally requires. + */ + public void addConfiguredOptions( final ExtensionSet extensionSet ) + { + m_optionals.add( extensionSet ); + } + + /** + * Adds an attribute that is to be put in main section of manifest. + * + * @param attribute an attribute that is to be put in main section of manifest. + */ + public void addConfiguredAttribute( final ExtraAttribute attribute ) + { + m_extraAttributes.add( attribute ); + } + + public void execute() + throws BuildException + { + validate(); + + final Manifest manifest = new Manifest(); + final Attributes attributes = manifest.getMainAttributes(); + + attributes.put( Attributes.Name.MANIFEST_VERSION, MANIFEST_VERSION ); + final String createdBy = "Apache Ant " + getProject().getProperty( "ant.version" ); + attributes.putValue( CREATED_BY, createdBy ); + + appendExtraAttributes( attributes ); + + if( null != m_extension ) + { + Extension.addExtension( m_extension, attributes ); + } + + //Add all the dependency data to manifest for dependencies + final ArrayList depends = toExtensions( m_dependencies ); + appendExtensionList( attributes, + Extension.EXTENSION_LIST, + "lib", + depends.size() ); + appendLibraryList( attributes, "lib", depends ); + + //Add all the dependency data to manifest for "optional" + //dependencies + final ArrayList option = toExtensions( m_optionals ); + appendExtensionList( attributes, + Extension.OPTIONAL_EXTENSION_LIST, + "opt", + option.size() ); + appendLibraryList( attributes, "opt", option ); + + try + { + final String message = "Generating manifest " + m_destfile.getAbsoluteFile(); + log( message, Project.MSG_INFO ); + writeManifest( manifest ); + } + catch( final IOException ioe ) + { + throw new BuildException( ioe.getMessage(), ioe ); + } + } + + /** + * Validate the tasks parameters. + * + * @throws BuildException if invalid parameters found + */ + private void validate() + throws BuildException + { + if( null == m_destfile ) + { + final String message = "Destfile attribute not specified."; + throw new BuildException( message ); + } + if( m_destfile.exists() && !m_destfile.isFile() ) + { + final String message = m_destfile + " is not a file."; + throw new BuildException( message ); + } + } + + /** + * Add any extra attributes to the manifest. + * + * @param attributes the manifest section to write + * attributes to + */ + private void appendExtraAttributes( final Attributes attributes ) + { + final Iterator iterator = m_extraAttributes.iterator(); + while( iterator.hasNext() ) + { + final ExtraAttribute attribute = + (ExtraAttribute)iterator.next(); + attributes.putValue( attribute.getName(), + attribute.getValue() ); + } + } + + /** + * Write out manifest to destfile. + * + * @param manifest the manifest + * @throws IOException if error writing file + */ + private void writeManifest( final Manifest manifest ) + throws IOException + { + FileOutputStream output = null; + try + { + output = new FileOutputStream( m_destfile ); + manifest.write( output ); + output.flush(); + } + finally + { + if( null != output ) + { + try + { + output.close(); + } + catch( IOException e ) + { + } + } + } + } + + /** + * Append specified extensions to specified attributes. + * Use the extensionKey to list the extensions, usually "Extension-List:" + * for required dependencies and "Optional-Extension-List:" for optional + * dependencies. NOTE: "Optional" dependencies are not part of the + * specification. + * + * @param attributes the attributes to add extensions to + * @param extensions the list of extensions + * @throws BuildException if an error occurs + */ + private void appendLibraryList( final Attributes attributes, + final String listPrefix, + final ArrayList extensions ) + throws BuildException + { + final int size = extensions.size(); + for( int i = 0; i < size; i++ ) + { + final Extension extension = (Extension)extensions.get( i ); + final String prefix = listPrefix + i + "-"; + Extension.addExtension( extension, prefix, attributes ); + } + } + + /** + * Append an attribute such as "Extension-List: lib0 lib1 lib2" + * using specified prefix and counting up to specified size. + * Also use specified extensionKey so that can generate list of + * optional dependencies aswell. + * + * @param size the number of librarys to list + * @param listPrefix the prefix for all librarys + * @param attributes the attributes to add key-value to + * @param extensionKey the key to use + */ + private void appendExtensionList( final Attributes attributes, + final Attributes.Name extensionKey, + final String listPrefix, + final int size ) + { + final StringBuffer sb = new StringBuffer(); + for( int i = 0; i < size; i++ ) + { + sb.append( listPrefix + i ); + sb.append( ' ' ); + } + + //add in something like + //"Extension-List: javahelp java3d" + attributes.put( extensionKey, sb.toString() ); + } + + /** + * Convert a list of ExtensionSet objects to extensions. + * + * @param extensionSets the list of ExtensionSets to add to list + * @throws BuildException if an error occurs + */ + private ArrayList toExtensions( final ArrayList extensionSets ) + throws BuildException + { + final ArrayList results = new ArrayList(); + + final int size = extensionSets.size(); + for( int i = 0; i < size; i++ ) + { + final ExtensionSet set = (ExtensionSet)extensionSets.get( i ); + final Extension[] extensions = set.toExtensions( getProject() ); + for( int j = 0; j < extensions.length; j++ ) + { + results.add( extensions[ j ] ); + } + } + + return results; + } +}