All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.jayway.maven.plugins.android.phase08preparepackage.DexMojo Maven / Gradle / Ivy

There is a newer version: 4.0.0-rc.2
Show newest version
/*
 * Copyright (C) 2009 Jayway AB
 * Copyright (C) 2007-2008 JVending Masa
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package com.jayway.maven.plugins.android.phase08preparepackage;

import java.io.File;
import java.io.IOException;
import java.util.ArrayList;
import java.util.HashSet;
import java.util.List;
import java.util.Set;

import org.apache.commons.io.FileUtils;
import org.apache.commons.lang.StringUtils;
import org.apache.maven.artifact.Artifact;
import org.apache.maven.model.Resource;
import org.apache.maven.plugin.MojoExecutionException;
import org.apache.maven.plugin.MojoFailureException;
import org.codehaus.plexus.archiver.ArchiverException;
import org.codehaus.plexus.archiver.jar.JarArchiver;
import org.codehaus.plexus.archiver.util.DefaultFileSet;

import com.jayway.maven.plugins.android.AbstractAndroidMojo;
import com.jayway.maven.plugins.android.CommandExecutor;
import com.jayway.maven.plugins.android.ExecutionException;
import com.jayway.maven.plugins.android.configuration.Dex;
import com.jayway.maven.plugins.android.phase04processclasses.ProguardMojo;

import static com.jayway.maven.plugins.android.common.AndroidExtension.AAR;


/**
 * Converts compiled Java classes to the Android dex format.
 * 
 * @author [email protected]
 * @goal dex
 * @phase prepare-package
 * @requiresDependencyResolution compile
 */
public class DexMojo extends AbstractAndroidMojo
{

    /**
     * Configuration for the dex command execution. It can be configured in the plugin configuration like so
     * 
     * 
     * <dex>
     *   <jvmArguments>
     *     <jvmArgument>-Xms256m</jvmArgument>
     *     <jvmArgument>-Xmx512m</jvmArgument>
     *   </jvmArguments>
     *   <coreLibrary>true|false</coreLibrary>
     *   <noLocals>true|false</noLocals>
     *   <optimize>true|false</optimize>
     *   <preDex>true|false</preDex>
     *   <preDexLibLocation>path to predexed libraries, defaults to target/dexedLibs</preDexLibLocation>
     *   <incremental>true|false</incremental>
     * </dex>
     * 
*

* or via properties dex.* or command line parameters android.dex.* * * @parameter */ private Dex dex; /** * Extra JVM Arguments. Using these you can e.g. increase memory for the jvm running the build. * * @parameter expression="${android.dex.jvmArguments}" default-value="-Xmx1024M" * @optional */ private String[] dexJvmArguments; /** * Decides whether to pass the --core-library flag to dx. * * @parameter expression="${android.dex.coreLibrary}" default-value="false" */ private boolean dexCoreLibrary; /** * Decides whether to pass the --no-locals flag to dx. * * @parameter expression="${android.dex.noLocals}" default-value="false" */ private boolean dexNoLocals; /** * Decides whether to pass the --no-optimize flag to dx. * * @parameter expression="${android.dex.optimize}" default-value="true" */ private boolean dexOptimize; /** * Decides whether to predex the jars. * * @parameter expression="${android.dex.predex}" default-value="false" */ private boolean dexPreDex; /** * Path to predexed libraries. * * @parameter expression="${android.dex.dexPreDexLibLocation}" default-value= * "${project.build.directory}${file.separator}dexedLibs" */ private String dexPreDexLibLocation; /** * Decides whether to pass the --incremental flag to dx. * * @parameter expression="${android.dex.incremental}" default-value="false" */ private boolean dexIncremental; private String[] parsedJvmArguments; private boolean parsedCoreLibrary; private boolean parsedNoLocals; private boolean parsedOptimize; private boolean parsedPreDex; private String parsedPreDexLibLocation; private boolean parsedIncremental; /** * @throws MojoExecutionException * @throws MojoFailureException */ @Override public void execute() throws MojoExecutionException, MojoFailureException { CommandExecutor executor = CommandExecutor.Factory.createDefaultCommmandExecutor(); executor.setLogger( this.getLog() ); File outputFile = new File( project.getBuild().getDirectory() + File.separator + "classes.dex" ); Set< File > inputFiles = getDexInputFiles(); parseConfiguration(); if ( generateApk ) { runDex( executor, outputFile, inputFiles ); } if ( attachJar ) { File jarFile = new File( project.getBuild().getDirectory() + File.separator + project.getBuild().getFinalName() + ".jar" ); projectHelper.attachArtifact( project, "jar", project.getArtifact().getClassifier(), jarFile ); } if ( attachSources ) { // Also attach an .apksources, containing sources from this project. final File apksources = createApkSourcesFile(); projectHelper.attachArtifact( project, "apksources", apksources ); } } /** * Gets the input files for dex. This is a combination of directories and jar files. * * @return */ private Set< File > getDexInputFiles() { Set< File > inputs = new HashSet< File >(); // ugly, don't know a better way to get this in mvn File proguardJar = new File( project.getBuild().getDirectory(), ProguardMojo.PROGUARD_OBFUSCATED_JAR ); getLog().debug( "Checking for existence of: " + proguardJar.toString() ); if ( proguardJar.exists() ) { // progurad has been run, use this jar getLog().debug( "Obfuscated jar exists, using that as input" ); inputs.add( proguardJar ); } else { getLog().debug( "Using non-obfuscated input" ); // no proguard, use original config inputs.add( new File( project.getBuild().getOutputDirectory() ) ); for ( Artifact artifact : getAllRelevantDependencyArtifacts() ) { if ( artifact.getType().equals( AAR ) ) { final String apkLibResDir = getLibraryUnpackDirectory( artifact ) + "/classes.jar"; if ( new File( apkLibResDir ).exists() ) { inputs.add( new File( apkLibResDir ) ); } } else if ( artifact.getType().equals( "so" ) || artifact.getType().equals( "a" ) ) { // Ignore native dependencies - no need for dexer to see those continue; } else { inputs.add( artifact.getFile().getAbsoluteFile() ); } } } return inputs; } private void parseConfiguration() { // config in pom found if ( dex != null ) { // the if statements make sure that properties/command line // parameter overrides configuration // and that the dafaults apply in all cases; if ( dex.getJvmArguments() == null ) { parsedJvmArguments = dexJvmArguments; } else { parsedJvmArguments = dex.getJvmArguments(); } if ( dex.isCoreLibrary() == null ) { parsedCoreLibrary = dexCoreLibrary; } else { parsedCoreLibrary = dex.isCoreLibrary(); } if ( dex.isNoLocals() == null ) { parsedNoLocals = dexNoLocals; } else { parsedNoLocals = dex.isNoLocals(); } if ( dex.isOptimize() == null ) { parsedOptimize = dexOptimize; } else { parsedOptimize = dex.isOptimize(); } if ( dex.isPreDex() == null ) { parsedPreDex = dexPreDex; } else { parsedPreDex = dex.isPreDex(); } if ( dex.getPreDexLibLocation() == null ) { parsedPreDexLibLocation = dexPreDexLibLocation; } else { parsedPreDexLibLocation = dex.getPreDexLibLocation(); } if ( dex.isIncremental() == null ) { parsedIncremental = dexIncremental; } else { parsedIncremental = dex.isIncremental(); } } else { parsedJvmArguments = dexJvmArguments; parsedCoreLibrary = dexCoreLibrary; parsedNoLocals = dexNoLocals; parsedOptimize = dexOptimize; parsedPreDex = dexPreDex; parsedPreDexLibLocation = dexPreDexLibLocation; parsedIncremental = dexIncremental; } } private Set< File > preDex( CommandExecutor executor, Set< File > inputFiles ) throws MojoExecutionException { Set< File > filtered = new HashSet< File >(); getLog().info( "Pre dex-ing libraries for faster dex-ing of the final application." ); for ( File inputFile : inputFiles ) { if ( inputFile.getName().matches( ".*\\.jar$" ) ) { List< String > commands = dexDefaultCommands(); File predexJar = predexJarPath( inputFile ); commands.add( "--output=" + predexJar.getAbsolutePath() ); commands.add( inputFile.getAbsolutePath() ); filtered.add( predexJar ); if ( !predexJar.isFile() || predexJar.lastModified() < inputFile.lastModified() ) { getLog().info( "Pre-dex ing jar: " + inputFile.getAbsolutePath() ); final String javaExecutable = getJavaExecutable().getAbsolutePath(); getLog().info( javaExecutable + " " + commands.toString() ); try { executor.executeCommand( javaExecutable, commands, project.getBasedir(), false ); } catch ( ExecutionException e ) { throw new MojoExecutionException( "", e ); } } } else { filtered.add( inputFile ); } } return filtered; } private File predexJarPath( File inputFile ) { final String slash = File.separator; final File predexLibsDirectory = new File( parsedPreDexLibLocation.trim() ); predexLibsDirectory.mkdirs(); return new File( predexLibsDirectory.getAbsolutePath() + slash + inputFile.getName() ); } private List< String > dexDefaultCommands() throws MojoExecutionException { List< String > commands = new ArrayList< String >(); if ( parsedJvmArguments != null ) { for ( String jvmArgument : parsedJvmArguments ) { // preserve backward compatibility allowing argument with or // without dash (e.g. Xmx512m as well as // -Xmx512m should work) (see // http://code.google.com/p/maven-android-plugin/issues/detail?id=153) if ( !jvmArgument.startsWith( "-" ) ) { jvmArgument = "-" + jvmArgument; } getLog().debug( "Adding jvm argument " + jvmArgument ); commands.add( jvmArgument ); } } commands.add( "-jar" ); commands.add( getAndroidSdk().getDxJarPath() ); commands.add( "--dex" ); return commands; } private void runDex( CommandExecutor executor, File outputFile, Set< File > inputFiles ) throws MojoExecutionException { List< String > commands = dexDefaultCommands(); Set< File > filteredFiles = inputFiles; if ( parsedPreDex ) { filteredFiles = preDex( executor, inputFiles ); } if ( !parsedOptimize ) { commands.add( "--no-optimize" ); } if ( parsedCoreLibrary ) { commands.add( "--core-library" ); } if ( parsedIncremental ) { commands.add( "--incremental" ); } commands.add( "--output=" + outputFile.getAbsolutePath() ); if ( parsedNoLocals ) { commands.add( "--no-locals" ); } for ( File inputFile : filteredFiles ) { getLog().debug( "Adding dex input: " + inputFile.getAbsolutePath() ); commands.add( inputFile.getAbsolutePath() ); } final String javaExecutable = getJavaExecutable().getAbsolutePath(); getLog().info( javaExecutable + " " + commands.toString() ); try { executor.executeCommand( javaExecutable, commands, project.getBasedir(), false ); } catch ( ExecutionException e ) { throw new MojoExecutionException( "", e ); } } /** * Figure out the full path to the current java executable. * * @return the full path to the current java executable. */ private static File getJavaExecutable() { final String javaHome = System.getProperty( "java.home" ); final String slash = File.separator; return new File( javaHome + slash + "bin" + slash + "java" ); } /** * @return * @throws MojoExecutionException */ protected File createApkSourcesFile() throws MojoExecutionException { final File apksources = new File( project.getBuild().getDirectory(), project.getBuild().getFinalName() + ".apksources" ); FileUtils.deleteQuietly( apksources ); try { JarArchiver jarArchiver = new JarArchiver(); jarArchiver.setDestFile( apksources ); addDirectory( jarArchiver, assetsDirectory, "assets" ); addDirectory( jarArchiver, resourceDirectory, "res" ); addDirectory( jarArchiver, sourceDirectory, "src/main/java" ); addJavaResources( jarArchiver, project.getBuild().getResources() ); jarArchiver.createArchive(); } catch ( ArchiverException e ) { throw new MojoExecutionException( "ArchiverException while creating .apksource file.", e ); } catch ( IOException e ) { throw new MojoExecutionException( "IOException while creating .apksource file.", e ); } return apksources; } /** * Makes sure the string ends with "/" * * @param prefix * any string, or null. * @return the prefix with a "/" at the end, never null. */ protected String endWithSlash( String prefix ) { prefix = StringUtils.defaultIfEmpty( prefix, "/" ); if ( !prefix.endsWith( "/" ) ) { prefix = prefix + "/"; } return prefix; } /** * Adds a directory to a {@link JarArchiver} with a directory prefix. * * @param jarArchiver * @param directory * The directory to add. * @param prefix * An optional prefix for where in the Jar file the directory's contents should go. */ protected void addDirectory( JarArchiver jarArchiver, File directory, String prefix ) { if ( directory != null && directory.exists() ) { final DefaultFileSet fileSet = new DefaultFileSet(); fileSet.setPrefix( endWithSlash( prefix ) ); fileSet.setDirectory( directory ); jarArchiver.addFileSet( fileSet ); } } /** * @param jarArchiver * @param javaResources */ protected void addJavaResources( JarArchiver jarArchiver, List< Resource > javaResources ) { for ( Resource javaResource : javaResources ) { addJavaResource( jarArchiver, javaResource ); } } /** * Adds a Java Resources directory (typically "src/main/resources") to a {@link JarArchiver}. * * @param jarArchiver * @param javaResource * The Java resource to add. */ protected void addJavaResource( JarArchiver jarArchiver, Resource javaResource ) { if ( javaResource != null ) { final File javaResourceDirectory = new File( javaResource.getDirectory() ); if ( javaResourceDirectory.exists() ) { final DefaultFileSet javaResourceFileSet = new DefaultFileSet(); javaResourceFileSet.setDirectory( javaResourceDirectory ); javaResourceFileSet.setPrefix( endWithSlash( "src/main/resources" ) ); jarArchiver.addFileSet( javaResourceFileSet ); } } } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy