<overlayable> tags can now have policy elements that indicate which
partition the overlay apk must reside on in order to be allowed to
overlay a resource. This change only adds parsing of <policy> and
encoding of policy in the proto ResourceTable. A later change will add
the encoding of policy and overlayable in the binary APK.
<overlayable>
<policy type="system|vendor|product|product_services|public" >
<item type="string" name="oof" />
</policy>
</overlayable>
Bug: 110869880
Test: make aapt2_tests
Change-Id: I8d4ed7b0e01f981149c6e3190af1681073b79b03
351 lines
14 KiB
C++
351 lines
14 KiB
C++
/*
|
|
* 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 "ResourceTable.h"
|
|
#include "Diagnostics.h"
|
|
#include "ResourceValues.h"
|
|
#include "test/Test.h"
|
|
#include "util/Util.h"
|
|
|
|
#include <algorithm>
|
|
#include <ostream>
|
|
#include <string>
|
|
|
|
using ::android::ConfigDescription;
|
|
using ::android::StringPiece;
|
|
using ::testing::Eq;
|
|
using ::testing::NotNull;
|
|
using ::testing::StrEq;
|
|
|
|
namespace aapt {
|
|
|
|
TEST(ResourceTableTest, FailToAddResourceWithBadName) {
|
|
ResourceTable table;
|
|
|
|
EXPECT_FALSE(table.AddResource(
|
|
test::ParseNameOrDie("android:id/hey,there"), ConfigDescription{}, "",
|
|
test::ValueBuilder<Id>().SetSource("test.xml", 21u).Build(),
|
|
test::GetDiagnostics()));
|
|
|
|
EXPECT_FALSE(table.AddResource(
|
|
test::ParseNameOrDie("android:id/hey:there"), ConfigDescription{}, "",
|
|
test::ValueBuilder<Id>().SetSource("test.xml", 21u).Build(),
|
|
test::GetDiagnostics()));
|
|
}
|
|
|
|
TEST(ResourceTableTest, AddResourceWithWeirdNameWhenAddingMangledResources) {
|
|
ResourceTable table;
|
|
|
|
EXPECT_TRUE(table.AddResourceMangled(
|
|
test::ParseNameOrDie("android:id/heythere "), ConfigDescription{}, "",
|
|
test::ValueBuilder<Id>().SetSource("test.xml", 21u).Build(), test::GetDiagnostics()));
|
|
}
|
|
|
|
TEST(ResourceTableTest, AddOneResource) {
|
|
ResourceTable table;
|
|
|
|
EXPECT_TRUE(table.AddResource(
|
|
test::ParseNameOrDie("android:attr/id"), ConfigDescription{}, "",
|
|
test::ValueBuilder<Id>().SetSource("test/path/file.xml", 23u).Build(),
|
|
test::GetDiagnostics()));
|
|
|
|
EXPECT_THAT(test::GetValue<Id>(&table, "android:attr/id"), NotNull());
|
|
}
|
|
|
|
TEST(ResourceTableTest, AddMultipleResources) {
|
|
ResourceTable table;
|
|
|
|
ConfigDescription config;
|
|
ConfigDescription language_config;
|
|
memcpy(language_config.language, "pl", sizeof(language_config.language));
|
|
|
|
EXPECT_TRUE(table.AddResource(
|
|
test::ParseNameOrDie("android:attr/layout_width"), config, "",
|
|
test::ValueBuilder<Id>().SetSource("test/path/file.xml", 10u).Build(),
|
|
test::GetDiagnostics()));
|
|
|
|
EXPECT_TRUE(table.AddResource(
|
|
test::ParseNameOrDie("android:attr/id"), config, "",
|
|
test::ValueBuilder<Id>().SetSource("test/path/file.xml", 12u).Build(),
|
|
test::GetDiagnostics()));
|
|
|
|
EXPECT_TRUE(table.AddResource(
|
|
test::ParseNameOrDie("android:string/ok"), config, "",
|
|
test::ValueBuilder<Id>().SetSource("test/path/file.xml", 14u).Build(),
|
|
test::GetDiagnostics()));
|
|
|
|
EXPECT_TRUE(table.AddResource(
|
|
test::ParseNameOrDie("android:string/ok"), language_config, "",
|
|
test::ValueBuilder<BinaryPrimitive>(android::Res_value{})
|
|
.SetSource("test/path/file.xml", 20u)
|
|
.Build(),
|
|
test::GetDiagnostics()));
|
|
|
|
EXPECT_THAT(test::GetValue<Id>(&table, "android:attr/layout_width"), NotNull());
|
|
EXPECT_THAT(test::GetValue<Id>(&table, "android:attr/id"), NotNull());
|
|
EXPECT_THAT(test::GetValue<Id>(&table, "android:string/ok"), NotNull());
|
|
EXPECT_THAT(test::GetValueForConfig<BinaryPrimitive>(&table, "android:string/ok", language_config), NotNull());
|
|
}
|
|
|
|
TEST(ResourceTableTest, OverrideWeakResourceValue) {
|
|
ResourceTable table;
|
|
|
|
ASSERT_TRUE(table.AddResource(test::ParseNameOrDie("android:attr/foo"), ConfigDescription{}, "",
|
|
test::AttributeBuilder().SetWeak(true).Build(),
|
|
test::GetDiagnostics()));
|
|
|
|
Attribute* attr = test::GetValue<Attribute>(&table, "android:attr/foo");
|
|
ASSERT_THAT(attr, NotNull());
|
|
EXPECT_TRUE(attr->IsWeak());
|
|
|
|
ASSERT_TRUE(table.AddResource(test::ParseNameOrDie("android:attr/foo"), ConfigDescription{}, "",
|
|
util::make_unique<Attribute>(), test::GetDiagnostics()));
|
|
|
|
attr = test::GetValue<Attribute>(&table, "android:attr/foo");
|
|
ASSERT_THAT(attr, NotNull());
|
|
EXPECT_FALSE(attr->IsWeak());
|
|
}
|
|
|
|
TEST(ResourceTableTest, AllowCompatibleDuplicateAttributes) {
|
|
ResourceTable table;
|
|
|
|
const ResourceName name = test::ParseNameOrDie("android:attr/foo");
|
|
Attribute attr_one(android::ResTable_map::TYPE_STRING);
|
|
attr_one.SetWeak(true);
|
|
Attribute attr_two(android::ResTable_map::TYPE_STRING | android::ResTable_map::TYPE_REFERENCE);
|
|
attr_two.SetWeak(true);
|
|
|
|
ASSERT_TRUE(table.AddResource(name, ConfigDescription{}, "",
|
|
util::make_unique<Attribute>(attr_one), test::GetDiagnostics()));
|
|
ASSERT_TRUE(table.AddResource(name, ConfigDescription{}, "",
|
|
util::make_unique<Attribute>(attr_two), test::GetDiagnostics()));
|
|
}
|
|
|
|
TEST(ResourceTableTest, ProductVaryingValues) {
|
|
ResourceTable table;
|
|
|
|
EXPECT_TRUE(table.AddResource(test::ParseNameOrDie("android:string/foo"),
|
|
test::ParseConfigOrDie("land"), "tablet",
|
|
util::make_unique<Id>(),
|
|
test::GetDiagnostics()));
|
|
EXPECT_TRUE(table.AddResource(test::ParseNameOrDie("android:string/foo"),
|
|
test::ParseConfigOrDie("land"), "phone",
|
|
util::make_unique<Id>(),
|
|
test::GetDiagnostics()));
|
|
|
|
EXPECT_THAT(test::GetValueForConfigAndProduct<Id>(&table, "android:string/foo",test::ParseConfigOrDie("land"), "tablet"), NotNull());
|
|
EXPECT_THAT(test::GetValueForConfigAndProduct<Id>(&table, "android:string/foo",test::ParseConfigOrDie("land"), "phone"), NotNull());
|
|
|
|
Maybe<ResourceTable::SearchResult> sr =
|
|
table.FindResource(test::ParseNameOrDie("android:string/foo"));
|
|
ASSERT_TRUE(sr);
|
|
std::vector<ResourceConfigValue*> values =
|
|
sr.value().entry->FindAllValues(test::ParseConfigOrDie("land"));
|
|
ASSERT_EQ(2u, values.size());
|
|
EXPECT_EQ(std::string("phone"), values[0]->product);
|
|
EXPECT_EQ(std::string("tablet"), values[1]->product);
|
|
}
|
|
|
|
static StringPiece LevelToString(Visibility::Level level) {
|
|
switch (level) {
|
|
case Visibility::Level::kPrivate:
|
|
return "private";
|
|
case Visibility::Level::kPublic:
|
|
return "private";
|
|
default:
|
|
return "undefined";
|
|
}
|
|
}
|
|
|
|
static ::testing::AssertionResult VisibilityOfResource(const ResourceTable& table,
|
|
const ResourceNameRef& name,
|
|
Visibility::Level level,
|
|
const StringPiece& comment) {
|
|
Maybe<ResourceTable::SearchResult> result = table.FindResource(name);
|
|
if (!result) {
|
|
return ::testing::AssertionFailure() << "no resource '" << name << "' found in table";
|
|
}
|
|
|
|
const Visibility& visibility = result.value().entry->visibility;
|
|
if (visibility.level != level) {
|
|
return ::testing::AssertionFailure() << "expected visibility " << LevelToString(level)
|
|
<< " but got " << LevelToString(visibility.level);
|
|
}
|
|
|
|
if (visibility.comment != comment) {
|
|
return ::testing::AssertionFailure() << "expected visibility comment '" << comment
|
|
<< "' but got '" << visibility.comment << "'";
|
|
}
|
|
return ::testing::AssertionSuccess();
|
|
}
|
|
|
|
TEST(ResourceTableTest, SetVisibility) {
|
|
using Level = Visibility::Level;
|
|
|
|
ResourceTable table;
|
|
const ResourceName name = test::ParseNameOrDie("android:string/foo");
|
|
|
|
Visibility visibility;
|
|
visibility.level = Visibility::Level::kPrivate;
|
|
visibility.comment = "private";
|
|
ASSERT_TRUE(table.SetVisibility(name, visibility, test::GetDiagnostics()));
|
|
ASSERT_TRUE(VisibilityOfResource(table, name, Level::kPrivate, "private"));
|
|
|
|
visibility.level = Visibility::Level::kUndefined;
|
|
visibility.comment = "undefined";
|
|
ASSERT_TRUE(table.SetVisibility(name, visibility, test::GetDiagnostics()));
|
|
ASSERT_TRUE(VisibilityOfResource(table, name, Level::kPrivate, "private"));
|
|
|
|
visibility.level = Visibility::Level::kPublic;
|
|
visibility.comment = "public";
|
|
ASSERT_TRUE(table.SetVisibility(name, visibility, test::GetDiagnostics()));
|
|
ASSERT_TRUE(VisibilityOfResource(table, name, Level::kPublic, "public"));
|
|
|
|
visibility.level = Visibility::Level::kPrivate;
|
|
visibility.comment = "private";
|
|
ASSERT_TRUE(table.SetVisibility(name, visibility, test::GetDiagnostics()));
|
|
ASSERT_TRUE(VisibilityOfResource(table, name, Level::kPublic, "public"));
|
|
}
|
|
|
|
TEST(ResourceTableTest, SetAllowNew) {
|
|
ResourceTable table;
|
|
const ResourceName name = test::ParseNameOrDie("android:string/foo");
|
|
|
|
AllowNew allow_new;
|
|
Maybe<ResourceTable::SearchResult> result;
|
|
|
|
allow_new.comment = "first";
|
|
ASSERT_TRUE(table.SetAllowNew(name, allow_new, test::GetDiagnostics()));
|
|
result = table.FindResource(name);
|
|
ASSERT_TRUE(result);
|
|
ASSERT_TRUE(result.value().entry->allow_new);
|
|
ASSERT_THAT(result.value().entry->allow_new.value().comment, StrEq("first"));
|
|
|
|
allow_new.comment = "second";
|
|
ASSERT_TRUE(table.SetAllowNew(name, allow_new, test::GetDiagnostics()));
|
|
result = table.FindResource(name);
|
|
ASSERT_TRUE(result);
|
|
ASSERT_TRUE(result.value().entry->allow_new);
|
|
ASSERT_THAT(result.value().entry->allow_new.value().comment, StrEq("second"));
|
|
}
|
|
|
|
TEST(ResourceTableTest, AddOverlayable) {
|
|
ResourceTable table;
|
|
const ResourceName name = test::ParseNameOrDie("android:string/foo");
|
|
|
|
Overlayable overlayable;
|
|
overlayable.policy = Overlayable::Policy::kProduct;
|
|
overlayable.comment = "first";
|
|
ASSERT_TRUE(table.AddOverlayable(name, overlayable, test::GetDiagnostics()));
|
|
Maybe<ResourceTable::SearchResult> result = table.FindResource(name);
|
|
ASSERT_TRUE(result);
|
|
ASSERT_THAT(result.value().entry->overlayable_declarations.size(), Eq(1));
|
|
ASSERT_THAT(result.value().entry->overlayable_declarations[0].comment, StrEq("first"));
|
|
ASSERT_THAT(result.value().entry->overlayable_declarations[0].policy,
|
|
Eq(Overlayable::Policy::kProduct));
|
|
|
|
Overlayable overlayable2;
|
|
overlayable2.comment = "second";
|
|
overlayable2.policy = Overlayable::Policy::kProductServices;
|
|
ASSERT_TRUE(table.AddOverlayable(name, overlayable2, test::GetDiagnostics()));
|
|
result = table.FindResource(name);
|
|
ASSERT_TRUE(result);
|
|
ASSERT_THAT(result.value().entry->overlayable_declarations.size(), Eq(2));
|
|
ASSERT_THAT(result.value().entry->overlayable_declarations[0].comment, StrEq("first"));
|
|
ASSERT_THAT(result.value().entry->overlayable_declarations[0].policy,
|
|
Eq(Overlayable::Policy::kProduct));
|
|
ASSERT_THAT(result.value().entry->overlayable_declarations[1].comment, StrEq("second"));
|
|
ASSERT_THAT(result.value().entry->overlayable_declarations[1].policy,
|
|
Eq(Overlayable::Policy::kProductServices));
|
|
}
|
|
|
|
TEST(ResourceTableTest, AddDuplicateOverlayableFail) {
|
|
ResourceTable table;
|
|
const ResourceName name = test::ParseNameOrDie("android:string/foo");
|
|
|
|
Overlayable overlayable;
|
|
overlayable.policy = Overlayable::Policy::kProduct;
|
|
ASSERT_TRUE(table.AddOverlayable(name, overlayable, test::GetDiagnostics()));
|
|
|
|
Overlayable overlayable2;
|
|
overlayable2.policy = Overlayable::Policy::kProduct;
|
|
ASSERT_FALSE(table.AddOverlayable(name, overlayable2, test::GetDiagnostics()));
|
|
}
|
|
|
|
TEST(ResourceTableTest, AddOverlayablePolicyAndNoneFirstFail) {
|
|
ResourceTable table;
|
|
const ResourceName name = test::ParseNameOrDie("android:string/foo");
|
|
|
|
ASSERT_TRUE(table.AddOverlayable(name, {}, test::GetDiagnostics()));
|
|
|
|
Overlayable overlayable2;
|
|
overlayable2.policy = Overlayable::Policy::kProduct;
|
|
ASSERT_FALSE(table.AddOverlayable(name, overlayable2, test::GetDiagnostics()));
|
|
}
|
|
|
|
TEST(ResourceTableTest, AddOverlayablePolicyAndNoneLastFail) {
|
|
ResourceTable table;
|
|
const ResourceName name = test::ParseNameOrDie("android:string/foo");
|
|
|
|
Overlayable overlayable;
|
|
overlayable.policy = Overlayable::Policy::kProduct;
|
|
ASSERT_TRUE(table.AddOverlayable(name, overlayable, test::GetDiagnostics()));
|
|
|
|
ASSERT_FALSE(table.AddOverlayable(name, {}, test::GetDiagnostics()));
|
|
}
|
|
|
|
TEST(ResourceTableTest, AllowDuplictaeResourcesNames) {
|
|
ResourceTable table(/* validate_resources */ false);
|
|
|
|
const ResourceName foo_name = test::ParseNameOrDie("android:bool/foo");
|
|
ASSERT_TRUE(table.AddResourceWithId(foo_name, ResourceId(0x7f0100ff), ConfigDescription{} , "",
|
|
test::BuildPrimitive(android::Res_value::TYPE_INT_BOOLEAN, 0),
|
|
test::GetDiagnostics()));
|
|
ASSERT_TRUE(table.AddResourceWithId(foo_name, ResourceId(0x7f010100), ConfigDescription{} , "",
|
|
test::BuildPrimitive(android::Res_value::TYPE_INT_BOOLEAN, 1),
|
|
test::GetDiagnostics()));
|
|
|
|
ASSERT_TRUE(table.SetVisibilityWithId(foo_name, Visibility{Visibility::Level::kPublic},
|
|
ResourceId(0x7f0100ff), test::GetDiagnostics()));
|
|
ASSERT_TRUE(table.SetVisibilityWithId(foo_name, Visibility{Visibility::Level::kPrivate},
|
|
ResourceId(0x7f010100), test::GetDiagnostics()));
|
|
|
|
auto package = table.FindPackageById(0x7f);
|
|
ASSERT_THAT(package, NotNull());
|
|
auto type = package->FindType(ResourceType::kBool);
|
|
ASSERT_THAT(type, NotNull());
|
|
|
|
auto entry1 = type->FindEntry("foo", 0x00ff);
|
|
ASSERT_THAT(entry1, NotNull());
|
|
ASSERT_THAT(entry1->id, Eq(0x00ff));
|
|
ASSERT_THAT(entry1->values[0], NotNull());
|
|
ASSERT_THAT(entry1->values[0]->value, NotNull());
|
|
ASSERT_THAT(ValueCast<BinaryPrimitive>(entry1->values[0]->value.get()), NotNull());
|
|
ASSERT_THAT(ValueCast<BinaryPrimitive>(entry1->values[0]->value.get())->value.data, Eq(0u));
|
|
ASSERT_THAT(entry1->visibility.level, Visibility::Level::kPublic);
|
|
|
|
auto entry2 = type->FindEntry("foo", 0x0100);
|
|
ASSERT_THAT(entry2, NotNull());
|
|
ASSERT_THAT(entry2->id, Eq(0x0100));
|
|
ASSERT_THAT(entry2->values[0], NotNull());
|
|
ASSERT_THAT(entry1->values[0]->value, NotNull());
|
|
ASSERT_THAT(ValueCast<BinaryPrimitive>(entry2->values[0]->value.get()), NotNull());
|
|
ASSERT_THAT(ValueCast<BinaryPrimitive>(entry2->values[0]->value.get())->value.data, Eq(1u));
|
|
ASSERT_THAT(entry2->visibility.level, Visibility::Level::kPrivate);
|
|
}
|
|
|
|
} // namespace aapt
|