com.android.build.gradle.tasks.PackageApplication Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of gradle-core Show documentation
Show all versions of gradle-core Show documentation
Core library to build Android Gradle plugin.
The newest version!
/*
* Copyright (C) 2016 The Android Open Source Project
*
* 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.android.build.gradle.tasks;
import com.android.SdkConstants;
import com.android.annotations.NonNull;
import com.android.annotations.Nullable;
import com.android.build.gradle.internal.incremental.DexPackagingPolicy;
import com.android.build.gradle.internal.incremental.FileType;
import com.android.build.gradle.internal.incremental.InstantRunPatchingPolicy;
import com.android.build.gradle.internal.scope.ConventionMappingHelper;
import com.android.build.gradle.internal.scope.PackagingScope;
import com.android.builder.profile.ProcessProfileWriter;
import com.android.ide.common.res2.FileStatus;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableSet;
import com.google.common.io.Files;
import com.google.wireless.android.sdk.stats.GradleBuildProjectMetrics;
import java.io.BufferedOutputStream;
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.util.Collections;
import java.util.Map;
import java.util.zip.ZipEntry;
import java.util.zip.ZipOutputStream;
import org.gradle.api.tasks.ParallelizableTask;
/**
* Task to package an Android application (APK).
*/
@ParallelizableTask
public class PackageApplication extends PackageAndroidArtifact {
@Override
protected boolean isIncremental() {
return true;
}
@Override
protected void doFullTaskAction() throws IOException {
super.doFullTaskAction();
recordMetrics();
}
@Override
protected void doIncrementalTaskAction(Map changedInputs) throws IOException {
super.doIncrementalTaskAction(changedInputs);
recordMetrics();
}
private File zipDexesForInstantRun(Iterable dexFolders,
ImmutableSet.Builder dexFoldersForApk)
throws IOException {
File tmpZipFile = new File(instantRunSupportDir, "classes.zip");
Files.createParentDirs(tmpZipFile);
ZipOutputStream zipFile = new ZipOutputStream(
new BufferedOutputStream(new FileOutputStream(tmpZipFile)));
// no need to compress a zip, the APK itself gets compressed.
zipFile.setLevel(0);
try {
for (File dexFolder : dexFolders) {
if (dexFolder.getName().contains(INSTANT_RUN_PACKAGES_PREFIX)) {
dexFoldersForApk.add(dexFolder);
} else {
for (File file : Files.fileTreeTraverser().breadthFirstTraversal(dexFolder)) {
if (file.isFile() && file.getName().endsWith(SdkConstants.DOT_DEX)) {
// There are several pieces of code in the runtime library which depends on
// this exact pattern, so it should not be changed without thorough testing
// (it's basically part of the contract).
String entryName = file.getParentFile().getName() + "-" + file.getName();
zipFile.putNextEntry(new ZipEntry(entryName));
try {
Files.copy(file, zipFile);
} finally {
zipFile.closeEntry();
}
}
}
}
}
} finally {
zipFile.close();
}
// now package that zip file as a zip since this is what the packager is expecting !
File finalResourceFile = new File(instantRunSupportDir, "resources.zip");
zipFile = new ZipOutputStream(new BufferedOutputStream(
new FileOutputStream(finalResourceFile)));
try {
zipFile.putNextEntry(new ZipEntry("instant-run.zip"));
try {
Files.copy(tmpZipFile, zipFile);
} finally {
zipFile.closeEntry();
}
} finally {
zipFile.close();
}
return finalResourceFile;
}
private void recordMetrics() {
long metricsStartTime = System.nanoTime();
GradleBuildProjectMetrics.Builder metrics = GradleBuildProjectMetrics.newBuilder();
Long apkSize = getSize(getOutputFile());
if (apkSize != null) {
metrics.setApkSize(apkSize);
}
Long resourcesApSize = getSize(getResourceFile());
if (resourcesApSize != null) {
metrics.setResourcesApSize(resourcesApSize);
}
metrics.setMetricsTimeNs(System.nanoTime() - metricsStartTime);
ProcessProfileWriter.getProject(getProject().getPath()).setMetrics(metrics);
}
@Nullable
private static Long getSize(@Nullable File file) {
if (file == null) {
return null;
}
try {
return java.nio.file.Files.size(file.toPath());
} catch (IOException e) {
return null;
}
}
// ----- ConfigAction -----
/**
* Configures the task to perform the "standard" packaging, including all
* files that should end up in the APK.
*/
public static class StandardConfigAction
extends PackageAndroidArtifact.ConfigAction {
public StandardConfigAction(
@NonNull PackagingScope scope,
@Nullable InstantRunPatchingPolicy patchingPolicy) {
super(scope, patchingPolicy);
}
@NonNull
@Override
public String getName() {
return packagingScope.getTaskName("package");
}
@NonNull
@Override
public Class getType() {
return PackageApplication.class;
}
@Override
public void execute(@NonNull final PackageApplication packageApplication) {
ConventionMappingHelper.map(
packageApplication,
"outputFile",
packagingScope::getOutputPackage);
super.execute(packageApplication);
}
}
/**
* Configures the task to only package resources and assets.
*/
public static class InstantRunResourcesConfigAction
extends PackageAndroidArtifact.ConfigAction {
@NonNull
private final File mOutputFile;
public InstantRunResourcesConfigAction(
@NonNull File outputFile,
@NonNull PackagingScope scope,
@Nullable InstantRunPatchingPolicy patchingPolicy) {
super(scope, patchingPolicy);
mOutputFile = outputFile;
}
@NonNull
@Override
public String getName() {
return packagingScope.getTaskName("packageInstantRunResources");
}
@NonNull
@Override
public Class getType() {
return PackageApplication.class;
}
@Override
public void execute(@NonNull final PackageApplication packageApplication) {
packageApplication.setOutputFile(mOutputFile);
super.execute(packageApplication);
packageApplication.instantRunFileType = FileType.RESOURCES;
// Don't try to add any special dex files to this zip.
packageApplication.dexPackagingPolicy = DexPackagingPolicy.STANDARD;
// Skip files which are not needed for hot/cold swap.
for (String prop : ImmutableList.of("dexFolders", "jniFolders", "javaResourceFiles")) {
ConventionMappingHelper.map(
packageApplication, prop, Collections::emptySet);
}
// Don't sign.
packageApplication.setSigningConfig(null);
}
}
}