blob: c2736191edf0d8db19150b761e6da72bdf37f2bc [file] [log] [blame]
// Copyright 2017 The Bazel Authors. All rights reserved.
//
// 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.google.devtools.build.lib.rules.android;
import static com.google.common.collect.ImmutableList.toImmutableList;
import static com.google.common.collect.Iterables.getOnlyElement;
import static com.google.common.truth.Truth.assertThat;
import static com.google.common.truth.Truth.assertWithMessage;
import static com.google.devtools.build.lib.actions.util.ActionsTestUtil.getFirstArtifactEndingWith;
import static com.google.devtools.build.lib.actions.util.ActionsTestUtil.prettyArtifactNames;
import static com.google.devtools.build.lib.rules.java.JavaCompileActionTestHelper.getJavacArguments;
import static com.google.devtools.build.lib.rules.java.JavaCompileActionTestHelper.getProcessorpath;
import static com.google.devtools.build.lib.testutil.MoreAsserts.assertThrows;
import com.google.common.base.Ascii;
import com.google.common.base.Joiner;
import com.google.common.base.Predicate;
import com.google.common.base.Predicates;
import com.google.common.collect.Collections2;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableMap;
import com.google.common.collect.ImmutableSet;
import com.google.common.collect.Iterables;
import com.google.common.collect.Streams;
import com.google.common.truth.Truth;
import com.google.devtools.build.lib.actions.Action;
import com.google.devtools.build.lib.actions.Artifact;
import com.google.devtools.build.lib.actions.util.ActionsTestUtil;
import com.google.devtools.build.lib.analysis.ConfiguredTarget;
import com.google.devtools.build.lib.analysis.FilesToRunProvider;
import com.google.devtools.build.lib.analysis.OutputGroupInfo;
import com.google.devtools.build.lib.analysis.actions.FileWriteAction;
import com.google.devtools.build.lib.analysis.actions.SpawnAction;
import com.google.devtools.build.lib.cmdline.RepositoryName;
import com.google.devtools.build.lib.collect.nestedset.NestedSet;
import com.google.devtools.build.lib.packages.BuildType;
import com.google.devtools.build.lib.packages.FileTarget;
import com.google.devtools.build.lib.packages.Rule;
import com.google.devtools.build.lib.packages.util.BazelMockAndroidSupport;
import com.google.devtools.build.lib.rules.android.AndroidRuleClasses.MultidexMode;
import com.google.devtools.build.lib.rules.android.deployinfo.AndroidDeployInfoOuterClass.AndroidDeployInfo;
import com.google.devtools.build.lib.rules.cpp.CppFileTypes;
import com.google.devtools.build.lib.rules.java.JavaCompilationArgsProvider;
import com.google.devtools.build.lib.rules.java.JavaCompileAction;
import com.google.devtools.build.lib.rules.java.JavaInfo;
import com.google.devtools.build.lib.rules.java.JavaSemantics;
import com.google.devtools.build.lib.skyframe.ConfiguredTargetAndData;
import com.google.devtools.build.lib.testutil.MoreAsserts;
import com.google.devtools.build.lib.util.FileType;
import com.google.devtools.build.lib.vfs.FileSystemUtils;
import com.google.devtools.build.lib.vfs.Path;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.Set;
import org.junit.Before;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runners.JUnit4;
/**
* A test for {@link com.google.devtools.build.lib.rules.android.AndroidBinary}.
*/
@RunWith(JUnit4.class)
public class AndroidBinaryTest extends AndroidBuildViewTestCase {
@Before
public void setupCcToolchain() throws Exception {
getAnalysisMock().ccSupport().setupCcToolchainConfigForCpu(mockToolsConfig, "armeabi-v7a");
}
@Before
public void setup() throws Exception {
scratch.file("java/android/BUILD",
"android_binary(name = 'app',",
" srcs = ['A.java'],",
" manifest = 'AndroidManifest.xml',",
" resource_files = glob(['res/**']),",
" )");
scratch.file("java/android/res/values/strings.xml",
"<resources><string name = 'hello'>Hello Android!</string></resources>");
scratch.file("java/android/A.java",
"package android; public class A {};");
setSkylarkSemanticsOptions("--experimental_google_legacy_api");
}
@Test
public void testAndroidSplitTransitionWithInvalidCpu() throws Exception {
scratch.file(
"test/skylark/my_rule.bzl",
"def impl(ctx): ",
" return []",
"my_rule = rule(",
" implementation = impl,",
" attrs = {",
" 'deps': attr.label_list(cfg = android_common.multi_cpu_configuration),",
" 'dep': attr.label(cfg = android_common.multi_cpu_configuration),",
" })");
scratch.file(
"test/skylark/BUILD",
"load('//test/skylark:my_rule.bzl', 'my_rule')",
"my_rule(name = 'test', deps = [':main'], dep = ':main')",
"cc_binary(name = 'main', srcs = ['main.c'])");
BazelMockAndroidSupport.setupNdk(mockToolsConfig);
// --android_cpu with --android_crosstool_top also triggers the split transition.
useConfiguration("--fat_apk_cpu=doesnotexist",
"--android_crosstool_top=//android/crosstool:everything");
AssertionError noToolchainError =
assertThrows(AssertionError.class, () -> getConfiguredTarget("//test/skylark:test"));
assertThat(noToolchainError)
.hasMessageThat()
.contains("does not contain a toolchain for cpu 'doesnotexist'");
}
@Test
public void testAssetsInExternalRepository() throws Exception {
FileSystemUtils.appendIsoLatin1(
scratch.resolve("WORKSPACE"), "local_repository(name='r', path='/r')");
scratch.file("/r/WORKSPACE");
scratch.file("/r/p/BUILD", "filegroup(name='assets', srcs=['a/b'])");
scratch.file("/r/p/a/b");
invalidatePackages();
scratchConfiguredTarget("java/a", "a",
"android_binary(",
" name = 'a',",
" srcs = ['A.java'],",
" manifest = 'AndroidManifest.xml',",
" assets = ['@r//p:assets'],",
" assets_dir = '')");
}
@Test
public void testMultidexModeAndMainDexProguardSpecs() throws Exception {
checkError("java/a", "a", "only allowed if 'multidex' is set to 'legacy'",
"android_binary(",
" name = 'a',",
" srcs = ['A.java'],",
" manifest = 'AndroidManifest.xml',",
" main_dex_proguard_specs = ['foo'])");
}
@Test
public void testAndroidManifestWithCustomName() throws Exception {
scratchConfiguredTarget(
"java/a",
"a",
"android_binary(",
" name = 'a',",
" srcs = ['A.java'],",
" manifest = 'SomeOtherAndroidManifest.xml')");
assertNoEvents();
}
@Test
public void testMainDexProguardSpecs() throws Exception {
useConfiguration("--noincremental_dexing");
ConfiguredTarget ct = scratchConfiguredTarget("java/a", "a",
"android_binary(",
" name = 'a',",
" srcs = ['A.java'],",
" manifest = 'AndroidManifest.xml',",
" multidex = 'legacy',",
" main_dex_proguard_specs = ['a.spec'])");
Artifact intermediateJar = artifactByPath(ImmutableList.of(getCompressedUnsignedApk(ct)),
".apk", ".dex.zip", ".dex.zip", "main_dex_list.txt", "_intermediate.jar");
List<String> args = getGeneratingSpawnActionArgs(intermediateJar);
MoreAsserts.assertContainsSublist(args, "-include", "java/a/a.spec");
assertThat(Joiner.on(" ").join(args)).doesNotContain("mainDexClasses.rules");
}
@Test
public void testMainDexListObfuscation() throws Exception {
useConfiguration("--noincremental_dexing");
scratch.file("/java/a/list.txt");
ConfiguredTarget ct =
scratchConfiguredTarget(
"java/a",
"a",
"android_binary(",
" name = 'a',",
" srcs = ['A.java'],",
" manifest = 'AndroidManifest.xml',",
" multidex = 'manual_main_dex',",
" proguard_generate_mapping = 1,",
" main_dex_list = 'list.txt')");
Artifact obfuscatedDexList =
artifactByPath(
ImmutableList.of(getCompressedUnsignedApk(ct)),
".apk",
".dex.zip",
".dex.zip",
"main_dex_list_obfuscated.txt");
List<String> args = getGeneratingSpawnActionArgs(obfuscatedDexList);
assertThat(args.get(0)).contains("dex_list_obfuscator");
MoreAsserts.assertContainsSublist(args, "--input", "java/a/list.txt");
}
@Test
public void testNonLegacyNativeDepsDoesNotPolluteDexSharding() throws Exception {
scratch.file("java/a/BUILD",
"android_binary(name = 'a',",
" manifest = 'AndroidManifest.xml',",
" multidex = 'native',",
" deps = [':cc'],",
" dex_shards = 2)",
"cc_library(name = 'cc',",
" srcs = ['cc.cc'])");
Artifact jarShard = artifactByPath(
ImmutableList.of(getCompressedUnsignedApk(getConfiguredTarget("//java/a:a"))),
".apk", "classes.dex.zip", "shard1.dex.zip", "shard1.jar.dex.zip");
Iterable<Artifact> shardInputs = getGeneratingAction(jarShard).getInputs();
assertThat(getFirstArtifactEndingWith(shardInputs, ".txt")).isNull();
}
@Test
public void testJavaPluginProcessorPath() throws Exception {
scratch.file("java/test/BUILD",
"java_library(name = 'plugin_dep',",
" srcs = [ 'ProcessorDep.java'])",
"java_plugin(name = 'plugin',",
" srcs = ['AnnotationProcessor.java'],",
" processor_class = 'com.google.process.stuff',",
" deps = [ ':plugin_dep' ])",
"android_binary(name = 'to_be_processed',",
" manifest = 'AndroidManifest.xml',",
" plugins = [':plugin'],",
" srcs = ['ToBeProcessed.java'])");
ConfiguredTarget target = getConfiguredTarget("//java/test:to_be_processed");
JavaCompileAction javacAction =
(JavaCompileAction) getGeneratingAction(getBinArtifact("libto_be_processed.jar", target));
assertThat(getProcessorNames(javacAction)).contains("com.google.process.stuff");
assertThat(getProcessorNames(javacAction)).hasSize(1);
assertThat(
ActionsTestUtil.baseArtifactNames(
getInputs(javacAction, getProcessorpath(javacAction))))
.containsExactly("libplugin.jar", "libplugin_dep.jar");
assertThat(
actionsTestUtil()
.predecessorClosureOf(getFilesToBuild(target), JavaSemantics.JAVA_SOURCE))
.isEqualTo("ToBeProcessed.java AnnotationProcessor.java ProcessorDep.java");
}
// Same test as above, enabling the plugin through the command line.
@Test
public void testPluginCommandLine() throws Exception {
scratch.file("java/test/BUILD",
"java_library(name = 'plugin_dep',",
" srcs = [ 'ProcessorDep.java'])",
"java_plugin(name = 'plugin',",
" srcs = ['AnnotationProcessor.java'],",
" processor_class = 'com.google.process.stuff',",
" deps = [ ':plugin_dep' ])",
"android_binary(name = 'to_be_processed',",
" manifest = 'AndroidManifest.xml',",
" srcs = ['ToBeProcessed.java'])");
useConfiguration("--plugin=//java/test:plugin");
ConfiguredTarget target = getConfiguredTarget("//java/test:to_be_processed");
JavaCompileAction javacAction =
(JavaCompileAction) getGeneratingAction(getBinArtifact("libto_be_processed.jar", target));
assertThat(getProcessorNames(javacAction)).contains("com.google.process.stuff");
assertThat(getProcessorNames(javacAction)).hasSize(1);
assertThat(
ActionsTestUtil.baseArtifactNames(
getInputs(javacAction, getProcessorpath(javacAction))))
.containsExactly("libplugin.jar", "libplugin_dep.jar");
assertThat(
actionsTestUtil()
.predecessorClosureOf(getFilesToBuild(target), JavaSemantics.JAVA_SOURCE))
.isEqualTo("ToBeProcessed.java AnnotationProcessor.java ProcessorDep.java");
}
@Test
public void testInvalidPlugin() throws Exception {
checkError("java/test", "lib",
// error:
getErrorMsgMisplacedRules("plugins", "android_binary",
"//java/test:lib", "java_library", "//java/test:not_a_plugin"),
// BUILD file:
"java_library(name = 'not_a_plugin',",
" srcs = [ 'NotAPlugin.java'])",
"android_binary(name = 'lib',",
" plugins = [':not_a_plugin'],",
" manifest = 'AndroidManifest.xml',",
" srcs = ['Lib.java'])");
}
@Test
public void testBaselineCoverageArtifacts() throws Exception {
useConfiguration("--collect_code_coverage");
ConfiguredTarget target = scratchConfiguredTarget("java/com/google/a", "bin",
"android_binary(name='bin', srcs=['Main.java'], manifest='AndroidManifest.xml')");
assertThat(baselineCoverageArtifactBasenames(target)).containsExactly("Main.java");
}
@Test
public void testSameSoFromMultipleDeps() throws Exception {
scratch.file("java/d/BUILD",
"genrule(name='genrule', srcs=[], outs=['genrule.so'], cmd='')",
"cc_library(name='cc1', srcs=[':genrule.so'])",
"cc_library(name='cc2', srcs=[':genrule.so'])",
"android_binary(name='ab', deps=[':cc1', ':cc2'], manifest='AndroidManifest.xml')");
getConfiguredTarget("//java/d:ab");
}
@Test
public void testSimpleBinary_desugarJava8() throws Exception {
useConfiguration("--experimental_desugar_for_android");
ConfiguredTarget binary = getConfiguredTarget("//java/android:app");
SpawnAction action = (SpawnAction) actionsTestUtil().getActionForArtifactEndingWith(
actionsTestUtil().artifactClosureOf(getFilesToBuild(binary)), "_deploy.jar");
assertThat(ActionsTestUtil.baseArtifactNames(action.getInputs()))
.contains("libapp.jar_desugared.jar");
assertThat(ActionsTestUtil.baseArtifactNames(action.getInputs()))
.doesNotContain("libapp.jar");
}
/**
* Tests that --experimental_check_desugar_deps causes the relevant flags to be set on desugaring
* and singlejar actions, and makes sure the deploy jar is built even when just building an APK.
*/
@Test
public void testSimpleBinary_checkDesugarDepsAlwaysHappens() throws Exception {
useConfiguration("--experimental_check_desugar_deps");
ConfiguredTarget binary = getConfiguredTarget("//java/android:app");
assertNoEvents();
// 1. Find app's deploy jar and make sure checking flags are set for it and its inputs
SpawnAction singlejar = (SpawnAction) actionsTestUtil().getActionForArtifactEndingWith(
getFilesToBuild(binary), "/app_deploy.jar");
assertThat(getGeneratingSpawnActionArgs(singlejar.getPrimaryOutput()))
.contains("--check_desugar_deps");
SpawnAction desugar =
(SpawnAction)
actionsTestUtil()
.getActionForArtifactEndingWith(singlejar.getInputs(), "/libapp.jar_desugared.jar");
assertThat(desugar).isNotNull();
assertThat(getGeneratingSpawnActionArgs(desugar.getPrimaryOutput()))
.contains("--emit_dependency_metadata_as_needed");
// 2. Make sure all APK outputs depend on the deploy Jar.
int found = 0;
for (Artifact built : getFilesToBuild(binary)) {
if (built.getExtension().equals("apk")) {
// If this assertion breaks then APK artifacts have stopped depending on deploy jars.
// If that's desired then we'll need to make sure dependency checking is done in another
// action that APK artifacts depend on, in addition to the check that happens when building
// deploy.jars, which we assert above.
assertWithMessage("%s dependency on deploy.jar", built.getFilename())
.that(actionsTestUtil().artifactClosureOf(built))
.contains(singlejar.getPrimaryOutput());
++found;
}
}
assertThat(found).isEqualTo(2 /* signed and unsigned apks */);
}
// regression test for #3169099
@Test
public void testBinarySrcs() throws Exception {
scratch.file("java/srcs/a.foo", "foo");
scratch.file("java/srcs/BUILD",
"android_binary(name = 'valid', manifest = 'AndroidManifest.xml', "
+ "srcs = ['a.java', 'b.srcjar', ':gvalid', ':gmix'])",
"android_binary(name = 'invalid', manifest = 'AndroidManifest.xml', "
+ "srcs = ['a.foo', ':ginvalid'])",
"android_binary(name = 'mix', manifest = 'AndroidManifest.xml', "
+ "srcs = ['a.java', 'a.foo'])",
"genrule(name = 'gvalid', srcs = ['a.java'], outs = ['b.java'], cmd = '')",
"genrule(name = 'ginvalid', srcs = ['a.java'], outs = ['b.foo'], cmd = '')",
"genrule(name = 'gmix', srcs = ['a.java'], outs = ['c.java', 'c.foo'], cmd = '')"
);
assertSrcsValidityForRuleType("//java/srcs", "android_binary", ".java or .srcjar");
}
// regression test for #3169095
@Test
public void testXmbInSrcs_NotPermittedButDoesNotThrow() throws Exception {
reporter.removeHandler(failFastHandler);
scratchConfiguredTarget("java/xmb", "a",
"android_binary(name = 'a', manifest = 'AndroidManifest.xml', srcs = ['a.xmb'])");
// We expect there to be an error here because a.xmb is not a valid src,
// and more importantly, no exception to have been thrown.
assertContainsEvent("in srcs attribute of android_binary rule //java/xmb:a: "
+ "target '//java/xmb:a.xmb' does not exist");
}
@Test
public void testNativeLibraryBasenameCollision() throws Exception {
reporter.removeHandler(failFastHandler); // expect errors
scratch.file("java/android/common/BUILD",
"cc_library(name = 'libcommon_armeabi',",
" srcs = ['armeabi/native.so'],)");
scratch.file("java/android/app/BUILD",
"cc_library(name = 'libnative',",
" srcs = ['native.so'],)",
"android_binary(name = 'b',",
" srcs = ['A.java'],",
" deps = [':libnative', '//java/android/common:libcommon_armeabi'],",
" manifest = 'AndroidManifest.xml',",
" )");
getConfiguredTarget("//java/android/app:b");
assertContainsEvent("Each library in the transitive closure must have a unique basename to "
+ "avoid name collisions when packaged into an apk, but two libraries have the basename "
+ "'native.so': java/android/common/armeabi/native.so and java/android/app/native.so");
}
private void setupNativeLibrariesForLinking() throws Exception {
scratch.file("java/android/common/BUILD",
"cc_library(name = 'common_native',",
" srcs = ['common.cc'],)",
"android_library(name = 'common',",
" exports = [':common_native'],)");
scratch.file("java/android/app/BUILD",
"cc_library(name = 'native',",
" srcs = ['native.cc'],)",
"android_binary(name = 'auto',",
" srcs = ['A.java'],",
" deps = [':native', '//java/android/common:common'],",
" manifest = 'AndroidManifest.xml',",
" )",
"android_binary(name = 'off',",
" srcs = ['A.java'],",
" deps = [':native', '//java/android/common:common'],",
" manifest = 'AndroidManifest.xml',",
" )");
}
private void assertNativeLibraryLinked(ConfiguredTarget target, String... srcNames) {
Artifact linkedLib = getOnlyElement(getNativeLibrariesInApk(target));
assertThat(linkedLib.getFilename())
.isEqualTo("lib" + target.getLabel().toPathFragment().getBaseName() + ".so");
assertThat(linkedLib.isSourceArtifact()).isFalse();
assertWithMessage("Native libraries were not linked to produce " + linkedLib)
.that(getGeneratingLabelForArtifact(linkedLib))
.isEqualTo(target.getLabel());
assertThat(artifactsToStrings(actionsTestUtil().artifactClosureOf(linkedLib)))
.containsAtLeastElementsIn(ImmutableSet.copyOf(Arrays.asList(srcNames)));
}
@Test
public void testNativeLibrary_LinksLibrariesWhenCodeIsPresent() throws Exception {
setupNativeLibrariesForLinking();
assertNativeLibraryLinked(getConfiguredTarget("//java/android/app:auto"),
"src java/android/common/common.cc", "src java/android/app/native.cc");
assertNativeLibraryLinked(getConfiguredTarget("//java/android/app:off"),
"src java/android/common/common.cc", "src java/android/app/native.cc");
}
@Test
public void testNativeLibrary_CopiesLibrariesDespiteExtraLayersOfIndirection() throws Exception {
scratch.file("java/android/app/BUILD",
"cc_library(name = 'native_dep',",
" srcs = ['dep.so'])",
"cc_library(name = 'native',",
" srcs = ['native_prebuilt.so'],",
" deps = [':native_dep'])",
"cc_library(name = 'native_wrapper',",
" deps = [':native'])",
"android_binary(name = 'app',",
" srcs = ['A.java'],",
" deps = [':native_wrapper'],",
" manifest = 'AndroidManifest.xml',",
" )");
assertNativeLibrariesCopiedNotLinked(getConfiguredTarget("//java/android/app:app"),
"src java/android/app/dep.so", "src java/android/app/native_prebuilt.so");
}
@Test
public void testNativeLibrary_CopiesLibrariesWrappedInCcLibraryWithSameName() throws Exception {
scratch.file("java/android/app/BUILD",
"cc_library(name = 'native',",
" srcs = ['libnative.so'])",
"android_binary(name = 'app',",
" srcs = ['A.java'],",
" deps = [':native'],",
" manifest = 'AndroidManifest.xml',",
" )");
assertNativeLibrariesCopiedNotLinked(getConfiguredTarget("//java/android/app:app"),
"src java/android/app/libnative.so");
}
@Test
public void testNativeLibrary_LinksWhenPrebuiltArchiveIsSupplied() throws Exception {
scratch.file("java/android/app/BUILD",
"cc_library(name = 'native_dep',",
" srcs = ['dep.lo'])",
"cc_library(name = 'native',",
" srcs = ['native_prebuilt.a'],",
" deps = [':native_dep'])",
"cc_library(name = 'native_wrapper',",
" deps = [':native'])",
"android_binary(name = 'app',",
" srcs = ['A.java'],",
" deps = [':native_wrapper'],",
" manifest = 'AndroidManifest.xml',",
" )");
assertNativeLibraryLinked(getConfiguredTarget("//java/android/app:app"),
"src java/android/app/native_prebuilt.a");
}
@Test
public void testNativeLibrary_CopiesFullLibrariesInIfsoMode() throws Exception {
useConfiguration("--interface_shared_objects");
scratch.file("java/android/app/BUILD",
"cc_library(name = 'native_dep',",
" srcs = ['dep.so'])",
"cc_library(name = 'native',",
" srcs = ['native.cc', 'native_prebuilt.so'],",
" deps = [':native_dep'])",
"android_binary(name = 'app',",
" srcs = ['A.java'],",
" deps = [':native'],",
" manifest = 'AndroidManifest.xml',",
" )");
ConfiguredTarget app = getConfiguredTarget("//java/android/app:app");
Iterable<Artifact> nativeLibraries = getNativeLibrariesInApk(app);
assertThat(artifactsToStrings(nativeLibraries))
.containsAtLeast("src java/android/app/native_prebuilt.so", "src java/android/app/dep.so");
assertThat(FileType.filter(nativeLibraries, CppFileTypes.INTERFACE_SHARED_LIBRARY))
.isEmpty();
}
@Test
public void testNativeLibrary_ProvidesLinkerScriptToLinkAction() throws Exception {
scratch.file("java/android/app/BUILD",
"cc_library(name = 'native',",
" srcs = ['native.cc'],",
" linkopts = ['-Wl,-version-script', '$(location jni.lds)'],",
" deps = ['jni.lds'],)",
"android_binary(name = 'app',",
" srcs = ['A.java'],",
" deps = [':native'],",
" manifest = 'AndroidManifest.xml',",
" )");
ConfiguredTarget app = getConfiguredTarget("//java/android/app:app");
Artifact copiedLib = getOnlyElement(getNativeLibrariesInApk(app));
Artifact linkedLib = getOnlyElement(getGeneratingAction(copiedLib).getInputs());
Iterable<Artifact> linkInputs = getGeneratingAction(linkedLib).getInputs();
assertThat(ActionsTestUtil.baseArtifactNames(linkInputs)).contains("jni.lds");
}
/** Regression test for http://b/33173461. */
@Test
public void testIncrementalDexingUsesDexArchives_binaryDependingOnAliasTarget()
throws Exception {
scratch.file(
"java/com/google/android/BUILD",
"android_library(",
" name = 'dep',",
" srcs = ['dep.java'],",
" resource_files = glob(['res/**']),",
" manifest = 'AndroidManifest.xml',",
")",
"alias(",
" name = 'alt',",
" actual = ':dep',",
")",
"android_binary(",
" name = 'top',",
" srcs = ['foo.java', 'bar.srcjar'],",
" multidex = 'native',",
" manifest = 'AndroidManifest.xml',",
" deps = [':alt',],",
")");
ConfiguredTarget topTarget = getConfiguredTarget("//java/com/google/android:top");
assertNoEvents();
Action shardAction =
getGeneratingAction(getBinArtifact("_dx/top/classes.jar", topTarget));
for (Artifact input : getNonToolInputs(shardAction)) {
String basename = input.getFilename();
// all jars are converted to dex archives
assertWithMessage(basename)
.that(!basename.contains(".jar") || basename.endsWith(".jar.dex.zip"))
.isTrue();
// all jars are desugared before being converted
if (basename.endsWith(".jar.dex.zip")) {
assertThat(getGeneratingAction(input).getPrimaryInput().getFilename())
.isEqualTo(basename.substring(0, basename.length() - ".jar.dex.zip".length())
+ ".jar_desugared.jar");
}
}
// Make sure exactly the dex archives generated for top and dependents appear. We also *don't*
// want neverlink and unused_dep to appear, and to be safe we do so by explicitly enumerating
// *all* expected input dex archives.
assertThat(
Iterables.filter(
ActionsTestUtil.baseArtifactNames(getNonToolInputs(shardAction)),
Predicates.containsPattern("\\.jar")))
.containsExactly(
// top's dex archives
"libtop.jar.dex.zip",
"top_resources.jar.dex.zip",
// dep's dex archives
"libdep.jar.dex.zip");
}
@Test
public void testIncrementalDexingDisabledWithBlacklistedDexopts() throws Exception {
// Even if we mark a dx flag as supported, incremental dexing isn't used with blacklisted
// dexopts (unless incremental_dexing attribute is set, which a different test covers)
useConfiguration("--incremental_dexing",
"--non_incremental_per_target_dexopts=--no-locals",
"--dexopts_supported_in_incremental_dexing=--no-locals");
scratch.file(
"java/com/google/android/BUILD",
"android_binary(",
" name = 'top',",
" srcs = ['foo.java', 'bar.srcjar'],",
" manifest = 'AndroidManifest.xml',",
" dexopts = ['--no-locals'],",
" dex_shards = 2,",
" multidex = 'native',",
")");
ConfiguredTarget topTarget = getConfiguredTarget("//java/com/google/android:top");
assertNoEvents();
Action shardAction = getGeneratingAction(getBinArtifact("_dx/top/shard1.jar", topTarget));
assertThat(
Iterables.filter(
ActionsTestUtil.baseArtifactNames(getNonToolInputs(shardAction)),
Predicates.containsPattern("\\.jar\\.dex\\.zip")))
.isEmpty(); // no dex archives are used
}
@Test
public void testIncrementalDexingDisabledWithProguard() throws Exception {
scratch.file(
"java/com/google/android/BUILD",
"android_binary(",
" name = 'top',",
" srcs = ['foo.java', 'bar.srcjar'],",
" manifest = 'AndroidManifest.xml',",
" proguard_specs = ['proguard.cfg'],",
")");
ConfiguredTarget topTarget = getConfiguredTarget("//java/com/google/android:top");
assertNoEvents();
Action dexAction = getGeneratingAction(getBinArtifact("_dx/top/classes.dex", topTarget));
assertThat(
Iterables.filter(
ActionsTestUtil.baseArtifactNames(dexAction.getInputs()),
Predicates.containsPattern("\\.jar")))
.containsExactly("top_proguard.jar", "dx_binary.jar"); // proguard output is used directly
}
@Test
public void testIncrementalDexing_incompatibleWithProguardWhenDisabled() throws Exception {
useConfiguration("--experimental_incremental_dexing_after_proguard=0"); // disable with Proguard
checkError("java/com/google/android", "top", "target cannot be incrementally dexed",
"android_binary(",
" name = 'top',",
" srcs = ['foo.java', 'bar.srcjar'],",
" manifest = 'AndroidManifest.xml',",
" proguard_specs = ['proguard.cfg'],",
" incremental_dexing = 1,",
")");
}
@Test
public void testIncrementalDexingAfterProguard_unsharded() throws Exception {
useConfiguration("--experimental_incremental_dexing_after_proguard=1");
// Use "legacy" multidex mode so we get a main dex list file and can test that it's passed to
// the splitter action (similar to _withDexShards below), unlike without the dex splitter where
// the main dex list goes to the merging action.
scratch.file(
"java/com/google/android/BUILD",
"android_binary(",
" name = 'top',",
" srcs = ['foo.java', 'bar.srcjar'],",
" manifest = 'AndroidManifest.xml',",
" incremental_dexing = 1,",
" multidex = 'legacy',",
" dexopts = ['--minimal-main-dex', '--positions=none'],",
" proguard_specs = ['b.pro'],",
")");
ConfiguredTarget topTarget = getConfiguredTarget("//java/com/google/android:top");
assertNoEvents();
SpawnAction shardAction =
getGeneratingSpawnAction(getBinArtifact("_dx/top/classes.dex.zip", topTarget));
assertThat(shardAction.getArguments()).contains("--main-dex-list");
assertThat(shardAction.getArguments()).contains("--minimal-main-dex");
assertThat(ActionsTestUtil.baseArtifactNames(getNonToolInputs(shardAction)))
.containsExactly("classes.jar", "main_dex_list.txt");
// --positions dexopt is supported after Proguard, even though not normally otherwise
assertThat(
paramFileArgsForAction(
getGeneratingSpawnAction(getBinArtifact("_dx/top/classes.jar", topTarget))))
.contains("--positions=none");
}
@Test
public void testIncrementalDexingAfterProguard_autoShardedMultidexAutoOptIn() throws Exception {
useConfiguration("--experimental_incremental_dexing_after_proguard=3",
"--experimental_incremental_dexing_after_proguard_by_default");
// Use "legacy" multidex mode so we get a main dex list file and can test that it's passed to
// the splitter action (similar to _withDexShards below), unlike without the dex splitter where
// the main dex list goes to the merging action.
scratch.file(
"java/com/google/android/BUILD",
"android_binary(",
" name = 'top',",
" srcs = ['foo.java', 'bar.srcjar'],",
" manifest = 'AndroidManifest.xml',",
" multidex = 'legacy',",
" dexopts = ['--minimal-main-dex', '--positions=none'],",
" proguard_specs = ['b.pro'],",
")"); // incremental_dexing = 1 attribute not needed
ConfiguredTarget topTarget = getConfiguredTarget("//java/com/google/android:top");
assertNoEvents();
SpawnAction splitAction =
getGeneratingSpawnAction(getTreeArtifact("dexsplits/top", topTarget));
assertThat(splitAction.getArguments()).contains("--main-dex-list");
assertThat(splitAction.getArguments()).contains("--minimal-main-dex");
assertThat(ActionsTestUtil.baseArtifactNames(getNonToolInputs(splitAction)))
.containsExactly("shard1.jar.dex.zip", "shard2.jar.dex.zip", "shard3.jar.dex.zip",
"main_dex_list.txt");
SpawnAction shuffleAction =
getGeneratingSpawnAction(getBinArtifact("_dx/top/shard1.jar", topTarget));
assertThat(shuffleAction.getArguments()).doesNotContain("--main-dex-list");
assertThat(ActionsTestUtil.baseArtifactNames(getNonToolInputs(shuffleAction)))
.containsExactly("top_proguard.jar");
// --positions dexopt is supported after Proguard, even though not normally otherwise
assertThat(
paramFileArgsForAction(
getGeneratingSpawnAction(getBinArtifact("_dx/top/shard3.jar.dex.zip", topTarget))))
.contains("--positions=none");
}
@Test
public void testIncrementalDexingAfterProguard_explicitDexShards() throws Exception {
useConfiguration("--experimental_incremental_dexing_after_proguard=2");
// Use "legacy" multidex mode so we get a main dex list file and can test that it's passed to
// the shardAction, not to the subsequent dexMerger action. Without dex_shards, main dex list
// file goes to the dexMerger instead (see _multidex test).
scratch.file(
"java/com/google/android/BUILD",
"android_binary(",
" name = 'top',",
" srcs = ['foo.java', 'bar.srcjar'],",
" manifest = 'AndroidManifest.xml',",
" dex_shards = 25,",
" incremental_dexing = 1,",
" multidex = 'legacy',",
" proguard_specs = ['b.pro'],",
")");
ConfiguredTarget topTarget = getConfiguredTarget("//java/com/google/android:top");
assertNoEvents();
SpawnAction shardAction =
getGeneratingSpawnAction(getBinArtifact("_dx/top/shard25.jar", topTarget));
assertThat(shardAction.getArguments()).contains("--main_dex_filter");
assertThat(ActionsTestUtil.baseArtifactNames(getNonToolInputs(shardAction)))
.containsExactly("top_proguard.jar", "main_dex_list.txt");
SpawnAction mergeAction =
getGeneratingSpawnAction(getBinArtifact("_dx/top/shard1.jar.dex.zip", topTarget));
assertThat(mergeAction.getArguments()).doesNotContain("--main-dex-list");
assertThat(ActionsTestUtil.baseArtifactNames(getNonToolInputs(mergeAction)))
.contains("shard1.jar");
}
@Test
public void testIncrementalDexingAfterProguard_autoShardedMonodex()
throws Exception {
useConfiguration("--experimental_incremental_dexing_after_proguard=3");
// Use "legacy" multidex mode so we get a main dex list file and can test that it's passed to
// the splitter action (similar to _withDexShards below), unlike without the dex splitter where
// the main dex list goes to the merging action.
scratch.file(
"java/com/google/android/BUILD",
"android_binary(",
" name = 'top',",
" srcs = ['foo.java', 'bar.srcjar'],",
" manifest = 'AndroidManifest.xml',",
" incremental_dexing = 1,",
" multidex = 'off',",
" proguard_specs = ['b.pro'],",
")");
ConfiguredTarget topTarget = getConfiguredTarget("//java/com/google/android:top");
assertNoEvents();
SpawnAction mergeAction =
getGeneratingSpawnAction(getBinArtifact("_dx/top/classes.dex.zip", topTarget));
assertThat(mergeAction.getArguments()).doesNotContain("--main-dex-list");
assertThat(ActionsTestUtil.baseArtifactNames(getNonToolInputs(mergeAction)))
.containsExactly("shard1.jar.dex.zip", "shard2.jar.dex.zip", "shard3.jar.dex.zip");
SpawnAction shuffleAction =
getGeneratingSpawnAction(getBinArtifact("_dx/top/shard1.jar", topTarget));
assertThat(shuffleAction.getArguments()).doesNotContain("--main-dex-list");
assertThat(ActionsTestUtil.baseArtifactNames(getNonToolInputs(shuffleAction)))
.containsExactly("top_proguard.jar");
}
@Test
public void testV1SigningMethod() throws Exception {
actualSignerToolTests("v1", "true", "false");
}
@Test
public void testV2SigningMethod() throws Exception {
actualSignerToolTests("v2", "false", "true");
}
@Test
public void testV1V2SigningMethod() throws Exception {
actualSignerToolTests("v1_v2", "true", "true");
}
private void actualSignerToolTests(String apkSigningMethod, String signV1, String signV2)
throws Exception {
scratch.file("java/com/google/android/hello/BUILD",
"android_binary(name = 'hello',",
" srcs = ['Foo.java'],",
" manifest = 'AndroidManifest.xml',)");
useConfiguration("--apk_signing_method=" + apkSigningMethod);
ConfiguredTarget binary = getConfiguredTarget("//java/com/google/android/hello:hello");
Set<Artifact> artifacts = actionsTestUtil().artifactClosureOf(getFilesToBuild(binary));
assertThat(getFirstArtifactEndingWith(artifacts, "signed_hello.apk")).isNull();
SpawnAction unsignedApkAction = (SpawnAction) actionsTestUtil()
.getActionForArtifactEndingWith(artifacts, "/hello_unsigned.apk");
assertThat(
Streams.stream(unsignedApkAction.getInputs())
.map(Artifact::getFilename)
.anyMatch(filename -> Ascii.toLowerCase(filename).contains("singlejar")))
.isTrue();
SpawnAction compressedUnsignedApkAction = (SpawnAction) actionsTestUtil()
.getActionForArtifactEndingWith(artifacts, "compressed_hello_unsigned.apk");
assertThat(
Streams.stream(compressedUnsignedApkAction.getInputs())
.map(Artifact::getFilename)
.anyMatch(filename -> Ascii.toLowerCase(filename).contains("singlejar")))
.isTrue();
SpawnAction zipalignAction = (SpawnAction) actionsTestUtil()
.getActionForArtifactEndingWith(artifacts, "zipaligned_hello.apk");
assertThat(zipalignAction.getCommandFilename()).endsWith("zipalign");
Artifact a = ActionsTestUtil.getFirstArtifactEndingWith(artifacts, "hello.apk");
assertThat(getGeneratingSpawnAction(a).getCommandFilename()).endsWith("ApkSignerBinary");
List<String> args = getGeneratingSpawnActionArgs(a);
assertThat(flagValue("--v1-signing-enabled", args)).isEqualTo(signV1);
assertThat(flagValue("--v2-signing-enabled", args)).isEqualTo(signV2);
}
@Test
public void testResourceShrinkingAction() throws Exception {
scratch.file("java/com/google/android/hello/BUILD",
"android_binary(name = 'hello',",
" srcs = ['Foo.java'],",
" manifest = 'AndroidManifest.xml',",
" inline_constants = 0,",
" resource_files = ['res/values/strings.xml'],",
" shrink_resources = 1,",
" resource_configuration_filters = ['en'],",
" proguard_specs = ['proguard-spec.pro'],)");
ConfiguredTarget binary = getConfiguredTarget("//java/com/google/android/hello:hello");
Set<Artifact> artifacts = actionsTestUtil().artifactClosureOf(getFilesToBuild(binary));
assertThat(artifacts)
.containsAtLeast(
getFirstArtifactEndingWith(artifacts, "resource_files.zip"),
getFirstArtifactEndingWith(artifacts, "proguard.jar"),
getFirstArtifactEndingWith(artifacts, "shrunk.ap_"));
List<String> processingArgs =
getGeneratingSpawnActionArgs(getFirstArtifactEndingWith(artifacts, "resource_files.zip"));
assertThat(flagValue("--resourcesOutput", processingArgs))
.endsWith("hello_files/resource_files.zip");
List<String> proguardArgs =
getGeneratingSpawnActionArgs(getFirstArtifactEndingWith(artifacts, "proguard.jar"));
assertThat(flagValue("-outjars", proguardArgs)).endsWith("hello_proguard.jar");
List<String> shrinkingArgs =
getGeneratingSpawnActionArgs(getFirstArtifactEndingWith(artifacts, "shrunk.ap_"));
assertThat(flagValue("--resources", shrinkingArgs))
.isEqualTo(flagValue("--resourcesOutput", processingArgs));
assertThat(flagValue("--shrunkJar", shrinkingArgs))
.isEqualTo(flagValue("-outjars", proguardArgs));
assertThat(flagValue("--proguardMapping", shrinkingArgs))
.isEqualTo(flagValue("-printmapping", proguardArgs));
assertThat(flagValue("--rTxt", shrinkingArgs))
.isEqualTo(flagValue("--rOutput", processingArgs));
assertThat(flagValue("--primaryManifest", shrinkingArgs))
.isEqualTo(flagValue("--manifestOutput", processingArgs));
assertThat(flagValue("--resourceConfigs", shrinkingArgs))
.isEqualTo(flagValue("--resourceConfigs", processingArgs));
List<String> packageArgs =
getGeneratingSpawnActionArgs(getFirstArtifactEndingWith(artifacts, "_hello_proguard.cfg"));
assertThat(flagValue("--tool", packageArgs)).isEqualTo("PACKAGE");
assertThat(packageArgs).doesNotContain("--conditionalKeepRules");
}
@Test
public void testResourceCycleShrinking() throws Exception {
useConfiguration("--experimental_android_resource_cycle_shrinking=true");
checkError(
"java/a",
"a",
"resource cycle shrinking can only be enabled for builds with aapt2",
"android_binary(",
" name = 'a',",
" srcs = ['A.java'],",
" manifest = 'AndroidManifest.xml',",
" resource_files = [ 'res/values/values.xml' ], ",
" shrink_resources = 1,",
")");
}
@Test
public void testResourceCycleShrinkingWithoutResourceShinking() throws Exception {
useConfiguration("--experimental_android_resource_cycle_shrinking=true");
checkError(
"java/a",
"a",
"resource cycle shrinking can only be enabled when resource shrinking is enabled",
"android_binary(",
" name = 'a',",
" srcs = ['A.java'],",
" manifest = 'AndroidManifest.xml',",
" resource_files = [ 'res/values/values.xml' ], ",
" shrink_resources = 0,",
")");
}
@Test
public void testResourceShrinking_RequiresProguard() throws Exception {
scratch.file("java/com/google/android/hello/BUILD",
"android_binary(name = 'hello',",
" srcs = ['Foo.java'],",
" manifest = 'AndroidManifest.xml',",
" inline_constants = 0,",
" resource_files = ['res/values/strings.xml'],",
" shrink_resources = 1,)");
ConfiguredTarget binary = getConfiguredTarget("//java/com/google/android/hello:hello");
Set<Artifact> artifacts = actionsTestUtil().artifactClosureOf(getFilesToBuild(binary));
assertThat(artifacts).containsNoneOf(
getFirstArtifactEndingWith(artifacts, "shrunk.jar"),
getFirstArtifactEndingWith(artifacts, "shrunk.ap_"));
}
@Test
public void testProguardExtraOutputs() throws Exception {
scratch.file(
"java/com/google/android/hello/BUILD",
"android_binary(name = 'b',",
" srcs = ['HelloApp.java'],",
" manifest = 'AndroidManifest.xml',",
" proguard_specs = ['proguard-spec.pro'])");
ConfiguredTarget output = getConfiguredTarget("//java/com/google/android/hello:b");
// Checks that ProGuard is called with the appropriate options.
Artifact a = getFirstArtifactEndingWith(getFilesToBuild(output), "_proguard.jar");
SpawnAction action = getGeneratingSpawnAction(a);
List<String> args = getGeneratingSpawnActionArgs(a);
// Assert that the ProGuard executable set in the android_sdk rule appeared in the command-line
// of the SpawnAction that generated the _proguard.jar.
assertThat(
Iterables.any(
args,
new Predicate<String>() {
@Override
public boolean apply(String s) {
return s.endsWith("ProGuard");
}
}))
.isTrue();
assertThat(args)
.containsAtLeast(
"-injars",
execPathEndingWith(action.getInputs(), "b_deploy.jar"),
"-printseeds",
execPathEndingWith(action.getOutputs(), "b_proguard.seeds"),
"-printusage",
execPathEndingWith(action.getOutputs(), "b_proguard.usage"))
.inOrder();
// Checks that the output files are produced.
assertProguardUsed(output);
assertThat(getBinArtifact("b_proguard.usage", output)).isNotNull();
assertThat(getBinArtifact("b_proguard.seeds", output)).isNotNull();
}
@Test
public void testProGuardExecutableMatchesConfiguration() throws Exception {
scratch.file("java/com/google/devtools/build/jkrunchy/BUILD",
"package(default_visibility=['//visibility:public'])",
"java_binary(name = 'jkrunchy',",
" srcs = glob(['*.java']),",
" main_class = 'com.google.devtools.build.jkrunchy.JKrunchyMain')");
useConfiguration("--proguard_top=//java/com/google/devtools/build/jkrunchy:jkrunchy");
scratch.file("java/com/google/android/hello/BUILD",
"android_binary(name = 'b',",
" srcs = ['HelloApp.java'],",
" manifest = 'AndroidManifest.xml',",
" proguard_specs = ['proguard-spec.pro'])");
ConfiguredTarget output = getConfiguredTarget("//java/com/google/android/hello:b_proguard.jar");
assertProguardUsed(output);
SpawnAction proguardAction = (SpawnAction) actionsTestUtil().getActionForArtifactEndingWith(
getFilesToBuild(output), "_proguard.jar");
Artifact jkrunchyExecutable =
getHostConfiguredTarget("//java/com/google/devtools/build/jkrunchy")
.getProvider(FilesToRunProvider.class)
.getExecutable();
assertWithMessage("ProGuard implementation was not correctly taken from the configuration")
.that(proguardAction.getCommandFilename())
.isEqualTo(jkrunchyExecutable.getExecPathString());
}
@Test
public void enforceProguardFileExtension_disabled_allowsOtherExtensions() throws Exception {
useConfiguration("--noenforce_proguard_file_extension");
scratch.file(
"java/android/hello/BUILD",
"android_binary(name = 'b',",
" srcs = ['HelloApp.java'],",
" manifest = 'AndroidManifest.xml',",
" proguard_specs = ['proguard-spec.pro'])");
ConfiguredTarget unused = getConfiguredTarget("//java/android/hello:b");
}
@Test
public void enforceProguardFileExtension_enabled_disallowsOtherExtensions() throws Exception {
useConfiguration("--enforce_proguard_file_extension");
scratch.file(
"java/android/hello/BUILD",
"android_binary(name = 'b',",
" srcs = ['HelloApp.java'],",
" manifest = 'AndroidManifest.xml',",
" proguard_specs = ['proguard-spec.pro'])");
AssertionError assertionError =
assertThrows(AssertionError.class, () -> getConfiguredTarget("//java/android/hello:b"));
assertThat(assertionError).hasMessageThat().contains("These files do not end in .pgcfg");
}
@Test
public void enforceProguardFileExtension_enabled_allowsPgcfg() throws Exception {
useConfiguration("--enforce_proguard_file_extension");
scratch.file(
"java/android/hello/BUILD",
"android_binary(name = 'b',",
" srcs = ['HelloApp.java'],",
" manifest = 'AndroidManifest.xml',",
" proguard_specs = [':proguard.pgcfg'])");
ConfiguredTarget unused = getConfiguredTarget("//java/android/hello:b");
}
@Test
public void enforceProguardFileExtension_enabled_ignoresThirdParty() throws Exception {
useConfiguration("--enforce_proguard_file_extension");
scratch.file(
"third_party/bar/BUILD", "licenses(['unencumbered'])", "exports_files(['proguard.pro'])");
scratch.file(
"java/android/hello/BUILD",
"android_binary(name = 'b',",
" srcs = ['HelloApp.java'],",
" manifest = 'AndroidManifest.xml',",
" proguard_specs = ['//third_party/bar:proguard.pro'])");
ConfiguredTarget unused = getConfiguredTarget("//java/android/hello:b");
}
@Test
public void testNeverlinkTransitivity() throws Exception {
useConfiguration("--android_fixed_resource_neverlinking");
scratch.file(
"java/com/google/android/neversayneveragain/BUILD",
"android_library(name = 'l1',",
" srcs = ['l1.java'],",
" manifest = 'AndroidManifest.xml',",
" resource_files = ['res/values/resource.xml'])",
"android_library(name = 'l2',",
" srcs = ['l2.java'],",
" deps = [':l1'],",
" neverlink = 1)",
"android_library(name = 'l3',",
" srcs = ['l3.java'],",
" deps = [':l2'])",
"android_library(name = 'l4',",
" srcs = ['l4.java'],",
" deps = [':l1'])",
"android_binary(name = 'b1',",
" srcs = ['b1.java'],",
" deps = [':l2'],",
" manifest = 'AndroidManifest.xml')",
"android_binary(name = 'b2',",
" srcs = ['b2.java'],",
" deps = [':l3'],",
" manifest = 'AndroidManifest.xml')",
"android_binary(name = 'b3',",
" srcs = ['b3.java'],",
" deps = [':l3', ':l4'],",
" manifest = 'AndroidManifest.xml')");
ConfiguredTarget b1 = getConfiguredTarget("//java/com/google/android/neversayneveragain:b1");
Action b1DeployAction = actionsTestUtil().getActionForArtifactEndingWith(
actionsTestUtil().artifactClosureOf(getFilesToBuild(b1)), "b1_deploy.jar");
List<String> b1Inputs = prettyArtifactNames(b1DeployAction.getInputs());
assertThat(b1Inputs).containsNoneOf(
"java/com/google/android/neversayneveragain/libl1.jar_desugared.jar",
"java/com/google/android/neversayneveragain/libl2.jar_desugared.jar",
"java/com/google/android/neversayneveragain/libl3.jar_desugared.jar",
"java/com/google/android/neversayneveragain/libl4.jar_desugared.jar");
assertThat(b1Inputs).contains(
"java/com/google/android/neversayneveragain/libb1.jar_desugared.jar");
assertThat(
resourceInputPaths(
"java/com/google/android/neversayneveragain", getValidatedResources(b1)))
.doesNotContain("res/values/resource.xml");
ConfiguredTarget b2 = getConfiguredTarget("//java/com/google/android/neversayneveragain:b2");
Action b2DeployAction = actionsTestUtil().getActionForArtifactEndingWith(
actionsTestUtil().artifactClosureOf(getFilesToBuild(b2)), "b2_deploy.jar");
List<String> b2Inputs = prettyArtifactNames(b2DeployAction.getInputs());
assertThat(b2Inputs).containsNoneOf(
"java/com/google/android/neversayneveragain/libl1.jar_desugared.jar",
"java/com/google/android/neversayneveragain/libl2.jar_desugared.jar",
"java/com/google/android/neversayneveragain/libl4.jar_desugared.jar");
assertThat(b2Inputs)
.containsAtLeast(
"java/com/google/android/neversayneveragain/_dx/l3/libl3.jar_desugared.jar",
"java/com/google/android/neversayneveragain/libb2.jar_desugared.jar");
assertThat(
resourceInputPaths(
"java/com/google/android/neversayneveragain", getValidatedResources(b2)))
.doesNotContain("res/values/resource.xml");
ConfiguredTarget b3 = getConfiguredTarget("//java/com/google/android/neversayneveragain:b3");
Action b3DeployAction = actionsTestUtil().getActionForArtifactEndingWith(
actionsTestUtil().artifactClosureOf(getFilesToBuild(b3)), "b3_deploy.jar");
List<String> b3Inputs = prettyArtifactNames(b3DeployAction.getInputs());
assertThat(b3Inputs)
.containsAtLeast(
"java/com/google/android/neversayneveragain/_dx/l1/libl1.jar_desugared.jar",
"java/com/google/android/neversayneveragain/_dx/l3/libl3.jar_desugared.jar",
"java/com/google/android/neversayneveragain/_dx/l4/libl4.jar_desugared.jar",
"java/com/google/android/neversayneveragain/libb3.jar_desugared.jar");
assertThat(b3Inputs)
.doesNotContain("java/com/google/android/neversayneveragain/libl2.jar_desugared.jar");
assertThat(
resourceInputPaths(
"java/com/google/android/neversayneveragain", getValidatedResources(b3)))
.contains("res/values/resource.xml");
}
@Test
public void testDexopts() throws Exception {
useConfiguration("--noincremental_dexing");
checkDexopts("[ '--opt1', '--opt2' ]", ImmutableList.of("--opt1", "--opt2"));
}
@Test
public void testDexoptsTokenization() throws Exception {
useConfiguration("--noincremental_dexing");
checkDexopts("[ '--opt1', '--opt2 tokenized' ]",
ImmutableList.of("--opt1", "--opt2", "tokenized"));
}
@Test
public void testDexoptsMakeVariableSubstitution() throws Exception {
useConfiguration("--noincremental_dexing");
checkDexopts("[ '--opt1', '$(COMPILATION_MODE)' ]", ImmutableList.of("--opt1", "fastbuild"));
}
private void checkDexopts(String dexopts, List<String> expectedArgs) throws Exception {
scratch.file("java/com/google/android/BUILD",
"android_binary(name = 'b',",
" srcs = ['dummy1.java'],",
" dexopts = " + dexopts + ",",
" manifest = 'AndroidManifest.xml')");
// Include arguments that are always included.
List<String> fixedArgs = ImmutableList.of("--num-threads=5");
expectedArgs = new ImmutableList.Builder<String>()
.addAll(fixedArgs).addAll(expectedArgs).build();
// Ensure that the args that immediately follow "--dex" match the expectation.
ConfiguredTarget binary = getConfiguredTarget("//java/com/google/android:b");
List<String> args =
getGeneratingSpawnActionArgs(
ActionsTestUtil.getFirstArtifactEndingWith(
actionsTestUtil().artifactClosureOf(getFilesToBuild(binary)), "classes.dex"));
int start = args.indexOf("--dex") + 1;
assertThat(start).isNotEqualTo(0);
int end = Math.min(args.size(), start + expectedArgs.size());
assertThat(args.subList(start, end)).isEqualTo(expectedArgs);
}
@Test
public void testDexMainListOpts() throws Exception {
checkDexMainListOpts("[ '--opt1', '--opt2' ]", "--opt1", "--opt2");
}
@Test
public void testDexMainListOptsTokenization() throws Exception {
checkDexMainListOpts("[ '--opt1', '--opt2 tokenized' ]", "--opt1", "--opt2", "tokenized");
}
@Test
public void testDexMainListOptsMakeVariableSubstitution() throws Exception {
checkDexMainListOpts("[ '--opt1', '$(COMPILATION_MODE)' ]", "--opt1", "fastbuild");
}
private void checkDexMainListOpts(String mainDexListOpts, String... expectedArgs)
throws Exception {
scratch.file("java/com/google/android/BUILD",
"android_binary(name = 'b',",
" srcs = ['dummy1.java'],",
" multidex = \"legacy\",",
" main_dex_list_opts = " + mainDexListOpts + ",",
" manifest = 'AndroidManifest.xml')");
// Ensure that the args that immediately follow the main class in the shell command
// match the expectation.
ConfiguredTarget binary = getConfiguredTarget("//java/com/google/android:b");
List<String> args =
getGeneratingSpawnActionArgs(
ActionsTestUtil.getFirstArtifactEndingWith(
actionsTestUtil().artifactClosureOf(getFilesToBuild(binary)), "main_dex_list.txt"));
// args: [ "bash", "-c", "java -cp dx.jar main opts other" ]
MoreAsserts.assertContainsSublist(args, expectedArgs);
}
@Test
public void omitResourcesInfoProviderFromAndroidBinary_enabled() throws Exception {
useConfiguration("--experimental_omit_resources_info_provider_from_android_binary");
ConfiguredTarget binary =
scratchConfiguredTarget(
"java/com/pkg/myapp",
"myapp",
"android_binary(",
" name = 'myapp',",
" manifest = 'AndroidManifest.xml',",
" resource_files = glob(['res/**/*']),",
")");
assertThat(binary.get(AndroidResourcesInfo.PROVIDER)).isNull();
}
@Test
public void omitResourcesInfoProviderFromAndroidBinary_disabled() throws Exception {
useConfiguration("--noexperimental_omit_resources_info_provider_from_android_binary");
ConfiguredTarget binary =
scratchConfiguredTarget(
"java/com/pkg/myapp",
"myapp",
"android_binary(",
" name = 'myapp',",
" manifest = 'AndroidManifest.xml',",
" resource_files = glob(['res/**/*']),",
")");
assertThat(binary.get(AndroidResourcesInfo.PROVIDER)).isNotNull();
}
@Test
public void testResourceConfigurationFilters() throws Exception {
scratch.file("java/com/google/android/BUILD",
"android_binary(name = 'b',",
" srcs = ['dummy1.java'],",
" manifest = 'AndroidManifest.xml',",
" resource_configuration_filters = [ 'en', 'fr'],)");
// Ensure that the args are present
ConfiguredTarget binary = getConfiguredTarget("//java/com/google/android:b");
List<String> args = resourceArguments(getValidatedResources(binary));
assertThat(flagValue("--resourceConfigs", args)).contains("en,fr");
}
@Test
public void testFilteredResourcesInvalidFilter() throws Exception {
String badQualifier = "invalid-qualifier";
checkError(
"java/r/android",
"r",
badQualifier,
"android_binary(name = 'r',",
" manifest = 'AndroidManifest.xml',",
" resource_files = ['res/values/foo.xml'],",
" resource_configuration_filters = ['" + badQualifier + "'])");
}
@Test
public void testFilteredResourcesInvalidResourceDir() throws Exception {
String badQualifierDir = "values-invalid-qualifier";
checkError(
"java/r/android",
"r",
badQualifierDir,
"android_binary(name = 'r',",
" manifest = 'AndroidManifest.xml',",
" resource_files = ['res/" + badQualifierDir + "/foo.xml'],",
" resource_configuration_filters = ['en'])");
}
/** Test that resources are not filtered in analysis under aapt2. */
@Test
public void testFilteredResourcesFilteringAapt2() throws Exception {
List<String> resources =
ImmutableList.of("res/values/foo.xml", "res/values-en/foo.xml", "res/values-fr/foo.xml");
String dir = "java/r/android";
mockAndroidSdkWithAapt2();
useConfiguration("--android_sdk=//sdk:sdk");
ConfiguredTarget binary =
scratchConfiguredTarget(
dir,
"r",
"android_binary(name = 'r',",
" manifest = 'AndroidManifest.xml',",
" resource_configuration_filters = ['', 'en, es, '],",
" aapt_version = 'aapt2',",
" densities = ['hdpi, , ', 'xhdpi'],",
" resource_files = ['" + Joiner.on("', '").join(resources) + "'])");
ValidatedAndroidResources directResources =
getValidatedResources(binary, /* transitive= */ false);
// Validate that the AndroidResourceProvider for this binary contains all values.
assertThat(resourceContentsPaths(dir, directResources)).containsExactlyElementsIn(resources);
// Validate that the input to resource processing contains all values.
assertThat(resourceInputPaths(dir, directResources)).containsAtLeastElementsIn(resources);
// Validate that the filters are correctly passed to the resource processing action
// This includes trimming whitespace and ignoring empty filters.
assertThat(resourceArguments(directResources)).contains("en,es");
assertThat(resourceArguments(directResources)).contains("hdpi,xhdpi");
}
@Test
public void testFilteredResourcesSimple() throws Exception {
testDirectResourceFiltering(
"en",
/* unexpectedQualifiers= */ ImmutableList.of("fr"),
/* expectedQualifiers= */ ImmutableList.of("en"));
}
@Test
public void testFilteredResourcesNoopFilter() throws Exception {
testDirectResourceFiltering(
/* filters= */ "",
/* unexpectedQualifiers= */ ImmutableList.<String>of(),
/* expectedQualifiers= */ ImmutableList.of("en", "fr"));
}
@Test
public void testFilteredResourcesMultipleFilters() throws Exception {
testDirectResourceFiltering(
"en,es",
/* unexpectedQualifiers= */ ImmutableList.of("fr"),
/* expectedQualifiers= */ ImmutableList.of("en", "es"));
}
@Test
public void testFilteredResourcesMultipleFiltersWithWhitespace() throws Exception {
testDirectResourceFiltering(
" en , es ",
/* unexpectedQualifiers= */ ImmutableList.of("fr"),
/* expectedQualifiers= */ ImmutableList.of("en", "es"));
}
@Test
public void testFilteredResourcesMultipleFilterStrings() throws Exception {
testDirectResourceFiltering(
"en', 'es",
/* unexpectedQualifiers= */ ImmutableList.of("fr"),
/* expectedQualifiers= */ ImmutableList.of("en", "es"));
}
@Test
public void testFilteredResourcesLocaleWithoutRegion() throws Exception {
testDirectResourceFiltering(
"en",
/* unexpectedQualifiers= */ ImmutableList.of("fr-rCA"),
/* expectedQualifiers= */ ImmutableList.of("en-rCA", "en-rUS", "en"));
}
@Test
public void testFilteredResourcesLocaleWithRegion() throws Exception {
testDirectResourceFiltering(
"en-rUS",
/* unexpectedQualifiers= */ ImmutableList.of("en-rGB"),
/* expectedQualifiers= */ ImmutableList.of("en-rUS", "en"));
}
@Test
public void testFilteredResourcesOldAaptLocale() throws Exception {
testDirectResourceFiltering(
"en_US,fr_CA",
/* unexpectedQualifiers= */ ImmutableList.of("en-rCA"),
/* expectedQualifiers = */ ImmutableList.of("en-rUS", "fr-rCA"));
}
@Test
public void testFilteredResourcesOldAaptLocaleOtherQualifiers() throws Exception {
testDirectResourceFiltering(
"mcc310-en_US-ldrtl,mcc311-mnc312-fr_CA",
/* unexpectedQualifiers= */ ImmutableList.of("en-rCA", "mcc312", "mcc311-mnc311"),
/* expectedQualifiers = */ ImmutableList.of("en-rUS", "fr-rCA", "mcc310", "mcc311-mnc312"));
}
@Test
public void testFilteredResourcesSmallestScreenWidth() throws Exception {
testDirectResourceFiltering(
"sw600dp",
/* unexpectedQualifiers= */ ImmutableList.of("sw700dp"),
/* expectedQualifiers= */ ImmutableList.of("sw500dp", "sw600dp"));
}
@Test
public void testFilteredResourcesScreenWidth() throws Exception {
testDirectResourceFiltering(
"w600dp",
/* unexpectedQualifiers= */ ImmutableList.of("w700dp"),
/* expectedQualifiers= */ ImmutableList.of("w500dp", "w600dp"));
}
@Test
public void testFilteredResourcesScreenHeight() throws Exception {
testDirectResourceFiltering(
"h600dp",
/* unexpectedQualifiers= */ ImmutableList.of("h700dp"),
/* expectedQualifiers= */ ImmutableList.of("h500dp", "h600dp"));
}
@Test
public void testFilteredResourcesScreenSize() throws Exception {
testDirectResourceFiltering(
"normal",
/* unexpectedQualifiers= */ ImmutableList.of("large", "xlarge"),
/* expectedQualifiers= */ ImmutableList.of("small", "normal"));
}
/** Tests that filtering on density is ignored to match aapt behavior. */
@Test
public void testFilteredResourcesDensity() throws Exception {
testDirectResourceFiltering(
"hdpi",
/* unexpectedQualifiers= */ ImmutableList.<String>of(),
/* expectedQualifiers= */ ImmutableList.of("ldpi", "mdpi", "hdpi", "xhdpi", "xxhdpi"));
}
/** Tests that filtering on API version is ignored to match aapt behavior. */
@Test
public void testFilteredResourcesApiVersion() throws Exception {
testDirectResourceFiltering(
"v4",
/* unexpectedQualifiers= */ ImmutableList.<String>of(),
/* expectedQualifiers= */ ImmutableList.of("v3", "v4", "v5"));
}
@Test
public void testFilteredResourcesRegularQualifiers() throws Exception {
// Include one value for each qualifier not tested above
String filters =
"mcc310-mnc004-ldrtl-long-round-port-car-night-notouch-keysexposed-nokeys-navexposed-nonav";
// In the qualifiers we expect to be removed, include one value that contradicts each qualifier
// of the filter
testDirectResourceFiltering(
filters,
/* unexpectedQualifiers= */ ImmutableList.of(
"mcc309",
"mnc03",
"ldltr",
"notlong",
"notround",
"land",
"watch",
"notnight",
"finger",
"keyshidden",
"qwerty",
"navhidden",
"dpad"),
/* expectedQualifiers= */ ImmutableList.of(filters));
}
@Test
public void testDensityFilteredResourcesSingleDensity() throws Exception {
testDensityResourceFiltering(
"hdpi", ImmutableList.of("ldpi", "mdpi", "xhdpi"), ImmutableList.of("hdpi"));
}
@Test
public void testDensityFilteredResourcesSingleClosestDensity() throws Exception {
testDensityResourceFiltering(
"hdpi", ImmutableList.of("ldpi", "mdpi"), ImmutableList.of("xhdpi"));
}
@Test
public void testDensityFilteredResourcesDifferingQualifiers() throws Exception {
testDensityResourceFiltering(
"hdpi",
ImmutableList.of("en-xhdpi", "fr"),
ImmutableList.of("en-hdpi", "fr-mdpi", "xhdpi"));
}
@Test
public void testDensityFilteredResourcesMultipleDensities() throws Exception {
testDensityResourceFiltering(
"hdpi,ldpi,xhdpi",
ImmutableList.of("mdpi", "xxhdpi"),
ImmutableList.of("ldpi", "hdpi", "xhdpi"));
}
@Test
public void testDensityFilteredResourcesDoubledDensity() throws Exception {
testDensityResourceFiltering(
"hdpi", ImmutableList.of("ldpi", "mdpi", "xhdpi"), ImmutableList.of("xxhdpi"));
}
@Test
public void testDensityFilteredResourcesDifferentRatiosSmallerCloser() throws Exception {
testDensityResourceFiltering("mdpi", ImmutableList.of("hdpi"), ImmutableList.of("ldpi"));
}
@Test
public void testDensityFilteredResourcesDifferentRatiosLargerCloser() throws Exception {
testDensityResourceFiltering("hdpi", ImmutableList.of("mdpi"), ImmutableList.of("xhdpi"));
}
@Test
public void testDensityFilteredResourcesSameRatioPreferLarger() throws Exception {
// xxhdpi is 480dpi and xxxhdpi is 640dpi.
// The ratios between 360dpi and 480dpi and between 480dpi and 640dpi are both 3:4.
testDensityResourceFiltering("xxhdpi", ImmutableList.of("360dpi"), ImmutableList.of("xxxhdpi"));
}
@Test
public void testDensityFilteredResourcesOptionsSmallerThanDesired() throws Exception {
testDensityResourceFiltering(
"xxxhdpi", ImmutableList.of("xhdpi", "mdpi", "ldpi"), ImmutableList.of("xxhdpi"));
}
@Test
public void testDensityFilteredResourcesOptionsLargerThanDesired() throws Exception {
testDensityResourceFiltering(
"ldpi", ImmutableList.of("xxxhdpi", "xxhdpi", "xhdpi"), ImmutableList.of("mdpi"));
}
@Test
public void testDensityFilteredResourcesSpecialValues() throws Exception {
testDensityResourceFiltering(
"xxxhdpi", ImmutableList.of("anydpi", "nodpi"), ImmutableList.of("ldpi"));
}
@Test
public void testDensityFilteredResourcesXml() throws Exception {
testDirectResourceFiltering(
/* resourceConfigurationFilters= */ "",
"hdpi",
ImmutableList.<String>of(),
ImmutableList.of("ldpi", "mdpi", "hdpi", "xhdpi"),
/* expectUnqualifiedResource= */ true,
"drawable",
"xml");
}
@Test
public void testDensityFilteredResourcesNotDrawable() throws Exception {
testDirectResourceFiltering(
/* resourceConfigurationFilters= */ "",
"hdpi",
ImmutableList.<String>of(),
ImmutableList.of("ldpi", "mdpi", "hdpi", "xhdpi"),
/* expectUnqualifiedResource= */ true,
"layout",
"png");
}
@Test
public void testQualifierAndDensityFilteredResources() throws Exception {
testDirectResourceFiltering(
"en,fr-mdpi",
"hdpi,ldpi",
ImmutableList.of("mdpi", "es-ldpi", "en-xxxhdpi", "fr-mdpi"),
ImmutableList.of("ldpi", "hdpi", "en-xhdpi", "fr-hdpi"),
/* expectUnqualifiedResource= */ false,
"drawable",
"png");
}
private void testDensityResourceFiltering(
String densities, List<String> unexpectedQualifiers, List<String> expectedQualifiers)
throws Exception {
testDirectResourceFiltering(
/* resourceConfigurationFilters= */ "",
densities,
unexpectedQualifiers,
expectedQualifiers,
/* expectUnqualifiedResource= */ false,
"drawable",
"png");
}
private void testDirectResourceFiltering(
String filters, List<String> unexpectedQualifiers, ImmutableList<String> expectedQualifiers)
throws Exception {
testDirectResourceFiltering(
filters,
/* densities= */ "",
unexpectedQualifiers,
expectedQualifiers,
/* expectUnqualifiedResource= */ true,
"drawable",
"png");
}
private void testDirectResourceFiltering(
String resourceConfigurationFilters,
String densities,
List<String> unexpectedQualifiers,
List<String> expectedQualifiers,
boolean expectUnqualifiedResource,
String folderType,
String suffix)
throws Exception {
List<String> unexpectedResources = new ArrayList<>();
for (String qualifier : unexpectedQualifiers) {
unexpectedResources.add("res/" + folderType + "-" + qualifier + "/foo." + suffix);
}
List<String> expectedResources = new ArrayList<>();
for (String qualifier : expectedQualifiers) {
expectedResources.add("res/" + folderType + "-" + qualifier + "/foo." + suffix);
}
String unqualifiedResource = "res/" + folderType + "/foo." + suffix;
if (expectUnqualifiedResource) {
expectedResources.add(unqualifiedResource);
} else {
unexpectedResources.add(unqualifiedResource);
}
// Default resources should never be filtered
expectedResources.add("res/values/foo.xml");
Iterable<String> allResources = Iterables.concat(unexpectedResources, expectedResources);
String dir = "java/r/android";
ConfiguredTarget binary =
scratchConfiguredTarget(
dir,
"r",
"android_binary(name = 'r',",
" manifest = 'AndroidManifest.xml',",
" resource_configuration_filters = ['" + resourceConfigurationFilters + "'],",
" densities = ['" + densities + "'],",
" resource_files = ['" + Joiner.on("', '").join(allResources) + "'])");
// No prefix should be added because of resource filtering.
assertThat(
getConfiguration(binary)
.getFragment(AndroidConfiguration.class)
.getOutputDirectoryName())
.isNull();
ValidatedAndroidResources directResources =
getValidatedResources(binary, /* transitive= */ false);
// Validate that the AndroidResourceProvider for this binary contains only the filtered values.
assertThat(resourceContentsPaths(dir, directResources))
.containsExactlyElementsIn(expectedResources);
// Validate that the input to resource processing contains only the filtered values.
assertThat(resourceInputPaths(dir, directResources))
.containsAtLeastElementsIn(expectedResources);
assertThat(resourceInputPaths(dir, directResources)).containsNoneIn(unexpectedResources);
// Only transitive resources need to be ignored when filtered,, and there aren't any here.
assertThat(resourceArguments(directResources)).doesNotContain("--prefilteredResources");
// Validate resource filters are passed to execution
List<String> args = resourceArguments(directResources);
// Remove whitespace and quotes from the resourceConfigurationFilters attribute value to get the
// value we expect to pass to the resource processing action
String fixedResourceConfigFilters = resourceConfigurationFilters.replaceAll("[ ']", "");
if (resourceConfigurationFilters.isEmpty()) {
assertThat(args).containsNoneOf("--resourceConfigs", fixedResourceConfigFilters);
} else {
assertThat(args).containsAtLeast("--resourceConfigs", fixedResourceConfigFilters).inOrder();
}
if (densities.isEmpty()) {
assertThat(args).containsNoneOf("--densities", densities);
} else {
// We still expect densities only for the purposes of adding information to manifests
assertThat(args).containsAtLeast("--densities", densities).inOrder();
}
}
@Test
public void testFilteredTransitiveResources() throws Exception {
String matchingResource = "res/values-en/foo.xml";
String unqualifiedResource = "res/values/foo.xml";
String notMatchingResource = "res/values-fr/foo.xml";
String dir = "java/r/android";
ConfiguredTarget binary =
scratchConfiguredTarget(
dir,
"r",
"android_library(name = 'lib',",
" manifest = 'AndroidManifest.xml',",
" resource_files = [",
" '" + matchingResource + "',",
" '" + unqualifiedResource + "',",
" '" + notMatchingResource + "'",
" ])",
"android_binary(name = 'r',",
" manifest = 'AndroidManifest.xml',",
" deps = [':lib'],",
" resource_configuration_filters = ['en'])");
ValidatedAndroidResources directResources =
getValidatedResources(binary, /* transitive= */ false);
ValidatedAndroidResources transitiveResources =
getValidatedResources(binary, /* transitive= */ true);
assertThat(resourceContentsPaths(dir, directResources)).isEmpty();
assertThat(resourceContentsPaths(dir, transitiveResources))
.containsExactly(matchingResource, unqualifiedResource);
assertThat(resourceInputPaths(dir, directResources))
.containsExactly(matchingResource, unqualifiedResource);
String[] flagValues =
flagValue("--prefilteredResources", resourceArguments(directResources)).split(",");
assertThat(flagValues).asList().containsExactly("values-fr/foo.xml");
}
@Test
public void testFilteredTransitiveResourcesDifferentDensities() throws Exception {
String dir = "java/r/android";
ConfiguredTarget binary =
scratchConfiguredTarget(
dir,
"bin",
"android_binary(name = 'bin',",
" manifest = 'AndroidManifest.xml',",
" resource_configuration_filters = ['en'],",
" densities = ['hdpi'],",
" deps = [':invalid', ':best', ':other', ':multiple'],",
")",
"android_library(name = 'invalid',",
" manifest = 'AndroidManifest.xml',",
" resource_files = ['invalid/res/drawable-fr-hdpi/foo.png'],",
")",
"android_library(name = 'best',",
" manifest = 'AndroidManifest.xml',",
" resource_files = ['best/res/drawable-en-hdpi/foo.png'],",
")",
"android_library(name = 'other',",
" manifest = 'AndroidManifest.xml',",
" resource_files = ['other/res/drawable-en-mdpi/foo.png'],",
")",
"android_library(name = 'multiple',",
" manifest = 'AndroidManifest.xml',",
" resource_files = [",
" 'multiple/res/drawable-en-hdpi/foo.png',",
" 'multiple/res/drawable-en-mdpi/foo.png'],",
")");
ValidatedAndroidResources directResources =
getValidatedResources(binary, /* transitive= */ false);
// All of the resources are transitive
assertThat(resourceContentsPaths(dir, directResources)).isEmpty();
// Only the best matches should be used. This includes multiple best matches, even given a
// single density filter, since they are each from a different dependency. This duplication
// would be resolved during merging.
assertThat(resourceInputPaths(dir, directResources))
.containsExactly(
"best/res/drawable-en-hdpi/foo.png", "multiple/res/drawable-en-hdpi/foo.png");
}
@Test
public void testFilteredResourcesAllFilteredOut() throws Exception {
String dir = "java/r/android";
final String keptBaseDir = "partly_filtered_dir";
String removedLibraryDir = "fully_filtered_library_dir";
String removedBinaryDir = "fully_filtered_binary_dir";
ConfiguredTarget binary =
scratchConfiguredTarget(
dir,
"bin",
"android_library(name = 'not_fully_filtered_lib',",
" manifest = 'AndroidManifest.xml',",
" resource_files = [",
// Some of the resources in this directory are kept:
" '" + keptBaseDir + "/values-es/foo.xml',",
" '" + keptBaseDir + "/values-fr/foo.xml',",
" '" + keptBaseDir + "/values-en/foo.xml',",
" ])",
"android_library(name = 'fully_filtered_lib',",
" manifest = 'AndroidManifest.xml',",
" resource_files = [",
// All of the resources in this directory are filtered out:
" '" + removedLibraryDir + "/values-es/foo.xml',",
" '" + removedLibraryDir + "/values-fr/foo.xml',",
" ])",
"android_binary(name = 'bin',",
" manifest = 'AndroidManifest.xml',",
" resource_configuration_filters = ['en'],",
" resource_files = [",
// All of the resources in this directory are filtered out:
" '" + removedBinaryDir + "/values-es/foo.xml',",
" '" + removedBinaryDir + "/values-fr/foo.xml',",
" ],",
" deps = [",
" ':fully_filtered_lib',",
" ':not_fully_filtered_lib',",
" ])");
List<String> resourceProcessingArgs =
getGeneratingSpawnActionArgs(getValidatedResources(binary).getRTxt());
assertThat(
Iterables.filter(
resourceProcessingArgs,
new Predicate<String>() {
@Override
public boolean apply(String input) {
return input.contains(keptBaseDir);
}
}))
.isNotEmpty();
for (String arg : resourceProcessingArgs) {
assertThat(arg).doesNotContain(removedLibraryDir);
assertThat(arg).doesNotContain(removedBinaryDir);
}
}
@Test
public void testFilterResourcesPseudolocalesPropagated() throws Exception {
String dir = "java/r/android";
ConfiguredTarget binary =
scratchConfiguredTarget(
dir,
"bin",
"android_binary(name = 'bin',",
" resource_files = glob(['res/**']),",
" resource_configuration_filters = ['en', 'en-rXA', 'ar-rXB'],",
" manifest = 'AndroidManifest.xml')");
List<String> resourceProcessingArgs =
getGeneratingSpawnActionArgs(getValidatedResources(binary).getRTxt());
assertThat(resourceProcessingArgs).containsAtLeast("--resourceConfigs", "ar-rXB,en,en-rXA");
}
@Test
public void testThrowOnResourceConflictFlagGetsPropagated() throws Exception {
scratch.file(
"java/r/android/BUILD",
"android_binary(",
" name = 'r',",
" manifest = 'AndroidManifest.xml',",
" resource_files = ['res/values/foo.xml'],",
")");
useConfiguration("--experimental_android_throw_on_resource_conflict");
ConfiguredTarget binary = getConfiguredTarget("//java/r/android:r");
List<String> resourceProcessingArgs = resourceArguments(getValidatedResources(binary));
assertThat(resourceProcessingArgs).contains("--throwOnResourceConflict");
}
/**
* Gets the paths of matching artifacts contained within a resource container
*
* @param dir the directory to look for artifacts in
* @param resource the container that contains eligible artifacts
* @return the paths to all artifacts from the input that are contained within the given
* directory, relative to that directory.
*/
private List<String> resourceContentsPaths(String dir, ValidatedAndroidResources resource) {
return pathsToArtifacts(dir, resource.getArtifacts());
}
/**
* Gets the paths of matching artifacts that are used as input to resource processing
*
* @param dir the directory to look for artifacts in
* @param resource the output from the resource processing that uses these artifacts as inputs
* @return the paths to all artifacts used as inputs to resource processing that are contained
* within the given directory, relative to that directory.
*/
private List<String> resourceInputPaths(String dir, ValidatedAndroidResources resource) {
return pathsToArtifacts(dir, resourceGeneratingAction(resource).getInputs());
}
/**
* Gets the paths of matching artifacts from an iterable
*
* @param dir the directory to look for artifacts in
* @param artifacts all available artifacts
* @return the paths to all artifacts from the input that are contained within the given
* directory, relative to that directory.
*/
private List<String> pathsToArtifacts(String dir, Iterable<Artifact> artifacts) {
List<String> paths = new ArrayList<>();
Path containingDir = rootDirectory;
for (String part : dir.split("/")) {
containingDir = containingDir.getChild(part);
}
for (Artifact a : artifacts) {
if (a.getPath().startsWith(containingDir)) {
paths.add(a.getPath().relativeTo(containingDir).toString());
}
}
return paths;
}
@Test
public void testInheritedRNotInRuntimeJars() throws Exception {
String dir = "java/r/android/";
scratch.file(
dir + "BUILD",
"android_library(name = 'sublib',",
" manifest = 'AndroidManifest.xml',",
" resource_files = glob(['res3/**']),",
" srcs =['sublib.java'],",
" )",
"android_library(name = 'lib',",
" manifest = 'AndroidManifest.xml',",
" resource_files = glob(['res2/**']),",
" deps = [':sublib'],",
" srcs =['lib.java'],",
" )",
"android_binary(name = 'bin',",
" manifest = 'AndroidManifest.xml',",
" resource_files = glob(['res/**']),",
" deps = [':lib'],",
" srcs =['bin.java'],",
" )");
Action deployJarAction =
getGeneratingAction(
getFileConfiguredTarget("//java/r/android:bin_deploy.jar").getArtifact());
List<String> inputs = ActionsTestUtil.baseArtifactNames(deployJarAction.getInputs());
assertThat(inputs)
.containsAtLeast(
"libsublib.jar_desugared.jar",
"liblib.jar_desugared.jar",
"libbin.jar_desugared.jar",
"bin_resources.jar_desugared.jar");
assertThat(inputs)
.containsNoneOf("lib_resources.jar_desugared.jar", "sublib_resources.jar_desugared.jar");
}
@Test
public void testLocalResourcesUseRClassGenerator() throws Exception {
scratch.file("java/r/android/BUILD",
"android_library(name = 'lib',",
" manifest = 'AndroidManifest.xml',",
" resource_files = glob(['res2/**']),",
" )",
"android_binary(name = 'r',",
" manifest = 'AndroidManifest.xml',",
" resource_files = glob(['res/**']),",
" deps = [':lib'],",
" )");
scratch.file("java/r/android/res2/values/strings.xml",
"<resources><string name = 'lib_string'>Libs!</string></resources>");
scratch.file("java/r/android/res/values/strings.xml",
"<resources><string name = 'hello'>Hello Android!</string></resources>");
Artifact jar = getResourceClassJar(getConfiguredTargetAndData("//java/r/android:r"));
assertThat(getGeneratingAction(jar).getMnemonic()).isEqualTo("RClassGenerator");
assertThat(getGeneratingSpawnActionArgs(jar))
.containsAtLeast("--primaryRTxt", "--primaryManifest", "--library", "--classJarOutput");
}
@Test
public void testLocalResourcesUseRClassGeneratorNoLibraries() throws Exception {
scratch.file("java/r/android/BUILD",
"android_binary(name = 'r',",
" manifest = 'AndroidManifest.xml',",
" resource_files = glob(['res/**']),",
" )");
scratch.file("java/r/android/res/values/strings.xml",
"<resources><string name = 'hello'>Hello Android!</string></resources>");
Artifact jar = getResourceClassJar(getConfiguredTargetAndData("//java/r/android:r"));
assertThat(getGeneratingAction(jar).getMnemonic()).isEqualTo("RClassGenerator");
List<String> args = getGeneratingSpawnActionArgs(jar);
assertThat(args).containsAtLeast("--primaryRTxt", "--primaryManifest", "--classJarOutput");
assertThat(args).doesNotContain("--libraries");
}
@Test
public void testUseRClassGeneratorCustomPackage() throws Exception {
scratch.file("java/r/android/BUILD",
"android_library(name = 'lib',",
" manifest = 'AndroidManifest.xml',",
" resource_files = glob(['res2/**']),",
" custom_package = 'com.lib.custom',",
" )",
"android_binary(name = 'r',",
" manifest = 'AndroidManifest.xml',",
" resource_files = glob(['res/**']),",
" custom_package = 'com.binary.custom',",
" deps = [':lib'],",
" )");
scratch.file("java/r/android/res2/values/strings.xml",
"<resources><string name = 'lib_string'>Libs!</string></resources>");
scratch.file("java/r/android/res/values/strings.xml",
"<resources><string name = 'hello'>Hello Android!</string></resources>");
ConfiguredTargetAndData binary = getConfiguredTargetAndData("//java/r/android:r");
Artifact jar = getResourceClassJar(binary);
assertThat(getGeneratingAction(jar).getMnemonic()).isEqualTo("RClassGenerator");
List<String> args = getGeneratingSpawnActionArgs(jar);
assertThat(args)
.containsAtLeast(
"--primaryRTxt",
"--primaryManifest",
"--library",
"--classJarOutput",
"--packageForR",
"com.binary.custom");
}
@Test
public void testUseRClassGeneratorMultipleDeps() throws Exception {
scratch.file(
"java/r/android/BUILD",
"android_library(name = 'lib1',",
" manifest = 'AndroidManifest.xml',",
" resource_files = glob(['res1/**']),",
" )",
"android_library(name = 'lib2',",
" manifest = 'AndroidManifest.xml',",
" resource_files = glob(['res2/**']),",
" )",
"android_binary(name = 'r',",
" manifest = 'AndroidManifest.xml',",
" resource_files = glob(['res/**']),",
" deps = [':lib1', ':lib2'],",
" )");
ConfiguredTargetAndData binary = getConfiguredTargetAndData("//java/r/android:r");
Artifact jar = getResourceClassJar(binary);
assertThat(getGeneratingAction(jar).getMnemonic()).isEqualTo("RClassGenerator");
List<String> args = getGeneratingSpawnActionArgs(jar);
AndroidResourcesInfo resourcesInfo =
binary.getConfiguredTarget().get(AndroidResourcesInfo.PROVIDER);
assertThat(resourcesInfo.getTransitiveAndroidResources()).hasSize(2);
ValidatedAndroidResources firstDep =
resourcesInfo.getTransitiveAndroidResources().toList().get(0);
ValidatedAndroidResources secondDep =
resourcesInfo.getTransitiveAndroidResources().toList().get(1);
assertThat(args)
.containsAtLeast(
"--primaryRTxt",
"--primaryManifest",
"--library",
firstDep.getRTxt().getExecPathString()
+ ","
+ firstDep.getManifest().getExecPathString(),
"--library",
secondDep.getRTxt().getExecPathString()
+ ","
+ secondDep.getManifest().getExecPathString(),
"--classJarOutput")
.inOrder();
}
@Test
public void testNoCrunchBinaryOnly() throws Exception {
scratch.file("java/r/android/BUILD",
"android_binary(name = 'r',",
" srcs = ['Foo.java'],",
" manifest = 'AndroidManifest.xml',",
" resource_files = ['res/drawable-hdpi-v4/foo.png',",
" 'res/drawable-hdpi-v4/bar.9.png'],",
" crunch_png = 0,",
" )");
ConfiguredTarget binary = getConfiguredTarget("//java/r/android:r");
List<String> args = getGeneratingSpawnActionArgs(getResourceApk(binary));
assertThat(args).contains("--useAaptCruncher=no");
}
@Test
public void testDoCrunch() throws Exception {
scratch.file("java/r/android/BUILD",
"android_binary(name = 'r',",
" srcs = ['Foo.java'],",
" manifest = 'AndroidManifest.xml',",
" resource_files = ['res/drawable-hdpi-v4/foo.png',",
" 'res/drawable-hdpi-v4/bar.9.png'],",
" crunch_png = 1,",
" )");
ConfiguredTarget binary = getConfiguredTarget("//java/r/android:r");
List<String> args = getGeneratingSpawnActionArgs(getResourceApk(binary));
assertThat(args).doesNotContain("--useAaptCruncher=no");
}
@Test
public void testDoCrunchDefault() throws Exception {
scratch.file("java/r/android/BUILD",
"android_binary(name = 'r',",
" srcs = ['Foo.java'],",
" manifest = 'AndroidManifest.xml',",
" resource_files = ['res/drawable-hdpi-v4/foo.png',",
" 'res/drawable-hdpi-v4/bar.9.png'],",
" )");
ConfiguredTarget binary = getConfiguredTarget("//java/r/android:r");
List<String> args = getGeneratingSpawnActionArgs(getResourceApk(binary));
assertThat(args).doesNotContain("--useAaptCruncher=no");
}
@Test
public void testNoCrunchWithAndroidLibraryNoBinaryResources() throws Exception {
scratch.file("java/r/android/BUILD",
"android_library(name = 'resources',",
" manifest = 'AndroidManifest.xml',",
" resource_files = ['res/values/strings.xml',",
" 'res/drawable-hdpi-v4/foo.png',",
" 'res/drawable-hdpi-v4/bar.9.png'],",
" )",
"android_binary(name = 'r',",
" srcs = ['Foo.java'],",
" manifest = 'AndroidManifest.xml',",
" deps = [':resources'],",
" crunch_png = 0,",
" )");
ConfiguredTarget binary = getConfiguredTarget("//java/r/android:r");
List<String> args = getGeneratingSpawnActionArgs(getResourceApk(binary));
assertThat(args).contains("--useAaptCruncher=no");
}
@Test
public void testNoCrunchWithMultidexNative() throws Exception {
scratch.file("java/r/android/BUILD",
"android_library(name = 'resources',",
" manifest = 'AndroidManifest.xml',",
" resource_files = ['res/values/strings.xml',",
" 'res/drawable-hdpi-v4/foo.png',",
" 'res/drawable-hdpi-v4/bar.9.png'],",
" )",
"android_binary(name = 'r',",
" srcs = ['Foo.java'],",
" manifest = 'AndroidManifest.xml',",
" deps = [':resources'],",
" multidex = 'native',",
" crunch_png = 0,",
" )");
ConfiguredTarget binary = getConfiguredTarget("//java/r/android:r");
List<String> args = getGeneratingSpawnActionArgs(getResourceApk(binary));
assertThat(args).contains("--useAaptCruncher=no");
}
@Test
public void testZipaligned() throws Exception {
ConfiguredTarget binary = getConfiguredTarget("//java/android:app");
Artifact a =
ActionsTestUtil.getFirstArtifactEndingWith(
actionsTestUtil().artifactClosureOf(getFilesToBuild(binary)), "zipaligned_app.apk");
SpawnAction action = getGeneratingSpawnAction(a);
assertThat(action.getMnemonic()).isEqualTo("AndroidZipAlign");
List<String> arguments = getGeneratingSpawnActionArgs(a);
assertThat(arguments).contains("-p");
assertThat(arguments).contains("4");
Artifact zipAlignTool =
getFirstArtifactEndingWith(action.getInputs(), "/zipalign");
assertThat(arguments).contains(zipAlignTool.getExecPathString());
Artifact unsignedApk =
getFirstArtifactEndingWith(action.getInputs(), "/app_unsigned.apk");
assertThat(arguments).contains(unsignedApk.getExecPathString());
Artifact zipalignedApk =
getFirstArtifactEndingWith(action.getOutputs(), "/zipaligned_app.apk");
assertThat(arguments).contains(zipalignedApk.getExecPathString());
}
@Test
public void testDeployInfo() throws Exception {
ConfiguredTarget binary = getConfiguredTarget("//java/android:app");
NestedSet<Artifact> outputGroup = getOutputGroup(binary, "android_deploy_info");
Artifact deployInfoArtifact = ActionsTestUtil
.getFirstArtifactEndingWith(outputGroup, "/deploy_info.deployinfo.pb");
assertThat(deployInfoArtifact).isNotNull();
AndroidDeployInfo deployInfo = getAndroidDeployInfo(deployInfoArtifact);
assertThat(deployInfo).isNotNull();
assertThat(deployInfo.getMergedManifest().getExecRootPath()).endsWith(
"/AndroidManifest.xml");
assertThat(deployInfo.getAdditionalMergedManifestsList()).isEmpty();
assertThat(deployInfo.getApksToDeploy(0).getExecRootPath()).endsWith("/app.apk");
}
/**
* Internal helper method: checks that dex sharding input and output is correct for
* different combinations of multidex mode and build with and without proguard.
*/
private void internalTestDexShardStructure(MultidexMode multidexMode, boolean proguard,
String nonProguardSuffix) throws Exception {
ConfiguredTarget target = getConfiguredTarget("//java/a:a");
assertNoEvents();
Action shardAction = getGeneratingAction(getBinArtifact("_dx/a/shard1.jar", target));
// Verify command line arguments
List<String> arguments = ((SpawnAction) shardAction).getRemainingArguments();
List<String> expectedArguments = new ArrayList<>();
Set<Artifact> artifacts = actionsTestUtil().artifactClosureOf(getFilesToBuild(target));
Artifact shard1 = getFirstArtifactEndingWith(artifacts, "shard1.jar");
Artifact shard2 = getFirstArtifactEndingWith(artifacts, "shard2.jar");
Artifact resourceJar = getFirstArtifactEndingWith(artifacts, "/java_resources.jar");
expectedArguments.add("--output_jar");
expectedArguments.add(shard1.getExecPathString());
expectedArguments.add("--output_jar");
expectedArguments.add(shard2.getExecPathString());
expectedArguments.add("--output_resources");
expectedArguments.add(resourceJar.getExecPathString());
if (multidexMode == MultidexMode.LEGACY) {
Artifact mainDexList = getFirstArtifactEndingWith(artifacts,
"main_dex_list.txt");
expectedArguments.add("--main_dex_filter");
expectedArguments.add(mainDexList.getExecPathString());
}
if (!proguard) {
expectedArguments.add("--input_jar");
expectedArguments.add(
getFirstArtifactEndingWith(artifacts, "a_resources.jar" + nonProguardSuffix)
.getExecPathString());
}
Artifact inputJar;
if (proguard) {
inputJar = getFirstArtifactEndingWith(artifacts, "a_proguard.jar");
} else {
inputJar = getFirstArtifactEndingWith(artifacts, "liba.jar" + nonProguardSuffix);
}
expectedArguments.add("--input_jar");
expectedArguments.add(inputJar.getExecPathString());
assertThat(arguments).containsExactlyElementsIn(expectedArguments).inOrder();
// Verify input and output artifacts
List<String> shardOutputs = ActionsTestUtil.baseArtifactNames(shardAction.getOutputs());
List<String> shardInputs = ActionsTestUtil.baseArtifactNames(shardAction.getInputs());
assertThat(shardOutputs)
.containsExactly("shard1.jar", "shard2.jar", "java_resources.jar");
if (multidexMode == MultidexMode.LEGACY) {
assertThat(shardInputs).contains("main_dex_list.txt");
} else {
assertThat(shardInputs).doesNotContain("main_dex_list.txt");
}
if (proguard) {
assertThat(shardInputs).contains("a_proguard.jar");
assertThat(shardInputs).doesNotContain("liba.jar" + nonProguardSuffix);
} else {
assertThat(shardInputs).contains("liba.jar" + nonProguardSuffix);
assertThat(shardInputs).doesNotContain("a_proguard.jar");
}
assertThat(shardInputs).doesNotContain("a_deploy.jar");
// Verify that dex compilation is followed by the correct merge operation
Action apkAction = getGeneratingAction(getFirstArtifactEndingWith(
actionsTestUtil().artifactClosureOf(getFilesToBuild(target)), "compressed_a_unsigned.apk"));
Action mergeAction = getGeneratingAction(getFirstArtifactEndingWith(
apkAction.getInputs(), "classes.dex.zip"));
Iterable<Artifact> dexShards = Iterables.filter(
mergeAction.getInputs(), ActionsTestUtil.getArtifactSuffixMatcher(".dex.zip"));
assertThat(ActionsTestUtil.baseArtifactNames(dexShards))
.containsExactly("shard1.dex.zip", "shard2.dex.zip");
}
@Test
public void testDexShardingNeedsMultidex() throws Exception {
scratch.file("java/a/BUILD",
"android_binary(",
" name='a',",
" srcs=['A.java'],",
" dex_shards=2,",
" manifest='AndroidManifest.xml')");
reporter.removeHandler(failFastHandler);
getConfiguredTarget("//java/a:a");
assertContainsEvent(".dex sharding is only available in multidex mode");
}
@Test
public void testDexShardingDoesNotWorkWithManualMultidex() throws Exception {
scratch.file("java/a/BUILD",
"android_binary(",
" name='a',",
" srcs=['A.java'],",
" dex_shards=2,",
" multidex='manual_main_dex',",
" main_dex_list='main_dex_list.txt',",
" manifest='AndroidManifest.xml')");
reporter.removeHandler(failFastHandler);
getConfiguredTarget("//java/a:a");
assertContainsEvent(".dex sharding is not available in manual multidex mode");
}
@Test
public void testDexShardingLegacyStructure() throws Exception {
useConfiguration("--noincremental_dexing");
scratch.file("java/a/BUILD",
"android_binary(",
" name='a',",
" srcs=['A.java'],",
" dex_shards=2,",
" multidex='legacy',",
" manifest='AndroidManifest.xml')");
internalTestDexShardStructure(MultidexMode.LEGACY, false, "_desugared.jar");
}
@Test
public void testDexShardingNativeStructure_withNoDesugaring() throws Exception {
useConfiguration("--noexperimental_desugar_for_android", "--noincremental_dexing");
scratch.file("java/a/BUILD",
"android_binary(",
" name='a',",
" srcs=['A.java'],",
" dex_shards=2,",
" multidex='native',",
" manifest='AndroidManifest.xml')");
internalTestDexShardStructure(MultidexMode.NATIVE, false, "");
}
@Test
public void testDexShardingNativeStructure() throws Exception {
useConfiguration("--noincremental_dexing");
scratch.file("java/a/BUILD",
"android_binary(",
" name='a',",
" srcs=['A.java'],",
" dex_shards=2,",
" multidex='native',",
" manifest='AndroidManifest.xml')");
internalTestDexShardStructure(MultidexMode.NATIVE, false, "_desugared.jar");
}
@Test
public void testDexShardingLegacyAndProguardStructure_withNoDesugaring() throws Exception {
useConfiguration("--noexperimental_desugar_for_android");
scratch.file("java/a/BUILD",
"android_binary(",
" name='a',",
" srcs=['A.java'],",
" dex_shards=2,",
" multidex='legacy',",
" manifest='AndroidManifest.xml',",
" proguard_specs=['proguard.cfg'])");
internalTestDexShardStructure(MultidexMode.LEGACY, true, "");
}
@Test
public void testDexShardingLegacyAndProguardStructure() throws Exception {
scratch.file("java/a/BUILD",
"android_binary(",
" name='a',",
" srcs=['A.java'],",
" dex_shards=2,",
" multidex='legacy',",
" manifest='AndroidManifest.xml',",
" proguard_specs=['proguard.cfg'])");
internalTestDexShardStructure(MultidexMode.LEGACY, true, "_desugared.jar");
}
@Test
public void testDexShardingNativeAndProguardStructure() throws Exception {
scratch.file("java/a/BUILD",
"android_binary(",
" name='a',",
" srcs=['A.java'],",
" dex_shards=2,",
" multidex='native',",
" manifest='AndroidManifest.xml',",
" proguard_specs=['proguard.cfg'])");
internalTestDexShardStructure(MultidexMode.NATIVE, true, "");
}
@Test
public void testIncrementalApkAndProguardBuildStructure() throws Exception {
scratch.file("java/a/BUILD",
"android_binary(",
" name='a',",
" srcs=['A.java'],",
" dex_shards=2,",
" multidex='native',",
" manifest='AndroidManifest.xml',",
" proguard_specs=['proguard.cfg'])");
ConfiguredTarget target = getConfiguredTarget("//java/a:a");
Action shardAction = getGeneratingAction(getBinArtifact("_dx/a/shard1.jar", target));
List<String> shardOutputs = ActionsTestUtil.baseArtifactNames(shardAction.getOutputs());
assertThat(shardOutputs).contains("java_resources.jar");
assertThat(shardOutputs).doesNotContain("a_deploy.jar");
}
@Test
public void testManualMainDexBuildStructure() throws Exception {
checkError("java/foo",
"maindex_nomultidex",
"Both \"main_dex_list\" and \"multidex='manual_main_dex'\" must be specified",
"android_binary(",
" name = 'maindex_nomultidex',",
" srcs = ['a.java'],",
" manifest = 'AndroidManifest.xml',",
" multidex = 'manual_main_dex')");
}
@Test
public void testMainDexListLegacyMultidex() throws Exception {
checkError("java/foo",
"maindex_nomultidex",
"Both \"main_dex_list\" and \"multidex='manual_main_dex'\" must be specified",
"android_binary(",
" name = 'maindex_nomultidex',",
" srcs = ['a.java'],",
" manifest = 'AndroidManifest.xml',",
" multidex = 'legacy',",
" main_dex_list = 'main_dex_list.txt')");
}
@Test
public void testMainDexListNativeMultidex() throws Exception {
checkError("java/foo",
"maindex_nomultidex",
"Both \"main_dex_list\" and \"multidex='manual_main_dex'\" must be specified",
"android_binary(",
" name = 'maindex_nomultidex',",
" srcs = ['a.java'],",
" manifest = 'AndroidManifest.xml',",
" multidex = 'native',",
" main_dex_list = 'main_dex_list.txt')");
}
@Test
public void testMainDexListNoMultidex() throws Exception {
checkError("java/foo",
"maindex_nomultidex",
"Both \"main_dex_list\" and \"multidex='manual_main_dex'\" must be specified",
"android_binary(",
" name = 'maindex_nomultidex',",
" srcs = ['a.java'],",
" manifest = 'AndroidManifest.xml',",
" main_dex_list = 'main_dex_list.txt')");
}
@Test
public void testMainDexListWithAndroidSdk() throws Exception {
scratch.file(
"sdk/BUILD",
"android_sdk(",
" name = 'sdk',",
" aapt = 'aapt',",
" adb = 'adb',",
" aidl = 'aidl',",
" android_jar = 'android.jar',",
" apksigner = 'apksigner',",
" dx = 'dx',",
" framework_aidl = 'framework_aidl',",
" main_dex_classes = 'main_dex_classes',",
" main_dex_list_creator = 'main_dex_list_creator',",
" proguard = 'proguard',",
" shrinked_android_jar = 'shrinked_android_jar',",
" zipalign = 'zipalign',",
" tags = ['__ANDROID_RULES_MIGRATION__'])");
scratch.file("java/a/BUILD",
"android_binary(",
" name = 'a',",
" srcs = ['A.java'],",
" manifest = 'AndroidManifest.xml',",
" multidex = 'legacy',",
" main_dex_list_opts = ['--hello', '--world'])");
useConfiguration("--android_sdk=//sdk:sdk");
ConfiguredTarget a = getConfiguredTarget("//java/a:a");
Artifact mainDexList = ActionsTestUtil.getFirstArtifactEndingWith(
actionsTestUtil().artifactClosureOf(getFilesToBuild(a)), "main_dex_list.txt");
List<String> args = getGeneratingSpawnActionArgs(mainDexList);
assertThat(args).containsAtLeast("--hello", "--world");
}
@Test
public void testMainDexAaptGenerationSupported() throws Exception {
useConfiguration("--android_sdk=//sdk:sdk", "--noincremental_dexing");
scratch.file(
"sdk/BUILD",
"android_sdk(",
" name = 'sdk',",
" build_tools_version = '24.0.0',",
" aapt = 'aapt',",
" adb = 'adb',",
" aidl = 'aidl',",
" android_jar = 'android.jar',",
" apksigner = 'apksigner',",
" dx = 'dx',",
" framework_aidl = 'framework_aidl',",
" main_dex_classes = 'main_dex_classes',",
" main_dex_list_creator = 'main_dex_list_creator',",
" proguard = 'proguard',",
" shrinked_android_jar = 'shrinked_android_jar',",
" zipalign = 'zipalign',",
" tags = ['__ANDROID_RULES_MIGRATION__'])");
scratch.file("java/a/BUILD",
"android_binary(",
" name = 'a',",
" srcs = ['A.java'],",
" manifest = 'AndroidManifest.xml',",
" multidex = 'legacy')");
ConfiguredTarget a = getConfiguredTarget("//java/a:a");
Artifact intermediateJar = artifactByPath(ImmutableList.of(getCompressedUnsignedApk(a)),
".apk", ".dex.zip", ".dex.zip", "main_dex_list.txt", "_intermediate.jar");
List<String> args = getGeneratingSpawnActionArgs(intermediateJar);
assertContainsSublist(
args,
ImmutableList.of(
"-include",
targetConfig.getBinFragment() + "/java/a/proguard/a/main_dex_a_proguard.cfg"));
}
@Test
public void testMainDexGenerationWithoutProguardMap() throws Exception {
useConfiguration("--noincremental_dexing");
scratchConfiguredTarget("java/foo", "abin",
"android_binary(",
" name = 'abin',",
" srcs = ['a.java'],",
" proguard_specs = [],",
" manifest = 'AndroidManifest.xml',",
" multidex = 'legacy',)");
ConfiguredTarget a = getConfiguredTarget("//java/foo:abin");
Artifact intermediateJar = artifactByPath(ImmutableList.of(getCompressedUnsignedApk(a)),
".apk", ".dex.zip", ".dex.zip", "main_dex_list.txt", "_intermediate.jar");
List<String> args = getGeneratingSpawnActionArgs(intermediateJar);
MoreAsserts.assertDoesNotContainSublist(
args,
"-previousobfuscationmap");
}
// regression test for b/14288948
@Test
public void testEmptyListAsProguardSpec() throws Exception {
scratch.file(
"java/foo/BUILD",
"android_binary(",
" name = 'abin',",
" srcs = ['a.java'],",
" proguard_specs = [],",
" manifest = 'AndroidManifest.xml')");
Rule rule = getTarget("//java/foo:abin").getAssociatedRule();
assertNoEvents();
ImmutableList<String> implicitOutputFilenames =
rule.getOutputFiles().stream().map(FileTarget::getName).collect(toImmutableList());
assertThat(implicitOutputFilenames).doesNotContain("abin_proguard.jar");
}
@Test
public void testConfigurableProguardSpecsEmptyList() throws Exception {
scratch.file(
"java/foo/BUILD",
"android_binary(",
" name = 'abin',",
" srcs = ['a.java'],",
" proguard_specs = select({",
" '" + BuildType.Selector.DEFAULT_CONDITION_KEY + "': [],",
" }),",
" manifest = 'AndroidManifest.xml')");
Rule rule = getTarget("//java/foo:abin").getAssociatedRule();
assertNoEvents();
ImmutableList<String> implicitOutputFilenames =
rule.getOutputFiles().stream().map(FileTarget::getName).collect(toImmutableList());
assertThat(implicitOutputFilenames).contains("abin_proguard.jar");
}
@Test
public void testConfigurableProguardSpecsEmptyListWithMapping() throws Exception {
scratchConfiguredTarget("java/foo", "abin",
"android_binary(",
" name = 'abin',",
" srcs = ['a.java'],",
" proguard_specs = select({",
" '" + BuildType.Selector.DEFAULT_CONDITION_KEY + "': [],",
" }),",
" proguard_generate_mapping = 1,",
" manifest = 'AndroidManifest.xml')");
assertNoEvents();
}
@Test
public void testResourcesWithConfigurationQualifier_LocalResources() throws Exception {
scratch.file("java/android/resources/BUILD",
"android_binary(name = 'r',",
" manifest = 'AndroidManifest.xml',",
" resource_files = glob(['res/**']),",
" )");
scratch.file("java/android/resources/res/values-en/strings.xml",
"<resources><string name = 'hello'>Hello Android!</string></resources>");
scratch.file("java/android/resources/res/values/strings.xml",
"<resources><string name = 'hello'>Hello Android!</string></resources>");
ConfiguredTarget resource = getConfiguredTarget("//java/android/resources:r");
List<String> args = getGeneratingSpawnActionArgs(getResourceApk(resource));
assertPrimaryResourceDirs(ImmutableList.of("java/android/resources/res"), args);
}
@Test
public void testResourcesInOtherPackage_exported_LocalResources() throws Exception {
scratch.file("java/android/resources/BUILD",
"android_binary(name = 'r',",
" manifest = 'AndroidManifest.xml',",
" resource_files = ['//java/resources/other:res/values/strings.xml'],",
" )");
scratch.file("java/resources/other/BUILD",
"exports_files(['res/values/strings.xml'])");
ConfiguredTarget resource = getConfiguredTarget("//java/android/resources:r");
List<String> args = getGeneratingSpawnActionArgs(getResourceApk(resource));
assertPrimaryResourceDirs(ImmutableList.of("java/resources/other/res"), args);
assertNoEvents();
}
@Test
public void testResourcesInOtherPackage_filegroup_LocalResources() throws Exception {
scratch.file("java/android/resources/BUILD",
"android_binary(name = 'r',",
" manifest = 'AndroidManifest.xml',",
" resource_files = ['//java/other/resources:fg'],",
" )");
scratch.file("java/other/resources/BUILD",
"filegroup(name = 'fg',",
" srcs = ['res/values/strings.xml'],",
")");
ConfiguredTarget resource = getConfiguredTarget("//java/android/resources:r");
List<String> args = getGeneratingSpawnActionArgs(getResourceApk(resource));
assertPrimaryResourceDirs(ImmutableList.of("java/other/resources/res"), args);
assertNoEvents();
}
@Test
public void testResourcesInOtherPackage_filegroupWithExternalSources_LocalResources()
throws Exception {
scratch.file("java/android/resources/BUILD",
"android_binary(name = 'r',",
" manifest = 'AndroidManifest.xml',",
" resource_files = [':fg'],",
" )",
"filegroup(name = 'fg',",
" srcs = ['//java/other/resources:res/values/strings.xml'])");
scratch.file("java/other/resources/BUILD",
"exports_files(['res/values/strings.xml'])");
ConfiguredTarget resource = getConfiguredTarget("//java/android/resources:r");
List<String> args = getGeneratingSpawnActionArgs(getResourceApk(resource));
assertPrimaryResourceDirs(ImmutableList.of("java/other/resources/res"), args);
assertNoEvents();
}
@Test
public void testMultipleDependentResourceDirectories_LocalResources()
throws Exception {
scratch.file("java/android/resources/d1/BUILD",
"android_library(name = 'd1',",
" manifest = 'AndroidManifest.xml',",
" resource_files = ['d1-res/values/strings.xml'],",
" )");
scratch.file("java/android/resources/d2/BUILD",
"android_library(name = 'd2',",
" manifest = 'AndroidManifest.xml',",
" resource_files = ['d2-res/values/strings.xml'],",
" )");
scratch.file("java/android/resources/BUILD",
"android_binary(name = 'r',",
" manifest = 'AndroidManifest.xml',",
" resource_files = ['bin-res/values/strings.xml'],",
" deps = [",
" '//java/android/resources/d1:d1','//java/android/resources/d2:d2'",
" ])");
ConfiguredTarget resource = getConfiguredTarget("//java/android/resources:r");
List<String> args = getGeneratingSpawnActionArgs(getResourceApk(resource));
assertPrimaryResourceDirs(ImmutableList.of("java/android/resources/bin-res"), args);
assertThat(getDirectDependentResourceDirs(args))
.containsAtLeast("java/android/resources/d1/d1-res", "java/android/resources/d2/d2-res");
assertNoEvents();
}
// Regression test for b/11924769
@Test
public void testResourcesInOtherPackage_doubleFilegroup_LocalResources() throws Exception {
scratch.file("java/android/resources/BUILD",
"android_binary(name = 'r',",
" manifest = 'AndroidManifest.xml',",
" resource_files = [':fg'],",
" )",
"filegroup(name = 'fg',",
" srcs = ['//java/other/resources:fg'])");
scratch.file("java/other/resources/BUILD",
"filegroup(name = 'fg',",
" srcs = ['res/values/strings.xml'],",
")");
ConfiguredTarget resource = getConfiguredTarget("//java/android/resources:r");
List<String> args = getGeneratingSpawnActionArgs(getResourceApk(resource));
assertPrimaryResourceDirs(ImmutableList.of("java/other/resources/res"), args);
assertNoEvents();
}
@Test
public void testManifestMissingFails_LocalResources() throws Exception {
checkError("java/android/resources", "r",
"manifest attribute of android_library rule //java/android/resources:r: manifest is "
+ "required when resource_files or assets are defined.",
"filegroup(name = 'b')",
"android_library(name = 'r',",
" resource_files = [':b'],",
" )");
}
@Test
public void testResourcesDoesNotMatchDirectoryLayout_BadFile_LocalResources() throws Exception {
checkError("java/android/resources", "r",
"'java/android/resources/res/somefile.xml' is not in the expected resource directory "
+ "structure of <resource directory>/{"
+ Joiner.on(',').join(AndroidResources.RESOURCE_DIRECTORY_TYPES) + "}",
"android_binary(name = 'r',",
" manifest = 'AndroidManifest.xml',",
" resource_files = ['res/somefile.xml', 'r/t/f/m/raw/fold']",
" )");
}
@Test
public void testResourcesDoesNotMatchDirectoryLayout_BadDirectory_LocalResources()
throws Exception {
checkError("java/android/resources",
"r",
"'java/android/resources/res/other/somefile.xml' is not in the expected resource directory "
+ "structure of <resource directory>/{"
+ Joiner.on(',').join(AndroidResources.RESOURCE_DIRECTORY_TYPES) + "}",
"android_binary(name = 'r',",
" manifest = 'AndroidManifest.xml',",
" resource_files = ['res/other/somefile.xml', 'r/t/f/m/raw/fold']",
" )");
}
@Test
public void testResourcesNotUnderCommonDirectoryFails_LocalResources() throws Exception {
checkError("java/android/resources", "r",
"'java/android/resources/r/t/f/m/raw/fold' (generated by '//java/android/resources:r/t/f/m/"
+ "raw/fold') is not in the same directory 'res' "
+ "(derived from java/android/resources/res/raw/speed). "
+ "All resources must share a common directory",
"android_binary(name = 'r',",
" manifest = 'AndroidManifest.xml',",
" resource_files = ['res/raw/speed', 'r/t/f/m/raw/fold']",
" )");
}
@Test
public void testAssetsAndNoAssetsDirFails_LocalResources() throws Exception {
scratch.file("java/android/resources/assets/values/strings.xml",
"<resources><string name = 'hello'>Hello Android!</string></resources>");
checkError("java/android/resources", "r",
"'assets' and 'assets_dir' should be either both empty or both non-empty",
"android_binary(name = 'r',",
" manifest = 'AndroidManifest.xml',",
" assets = glob(['assets/**']),",
" )");
}
@Test
public void testAssetsDirAndNoAssetsFails_LocalResources() throws Exception {
checkError("java/cpp/android", "r",
"'assets' and 'assets_dir' should be either both empty or both non-empty",
"android_binary(name = 'r',",
" manifest = 'AndroidManifest.xml',",
" assets_dir = 'assets',",
" )");
}
@Test
public void testAssetsNotUnderAssetsDirFails_LocalResources() throws Exception {
checkError("java/android/resources", "r",
"'java/android/resources/r/t/f/m' (generated by '//java/android/resources:r/t/f/m') "
+ "is not beneath 'assets'",
"android_binary(name = 'r',",
" manifest = 'AndroidManifest.xml',",
" assets_dir = 'assets',",
" assets = ['assets/valuable', 'r/t/f/m']",
" )");
}
@Test
public void testFileLocation_LocalResources() throws Exception {
scratch.file("java/android/resources/BUILD",
"android_binary(name = 'r',",
" srcs = ['Foo.java'],",
" manifest = 'AndroidManifest.xml',",
" resource_files = ['res/values/strings.xml'],",
" )");
ConfiguredTarget r = getConfiguredTarget("//java/android/resources:r");
assertThat(getFirstArtifactEndingWith(getFilesToBuild(r), ".apk").getRoot())
.isEqualTo(getTargetConfiguration().getBinDirectory(RepositoryName.MAIN));
}
@Test
public void testCustomPackage_LocalResources() throws Exception {
scratch.file("a/r/BUILD",
"android_binary(name = 'r',",
" srcs = ['Foo.java'],",
" custom_package = 'com.google.android.bar',",
" manifest = 'AndroidManifest.xml',",
" resource_files = ['res/values/strings.xml'],",
" )");
ConfiguredTarget r = getConfiguredTarget("//a/r:r");
assertNoEvents();
List<String> args = getGeneratingSpawnActionArgs(getResourceApk(r));
assertContainsSublist(args, ImmutableList.of("--packageForR", "com.google.android.bar"));
}
@Test
public void testCustomJavacopts() throws Exception {
scratch.file("java/foo/A.java", "foo");
scratch.file("java/foo/BUILD",
"android_binary(name = 'a', manifest = 'AndroidManifest.xml', ",
" srcs = ['A.java'], javacopts = ['-g:lines,source'])");
Artifact deployJar = getFileConfiguredTarget("//java/foo:a_deploy.jar").getArtifact();
Action deployAction = getGeneratingAction(deployJar);
JavaCompileAction javacAction =
(JavaCompileAction)
actionsTestUtil()
.getActionForArtifactEndingWith(
actionsTestUtil().artifactClosureOf(deployAction.getInputs()), "liba.jar");
assertThat(getJavacArguments(javacAction)).contains("-g:lines,source");
}
@Test
public void testFixDepsToolFlag() throws Exception {
useConfiguration("--experimental_fix_deps_tool=autofixer");
scratch.file("java/foo/A.java", "foo");
scratch.file(
"java/foo/BUILD",
"android_binary(name = 'a', manifest = 'AndroidManifest.xml', ",
" srcs = ['A.java'])");
Iterable<String> commandLine =
getJavacArguments(
((JavaCompileAction)
actionsTestUtil()
.getActionForArtifactEndingWith(
actionsTestUtil()
.artifactClosureOf(
getGeneratingAction(
getFileConfiguredTarget("//java/foo:a_deploy.jar")
.getArtifact())
.getInputs()),
"liba.jar")));
assertThat(commandLine).containsAtLeast("--experimental_fix_deps_tool", "autofixer").inOrder();
}
@Test
public void testFixDepsToolFlagEmpty() throws Exception {
scratch.file("java/foo/A.java", "foo");
scratch.file(
"java/foo/BUILD",
"android_binary(name = 'a', manifest = 'AndroidManifest.xml', ",
" srcs = ['A.java'])");
Iterable<String> commandLine =
getJavacArguments(
((JavaCompileAction)
actionsTestUtil()
.getActionForArtifactEndingWith(
actionsTestUtil()
.artifactClosureOf(
getGeneratingAction(
getFileConfiguredTarget("//java/foo:a_deploy.jar")
.getArtifact())
.getInputs()),
"liba.jar")));
assertThat(commandLine).containsAtLeast("--experimental_fix_deps_tool", "add_dep").inOrder();
}
@Test
public void testAndroidBinaryExportsJavaCompilationArgsProvider() throws Exception {
scratch.file("java/foo/A.java", "foo");
scratch.file("java/foo/BUILD",
"android_binary(name = 'a', manifest = 'AndroidManifest.xml', ",
" srcs = ['A.java'], javacopts = ['-g:lines,source'])");
final JavaCompilationArgsProvider provider = JavaInfo
.getProvider(JavaCompilationArgsProvider.class, getConfiguredTarget("//java/foo:a"));
assertThat(provider).isNotNull();
}
@Test
public void testNoApplicationId_LocalResources() throws Exception {
scratch.file("java/a/r/BUILD",
"android_binary(name = 'r',",
" srcs = ['Foo.java'],",
" manifest = 'AndroidManifest.xml',",
" resource_files = ['res/values/strings.xml'],",
" )");
ConfiguredTarget r = getConfiguredTarget("//java/a/r:r");
assertNoEvents();
List<String> args = getGeneratingSpawnActionArgs(getResourceApk(r));
Truth.assertThat(args).doesNotContain("--applicationId");
}
@Test
public void testDisallowPrecompiledJars() throws Exception {
checkError("java/precompiled", "binary",
// messages:
"does not produce any android_binary srcs files (expected .java or .srcjar)",
// build file:
"android_binary(name = 'binary',",
" manifest='AndroidManifest.xml',",
" srcs = [':jar'])",
"filegroup(name = 'jar',",
" srcs = ['lib.jar'])");
}
@Test
public void testDesugarJava8Libs_noProguard() throws Exception {
useConfiguration("--experimental_desugar_java8_libs");
scratch.file(
"java/com/google/android/BUILD",
"android_binary(",
" name = 'foo',",
" srcs = ['foo.java'],",
" manifest = 'AndroidManifest.xml',",
" multidex = 'native',",
")");
ConfiguredTarget top = getConfiguredTarget("//java/com/google/android:foo");
Artifact artifact = getBinArtifact("_dx/foo/_final_classes.dex.zip", top);
assertWithMessage("_final_classes.dex.zip").that(artifact).isNotNull();
Action generatingAction = getGeneratingAction(artifact);
assertThat(ActionsTestUtil.baseArtifactNames(generatingAction.getInputs()))
.containsAtLeast("classes.dex.zip", /*canned*/ "java8_legacy.dex.zip");
}
@Test
public void testDesugarJava8Libs_withProguard() throws Exception {
useConfiguration("--experimental_desugar_java8_libs");
scratch.file(
"java/com/google/android/BUILD",
"android_binary(",
" name = 'foo',",
" srcs = ['foo.java'],",
" manifest = 'AndroidManifest.xml',",
" multidex = 'native',",
" proguard_specs = ['foo.cfg'],",
")");
ConfiguredTarget top = getConfiguredTarget("//java/com/google/android:foo");
Artifact artifact = getBinArtifact("_dx/foo/_final_classes.dex.zip", top);
assertWithMessage("_final_classes.dex.zip").that(artifact).isNotNull();
Action generatingAction = getGeneratingAction(artifact);
assertThat(ActionsTestUtil.baseArtifactNames(generatingAction.getInputs()))
.containsAtLeast("classes.dex.zip", /*built*/ "_java8_legacy.dex.zip");
}
@Test
public void testDesugarJava8Libs_noMultidexError() throws Exception {
useConfiguration("--experimental_desugar_java8_libs");
checkError(/*packageName=*/ "java/com/google/android", /*ruleName=*/ "foo",
/*expectedErrorMessage=*/ "multidex",
"android_binary(",
" name = 'foo',",
" srcs = ['foo.java'],",
" manifest = 'AndroidManifest.xml',",
")");
}
@Test
public void testApplyProguardMapping() throws Exception {
scratch.file(
"java/com/google/android/BUILD",
"android_binary(",
" name = 'foo',",
" srcs = ['foo.java'],",
" proguard_apply_mapping = 'proguard.map',",
" proguard_specs = ['foo.pro'],",
" manifest = 'AndroidManifest.xml',",
")");
ConfiguredTarget ct = getConfiguredTarget("//java/com/google/android:foo");
Artifact artifact = artifactByPath(getFilesToBuild(ct), "_proguard.jar");
Action generatingAction = getGeneratingAction(artifact);
assertThat(Artifact.toExecPaths(generatingAction.getInputs()))
.contains("java/com/google/android/proguard.map");
// Cannot use assertThat().containsAllOf().inOrder() as that does not assert that the elements
// are consecutive.
MoreAsserts.assertContainsSublist(getGeneratingSpawnActionArgs(artifact),
"-applymapping", "java/com/google/android/proguard.map");
}
@Test
public void testApplyProguardMappingWithNoSpec() throws Exception {
checkError(
"java/com/google/android", "foo",
// messages:
"'proguard_apply_mapping' can only be used when 'proguard_specs' is also set",
// build file:
"android_binary(",
" name = 'foo',",
" srcs = ['foo.java'],",
" proguard_apply_mapping = 'proguard.map',",
" manifest = 'AndroidManifest.xml',",
")");
}
@Test
public void testApplyProguardDictionary() throws Exception {
scratch.file(
"java/com/google/android/BUILD",
"android_binary(",
" name = 'foo',",
" srcs = ['foo.java'],",
" proguard_apply_dictionary = 'dictionary.txt',",
" proguard_specs = ['foo.pro'],",
" manifest = 'AndroidManifest.xml',",
")");
ConfiguredTarget ct = getConfiguredTarget("//java/com/google/android:foo");
Artifact artifact = artifactByPath(getFilesToBuild(ct), "_proguard.jar");
Action generatingAction = getGeneratingAction(artifact);
assertThat(Artifact.toExecPaths(generatingAction.getInputs()))
.contains("java/com/google/android/dictionary.txt");
// Cannot use assertThat().containsAllOf().inOrder() as that does not assert that the elements
// are consecutive.
MoreAsserts.assertContainsSublist(getGeneratingSpawnActionArgs(artifact),
"-obfuscationdictionary", "java/com/google/android/dictionary.txt");
MoreAsserts.assertContainsSublist(getGeneratingSpawnActionArgs(artifact),
"-classobfuscationdictionary", "java/com/google/android/dictionary.txt");
MoreAsserts.assertContainsSublist(getGeneratingSpawnActionArgs(artifact),
"-packageobfuscationdictionary", "java/com/google/android/dictionary.txt");
}
@Test
public void testApplyProguardDictionaryWithNoSpec() throws Exception {
checkError(
"java/com/google/android",
"foo",
// messages:
"'proguard_apply_dictionary' can only be used when 'proguard_specs' is also set",
// build file:
"android_binary(",
" name = 'foo',",
" srcs = ['foo.java'],",
" proguard_apply_dictionary = 'dictionary.txt',",
" manifest = 'AndroidManifest.xml',",
")");
}
@Test
public void testFeatureFlagsAttributeSetsSelectInDependency() throws Exception {
useConfiguration(
"--experimental_dynamic_configs=notrim",
"--enforce_transitive_configs_for_config_feature_flag");
scratch.file(
"java/com/foo/BUILD",
"config_feature_flag(",
" name = 'flag1',",
" allowed_values = ['on', 'off'],",
" default_value = 'off',",
")",
"config_setting(",
" name = 'flag1@on',",
" flag_values = {':flag1': 'on'},",
" transitive_configs = [':flag1'],",
")",
"config_feature_flag(",
" name = 'flag2',",
" allowed_values = ['on', 'off'],",
" default_value = 'off',",
")",
"config_setting(",
" name = 'flag2@on',",
" flag_values = {':flag2': 'on'},",
" transitive_configs = [':flag2'],",
")",
"android_library(",
" name = 'lib',",
" srcs = select({",
" ':flag1@on': ['Flag1On.java'],",
" '//conditions:default': ['Flag1Off.java'],",
" }) + select({",
" ':flag2@on': ['Flag2On.java'],",
" '//conditions:default': ['Flag2Off.java'],",
" }),",
" transitive_configs = [':flag1', ':flag2'],",
")",
"android_binary(",
" name = 'foo',",
" manifest = 'AndroidManifest.xml',",
" deps = [':lib'],",
" feature_flags = {",
" 'flag1': 'on',",
" },",
" transitive_configs = [':flag1', ':flag2'],",
")");
ConfiguredTarget binary = getConfiguredTarget("//java/com/foo");
List<String> inputs =
prettyArtifactNames(actionsTestUtil().artifactClosureOf(getFinalUnsignedApk(binary)));
assertThat(inputs).containsAtLeast("java/com/foo/Flag1On.java", "java/com/foo/Flag2Off.java");
assertThat(inputs).containsNoneOf("java/com/foo/Flag1Off.java", "java/com/foo/Flag2On.java");
}
@Test
public void testFeatureFlagsAttributeSetsSelectInBinary() throws Exception {
useConfiguration(
"--experimental_dynamic_configs=notrim",
"--enforce_transitive_configs_for_config_feature_flag");
scratch.file(
"java/com/foo/BUILD",
"config_feature_flag(",
" name = 'flag1',",
" allowed_values = ['on', 'off'],",
" default_value = 'off',",
")",
"config_setting(",
" name = 'flag1@on',",
" flag_values = {':flag1': 'on'},",
" transitive_configs = [':flag1'],",
")",
"config_feature_flag(",
" name = 'flag2',",
" allowed_values = ['on', 'off'],",
" default_value = 'off',",
")",
"config_setting(",
" name = 'flag2@on',",
" flag_values = {':flag2': 'on'},",
" transitive_configs = [':flag2'],",
")",
"android_binary(",
" name = 'foo',",
" manifest = 'AndroidManifest.xml',",
" srcs = select({",
" ':flag1@on': ['Flag1On.java'],",
" '//conditions:default': ['Flag1Off.java'],",
" }) + select({",
" ':flag2@on': ['Flag2On.java'],",
" '//conditions:default': ['Flag2Off.java'],",
" }),",
" feature_flags = {",
" 'flag1': 'on',",
" },",
" transitive_configs = [':flag1', ':flag2'],",
")");
ConfiguredTarget binary = getConfiguredTarget("//java/com/foo");
List<String> inputs =
prettyArtifactNames(actionsTestUtil().artifactClosureOf(getFinalUnsignedApk(binary)));
assertThat(inputs).containsAtLeast("java/com/foo/Flag1On.java", "java/com/foo/Flag2Off.java");
assertThat(inputs).containsNoneOf("java/com/foo/Flag1Off.java", "java/com/foo/Flag2On.java");
}
@Test
public void testFeatureFlagsAttributeSetsSelectInBinaryAlias() throws Exception {
useConfiguration(
"--experimental_dynamic_configs=notrim",
"--enforce_transitive_configs_for_config_feature_flag");
scratch.file(
"java/com/foo/BUILD",
"config_feature_flag(",
" name = 'flag1',",
" allowed_values = ['on', 'off'],",
" default_value = 'off',",
")",
"config_setting(",
" name = 'flag1@on',",
" flag_values = {':flag1': 'on'},",
" transitive_configs = [':flag1'],",
")",
"android_binary(",
" name = 'foo',",
" manifest = 'AndroidManifest.xml',",
" srcs = select({",
" ':flag1@on': ['Flag1On.java'],",
" '//conditions:default': ['Flag1Off.java'],",
" }),",
" feature_flags = {",
" 'flag1': 'on',",
" },",
" transitive_configs = [':flag1'],",
")",
"alias(",
" name = 'alias',",
" actual = ':foo',",
")");
ConfiguredTarget binary = getConfiguredTarget("//java/com/foo:alias");
List<String> inputs =
prettyArtifactNames(actionsTestUtil().artifactClosureOf(getFinalUnsignedApk(binary)));
assertThat(inputs).contains("java/com/foo/Flag1On.java");
assertThat(inputs).doesNotContain("java/com/foo/Flag1Off.java");
}
@Test
public void testFeatureFlagsAttributeFailsAnalysisIfFlagValueIsInvalid() throws Exception {
reporter.removeHandler(failFastHandler);
useConfiguration(
"--experimental_dynamic_configs=on",
"--enforce_transitive_configs_for_config_feature_flag");
scratch.file(
"java/com/foo/BUILD",
"config_feature_flag(",
" name = 'flag1',",
" allowed_values = ['on', 'off'],",
" default_value = 'off',",
")",
"config_setting(",
" name = 'flag1@on',",
" flag_values = {':flag1': 'on'},",
" transitive_configs = [':flag1'],",
")",
"android_library(",
" name = 'lib',",
" srcs = select({",
" ':flag1@on': ['Flag1On.java'],",
" '//conditions:default': ['Flag1Off.java'],",
" }),",
" transitive_configs = [':flag1'],",
")",
"android_binary(",
" name = 'foo',",
" manifest = 'AndroidManifest.xml',",
" deps = [':lib'],",
" feature_flags = {",
" 'flag1': 'invalid',",
" },",
" transitive_configs = [':flag1'],",
")");
assertThat(getConfiguredTarget("//java/com/foo")).isNull();
assertContainsEvent(
"in config_feature_flag rule //java/com/foo:flag1: "
+ "value must be one of [\"off\", \"on\"], but was \"invalid\"");
}
@Test
public void testFeatureFlagsAttributeFailsAnalysisIfFlagValueIsInvalidEvenIfNotUsed()
throws Exception {
reporter.removeHandler(failFastHandler);
useConfiguration(
"--experimental_dynamic_configs=on",
"--enforce_transitive_configs_for_config_feature_flag");
scratch.file(
"java/com/foo/BUILD",
"config_feature_flag(",
" name = 'flag1',",
" allowed_values = ['on', 'off'],",
" default_value = 'off',",
")",
"config_setting(",
" name = 'flag1@on',",
" flag_values = {':flag1': 'on'},",
" transitive_configs = [':flag1'],",
")",
"android_binary(",
" name = 'foo',",
" manifest = 'AndroidManifest.xml',",
" feature_flags = {",
" 'flag1': 'invalid',",
" },",
" transitive_configs = [':flag1'],",
")");
assertThat(getConfiguredTarget("//java/com/foo")).isNull();
assertContainsEvent(
"in config_feature_flag rule //java/com/foo:flag1: "
+ "value must be one of [\"off\", \"on\"], but was \"invalid\"");
}
@Test
public void testFeatureFlagsAttributeFailsAnalysisIfFlagIsAliased()
throws Exception {
reporter.removeHandler(failFastHandler);
useConfiguration(
"--experimental_dynamic_configs=notrim",
"--enforce_transitive_configs_for_config_feature_flag");
scratch.file(
"java/com/foo/BUILD",
"config_feature_flag(",
" name = 'flag1',",
" allowed_values = ['on', 'off'],",
" default_value = 'off',",
")",
"alias(",
" name = 'alias',",
" actual = 'flag1',",
" transitive_configs = [':flag1'],",
")",
"android_binary(",
" name = 'foo',",
" manifest = 'AndroidManifest.xml',",
" feature_flags = {",
" 'alias': 'on',",
" },",
" transitive_configs = [':flag1'],",
")");
assertThat(getConfiguredTarget("//java/com/foo")).isNull();
assertContainsEvent(
"in feature_flags attribute of android_binary rule //java/com/foo:foo: "
+ "Feature flags must be named directly, not through aliases; "
+ "use '//java/com/foo:flag1', not '//java/com/foo:alias'");
}
@Test
public void testFeatureFlagsAttributeSetsFeatureFlagProviderValues() throws Exception {
useConfiguration(
"--experimental_dynamic_configs=notrim",
"--enforce_transitive_configs_for_config_feature_flag");
scratch.file(
"java/com/foo/reader.bzl",
"def _impl(ctx):",
" ctx.actions.write(",
" ctx.outputs.java,",
" '\\n'.join([",
" str(target.label) + ': ' + target[config_common.FeatureFlagInfo].value",
" for target in ctx.attr.flags]))",
" return [DefaultInfo(files=depset([ctx.outputs.java]))]",
"flag_reader = rule(",
" implementation=_impl,",
" attrs={'flags': attr.label_list(providers=[config_common.FeatureFlagInfo])},",
" outputs={'java': '%{name}.java'},",
")");
scratch.file(
"java/com/foo/BUILD",
"load('//java/com/foo:reader.bzl', 'flag_reader')",
"config_feature_flag(",
" name = 'flag1',",
" allowed_values = ['on', 'off'],",
" default_value = 'off',",
")",
"config_feature_flag(",
" name = 'flag2',",
" allowed_values = ['on', 'off'],",
" default_value = 'off',",
")",
"flag_reader(",
" name = 'FooFlags',",
" flags = [':flag1', ':flag2'],",
" transitive_configs = [':flag1', ':flag2'],",
")",
"android_binary(",
" name = 'foo',",
" manifest = 'AndroidManifest.xml',",
" srcs = [':FooFlags.java'],",
" feature_flags = {",
" 'flag1': 'on',",
" },",
" transitive_configs = [':flag1', ':flag2'],",
")");
Artifact flagList =
getFirstArtifactEndingWith(
actionsTestUtil()
.artifactClosureOf(getFinalUnsignedApk(getConfiguredTarget("//java/com/foo"))),
"/FooFlags.java");
FileWriteAction action = (FileWriteAction) getGeneratingAction(flagList);
assertThat(action.getFileContents())
.isEqualTo("//java/com/foo:flag1: on\n//java/com/foo:flag2: off");
}
@Test
public void testNocompressExtensions() throws Exception {
scratch.file(
"java/r/android/BUILD",
"android_binary(",
" name = 'r',",
" srcs = ['Foo.java'],",
" manifest = 'AndroidManifest.xml',",
" resource_files = ['res/raw/foo.apk'],",
" nocompress_extensions = ['.apk', '.so'],",
")");
ConfiguredTarget binary = getConfiguredTarget("//java/r/android:r");
ValidatedAndroidResources resource = getValidatedResources(binary);
List<String> args = resourceArguments(resource);
Artifact inputManifest =
getFirstArtifactEndingWith(
getGeneratingSpawnAction(resource.getManifest()).getInputs(), "AndroidManifest.xml");
assertContainsSublist(
args,
ImmutableList.of(
"--primaryData", "java/r/android/res::" + inputManifest.getExecPathString()));
assertThat(args).contains("--uncompressedExtensions");
assertThat(args.get(args.indexOf("--uncompressedExtensions") + 1)).isEqualTo(".apk,.so");
assertThat(getGeneratingSpawnActionArgs(getCompressedUnsignedApk(binary)))
.containsAtLeast("--nocompress_suffixes", ".apk", ".so")
.inOrder();
assertThat(getGeneratingSpawnActionArgs(getFinalUnsignedApk(binary)))
.containsAtLeast("--nocompress_suffixes", ".apk", ".so")
.inOrder();
}
@Test
public void testFeatureFlagPolicyMustContainRuleToUseFeatureFlags() throws Exception {
reporter.removeHandler(failFastHandler); // expecting an error
useConfiguration("--enforce_transitive_configs_for_config_feature_flag");
scratch.overwriteFile(
"tools/whitelists/config_feature_flag/BUILD",
"package_group(",
" name = 'config_feature_flag',",
" packages = ['//flag'])");
scratch.file(
"flag/BUILD",
"config_feature_flag(",
" name = 'flag',",
" allowed_values = ['right', 'wrong'],",
" default_value = 'right',",
" visibility = ['//java/com/google/android/foo:__pkg__'],",
")");
scratch.file(
"java/com/google/android/foo/BUILD",
"android_binary(",
" name = 'foo',",
" manifest = 'AndroidManifest.xml',",
" srcs = [':FooFlags.java'],",
" feature_flags = {",
" '//flag:flag': 'right',",
" },",
" transitive_configs = ['//flag:flag'],",
")");
assertThat(getConfiguredTarget("//java/com/google/android/foo:foo")).isNull();
assertContainsEvent(
"in feature_flags attribute of android_binary rule //java/com/google/android/foo:foo: "
+ "the feature_flags attribute is not available in package "
+ "'java/com/google/android/foo'");
}
@Test
public void testFeatureFlagPolicyDoesNotBlockRuleIfInPolicy() throws Exception {
useConfiguration("--enforce_transitive_configs_for_config_feature_flag");
scratch.overwriteFile(
"tools/whitelists/config_feature_flag/BUILD",
"package_group(",
" name = 'config_feature_flag',",
" packages = ['//flag', '//java/com/google/android/foo'])");
scratch.file(
"flag/BUILD",
"config_feature_flag(",
" name = 'flag',",
" allowed_values = ['right', 'wrong'],",
" default_value = 'right',",
" visibility = ['//java/com/google/android/foo:__pkg__'],",
")");
scratch.file(
"java/com/google/android/foo/BUILD",
"android_binary(",
" name = 'foo',",
" manifest = 'AndroidManifest.xml',",
" srcs = [':FooFlags.java'],",
" feature_flags = {",
" '//flag:flag': 'right',",
" },",
" transitive_configs = ['//flag:flag'],",
")");
assertThat(getConfiguredTarget("//java/com/google/android/foo:foo")).isNotNull();
assertNoEvents();
}
@Test
public void testFeatureFlagPolicyIsNotUsedIfFlagValuesNotUsed() throws Exception {
scratch.overwriteFile(
"tools/whitelists/config_feature_flag/BUILD",
"package_group(",
" name = 'config_feature_flag',",
" packages = ['*super* busted package group'])");
scratch.file(
"java/com/google/android/foo/BUILD",
"android_binary(",
" name = 'foo',",
" manifest = 'AndroidManifest.xml',",
" srcs = [':FooFlags.java'],",
")");
assertThat(getConfiguredTarget("//java/com/google/android/foo:foo")).isNotNull();
// the package_group is busted, so we would have failed to get this far if we depended on it
assertNoEvents();
// sanity check time: does this test actually test what we're testing for?
reporter.removeHandler(failFastHandler);
assertThat(getConfiguredTarget("//tools/whitelists/config_feature_flag:config_feature_flag"))
.isNull();
assertContainsEvent("*super* busted package group");
}
@Test
public void testAapt2WithoutAndroidSdk() throws Exception {
useConfiguration("--android_aapt=aapt2");
checkError(
"java/a",
"a",
"aapt2 processing requested but not available on the android_sdk",
"android_binary(",
" name = 'a',",
" srcs = ['A.java'],",
" manifest = 'AndroidManifest.xml',",
" resource_files = [ 'res/values/values.xml' ], ",
" aapt_version = 'aapt2'",
")");
}
@Test
public void testAapt2FlagWithoutAndroidSdk() throws Exception {
useConfiguration("--android_aapt=aapt2");
checkError(
"java/a",
"a",
"aapt2 processing requested but not available on the android_sdk",
"android_binary(",
" name = 'a',",
" srcs = ['A.java'],",
" manifest = 'AndroidManifest.xml',",
" resource_files = [ 'res/values/values.xml' ], ",
")");
}
@Test
public void testAapt2WithAndroidSdk() throws Exception {
mockAndroidSdkWithAapt2();
scratch.file(
"java/a/BUILD",
"android_binary(",
" name = 'a',",
" srcs = ['A.java'],",
" manifest = 'AndroidManifest.xml',",
" resource_files = [ 'res/values/values.xml' ], ",
" aapt_version = 'aapt2'",
")");
useConfiguration("--android_sdk=//sdk:sdk");
ConfiguredTarget a = getConfiguredTarget("//java/a:a");
Artifact apk = getImplicitOutputArtifact(a, AndroidRuleClasses.ANDROID_RESOURCES_APK);
assertThat(getGeneratingSpawnActionArgs(apk))
.containsAtLeast("--aapt2", "sdk/aapt2", "--tool", "AAPT2_PACKAGE");
}
@Test
public void testAapt2WithAndroidSdkAndDependencies() throws Exception {
mockAndroidSdkWithAapt2();
scratch.file(
"java/b/BUILD",
"android_library(",
" name = 'b',",
" srcs = ['B.java'],",
" manifest = 'AndroidManifest.xml',",
" resource_files = [ 'res/values/values.xml' ], ",
")");
scratch.file(
"java/a/BUILD",
"android_binary(",
" name = 'a',",
" srcs = ['A.java'],",
" manifest = 'AndroidManifest.xml',",
" deps = [ '//java/b:b' ],",
" resource_files = [ 'res/values/values.xml' ], ",
" aapt_version = 'aapt2'",
")");
useConfiguration("--android_sdk=//sdk:sdk");
ConfiguredTarget a = getConfiguredTarget("//java/a:a");
ConfiguredTarget b = getDirectPrerequisite(a, "//java/b:b");
Artifact classJar =
getImplicitOutputArtifact(a, AndroidRuleClasses.ANDROID_RESOURCES_CLASS_JAR);
Artifact rTxt = getImplicitOutputArtifact(a, AndroidRuleClasses.ANDROID_R_TXT);
Artifact apk = getImplicitOutputArtifact(a, AndroidRuleClasses.ANDROID_RESOURCES_APK);
SpawnAction apkAction = getGeneratingSpawnAction(apk);
assertThat(getGeneratingSpawnActionArgs(apk))
.containsAtLeast("--aapt2", "sdk/aapt2", "--tool", "AAPT2_PACKAGE");
assertThat(apkAction.getInputs())
.contains(
getImplicitOutputArtifact(b, AndroidRuleClasses.ANDROID_COMPILED_SYMBOLS));
SpawnAction classAction = getGeneratingSpawnAction(classJar);
assertThat(classAction.getInputs())
.containsAtLeast(
rTxt, getImplicitOutputArtifact(b, AndroidRuleClasses.ANDROID_RESOURCES_AAPT2_R_TXT));
}
@Test
public void testAapt2ResourceShrinkingAction() throws Exception {
mockAndroidSdkWithAapt2();
scratch.file(
"java/com/google/android/hello/BUILD",
"android_binary(name = 'hello',",
" srcs = ['Foo.java'],",
" manifest = 'AndroidManifest.xml',",
" inline_constants = 0,",
" aapt_version='aapt2',",
" resource_files = ['res/values/strings.xml'],",
" shrink_resources = 1,",
" proguard_specs = ['proguard-spec.pro'],)");
useConfiguration("--android_sdk=//sdk:sdk");
ConfiguredTargetAndData targetAndData =
getConfiguredTargetAndData("//java/com/google/android/hello:hello");
ConfiguredTarget binary = targetAndData.getConfiguredTarget();
Artifact jar = getResourceClassJar(targetAndData);
assertThat(getGeneratingAction(jar).getMnemonic()).isEqualTo("RClassGenerator");
assertThat(getGeneratingSpawnActionArgs(jar)).contains("--finalFields");
Set<Artifact> artifacts = actionsTestUtil().artifactClosureOf(getFilesToBuild(binary));
assertThat(artifacts)
.containsAtLeast(
getFirstArtifactEndingWith(artifacts, "resource_files.zip"),
getFirstArtifactEndingWith(artifacts, "proguard.jar"),
getFirstArtifactEndingWith(artifacts, "shrunk.ap_"));
List<String> processingArgs =
getGeneratingSpawnActionArgs(getFirstArtifactEndingWith(artifacts, "resource_files.zip"));
assertThat(flagValue("--resourcesOutput", processingArgs))
.endsWith("hello_files/resource_files.zip");
List<String> proguardArgs =
getGeneratingSpawnActionArgs(getFirstArtifactEndingWith(artifacts, "proguard.jar"));
assertThat(flagValue("-outjars", proguardArgs)).endsWith("hello_proguard.jar");
List<String> shrinkingArgs =
getGeneratingSpawnActionArgs(getFirstArtifactEndingWith(artifacts, "shrunk.ap_"));
assertThat(flagValue("--tool", shrinkingArgs)).isEqualTo("SHRINK_AAPT2");
assertThat(flagValue("--aapt2", shrinkingArgs)).isEqualTo(flagValue("--aapt2", processingArgs));
assertThat(flagValue("--resources", shrinkingArgs))
.isEqualTo(flagValue("--resourcesOutput", processingArgs));
assertThat(flagValue("--shrunkJar", shrinkingArgs))
.isEqualTo(flagValue("-outjars", proguardArgs));
assertThat(flagValue("--proguardMapping", shrinkingArgs))
.isEqualTo(flagValue("-printmapping", proguardArgs));
assertThat(flagValue("--rTxt", shrinkingArgs))
.isEqualTo(flagValue("--rOutput", processingArgs));
List<String> packageArgs =
getGeneratingSpawnActionArgs(getFirstArtifactEndingWith(artifacts, "_hello_proguard.cfg"));
assertThat(flagValue("--tool", packageArgs)).isEqualTo("AAPT2_PACKAGE");
assertThat(packageArgs).doesNotContain("--conditionalKeepRules");
}
@Test
public void testAapt2ResourceCycleShrinking() throws Exception {
mockAndroidSdkWithAapt2();
useConfiguration(
"--android_sdk=//sdk:sdk", "--experimental_android_resource_cycle_shrinking=true");
scratch.file(
"java/com/google/android/hello/BUILD",
"android_binary(name = 'hello',",
" srcs = ['Foo.java'],",
" manifest = 'AndroidManifest.xml',",
" inline_constants = 0,",
" aapt_version='aapt2',",
" resource_files = ['res/values/strings.xml'],",
" shrink_resources = 1,",
" proguard_specs = ['proguard-spec.pro'],)");
ConfiguredTargetAndData targetAndData =
getConfiguredTargetAndData("//java/com/google/android/hello:hello");
ConfiguredTarget binary = targetAndData.getConfiguredTarget();
Artifact jar = getResourceClassJar(targetAndData);
assertThat(getGeneratingAction(jar).getMnemonic()).isEqualTo("RClassGenerator");
assertThat(getGeneratingSpawnActionArgs(jar)).contains("--nofinalFields");
Set<Artifact> artifacts = actionsTestUtil().artifactClosureOf(getFilesToBuild(binary));
List<String> packageArgs =
getGeneratingSpawnActionArgs(getFirstArtifactEndingWith(artifacts, "_hello_proguard.cfg"));
assertThat(flagValue("--tool", packageArgs)).isEqualTo("AAPT2_PACKAGE");
assertThat(packageArgs).contains("--conditionalKeepRules");
}
@Test
public void testAapt2ResourceCycleShinkingWithoutResourceShrinking() throws Exception {
mockAndroidSdkWithAapt2();
useConfiguration(
"--android_sdk=//sdk:sdk", "--experimental_android_resource_cycle_shrinking=true");
checkError(
"java/a",
"a",
"resource cycle shrinking can only be enabled when resource shrinking is enabled",
"android_binary(",
" name = 'a',",
" srcs = ['A.java'],",
" manifest = 'AndroidManifest.xml',",
" resource_files = [ 'res/values/values.xml' ], ",
" shrink_resources = 0,",
" aapt_version = 'aapt2'",
")");
}
@Test
public void testDebugKeyExistsBinary() throws Exception {
String debugKeyTarget = "//java/com/google/android/hello:debug_keystore";
scratch.file("java/com/google/android/hello/debug_keystore", "A debug key");
scratch.file(
"java/com/google/android/hello/BUILD",
"android_binary(name = 'b',",
" srcs = ['HelloApp.java'],",
" manifest = 'AndroidManifest.xml',",
" debug_key = '" + debugKeyTarget + "')");
checkDebugKey(debugKeyTarget, true);
}
@Test
public void testDebugKeyNotExistsBinary() throws Exception {
String debugKeyTarget = "//java/com/google/android/hello:debug_keystore";
scratch.file(
"java/com/google/android/hello/BUILD",
"android_binary(name = 'b',",
" srcs = ['HelloApp.java'],",
" manifest = 'AndroidManifest.xml')");
checkDebugKey(debugKeyTarget, false);
}
@Test
public void testOnlyProguardSpecs() throws Exception {
scratch.file("java/com/google/android/hello/BUILD",
"android_library(name = 'l2',",
" srcs = ['MoreMaps.java'],",
" neverlink = 1)",
"android_library(name = 'l3',",
" idl_srcs = ['A.aidl'],",
" deps = [':l2'])",
"android_library(name = 'l4',",
" srcs = ['SubMoreMaps.java'],",
" neverlink = 1)",
"android_binary(name = 'b',",
" srcs = ['HelloApp.java'],",
" deps = [':l3', ':l4'],",
" manifest = 'AndroidManifest.xml',",
" proguard_specs = ['proguard-spec.pro', 'proguard-spec1.pro',",
" 'proguard-spec2.pro'])");
checkProguardUse(
"//java/com/google/android/hello:b", "b_proguard.jar", false, null,
targetConfig.getBinFragment()
+ "/java/com/google/android/hello/proguard/b/legacy_b_combined_library_jars.jar");
}
@Test
public void testOnlyProguardSpecsProguardJar() throws Exception {
scratch.file("java/com/google/android/hello/BUILD",
"android_library(name = 'l2',",
" srcs = ['MoreMaps.java'],",
" neverlink = 1)",
"android_library(name = 'l3',",
" idl_srcs = ['A.aidl'],",
" deps = [':l2'])",
"android_library(name = 'l4',",
" srcs = ['SubMoreMaps.java'],",
" neverlink = 1)",
"android_binary(name = 'b',",
" srcs = ['HelloApp.java'],",
" deps = [':l3', ':l4'],",
" manifest = 'AndroidManifest.xml',",
" proguard_generate_mapping = 1,",
" proguard_specs = ['proguard-spec.pro', 'proguard-spec1.pro',",
" 'proguard-spec2.pro'])");
ConfiguredTarget output = getConfiguredTarget("//java/com/google/android/hello:b_proguard.jar");
assertProguardUsed(output);
output = getConfiguredTarget("//java/com/google/android/hello:b_proguard.map");
assertWithMessage("proguard.map is not in the rule output")
.that(
actionsTestUtil()
.getActionForArtifactEndingWith(getFilesToBuild(output), "_proguard.map"))
.isNotNull();
}
@Test
public void testCommandLineForMultipleProguardSpecs() throws Exception {
scratch.file("java/com/google/android/hello/BUILD",
"android_library(name = 'l1',",
" srcs = ['Maps.java'],",
" neverlink = 1)",
"android_binary(name = 'b',",
" srcs = ['HelloApp.java'],",
" deps = [':l1'],",
" manifest = 'AndroidManifest.xml',",
" proguard_specs = ['proguard-spec.pro', 'proguard-spec1.pro',",
" 'proguard-spec2.pro'])");
ConfiguredTarget binary = getConfiguredTarget("//java/com/google/android/hello:b");
SpawnAction action = (SpawnAction) actionsTestUtil().getActionForArtifactEndingWith(
getFilesToBuild(binary), "_proguard.jar");
assertWithMessage("Proguard action does not contain expected inputs.")
.that(prettyArtifactNames(action.getInputs()))
.containsAtLeast(
"java/com/google/android/hello/proguard-spec.pro",
"java/com/google/android/hello/proguard-spec1.pro",
"java/com/google/android/hello/proguard-spec2.pro");
assertThat(action.getArguments())
.containsExactly(
getProguardBinary().getExecPathString(),
"-forceprocessing",
"-injars",
execPathEndingWith(action.getInputs(), "b_deploy.jar"),
"-outjars",
execPathEndingWith(action.getOutputs(), "b_proguard.jar"),
// Only one combined library jar
"-libraryjars",
execPathEndingWith(action.getInputs(), "legacy_b_combined_library_jars.jar"),
"@" + execPathEndingWith(action.getInputs(), "b_proguard.cfg"),
"@java/com/google/android/hello/proguard-spec.pro",
"@java/com/google/android/hello/proguard-spec1.pro",
"@java/com/google/android/hello/proguard-spec2.pro",
"-printseeds",
execPathEndingWith(action.getOutputs(), "_proguard.seeds"),
"-printusage",
execPathEndingWith(action.getOutputs(), "_proguard.usage"),
"-printconfiguration",
execPathEndingWith(action.getOutputs(), "_proguard.config"))
.inOrder();
}
/** Regression test for b/17790639 */
@Test
public void testNoDuplicatesInProguardCommand() throws Exception {
scratch.file("java/com/google/android/hello/BUILD",
"android_library(name = 'l1',",
" srcs = ['Maps.java'],",
" neverlink = 1)",
"android_binary(name = 'b',",
" srcs = ['HelloApp.java'],",
" deps = [':l1'],",
" manifest = 'AndroidManifest.xml',",
" proguard_specs = ['proguard-spec.pro', 'proguard-spec1.pro',",
" 'proguard-spec2.pro'])");
ConfiguredTarget binary = getConfiguredTarget("//java/com/google/android/hello:b");
SpawnAction action = (SpawnAction) actionsTestUtil().getActionForArtifactEndingWith(
getFilesToBuild(binary), "_proguard.jar");
assertThat(action.getArguments())
.containsExactly(
getProguardBinary().getExecPathString(),
"-forceprocessing",
"-injars",
execPathEndingWith(action.getInputs(), "b_deploy.jar"),
"-outjars",
execPathEndingWith(action.getOutputs(), "b_proguard.jar"),
// Only one combined library jar
"-libraryjars",
execPathEndingWith(action.getInputs(), "legacy_b_combined_library_jars.jar"),
"@" + execPathEndingWith(action.getInputs(), "b_proguard.cfg"),
"@java/com/google/android/hello/proguard-spec.pro",
"@java/com/google/android/hello/proguard-spec1.pro",
"@java/com/google/android/hello/proguard-spec2.pro",
"-printseeds",
execPathEndingWith(action.getOutputs(), "_proguard.seeds"),
"-printusage",
execPathEndingWith(action.getOutputs(), "_proguard.usage"),
"-printconfiguration",
execPathEndingWith(action.getOutputs(), "_proguard.config"))
.inOrder();
}
@Test
public void testProguardMapping() throws Exception {
scratch.file("java/com/google/android/hello/BUILD",
"android_binary(name = 'b',",
" srcs = ['HelloApp.java'],",
" manifest = 'AndroidManifest.xml',",
" proguard_specs = ['proguard-spec.pro'],",
" proguard_generate_mapping = 1)");
checkProguardUse(
"//java/com/google/android/hello:b", "b_proguard.jar", true, null, getAndroidJarPath());
}
@Test
public void testProguardMappingProvider() throws Exception {
scratch.file("java/com/google/android/hello/BUILD",
"android_library(name = 'l2',",
" srcs = ['MoreMaps.java'],",
" neverlink = 1)",
"android_library(name = 'l3',",
" idl_srcs = ['A.aidl'],",
" deps = [':l2'])",
"android_library(name = 'l4',",
" srcs = ['SubMoreMaps.java'],",
" neverlink = 1)",
"android_binary(name = 'b1',",
" srcs = ['HelloApp.java'],",
" deps = [':l3', ':l4'],",
" manifest = 'AndroidManifest.xml',",
" proguard_generate_mapping = 1,",
" proguard_specs = ['proguard-spec.pro', 'proguard-spec1.pro',",
" 'proguard-spec2.pro'])",
"android_binary(name = 'b2',",
" srcs = ['HelloApp.java'],",
" deps = [':l3', ':l4'],",
" manifest = 'AndroidManifest.xml',",
" proguard_specs = ['proguard-spec.pro', 'proguard-spec1.pro',",
" 'proguard-spec2.pro'])");
ConfiguredTarget output = getConfiguredTarget("//java/com/google/android/hello:b1");
assertProguardUsed(output);
Artifact mappingArtifact = getBinArtifact("b1_proguard.map", output);
ProguardMappingProvider mappingProvider = output.get(ProguardMappingProvider.PROVIDER);
assertThat(mappingProvider.getProguardMapping()).isEqualTo(mappingArtifact);
output = getConfiguredTarget("//java/com/google/android/hello:b2");
assertProguardUsed(output);
assertThat(output.get(ProguardMappingProvider.PROVIDER)).isNull();
}
@Test
public void testLegacyOptimizationModeUsesExtraProguardSpecs() throws Exception {
useConfiguration("--extra_proguard_specs=java/com/google/android/hello:extra.pro");
scratch.file("java/com/google/android/hello/BUILD",
"exports_files(['extra.pro'])",
"android_binary(name = 'b',",
" srcs = ['HelloApp.java'],",
" manifest = 'AndroidManifest.xml',",
" proguard_specs = ['proguard-spec.pro'])");
checkProguardUse(
"//java/com/google/android/hello:b", "b_proguard.jar", false, null, getAndroidJarPath());
SpawnAction action = (SpawnAction) actionsTestUtil().getActionForArtifactEndingWith(
getFilesToBuild(getConfiguredTarget("//java/com/google/android/hello:b")), "_proguard.jar");
assertThat(prettyArtifactNames(action.getInputs())).containsNoDuplicates();
assertThat(Collections2.filter(action.getArguments(), arg -> arg.startsWith("@")))
.containsExactly(
"@" + execPathEndingWith(action.getInputs(), "/proguard-spec.pro"),
"@" + execPathEndingWith(action.getInputs(), "/_b_proguard.cfg"),
"@java/com/google/android/hello/extra.pro");
}
@Test
public void testExtraProguardSpecsDontDuplicateProguardInputFiles() throws Exception {
useConfiguration("--extra_proguard_specs=java/com/google/android/hello:proguard-spec.pro");
scratch.file("java/com/google/android/hello/BUILD",
"android_binary(name = 'b',",
" srcs = ['HelloApp.java'],",
" manifest = 'AndroidManifest.xml',",
" proguard_specs = ['proguard-spec.pro'])");
checkProguardUse(
"//java/com/google/android/hello:b", "b_proguard.jar", false, null, getAndroidJarPath());
SpawnAction action = (SpawnAction) actionsTestUtil().getActionForArtifactEndingWith(
getFilesToBuild(getConfiguredTarget("//java/com/google/android/hello:b")), "_proguard.jar");
assertThat(prettyArtifactNames(action.getInputs())).containsNoDuplicates();
assertThat(Collections2.filter(action.getArguments(), arg -> arg.startsWith("@")))
.containsExactly(
"@java/com/google/android/hello/proguard-spec.pro",
"@" + execPathEndingWith(action.getInputs(), "/_b_proguard.cfg"));
}
@Test
public void testLegacyLinkingProguardNotUsedWithoutSpecOnBinary() throws Exception {
useConfiguration(
"--java_optimization_mode=legacy",
"--extra_proguard_specs=//java/com/google/android/hello:ignored.pro");
scratch.file("java/com/google/android/hello/BUILD",
"exports_files(['ignored.pro'])",
"android_library(name = 'l2',",
" srcs = ['MoreMaps.java'],",
" neverlink = 1)",
"android_library(name = 'l3',",
" idl_srcs = ['A.aidl'],",
// Having a library spec should not trigger proguard on the binary target.
" proguard_specs = ['library_spec.cfg'],",
" deps = [':l2'])",
"android_library(name = 'l4',",
" srcs = ['SubMoreMaps.java'],",
" neverlink = 1)",
"android_binary(name = 'b',",
" srcs = ['HelloApp.java'],",
" deps = [':l3', ':l4'],",
" manifest = 'AndroidManifest.xml',)");
assertProguardNotUsed(getConfiguredTarget("//java/com/google/android/hello:b"));
}
@Test
public void testFullOptimizationModeForcesProguard() throws Exception {
useConfiguration("--java_optimization_mode=optimize_minify");
SpawnAction action = testProguardOptimizationMode();
// Expect spec generated from resources as well as library's (validated) spec as usual
assertThat(Collections2.filter(action.getArguments(), arg -> arg.startsWith("@")))
.containsExactly(
"@" + execPathEndingWith(action.getInputs(), "/_b_proguard.cfg"),
"@" + execPathEndingWith(action.getInputs(), "library_spec.cfg_valid"));
}
@Test
public void testOptimizingModesIncludeExtraProguardSpecs() throws Exception {
useConfiguration(
"--java_optimization_mode=fast_minify",
"--extra_proguard_specs=//java/com/google/android/hello:extra.pro");
SpawnAction action = testProguardOptimizationMode();
assertThat(action.getArguments()).contains("@java/com/google/android/hello/extra.pro");
}
@Test
public void testRenameModeForcesProguardWithSpecForMode() throws Exception {
testProguardPartialOptimizationMode("rename", "-dontshrink\n-dontoptimize\n");
}
@Test
public void testMinimizingModeForcesProguardWithSpecForMode() throws Exception {
testProguardPartialOptimizationMode("fast_minify", "-dontoptimize\n");
}
public void testProguardPartialOptimizationMode(String mode, String expectedSpecForMode)
throws Exception {
useConfiguration("--java_optimization_mode=" + mode);
SpawnAction action = testProguardOptimizationMode();
// Expect spec generated from resources, library's (validated) spec, and spec for mode
String modeSpecFileSuffix = "/" + mode + "_b_proguard.cfg";
assertThat(Collections2.filter(action.getArguments(), arg -> arg.startsWith("@")))
.containsExactly(
"@" + execPathEndingWith(action.getInputs(), modeSpecFileSuffix),
"@" + execPathEndingWith(action.getInputs(), "/_b_proguard.cfg"),
"@" + execPathEndingWith(action.getInputs(), "library_spec.cfg_valid"));
FileWriteAction modeSpec = (FileWriteAction) actionsTestUtil().getActionForArtifactEndingWith(
action.getInputs(), modeSpecFileSuffix);
assertThat(modeSpec.getFileContents()).isEqualTo(expectedSpecForMode);
}
public SpawnAction testProguardOptimizationMode() throws Exception {
scratch.file("java/com/google/android/hello/BUILD",
"exports_files(['extra.pro'])",
"android_library(name = 'l',",
" idl_srcs = ['A.aidl'],",
" proguard_specs = ['library_spec.cfg'])",
"android_binary(name = 'b',",
" srcs = ['HelloApp.java'],",
" deps = [':l'],",
" manifest = 'AndroidManifest.xml',)");
checkProguardUse(
"//java/com/google/android/hello:b",
"b_proguard.jar",
/*expectMapping*/ true,
/*passes*/ null,
getAndroidJarPath());
SpawnAction action = (SpawnAction) actionsTestUtil().getActionForArtifactEndingWith(
getFilesToBuild(getConfiguredTarget("//java/com/google/android/hello:b")), "_proguard.jar");
assertThat(prettyArtifactNames(action.getInputs())).containsNoDuplicates();
return action;
}
@Test
public void testProguardSpecFromLibraryUsedInBinary() throws Exception {
scratch.file("java/com/google/android/hello/BUILD",
"android_library(name = 'l2',",
" srcs = ['MoreMaps.java'],",
" proguard_specs = ['library_spec.cfg'])",
"android_library(name = 'l3',",
" idl_srcs = ['A.aidl'],",
" proguard_specs = ['library_spec.cfg'],",
" deps = [':l2'])",
"android_library(name = 'l4',",
" srcs = ['SubMoreMaps.java'],",
" neverlink = 1)",
"android_binary(name = 'b',",
" srcs = ['HelloApp.java'],",
" deps = [':l3', ':l4'],",
" proguard_specs = ['proguard-spec.pro'],",
" manifest = 'AndroidManifest.xml',)");
assertProguardUsed(getConfiguredTarget("//java/com/google/android/hello:b"));
assertProguardGenerated(getConfiguredTarget("//java/com/google/android/hello:b"));
SpawnAction action = (SpawnAction) actionsTestUtil().getActionForArtifactEndingWith(
getFilesToBuild(getConfiguredTarget("//java/com/google/android/hello:b")), "_proguard.jar");
assertThat(prettyArtifactNames(action.getInputs()))
.contains("java/com/google/android/hello/proguard-spec.pro");
assertThat(prettyArtifactNames(action.getInputs()))
.contains(
"java/com/google/android/hello/validated_proguard/l2/java/com/google/android/hello/library_spec.cfg_valid");
assertThat(prettyArtifactNames(action.getInputs())).containsNoDuplicates();
}
@Test
public void testResourcesUsedInProguardGenerate() throws Exception {
scratch.file("java/com/google/android/hello/BUILD",
"android_binary(name = 'b',",
" srcs = ['HelloApp.java'],",
" manifest = 'AndroidManifest.xml',",
" resource_files = ['res/values/strings.xml'],",
" proguard_specs = ['proguard-spec.pro', 'proguard-spec1.pro',",
" 'proguard-spec2.pro'])");
scratch.file("java/com/google/android/hello/res/values/strings.xml",
"<resources><string name = 'hello'>Hello Android!</string></resources>");
ConfiguredTarget binary = getConfiguredTarget("//java/com/google/android/hello:b");
SpawnAction action = (SpawnAction) actionsTestUtil().getActionForArtifactEndingWith(
actionsTestUtil().artifactClosureOf(getFilesToBuild(binary)), "_proguard.cfg");
assertProguardGenerated(binary);
assertWithMessage("Generate proguard action does not contain expected input.")
.that(prettyArtifactNames(action.getInputs()))
.contains("java/com/google/android/hello/res/values/strings.xml");
}
@Test
public void testUseSingleJarForLibraryJars() throws Exception {
scratch.file("java/com/google/android/hello/BUILD",
"android_library(name = 'l1',",
" srcs = ['Maps.java'],",
" neverlink = 1)",
"android_binary(name = 'b',",
" srcs = ['HelloApp.java'],",
" deps = [':l1'],",
" manifest = 'AndroidManifest.xml',",
" proguard_specs = ['proguard-spec.pro', 'proguard-spec1.pro',",
" 'proguard-spec2.pro'])");
ConfiguredTarget binary = getConfiguredTarget("//java/com/google/android/hello:b");
SpawnAction action = (SpawnAction) actionsTestUtil().getActionForArtifactEndingWith(
getFilesToBuild(binary), "_proguard.jar");
checkProguardLibJars(action, targetConfig.getBinFragment()
+ "/java/com/google/android/hello/proguard/b/legacy_b_combined_library_jars.jar");
}
@Test
public void testOnlyOneLibraryJar() throws Exception {
scratch.file("java/com/google/android/hello/BUILD",
"android_binary(name = 'b',",
" srcs = ['HelloApp.java'],",
" manifest = 'AndroidManifest.xml',",
" proguard_specs = ['proguard-spec.pro'],",
" proguard_generate_mapping = 1)");
ConfiguredTarget binary = getConfiguredTarget("//java/com/google/android/hello:b");
SpawnAction action = (SpawnAction) actionsTestUtil().getActionForArtifactEndingWith(
getFilesToBuild(binary), "_proguard.jar");
checkProguardLibJars(action, getAndroidJarPath());
}
@Test
public void testApkInfoAccessibleFromSkylark() throws Exception {
scratch.file(
"java/com/google/android/BUILD",
"load(':postprocess.bzl', 'postprocess')",
"android_binary(name = 'b1',",
" srcs = ['b1.java'],",
" manifest = 'AndroidManifest.xml')",
"postprocess(name = 'postprocess', dep = ':b1')");
scratch.file(
"java/com/google/android/postprocess.bzl",
"def _impl(ctx):",
" return [DefaultInfo(files=depset([ctx.attr.dep[ApkInfo].signed_apk]))]",
"postprocess = rule(implementation=_impl,",
" attrs={'dep': attr.label(providers=[ApkInfo])})");
ConfiguredTarget postprocess = getConfiguredTarget("//java/com/google/android:postprocess");
assertThat(postprocess).isNotNull();
assertThat(
prettyArtifactNames(postprocess.getProvider(FilesToRunProvider.class).getFilesToRun()))
.containsExactly("java/com/google/android/b1.apk");
}
@Test
public void testInstrumentationInfoAccessibleFromSkylark() throws Exception {
scratch.file(
"java/com/google/android/instr/BUILD",
"load(':instr.bzl', 'instr')",
"android_binary(name = 'b1',",
" srcs = ['b1.java'],",
" instruments = ':b2',",
" manifest = 'AndroidManifest.xml')",
"android_binary(name = 'b2',",
" srcs = ['b2.java'],",
" manifest = 'AndroidManifest.xml')",
"instr(name = 'instr', dep = ':b1')");
scratch.file(
"java/com/google/android/instr/instr.bzl",
"def _impl(ctx):",
" target = ctx.attr.dep[AndroidInstrumentationInfo].target.signed_apk",
" return [DefaultInfo(files=depset([target]))]",
"instr = rule(implementation=_impl,",
" attrs={'dep': attr.label(providers=[AndroidInstrumentationInfo])})");
ConfiguredTarget instr = getConfiguredTarget("//java/com/google/android/instr");
assertThat(instr).isNotNull();
assertThat(prettyArtifactNames(instr.getProvider(FilesToRunProvider.class).getFilesToRun()))
.containsExactly("java/com/google/android/instr/b2.apk");
}
@Test
public void testInstrumentationInfoCreatableFromSkylark() throws Exception {
scratch.file(
"java/com/google/android/instr/BUILD",
"load(':instr.bzl', 'instr')",
"android_binary(name = 'b1',",
" srcs = ['b1.java'],",
" instruments = ':b2',",
" manifest = 'AndroidManifest.xml')",
"android_binary(name = 'b2',",
" srcs = ['b2.java'],",
" manifest = 'AndroidManifest.xml')",
"instr(name = 'instr', dep = ':b1')");
scratch.file(
"java/com/google/android/instr/instr.bzl",
"def _impl(ctx):",
" target = ctx.attr.dep[AndroidInstrumentationInfo].target",
" return [AndroidInstrumentationInfo(target=target)]",
"instr = rule(implementation=_impl,",
" attrs={'dep': attr.label(providers=[AndroidInstrumentationInfo])})");
ConfiguredTarget instr = getConfiguredTarget("//java/com/google/android/instr");
assertThat(instr).isNotNull();
assertThat(instr.get(AndroidInstrumentationInfo.PROVIDER).getTarget().getApk().prettyPrint())
.isEqualTo("java/com/google/android/instr/b2.apk");
}
@Test
public void testInstrumentationInfoProviderHasApks() throws Exception {
scratch.file(
"java/com/google/android/instr/BUILD",
"android_binary(name = 'b1',",
" srcs = ['b1.java'],",
" instruments = ':b2',",
" manifest = 'AndroidManifest.xml')",
"android_binary(name = 'b2',",
" srcs = ['b2.java'],",
" manifest = 'AndroidManifest.xml')");
ConfiguredTarget b1 = getConfiguredTarget("//java/com/google/android/instr:b1");
AndroidInstrumentationInfo provider = b1.get(AndroidInstrumentationInfo.PROVIDER);
assertThat(provider.getTarget()).isNotNull();
assertThat(provider.getTarget().getApk().prettyPrint())
.isEqualTo("java/com/google/android/instr/b2.apk");
}
@Test
public void testNoInstrumentationInfoProviderIfNotInstrumenting() throws Exception {
scratch.file(
"java/com/google/android/instr/BUILD",
"android_binary(name = 'b1',",
" srcs = ['b1.java'],",
" manifest = 'AndroidManifest.xml')");
ConfiguredTarget b1 = getConfiguredTarget("//java/com/google/android/instr:b1");
AndroidInstrumentationInfo provider = b1.get(AndroidInstrumentationInfo.PROVIDER);
assertThat(provider).isNull();
}
@Test
public void testFilterActionWithInstrumentedBinary() throws Exception {
scratch.file(
"java/com/google/android/instr/BUILD",
"android_binary(name = 'b1',",
" srcs = ['b1.java'],",
" instruments = ':b2',",
" manifest = 'AndroidManifest.xml')",
"android_binary(name = 'b2',",
" srcs = ['b2.java'],",
" manifest = 'AndroidManifest.xml')");
ConfiguredTarget b1 = getConfiguredTarget("//java/com/google/android/instr:b1");
SpawnAction action =
(SpawnAction)
actionsTestUtil().getActionForArtifactEndingWith(getFilesToBuild(b1), "_filtered.jar");
assertThat(action.getArguments())
.containsAtLeast(
"--inputZip",
getFirstArtifactEndingWith(action.getInputs(), "b1_deploy.jar").getExecPathString(),
"--filterZips",
getFirstArtifactEndingWith(action.getInputs(), "b2_deploy.jar").getExecPathString(),
"--outputZip",
getFirstArtifactEndingWith(action.getOutputs(), "b1_filtered.jar").getExecPathString(),
"--filterTypes",
".class",
"--checkHashMismatch",
"IGNORE",
"--explicitFilters",
"/BR\\.class$,/databinding/[^/]+Binding\\.class$,R\\.class,R\\$.*\\.class",
"--outputMode",
"DONT_CARE");
}
/**
* 'proguard_specs' attribute gets read by an implicit outputs function: the
* current heuristic is that if this attribute is configurable, we assume its
* contents are non-empty and thus create the mybinary_proguard.jar output.
* Test that here.
*/
@Test
public void testConfigurableProguardSpecs() throws Exception {
scratch.file("conditions/BUILD",
"config_setting(",
" name = 'a',",
" values = {'test_arg': 'a'})",
"config_setting(",
" name = 'b',",
" values = {'test_arg': 'b'})");
scratchConfiguredTarget("java/foo", "abin",
"android_binary(",
" name = 'abin',",
" srcs = ['a.java'],",
" proguard_specs = select({",
" '//conditions:a': [':file1.pro'],",
" '//conditions:b': [],",
" '//conditions:default': [':file3.pro'],",
" }) + [",
// Add a long list here as a regression test for b/68238721
" 'file4.pro',",
" 'file5.pro',",
" 'file6.pro',",
" 'file7.pro',",
" 'file8.pro',",
" ],",
" manifest = 'AndroidManifest.xml')");
checkProguardUse(
"//java/foo:abin",
"abin_proguard.jar",
/*expectMapping=*/ false,
/*passes=*/ null,
getAndroidJarPath());
}
@Test
public void testSkipParsingActionFlagGetsPropagated() throws Exception {
mockAndroidSdkWithAapt2();
scratch.file(
"java/b/BUILD",
"android_library(",
" name = 'b',",
" srcs = ['B.java'],",
" manifest = 'AndroidManifest.xml',",
" resource_files = [ 'res/values/values.xml' ], ",
")");
scratch.file(
"java/a/BUILD",
"android_binary(",
" name = 'a',",
" srcs = ['A.java'],",
" manifest = 'AndroidManifest.xml',",
" deps = [ '//java/b:b' ],",
" resource_files = [ 'res/values/values.xml' ], ",
" aapt_version = 'aapt2'",
")");
useConfiguration("--android_sdk=//sdk:sdk", "--experimental_skip_parsing_action");
ConfiguredTarget a = getConfiguredTarget("//java/a:a");
ConfiguredTarget b = getDirectPrerequisite(a, "//java/b:b");
List<String> resourceProcessingArgs =
getGeneratingSpawnActionArgs(getValidatedResources(a).getApk());
assertThat(resourceProcessingArgs).contains("AAPT2_PACKAGE");
String directData =
resourceProcessingArgs.get(
resourceProcessingArgs.indexOf("--directData") + 1);
assertThat(directData).contains("symbols.zip");
assertThat(directData).doesNotContain("merged.bin");
assertThat(resourceProcessingArgs).contains("--useCompiledResourcesForMerge");
List<String> resourceMergingArgs =
getGeneratingSpawnActionArgs(getValidatedResources(b).getJavaClassJar());
assertThat(resourceMergingArgs).contains("MERGE_COMPILED");
}
@Test
public void alwaysSkipParsingActionWithAapt2() throws Exception {
mockAndroidSdkWithAapt2();
scratch.file(
"java/b/BUILD",
"android_library(",
" name = 'b',",
" srcs = ['B.java'],",
" manifest = 'AndroidManifest.xml',",
" resource_files = [ 'res/values/values.xml' ], ",
")");
scratch.file(
"java/a/BUILD",
"android_binary(",
" name = 'a',",
" srcs = ['A.java'],",
" manifest = 'AndroidManifest.xml',",
" deps = [ '//java/b:b' ],",
" resource_files = [ 'res/values/values.xml' ], ",
" aapt_version = 'aapt2'",
")");
useConfiguration("--android_sdk=//sdk:sdk");
ConfiguredTarget a = getConfiguredTarget("//java/a:a");
ConfiguredTarget b = getDirectPrerequisite(a, "//java/b:b");
List<String> resourceProcessingArgs =
getGeneratingSpawnActionArgs(getValidatedResources(a).getApk());
assertThat(resourceProcessingArgs).contains("AAPT2_PACKAGE");
String directData =
resourceProcessingArgs.get(resourceProcessingArgs.indexOf("--directData") + 1);
assertThat(directData).contains("symbols.zip");
assertThat(directData).doesNotContain("merged.bin");
assertThat(resourceProcessingArgs).contains("--useCompiledResourcesForMerge");
// Libraries will still need to merge the xml until skip parsing is on by default.
List<String> resourceMergingArgs =
getGeneratingSpawnActionArgs(getValidatedResources(b).getJavaClassJar());
assertThat(resourceMergingArgs).contains("MERGE_COMPILED");
}
@Test
public void testAapt1BuildsWithAapt2Sdk() throws Exception {
mockAndroidSdkWithAapt2();
scratch.file(
"java/b/BUILD",
"android_library(",
" name = 'b',",
" srcs = ['B.java'],",
" manifest = 'AndroidManifest.xml',",
" resource_files = [ 'res/values/values.xml' ], ",
")");
scratch.file(
"java/a/BUILD",
"android_binary(",
" name = 'a',",
" srcs = ['A.java'],",
" manifest = 'AndroidManifest.xml',",
" deps = [ '//java/b:b' ],",
" resource_files = [ 'res/values/values.xml' ], ",
" aapt_version = 'aapt'",
")");
useConfiguration("--android_sdk=//sdk:sdk", "--experimental_skip_parsing_action");
ConfiguredTarget a = getConfiguredTarget("//java/a:a");
ConfiguredTarget b = getDirectPrerequisite(a, "//java/b:b");
List<String> resourceProcessingArgs =
getGeneratingSpawnActionArgs(getValidatedResources(a).getRTxt());
assertThat(resourceProcessingArgs).contains("PACKAGE");
String directData =
resourceProcessingArgs.get(
resourceProcessingArgs.indexOf("--directData") + 1);
assertThat(directData).doesNotContain("symbols.zip");
assertThat(directData).contains("merged.bin");
List<String> compiledResourceMergingArgs =
getGeneratingSpawnActionArgs(getValidatedResources(b).getJavaClassJar());
assertThat(compiledResourceMergingArgs).contains("MERGE_COMPILED");
Set<Artifact> artifacts = actionsTestUtil().artifactClosureOf(getFilesToBuild(b));
List<String> parsedResourceMergingArgs =
getGeneratingSpawnActionArgs(getFirstArtifactEndingWith(artifacts, "resource_files.zip"));
assertThat(parsedResourceMergingArgs).contains("MERGE");
}
@Test
public void skylarkJavaInfoToAndroidBinaryAttributes() throws Exception {
scratch.file(
"java/r/android/extension.bzl",
"def _impl(ctx):",
" dep_params = ctx.attr.dep[JavaInfo]",
" return [dep_params]",
"my_rule = rule(",
" _impl,",
" attrs = {",
" 'dep': attr.label(),",
" },",
")");
scratch.file(
"java/r/android/BUILD",
"load(':extension.bzl', 'my_rule')",
"android_library(",
" name = 'al_bottom_for_deps',",
" srcs = ['java/A.java'],",
")",
"my_rule(",
" name = 'mya',",
" dep = ':al_bottom_for_deps',",
")",
"android_binary(",
" name = 'foo_app',",
" srcs = ['java/B.java'],",
" deps = [':mya'],",
" manifest = 'AndroidManifest.xml',",
// TODO(b/75051107): Remove the following line when fixed.
" incremental_dexing = 0,",
")");
// Test that all bottom jars are on the runtime classpath of the app.
ConfiguredTarget target = getConfiguredTarget("//java/r/android:foo_app");
ImmutableList<Artifact> transitiveSrcJars =
OutputGroupInfo.get(target).getOutputGroup(JavaSemantics.SOURCE_JARS_OUTPUT_GROUP).toList();
assertThat(ActionsTestUtil.baseArtifactNames(transitiveSrcJars)).containsExactly(
"libal_bottom_for_deps-src.jar",
"libfoo_app-src.jar");
}
@Test
public void androidManifestMergerOrderAlphabetical_MergeesSortedByExecPath() throws Exception {
// Hack: Avoid the Android split transition by turning off fat_apk_cpu/android_cpu.
// This is necessary because the transition would change the configuration directory, causing
// the manifest paths in the assertion not to match.
// TODO(mstaib): Get the library manifests in the same configuration as the binary gets them.
useConfiguration(
"--fat_apk_cpu=", "--android_cpu=", "--android_manifest_merger_order=alphabetical");
scratch.overwriteFile(
"java/android/BUILD",
"android_library(",
" name = 'core',",
" manifest = 'core/AndroidManifest.xml',",
" exports_manifest = 1,",
" resource_files = ['core/res/values/strings.xml'],",
")",
"android_library(",
" name = 'utility',",
" manifest = 'utility/AndroidManifest.xml',",
" exports_manifest = 1,",
" resource_files = ['utility/res/values/values.xml'],",
" deps = ['//java/common:common'],",
")");
scratch.file(
"java/binary/BUILD",
"android_binary(",
" name = 'application',",
" srcs = ['App.java'],",
" manifest = 'app/AndroidManifest.xml',",
" deps = [':library'],",
")",
"android_library(",
" name = 'library',",
" manifest = 'library/AndroidManifest.xml',",
" exports_manifest = 1,",
" deps = ['//java/common:theme', '//java/android:utility'],",
")");
scratch.file(
"java/common/BUILD",
"android_library(",
" name = 'common',",
" manifest = 'common/AndroidManifest.xml',",
" exports_manifest = 1,",
" resource_files = ['common/res/values/common.xml'],",
" deps = ['//java/android:core'],",
")",
"android_library(",
" name = 'theme',",
" manifest = 'theme/AndroidManifest.xml',",
" exports_manifest = 1,",
" resource_files = ['theme/res/values/values.xml'],",
")");
Artifact androidCoreManifest = getLibraryManifest(getConfiguredTarget("//java/android:core"));
Artifact androidUtilityManifest =
getLibraryManifest(getConfiguredTarget("//java/android:utility"));
Artifact binaryLibraryManifest =
getLibraryManifest(getConfiguredTarget("//java/binary:library"));
Artifact commonManifest = getLibraryManifest(getConfiguredTarget("//java/common:common"));
Artifact commonThemeManifest = getLibraryManifest(getConfiguredTarget("//java/common:theme"));
assertThat(getBinaryMergeeManifests(getConfiguredTarget("//java/binary:application")))
.containsExactlyEntriesIn(
ImmutableMap.of(
androidCoreManifest.getExecPath().toString(), "//java/android:core",
androidUtilityManifest.getExecPath().toString(), "//java/android:utility",
binaryLibraryManifest.getExecPath().toString(), "//java/binary:library",
commonManifest.getExecPath().toString(), "//java/common:common",
commonThemeManifest.getExecPath().toString(), "//java/common:theme"))
.inOrder();
}
@Test
public void androidManifestMergerOrderAlphabeticalByConfiguration_MergeesSortedByPathInBinOrGen()
throws Exception {
// Hack: Avoid the Android split transition by turning off fat_apk_cpu/android_cpu.
// This is necessary because the transition would change the configuration directory, causing
// the manifest paths in the assertion not to match.
// TODO(mstaib): Get the library manifests in the same configuration as the binary gets them.
useConfiguration(
"--fat_apk_cpu=",
"--android_cpu=",
"--android_manifest_merger_order=alphabetical_by_configuration");
scratch.overwriteFile(
"java/android/BUILD",
"android_library(",
" name = 'core',",
" manifest = 'core/AndroidManifest.xml',",
" exports_manifest = 1,",
" resource_files = ['core/res/values/strings.xml'],",
")",
"android_library(",
" name = 'utility',",
" manifest = 'utility/AndroidManifest.xml',",
" exports_manifest = 1,",
" resource_files = ['utility/res/values/values.xml'],",
" deps = ['//java/common:common'],",
" transitive_configs = ['//flags:a', '//flags:b'],",
")");
scratch.file(
"java/binary/BUILD",
"android_binary(",
" name = 'application',",
" srcs = ['App.java'],",
" manifest = 'app/AndroidManifest.xml',",
" deps = [':library'],",
" feature_flags = {",
" '//flags:a': 'on',",
" '//flags:b': 'on',",
" '//flags:c': 'on',",
" },",
" transitive_configs = ['//flags:a', '//flags:b', '//flags:c'],",
")",
"android_library(",
" name = 'library',",
" manifest = 'library/AndroidManifest.xml',",
" exports_manifest = 1,",
" deps = ['//java/common:theme', '//java/android:utility'],",
" transitive_configs = ['//flags:a', '//flags:b', '//flags:c'],",
")");
scratch.file(
"java/common/BUILD",
"android_library(",
" name = 'common',",
" manifest = 'common/AndroidManifest.xml',",
" exports_manifest = 1,",
" resource_files = ['common/res/values/common.xml'],",
" deps = ['//java/android:core'],",
" transitive_configs = ['//flags:a'],",
")",
"android_library(",
" name = 'theme',",
" manifest = 'theme/AndroidManifest.xml',",
" exports_manifest = 1,",
" resource_files = ['theme/res/values/values.xml'],",
" transitive_configs = ['//flags:a', '//flags:b', '//flags:c'],",
")");
scratch.file(
"flags/BUILD",
"config_feature_flag(",
" name = 'a',",
" allowed_values = ['on', 'off'],",
" default_value = 'off',",
")",
"config_feature_flag(",
" name = 'b',",
" allowed_values = ['on', 'off'],",
" default_value = 'off',",
")",
"config_feature_flag(",
" name = 'c',",
" allowed_values = ['on', 'off'],",
" default_value = 'off',",
")");
assertThat(getBinaryMergeeManifests(getConfiguredTarget("//java/binary:application")).values())
.containsExactly(
"//java/android:core",
"//java/android:utility",
"//java/binary:library",
"//java/common:common",
"//java/common:theme")
.inOrder();
}
// DEPENDENCY order is not tested; the incorrect order of dependencies means the test would
// have to enforce incorrect behavior.
// TODO(b/117338320): Add a test when dependency order is fixed.
}