blob: 63a84abfc6b06b5eb62725ae401c077aa5825c17 [file] [log] [blame]
/*
* Copyright (C) 2012 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.annotations.NonNull
import com.android.build.gradle.internal.dependency.ManifestDependencyImpl
import com.android.build.gradle.internal.scope.ConventionMappingHelper
import com.android.build.gradle.internal.scope.TaskConfigAction
import com.android.build.gradle.internal.scope.VariantOutputScope
import com.android.build.gradle.internal.scope.VariantScope
import com.android.build.gradle.internal.tasks.PrepareDependenciesTask
import com.android.build.gradle.internal.variant.ApkVariantOutputData
import com.android.build.gradle.internal.variant.BaseVariantData
import com.android.build.gradle.internal.variant.BaseVariantOutputData
import com.android.builder.core.VariantConfiguration
import com.android.builder.dependency.LibraryDependency
import com.android.builder.model.AndroidProject
import com.android.manifmerger.ManifestMerger2
import com.google.common.collect.Lists
import org.gradle.api.tasks.Input
import org.gradle.api.tasks.InputFile
import org.gradle.api.tasks.InputFiles
import org.gradle.api.tasks.Optional
import org.gradle.api.tasks.ParallelizableTask
import static com.android.builder.model.AndroidProject.FD_INTERMEDIATES
import static com.android.builder.model.AndroidProject.FD_OUTPUTS
/**
* A task that processes the manifest
*/
@ParallelizableTask
public class MergeManifests extends ManifestProcessorTask {
// ----- PRIVATE TASK API -----
@InputFile
File getMainManifest() {
return variantConfiguration.getMainManifest();
}
@InputFiles
List<File> getManifestOverlays() {
return variantConfiguration.getManifestOverlays();
}
@Input @Optional
String getPackageOverride() {
return variantConfiguration.getIdOverride();
}
@Input
int getVersionCode() {
if (variantOutputData!= null) {
return variantOutputData.versionCode
}
return variantConfiguration.versionCode;
}
@Input @Optional
String getVersionName() {
if (variantOutputData!= null) {
return variantOutputData.versionName
}
return variantConfiguration.getVersionName();
}
@Input @Optional
String minSdkVersion
@Input @Optional
String targetSdkVersion
@Input @Optional
Integer maxSdkVersion
@Input @Optional
File reportFile
/**
* Return a serializable version of our map of key value pairs for placeholder substitution.
* This serialized form is only used by gradle to compare past and present tasks to determine
* whether a task need to be re-run or not.
*/
@Input @Optional
String getManifestPlaceholders() {
return serializeMap(variantConfiguration.getManifestPlaceholders());
}
VariantConfiguration variantConfiguration
ApkVariantOutputData variantOutputData
List<ManifestDependencyImpl> libraries
/**
* since libraries above can't return it's input files (@Nested doesn't
* work on lists), so do a method that will gather them and return them.
*/
@InputFiles
List<File> getLibraryManifests() {
List<ManifestDependencyImpl> libs = getLibraries()
if (libs == null || libs.isEmpty()) {
return Collections.emptyList();
}
List<File> files = Lists.newArrayListWithCapacity(libs.size() * 2)
for (ManifestDependencyImpl mdi : libs) {
files.addAll(mdi.getAllManifests())
}
return files;
}
@Override
protected void doFullTaskAction() {
getBuilder().mergeManifests(
getMainManifest(),
getManifestOverlays(),
getLibraries(),
getPackageOverride(),
getVersionCode(),
getVersionName(),
getMinSdkVersion(),
getTargetSdkVersion(),
getMaxSdkVersion(),
getManifestOutputFile().absolutePath,
// no appt friendly merged manifest file necessary for applications.
null /* aaptFriendlyManifestOutputFile */ ,
ManifestMerger2.MergeType.APPLICATION,
variantConfiguration.getManifestPlaceholders(),
getReportFile())
}
// ----- ConfigAction -----
public static class ConfigAction implements TaskConfigAction<MergeManifests> {
VariantOutputScope scope
ConfigAction(VariantOutputScope scope) {
this.scope = scope
}
@Override
String getName() {
return scope.getTaskName("process", "Manifest")
}
@Override
Class<MergeManifests> getType() {
return MergeManifests
}
@Override
void execute(MergeManifests processManifestTask) {
BaseVariantOutputData variantOutputData = scope.variantOutputData
BaseVariantData<? extends BaseVariantOutputData> variantData =
scope.variantScope.variantData
VariantConfiguration config = variantData.getVariantConfiguration()
variantOutputData.manifestProcessorTask = processManifestTask
processManifestTask.androidBuilder = scope.globalScope.androidBuilder
processManifestTask.dependsOn variantData.prepareDependenciesTask
if (variantData.generateApkDataTask != null) {
processManifestTask.dependsOn variantData.generateApkDataTask
}
if (scope.compatibleScreensManifestTask != null) {
processManifestTask.dependsOn scope.compatibleScreensManifestTask.name
}
processManifestTask.variantConfiguration = config
if (variantOutputData instanceof ApkVariantOutputData) {
processManifestTask.variantOutputData =
variantOutputData as ApkVariantOutputData
}
ConventionMappingHelper.map(processManifestTask, "libraries") {
List<ManifestDependencyImpl> manifests =
getManifestDependencies(config.directLibraries)
if (variantData.generateApkDataTask != null &&
variantData.getVariantConfiguration().getBuildType().
isEmbedMicroApp()) {
manifests.add(new ManifestDependencyImpl(
variantData.generateApkDataTask.getManifestFile(), []))
}
if (scope.compatibleScreensManifestTask != null) {
manifests.add(new ManifestDependencyImpl(
scope.getCompatibleScreensManifestFile(), []))
}
return manifests
}
ConventionMappingHelper.map(processManifestTask, "minSdkVersion") {
if (scope.globalScope.androidBuilder.isPreviewTarget()) {
return scope.globalScope.androidBuilder.getTargetCodename()
}
config.mergedFlavor.minSdkVersion?.apiString
}
ConventionMappingHelper.map(processManifestTask, "targetSdkVersion") {
if (scope.globalScope.androidBuilder.isPreviewTarget()) {
return scope.globalScope.androidBuilder.getTargetCodename()
}
return config.mergedFlavor.targetSdkVersion?.apiString
}
ConventionMappingHelper.map(processManifestTask, "maxSdkVersion") {
if (scope.globalScope.androidBuilder.isPreviewTarget()) {
return null
}
return config.mergedFlavor.maxSdkVersion
}
ConventionMappingHelper.map(processManifestTask, "manifestOutputFile") {
scope.getManifestOutputFile()
}
ConventionMappingHelper.map(processManifestTask, "reportFile") {
new File(
"${scope.getGlobalScope().getBuildDir()}/${FD_OUTPUTS}/logs/manifest-merger-${config.baseName}-report.txt")
}
}
@NonNull
private static List<ManifestDependencyImpl> getManifestDependencies(
List<LibraryDependency> libraries) {
List<ManifestDependencyImpl> list = Lists.newArrayListWithCapacity(libraries.size())
for (LibraryDependency lib : libraries) {
// get the dependencies
List<ManifestDependencyImpl> children = getManifestDependencies(lib.dependencies)
list.add(new ManifestDependencyImpl(lib.getName(), lib.manifest, children))
}
return list
}
}
}