/* * Copyright (C) 2015 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 <test/Context.h> #include "link/Linkers.h" #include "test/Test.h" namespace aapt { class XmlReferenceLinkerTest : public ::testing::Test { public: void SetUp() override { mContext = test::ContextBuilder() .setCompilationPackage(u"com.app.test") .setNameManglerPolicy( NameManglerPolicy{ u"com.app.test", { u"com.android.support" } }) .addSymbolSource(test::StaticSymbolSourceBuilder() .addPublicSymbol(u"@android:attr/layout_width", ResourceId(0x01010000), test::AttributeBuilder() .setTypeMask(android::ResTable_map::TYPE_ENUM | android::ResTable_map::TYPE_DIMENSION) .addItem(u"match_parent", 0xffffffff) .build()) .addPublicSymbol(u"@android:attr/background", ResourceId(0x01010001), test::AttributeBuilder() .setTypeMask(android::ResTable_map::TYPE_COLOR).build()) .addPublicSymbol(u"@android:attr/attr", ResourceId(0x01010002), test::AttributeBuilder().build()) .addPublicSymbol(u"@android:attr/text", ResourceId(0x01010003), test::AttributeBuilder() .setTypeMask(android::ResTable_map::TYPE_STRING) .build()) // Add one real symbol that was introduces in v21 .addPublicSymbol(u"@android:attr/colorAccent", ResourceId(0x01010435), test::AttributeBuilder().build()) // Private symbol. .addSymbol(u"@android:color/hidden", ResourceId(0x01020001)) .addPublicSymbol(u"@android:id/id", ResourceId(0x01030000)) .addSymbol(u"@com.app.test:id/id", ResourceId(0x7f030000)) .addSymbol(u"@com.app.test:color/green", ResourceId(0x7f020000)) .addSymbol(u"@com.app.test:color/red", ResourceId(0x7f020001)) .addSymbol(u"@com.app.test:attr/colorAccent", ResourceId(0x7f010000), test::AttributeBuilder() .setTypeMask(android::ResTable_map::TYPE_COLOR).build()) .addPublicSymbol(u"@com.app.test:attr/com.android.support$colorAccent", ResourceId(0x7f010001), test::AttributeBuilder() .setTypeMask(android::ResTable_map::TYPE_COLOR).build()) .addPublicSymbol(u"@com.app.test:attr/attr", ResourceId(0x7f010002), test::AttributeBuilder().build()) .build()) .build(); } protected: std::unique_ptr<IAaptContext> mContext; }; TEST_F(XmlReferenceLinkerTest, LinkBasicAttributes) { std::unique_ptr<xml::XmlResource> doc = test::buildXmlDomForPackageName(mContext.get(), R"EOF( <View xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:background="@color/green" android:text="hello" class="hello" />)EOF"); XmlReferenceLinker linker; ASSERT_TRUE(linker.consume(mContext.get(), doc.get())); xml::Element* viewEl = xml::findRootElement(doc.get()); ASSERT_NE(viewEl, nullptr); xml::Attribute* xmlAttr = viewEl->findAttribute(u"http://schemas.android.com/apk/res/android", u"layout_width"); ASSERT_NE(xmlAttr, nullptr); AAPT_ASSERT_TRUE(xmlAttr->compiledAttribute); AAPT_ASSERT_TRUE(xmlAttr->compiledAttribute.value().id); EXPECT_EQ(xmlAttr->compiledAttribute.value().id.value(), ResourceId(0x01010000)); ASSERT_NE(xmlAttr->compiledValue, nullptr); ASSERT_NE(valueCast<BinaryPrimitive>(xmlAttr->compiledValue.get()), nullptr); xmlAttr = viewEl->findAttribute(u"http://schemas.android.com/apk/res/android", u"background"); ASSERT_NE(xmlAttr, nullptr); AAPT_ASSERT_TRUE(xmlAttr->compiledAttribute); AAPT_ASSERT_TRUE(xmlAttr->compiledAttribute.value().id); EXPECT_EQ(xmlAttr->compiledAttribute.value().id.value(), ResourceId(0x01010001)); ASSERT_NE(xmlAttr->compiledValue, nullptr); Reference* ref = valueCast<Reference>(xmlAttr->compiledValue.get()); ASSERT_NE(ref, nullptr); AAPT_ASSERT_TRUE(ref->name); EXPECT_EQ(ref->name.value(), test::parseNameOrDie(u"@color/green")); // Make sure the name // didn't change. AAPT_ASSERT_TRUE(ref->id); EXPECT_EQ(ref->id.value(), ResourceId(0x7f020000)); xmlAttr = viewEl->findAttribute(u"http://schemas.android.com/apk/res/android", u"text"); ASSERT_NE(xmlAttr, nullptr); AAPT_ASSERT_TRUE(xmlAttr->compiledAttribute); ASSERT_FALSE(xmlAttr->compiledValue); // Strings don't get compiled for memory sake. xmlAttr = viewEl->findAttribute(u"", u"class"); ASSERT_NE(xmlAttr, nullptr); AAPT_ASSERT_FALSE(xmlAttr->compiledAttribute); ASSERT_EQ(xmlAttr->compiledValue, nullptr); } TEST_F(XmlReferenceLinkerTest, PrivateSymbolsAreNotLinked) { std::unique_ptr<xml::XmlResource> doc = test::buildXmlDomForPackageName(mContext.get(), R"EOF( <View xmlns:android="http://schemas.android.com/apk/res/android" android:colorAccent="@android:color/hidden" />)EOF"); XmlReferenceLinker linker; ASSERT_FALSE(linker.consume(mContext.get(), doc.get())); } TEST_F(XmlReferenceLinkerTest, PrivateSymbolsAreLinkedWhenReferenceHasStarPrefix) { std::unique_ptr<xml::XmlResource> doc = test::buildXmlDomForPackageName(mContext.get(), R"EOF( <View xmlns:android="http://schemas.android.com/apk/res/android" android:colorAccent="@*android:color/hidden" />)EOF"); XmlReferenceLinker linker; ASSERT_TRUE(linker.consume(mContext.get(), doc.get())); } TEST_F(XmlReferenceLinkerTest, SdkLevelsAreRecorded) { std::unique_ptr<xml::XmlResource> doc = test::buildXmlDomForPackageName(mContext.get(), R"EOF( <View xmlns:android="http://schemas.android.com/apk/res/android" android:colorAccent="#ffffff" />)EOF"); XmlReferenceLinker linker; ASSERT_TRUE(linker.consume(mContext.get(), doc.get())); EXPECT_TRUE(linker.getSdkLevels().count(21) == 1); } TEST_F(XmlReferenceLinkerTest, LinkMangledAttributes) { std::unique_ptr<xml::XmlResource> doc = test::buildXmlDomForPackageName(mContext.get(), R"EOF( <View xmlns:support="http://schemas.android.com/apk/res/com.android.support" support:colorAccent="#ff0000" />)EOF"); XmlReferenceLinker linker; ASSERT_TRUE(linker.consume(mContext.get(), doc.get())); xml::Element* viewEl = xml::findRootElement(doc.get()); ASSERT_NE(viewEl, nullptr); xml::Attribute* xmlAttr = viewEl->findAttribute( u"http://schemas.android.com/apk/res/com.android.support", u"colorAccent"); ASSERT_NE(xmlAttr, nullptr); AAPT_ASSERT_TRUE(xmlAttr->compiledAttribute); AAPT_ASSERT_TRUE(xmlAttr->compiledAttribute.value().id); EXPECT_EQ(xmlAttr->compiledAttribute.value().id.value(), ResourceId(0x7f010001)); ASSERT_NE(valueCast<BinaryPrimitive>(xmlAttr->compiledValue.get()), nullptr); } TEST_F(XmlReferenceLinkerTest, LinkAutoResReference) { std::unique_ptr<xml::XmlResource> doc = test::buildXmlDomForPackageName(mContext.get(), R"EOF( <View xmlns:app="http://schemas.android.com/apk/res-auto" app:colorAccent="@app:color/red" />)EOF"); XmlReferenceLinker linker; ASSERT_TRUE(linker.consume(mContext.get(), doc.get())); xml::Element* viewEl = xml::findRootElement(doc.get()); ASSERT_NE(viewEl, nullptr); xml::Attribute* xmlAttr = viewEl->findAttribute(u"http://schemas.android.com/apk/res-auto", u"colorAccent"); ASSERT_NE(xmlAttr, nullptr); AAPT_ASSERT_TRUE(xmlAttr->compiledAttribute); AAPT_ASSERT_TRUE(xmlAttr->compiledAttribute.value().id); EXPECT_EQ(xmlAttr->compiledAttribute.value().id.value(), ResourceId(0x7f010000)); Reference* ref = valueCast<Reference>(xmlAttr->compiledValue.get()); ASSERT_NE(ref, nullptr); AAPT_ASSERT_TRUE(ref->name); AAPT_ASSERT_TRUE(ref->id); EXPECT_EQ(ref->id.value(), ResourceId(0x7f020001)); } TEST_F(XmlReferenceLinkerTest, LinkViewWithShadowedPackageAlias) { std::unique_ptr<xml::XmlResource> doc = test::buildXmlDomForPackageName(mContext.get(), R"EOF( <View xmlns:app="http://schemas.android.com/apk/res/android" app:attr="@app:id/id"> <View xmlns:app="http://schemas.android.com/apk/res/com.app.test" app:attr="@app:id/id"/> </View>)EOF"); XmlReferenceLinker linker; ASSERT_TRUE(linker.consume(mContext.get(), doc.get())); xml::Element* viewEl = xml::findRootElement(doc.get()); ASSERT_NE(viewEl, nullptr); // All attributes and references in this element should be referring to "android" (0x01). xml::Attribute* xmlAttr = viewEl->findAttribute(u"http://schemas.android.com/apk/res/android", u"attr"); ASSERT_NE(xmlAttr, nullptr); AAPT_ASSERT_TRUE(xmlAttr->compiledAttribute); AAPT_ASSERT_TRUE(xmlAttr->compiledAttribute.value().id); EXPECT_EQ(xmlAttr->compiledAttribute.value().id.value(), ResourceId(0x01010002)); Reference* ref = valueCast<Reference>(xmlAttr->compiledValue.get()); ASSERT_NE(ref, nullptr); AAPT_ASSERT_TRUE(ref->id); EXPECT_EQ(ref->id.value(), ResourceId(0x01030000)); ASSERT_FALSE(viewEl->getChildElements().empty()); viewEl = viewEl->getChildElements().front(); ASSERT_NE(viewEl, nullptr); // All attributes and references in this element should be referring to "com.app.test" (0x7f). xmlAttr = viewEl->findAttribute(u"http://schemas.android.com/apk/res/com.app.test", u"attr"); ASSERT_NE(xmlAttr, nullptr); AAPT_ASSERT_TRUE(xmlAttr->compiledAttribute); AAPT_ASSERT_TRUE(xmlAttr->compiledAttribute.value().id); EXPECT_EQ(xmlAttr->compiledAttribute.value().id.value(), ResourceId(0x7f010002)); ref = valueCast<Reference>(xmlAttr->compiledValue.get()); ASSERT_NE(ref, nullptr); AAPT_ASSERT_TRUE(ref->id); EXPECT_EQ(ref->id.value(), ResourceId(0x7f030000)); } TEST_F(XmlReferenceLinkerTest, LinkViewWithLocalPackageAndAliasOfTheSameName) { std::unique_ptr<xml::XmlResource> doc = test::buildXmlDomForPackageName(mContext.get(), R"EOF( <View xmlns:android="http://schemas.android.com/apk/res/com.app.test" android:attr="@id/id"/>)EOF"); XmlReferenceLinker linker; ASSERT_TRUE(linker.consume(mContext.get(), doc.get())); xml::Element* viewEl = xml::findRootElement(doc.get()); ASSERT_NE(viewEl, nullptr); // All attributes and references in this element should be referring to "com.app.test" (0x7f). xml::Attribute* xmlAttr = viewEl->findAttribute( u"http://schemas.android.com/apk/res/com.app.test", u"attr"); ASSERT_NE(xmlAttr, nullptr); AAPT_ASSERT_TRUE(xmlAttr->compiledAttribute); AAPT_ASSERT_TRUE(xmlAttr->compiledAttribute.value().id); EXPECT_EQ(xmlAttr->compiledAttribute.value().id.value(), ResourceId(0x7f010002)); Reference* ref = valueCast<Reference>(xmlAttr->compiledValue.get()); ASSERT_NE(ref, nullptr); AAPT_ASSERT_TRUE(ref->id); EXPECT_EQ(ref->id.value(), ResourceId(0x7f030000)); } } // namespace aapt