/* * Copyright (C) 2017 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include "Util.h" #include "android-base/stringprintf.h" #include "AppInfo.h" #include "split/TableSplitter.h" #include "test/Builders.h" #include "test/Test.h" #include "util/Files.h" using ::android::ConfigDescription; using testing::Pair; using testing::UnorderedElementsAre; namespace aapt { #ifdef _WIN32 #define CREATE_PATH(path) android::base::StringPrintf(";%s", path) #else #define CREATE_PATH(path) android::base::StringPrintf(":%s", path) #endif #define EXPECT_CONFIG_EQ(constraints, config) \ EXPECT_EQ(constraints.configs.size(), 1); \ EXPECT_EQ(*constraints.configs.begin(), config); \ constraints.configs.clear(); TEST(UtilTest, SplitNamesAreSanitized) { AppInfo app_info{"com.pkg"}; SplitConstraints split_constraints{ {test::ParseConfigOrDie("en-rUS-land"), test::ParseConfigOrDie("b+sr+Latn")}}; const auto doc = GenerateSplitManifest(app_info, split_constraints); const auto &root = doc->root; EXPECT_EQ(root->name, "manifest"); // split names cannot contain hyphens or plus signs. EXPECT_EQ(root->FindAttribute("", "split")->value, "config.b_sr_Latn_en_rUS_land"); // but we should use resource qualifiers verbatim in 'targetConfig'. EXPECT_EQ(root->FindAttribute("", "targetConfig")->value, "b+sr+Latn,en-rUS-land"); } TEST (UtilTest, LongVersionCodeDefined) { auto doc = test::BuildXmlDom(R"( )"); SetLongVersionCode(doc->root.get(), 42); auto version_code = doc->root->FindAttribute(xml::kSchemaAndroid, "versionCode"); ASSERT_NE(version_code, nullptr); EXPECT_EQ(version_code->value, "0x0000002a"); ASSERT_NE(version_code->compiled_value, nullptr); auto compiled_version_code = ValueCast(version_code->compiled_value.get()); ASSERT_NE(compiled_version_code, nullptr); EXPECT_EQ(compiled_version_code->value.data, 42U); auto version_code_major = doc->root->FindAttribute(xml::kSchemaAndroid, "versionCodeMajor"); EXPECT_EQ(version_code_major, nullptr); } TEST (UtilTest, LongVersionCodeUndefined) { auto doc = test::BuildXmlDom(R"( )"); SetLongVersionCode(doc->root.get(), 420000000000); auto version_code = doc->root->FindAttribute(xml::kSchemaAndroid, "versionCode"); ASSERT_NE(version_code, nullptr); EXPECT_EQ(version_code->value, "0xc9f36800"); ASSERT_NE(version_code->compiled_value, nullptr); auto compiled_version_code = ValueCast(version_code->compiled_value.get()); ASSERT_NE(compiled_version_code, nullptr); EXPECT_EQ(compiled_version_code->value.data, 0xc9f36800); auto version_code_major = doc->root->FindAttribute(xml::kSchemaAndroid, "versionCodeMajor"); ASSERT_NE(version_code_major, nullptr); EXPECT_EQ(version_code_major->value, "0x00000061"); ASSERT_NE(version_code_major->compiled_value, nullptr); auto compiled_version_code_major = ValueCast( version_code_major->compiled_value.get()); ASSERT_NE(compiled_version_code_major, nullptr); EXPECT_EQ(compiled_version_code_major->value.data, 0x61); } TEST (UtilTest, ParseSplitParameters) { android::IDiagnostics* diagnostics = test::ContextBuilder().Build().get()->GetDiagnostics(); std::string path; SplitConstraints constraints; ConfigDescription expected_configuration; // ========== Test IMSI ========== // mcc: 'mcc[0-9]{3}' // mnc: 'mnc[0-9]{1,3}' ASSERT_TRUE(ParseSplitParameter(CREATE_PATH("mcc310"), diagnostics, &path, &constraints)); expected_configuration = test::ConfigDescriptionBuilder() .setMcc(0x0136) .Build(); EXPECT_CONFIG_EQ(constraints, expected_configuration); ASSERT_TRUE(ParseSplitParameter(CREATE_PATH("mcc310-mnc004"), diagnostics, &path, &constraints)); expected_configuration = test::ConfigDescriptionBuilder() .setMcc(0x0136) .setMnc(0x0004) .Build(); EXPECT_CONFIG_EQ(constraints, expected_configuration); ASSERT_TRUE(ParseSplitParameter(CREATE_PATH("mcc310-mnc000"), diagnostics, &path, &constraints)); expected_configuration = test::ConfigDescriptionBuilder() .setMcc(0x0136) .setMnc(0xFFFF) .Build(); EXPECT_CONFIG_EQ(constraints, expected_configuration); // ========== Test LOCALE ========== // locale: '[a-z]{2,3}(-r[a-z]{2})?' // locale: 'b+[a-z]{2,3}(+[a-z[0-9]]{2})?' ASSERT_TRUE(ParseSplitParameter(CREATE_PATH("es"), diagnostics, &path, &constraints)); expected_configuration = test::ConfigDescriptionBuilder() .setLanguage(0x6573) .Build(); EXPECT_CONFIG_EQ(constraints, expected_configuration); ASSERT_TRUE(ParseSplitParameter(CREATE_PATH("fr-rCA"), diagnostics, &path, &constraints)); expected_configuration = test::ConfigDescriptionBuilder() .setLanguage(0x6672) .setCountry(0x4341) .Build(); EXPECT_CONFIG_EQ(constraints, expected_configuration); ASSERT_TRUE(ParseSplitParameter(CREATE_PATH("b+es+419"), diagnostics, &path, &constraints)); expected_configuration = test::ConfigDescriptionBuilder() .setLanguage(0x6573) .setCountry(0xA424) .Build(); EXPECT_CONFIG_EQ(constraints, expected_configuration); // ========== Test SCREEN_TYPE ========== // orientation: '(port|land|square)' // touchscreen: '(notouch|stylus|finger)' // density" '(anydpi|nodpi|ldpi|mdpi|tvdpi|hdpi|xhdpi|xxhdpi|xxxhdpi|[0-9]*dpi)' ASSERT_TRUE(ParseSplitParameter(CREATE_PATH("square"), diagnostics, &path, &constraints)); expected_configuration = test::ConfigDescriptionBuilder() .setOrientation(0x03) .Build(); EXPECT_CONFIG_EQ(constraints, expected_configuration); ASSERT_TRUE(ParseSplitParameter(CREATE_PATH("stylus"), diagnostics, &path, &constraints)); expected_configuration = test::ConfigDescriptionBuilder() .setTouchscreen(0x02) .Build(); EXPECT_CONFIG_EQ(constraints, expected_configuration); ASSERT_TRUE(ParseSplitParameter(CREATE_PATH("xxxhdpi"), diagnostics, &path, &constraints)); expected_configuration = test::ConfigDescriptionBuilder() .setDensity(0x0280) .setSdkVersion(0x0004) // version [any density requires donut] .Build(); EXPECT_CONFIG_EQ(constraints, expected_configuration); ASSERT_TRUE(ParseSplitParameter(CREATE_PATH("land-xhdpi-finger"), diagnostics, &path, &constraints)); expected_configuration = test::ConfigDescriptionBuilder() .setOrientation(0x02) .setTouchscreen(0x03) .setDensity(0x0140) .setSdkVersion(0x0004) // version [any density requires donut] .Build(); EXPECT_CONFIG_EQ(constraints, expected_configuration); // ========== Test INPUT ========== // keyboard: '(nokeys|qwerty|12key)' // navigation: '(nonav|dpad|trackball|wheel)' // inputFlags: '(keysexposed|keyshidden|keyssoft)' // inputFlags: '(navexposed|navhidden)' ASSERT_TRUE(ParseSplitParameter(CREATE_PATH("qwerty"), diagnostics, &path, &constraints)); expected_configuration = test::ConfigDescriptionBuilder() .setKeyboard(0x02) .Build(); EXPECT_CONFIG_EQ(constraints, expected_configuration); ASSERT_TRUE(ParseSplitParameter(CREATE_PATH("dpad"), diagnostics, &path, &constraints)); expected_configuration = test::ConfigDescriptionBuilder() .setNavigation(0x02) .Build(); EXPECT_CONFIG_EQ(constraints, expected_configuration); ASSERT_TRUE(ParseSplitParameter(CREATE_PATH("keyssoft-navhidden"), diagnostics, &path, &constraints)); expected_configuration = test::ConfigDescriptionBuilder() .setInputFlags(0x0B) .Build(); EXPECT_CONFIG_EQ(constraints, expected_configuration); ASSERT_TRUE(ParseSplitParameter(CREATE_PATH("keyshidden-nokeys-navexposed-trackball"), diagnostics, &path, &constraints)); expected_configuration = test::ConfigDescriptionBuilder() .setKeyboard(0x01) .setNavigation(0x03) .setInputFlags(0x06) .Build(); EXPECT_CONFIG_EQ(constraints, expected_configuration); // ========== Test SCREEN_SIZE ========== // screenWidth/screenHeight: '[0-9]+x[0-9]+' ASSERT_TRUE(ParseSplitParameter(CREATE_PATH("1920x1080"), diagnostics, &path, &constraints)); expected_configuration = test::ConfigDescriptionBuilder() .setScreenWidth(0x0780) .setScreenHeight(0x0438) .Build(); EXPECT_CONFIG_EQ(constraints, expected_configuration); // ========== Test VERSION ========== // version 'v[0-9]+' // ========== Test SCREEN_CONFIG ========== // screenLayout [direction]: '(ldltr|ldrtl)' // screenLayout [size]: '(small|normal|large|xlarge)' // screenLayout [long]: '(long|notlong)' // uiMode [type]: '(desk|car|television|appliance|watch|vrheadset)' // uiMode [night]: '(night|notnight)' // smallestScreenWidthDp: 'sw[0-9]dp' ASSERT_TRUE(ParseSplitParameter(CREATE_PATH("ldrtl"), diagnostics, &path, &constraints)); expected_configuration = test::ConfigDescriptionBuilder() .setScreenLayout(0x80) .Build(); EXPECT_CONFIG_EQ(constraints, expected_configuration); ASSERT_TRUE(ParseSplitParameter(CREATE_PATH("small"), diagnostics, &path, &constraints)); expected_configuration = test::ConfigDescriptionBuilder() .setScreenLayout(0x01) .setSdkVersion(0x0004) // screenLayout (size) requires donut .Build(); EXPECT_CONFIG_EQ(constraints, expected_configuration); ASSERT_TRUE(ParseSplitParameter(CREATE_PATH("notlong"), diagnostics, &path, &constraints)); expected_configuration = test::ConfigDescriptionBuilder() .setScreenLayout(0x10) .setSdkVersion(0x0004) // screenLayout (long) requires donut .Build(); EXPECT_CONFIG_EQ(constraints, expected_configuration); ASSERT_TRUE(ParseSplitParameter(CREATE_PATH("ldltr-normal-long"), diagnostics, &path, &constraints)); expected_configuration = test::ConfigDescriptionBuilder() .setScreenLayout(0x62) .setSdkVersion(0x0004) // screenLayout (size|long) requires donut .Build(); EXPECT_CONFIG_EQ(constraints, expected_configuration); ASSERT_TRUE(ParseSplitParameter(CREATE_PATH("car"), diagnostics, &path, &constraints)); expected_configuration = test::ConfigDescriptionBuilder() .setUiMode(0x03) .setSdkVersion(0x0008) // uiMode requires froyo .Build(); EXPECT_CONFIG_EQ(constraints, expected_configuration); ASSERT_TRUE(ParseSplitParameter(CREATE_PATH("vrheadset"), diagnostics, &path, &constraints)); expected_configuration = test::ConfigDescriptionBuilder() .setUiMode(0x07) .setSdkVersion(0x001A) // uiMode 'vrheadset' requires oreo .Build(); EXPECT_CONFIG_EQ(constraints, expected_configuration); ASSERT_TRUE(ParseSplitParameter(CREATE_PATH("television-night"), diagnostics, &path, &constraints)); expected_configuration = test::ConfigDescriptionBuilder() .setUiMode(0x24) .setSdkVersion(0x0008) // uiMode requires froyo .Build(); EXPECT_CONFIG_EQ(constraints, expected_configuration); ASSERT_TRUE(ParseSplitParameter(CREATE_PATH("sw1920dp"), diagnostics, &path, &constraints)); expected_configuration = test::ConfigDescriptionBuilder() .setSmallestScreenWidthDp(0x0780) .setSdkVersion(0x000D) // smallestScreenWidthDp requires honeycomb mr2 .Build(); EXPECT_CONFIG_EQ(constraints, expected_configuration); // ========== Test SCREEN_SIZE_DP ========== // screenWidthDp: 'w[0-9]dp' // screenHeightDp: 'h[0-9]dp' ASSERT_TRUE(ParseSplitParameter(CREATE_PATH("w1920dp"), diagnostics, &path, &constraints)); expected_configuration = test::ConfigDescriptionBuilder() .setScreenWidthDp(0x0780) .setSdkVersion(0x000D) // screenWidthDp requires honeycomb mr2 .Build(); EXPECT_CONFIG_EQ(constraints, expected_configuration); ASSERT_TRUE(ParseSplitParameter(CREATE_PATH("h1080dp"), diagnostics, &path, &constraints)); expected_configuration = test::ConfigDescriptionBuilder() .setScreenHeightDp(0x0438) .setSdkVersion(0x000D) // screenHeightDp requires honeycomb mr2 .Build(); EXPECT_CONFIG_EQ(constraints, expected_configuration); // ========== Test SCREEN_CONFIG_2 ========== // screenLayout2: '(round|notround)' // colorMode: '(widecg|nowidecg)' // colorMode: '(highhdr|lowdr)' ASSERT_TRUE(ParseSplitParameter(CREATE_PATH("round"), diagnostics, &path, &constraints)); expected_configuration = test::ConfigDescriptionBuilder() .setScreenLayout2(0x02) .setSdkVersion(0x0017) // screenLayout2 (round) requires marshmallow .Build(); EXPECT_CONFIG_EQ(constraints, expected_configuration); ASSERT_TRUE(ParseSplitParameter(CREATE_PATH("widecg-highdr"), diagnostics, &path, &constraints)); expected_configuration = test::ConfigDescriptionBuilder() .setColorMode(0x0A) .setSdkVersion(0x001A) // colorMode (hdr|colour gamut) requires oreo .Build(); EXPECT_CONFIG_EQ(constraints, expected_configuration); } TEST(UtilTest, ParseFeatureFlagsParameter_Empty) { auto diagnostics = test::ContextBuilder().Build()->GetDiagnostics(); FeatureFlagValues feature_flag_values; ASSERT_TRUE(ParseFeatureFlagsParameter("", diagnostics, &feature_flag_values)); EXPECT_TRUE(feature_flag_values.empty()); } TEST(UtilTest, ParseFeatureFlagsParameter_TooManyParts) { auto diagnostics = test::ContextBuilder().Build()->GetDiagnostics(); FeatureFlagValues feature_flag_values; ASSERT_FALSE(ParseFeatureFlagsParameter("foo=bar=baz", diagnostics, &feature_flag_values)); } TEST(UtilTest, ParseFeatureFlagsParameter_NoNameGiven) { auto diagnostics = test::ContextBuilder().Build()->GetDiagnostics(); FeatureFlagValues feature_flag_values; ASSERT_FALSE(ParseFeatureFlagsParameter("foo=true,=false", diagnostics, &feature_flag_values)); } TEST(UtilTest, ParseFeatureFlagsParameter_InvalidValue) { auto diagnostics = test::ContextBuilder().Build()->GetDiagnostics(); FeatureFlagValues feature_flag_values; ASSERT_FALSE(ParseFeatureFlagsParameter("foo=true,bar=42", diagnostics, &feature_flag_values)); } TEST(UtilTest, ParseFeatureFlagsParameter_DuplicateFlag) { auto diagnostics = test::ContextBuilder().Build()->GetDiagnostics(); FeatureFlagValues feature_flag_values; ASSERT_TRUE( ParseFeatureFlagsParameter("foo=true,bar=true,foo=false", diagnostics, &feature_flag_values)); EXPECT_THAT(feature_flag_values, UnorderedElementsAre(Pair("foo", std::optional(false)), Pair("bar", std::optional(true)))); } TEST(UtilTest, ParseFeatureFlagsParameter_Valid) { auto diagnostics = test::ContextBuilder().Build()->GetDiagnostics(); FeatureFlagValues feature_flag_values; ASSERT_TRUE(ParseFeatureFlagsParameter("foo= true, bar =FALSE,baz=, quux", diagnostics, &feature_flag_values)); EXPECT_THAT(feature_flag_values, UnorderedElementsAre(Pair("foo", std::optional(true)), Pair("bar", std::optional(false)), Pair("baz", std::nullopt), Pair("quux", std::nullopt))); } TEST (UtilTest, AdjustSplitConstraintsForMinSdk) { std::unique_ptr context = test::ContextBuilder().Build(); android::IDiagnostics* diagnostics = context.get()->GetDiagnostics(); std::vector test_constraints; std::string path; test_constraints.push_back({}); ASSERT_TRUE(ParseSplitParameter(CREATE_PATH("v7"), diagnostics, &path, &test_constraints.back())); test_constraints.push_back({}); ASSERT_TRUE(ParseSplitParameter(CREATE_PATH("xhdpi"), diagnostics, &path, &test_constraints.back())); EXPECT_EQ(test_constraints.size(), 2); EXPECT_EQ(test_constraints[0].name, "v7"); EXPECT_EQ(test_constraints[0].configs.size(), 1); EXPECT_NE(*test_constraints[0].configs.begin(), ConfigDescription::DefaultConfig()); EXPECT_EQ(test_constraints[1].name, "xhdpi"); EXPECT_EQ(test_constraints[1].configs.size(), 1); EXPECT_NE(*test_constraints[0].configs.begin(), ConfigDescription::DefaultConfig()); auto adjusted_contraints = AdjustSplitConstraintsForMinSdk(26, test_constraints); EXPECT_EQ(adjusted_contraints.size(), 2); EXPECT_EQ(adjusted_contraints[0].name, "v7"); EXPECT_EQ(adjusted_contraints[0].configs.size(), 0); EXPECT_EQ(adjusted_contraints[1].name, "xhdpi"); EXPECT_EQ(adjusted_contraints[1].configs.size(), 1); EXPECT_NE(*adjusted_contraints[1].configs.begin(), ConfigDescription::DefaultConfig()); } TEST (UtilTest, RegularExperssionsSimple) { std::string valid(".bc$"); std::regex expression = GetRegularExpression(valid); EXPECT_TRUE(std::regex_search("file.abc", expression)); EXPECT_TRUE(std::regex_search("file.123bc", expression)); EXPECT_FALSE(std::regex_search("abc.zip", expression)); } TEST (UtilTest, RegularExpressionComplex) { std::string valid("\\.(d|D)(e|E)(x|X)$"); std::regex expression = GetRegularExpression(valid); EXPECT_TRUE(std::regex_search("file.dex", expression)); EXPECT_TRUE(std::regex_search("file.DEX", expression)); EXPECT_TRUE(std::regex_search("file.dEx", expression)); EXPECT_FALSE(std::regex_search("file.dexx", expression)); EXPECT_FALSE(std::regex_search("dex.file", expression)); EXPECT_FALSE(std::regex_search("file.adex", expression)); } TEST (UtilTest, RegularExpressionNonEnglish) { std::string valid("\\.(k|K)(o|O)(ń|Ń)(c|C)(ó|Ó)(w|W)(k|K)(a|A)$"); std::regex expression = GetRegularExpression(valid); EXPECT_TRUE(std::regex_search("file.końcówka", expression)); EXPECT_TRUE(std::regex_search("file.KOŃCÓWKA", expression)); EXPECT_TRUE(std::regex_search("file.kOńcÓwkA", expression)); EXPECT_FALSE(std::regex_search("file.koncowka", expression)); } TEST(UtilTest, ParseConfigWithDirectives) { const std::string& content = R"( bool/remove_me#remove bool/keep_name#no_collapse layout/keep_path#no_path_shorten string/foo#no_obfuscate dimen/bar#no_obfuscate layout/keep_name_and_path#no_collapse,no_path_shorten )"; aapt::test::Context context; std::unordered_set resource_exclusion; std::set name_collapse_exemptions; std::set path_shorten_exemptions; EXPECT_TRUE(ParseResourceConfig(content, &context, resource_exclusion, name_collapse_exemptions, path_shorten_exemptions)); EXPECT_THAT(name_collapse_exemptions, UnorderedElementsAre(ResourceName({}, ResourceType::kString, "foo"), ResourceName({}, ResourceType::kDimen, "bar"), ResourceName({}, ResourceType::kBool, "keep_name"), ResourceName({}, ResourceType::kLayout, "keep_name_and_path"))); EXPECT_THAT(path_shorten_exemptions, UnorderedElementsAre(ResourceName({}, ResourceType::kLayout, "keep_path"), ResourceName({}, ResourceType::kLayout, "keep_name_and_path"))); EXPECT_THAT(resource_exclusion, UnorderedElementsAre(ResourceName({}, ResourceType::kBool, "remove_me"))); } TEST(UtilTest, ParseConfigResourceWithPackage) { const std::string& content = R"( package:bool/remove_me#remove )"; aapt::test::Context context; std::unordered_set resource_exclusion; std::set name_collapse_exemptions; std::set path_shorten_exemptions; EXPECT_FALSE(ParseResourceConfig(content, &context, resource_exclusion, name_collapse_exemptions, path_shorten_exemptions)); } TEST(UtilTest, ParseConfigInvalidName) { const std::string& content = R"( package:bool/1231#remove )"; aapt::test::Context context; std::unordered_set resource_exclusion; std::set name_collapse_exemptions; std::set path_shorten_exemptions; EXPECT_FALSE(ParseResourceConfig(content, &context, resource_exclusion, name_collapse_exemptions, path_shorten_exemptions)); } TEST(UtilTest, ParseConfigNoHash) { const std::string& content = R"( package:bool/my_bool )"; aapt::test::Context context; std::unordered_set resource_exclusion; std::set name_collapse_exemptions; std::set path_shorten_exemptions; EXPECT_FALSE(ParseResourceConfig(content, &context, resource_exclusion, name_collapse_exemptions, path_shorten_exemptions)); } } // namespace aapt