/* * 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 "compile/IdAssigner.h" #include "test/Test.h" namespace aapt { struct IdAssignerTests : public ::testing::Test { void SetUp() override { context = test::ContextBuilder().SetCompilationPackage("android").SetPackageId(0x01).Build(); } std::unique_ptr context; }; ::testing::AssertionResult VerifyIds(ResourceTable* table); TEST_F(IdAssignerTests, AssignIds) { auto table = test::ResourceTableBuilder() .AddSimple("android:attr/foo") .AddSimple("android:attr/bar") .AddSimple("android:id/foo") .Build(); IdAssigner assigner; ASSERT_TRUE(assigner.Consume(context.get(), table.get())); ASSERT_TRUE(VerifyIds(table.get())); } TEST_F(IdAssignerTests, AssignIdsWithReservedIds) { auto table = test::ResourceTableBuilder() .AddSimple("android:id/foo", ResourceId(0x01010000)) .AddSimple("android:dimen/two") .AddSimple("android:integer/three") .AddSimple("android:string/five") .AddSimple("android:attr/fun", ResourceId(0x01040000)) .AddSimple("android:attr/foo", ResourceId(0x01040006)) .AddSimple("android:attr/bar") .AddSimple("android:attr/baz") .Build(); IdAssigner assigner; ASSERT_TRUE(assigner.Consume(context.get(), table.get())); ASSERT_TRUE(VerifyIds(table.get())); std::optional maybe_result; // Expect to fill in the gaps between 0x0101XXXX and 0x0104XXXX. maybe_result = table->FindResource(test::ParseNameOrDie("android:dimen/two")); ASSERT_TRUE(maybe_result); EXPECT_EQ(0x01020000, maybe_result.value().entry->id); maybe_result = table->FindResource(test::ParseNameOrDie("android:integer/three")); ASSERT_TRUE(maybe_result); EXPECT_EQ(0x01030000, maybe_result.value().entry->id); // Expect to bypass the reserved 0x0104XXXX IDs and use the next 0x0105XXXX // IDs. maybe_result = table->FindResource(test::ParseNameOrDie("android:string/five")); ASSERT_TRUE(maybe_result); EXPECT_EQ(0x01050000, maybe_result.value().entry->id); // Expect to fill in the gaps between 0x01040000 and 0x01040006. maybe_result = table->FindResource(test::ParseNameOrDie("android:attr/bar")); ASSERT_TRUE(maybe_result); EXPECT_EQ(0x01040001, maybe_result.value().entry->id); maybe_result = table->FindResource(test::ParseNameOrDie("android:attr/baz")); ASSERT_TRUE(maybe_result); EXPECT_EQ(0x01040002, maybe_result.value().entry->id); } TEST_F(IdAssignerTests, FailWhenNonUniqueIdsAssigned) { auto table = test::ResourceTableBuilder() .AddSimple("android:attr/foo", ResourceId(0x01040006)) .AddSimple("android:attr/bar", ResourceId(0x01040006)) .Build(); IdAssigner assigner; ASSERT_FALSE(assigner.Consume(context.get(), table.get())); } TEST_F(IdAssignerTests, FailWhenNonUniqueTypeIdsAssigned) { auto table = test::ResourceTableBuilder() .AddSimple("android:string/foo", ResourceId(0x01040000)) .AddSimple("android:attr/bar", ResourceId(0x01040006)) .Build(); IdAssigner assigner; ASSERT_FALSE(assigner.Consume(context.get(), table.get())); } TEST_F(IdAssignerTests, FailWhenTypeHasTwoNonStagedIds) { auto table = test::ResourceTableBuilder() .AddSimple("android:attr/foo", ResourceId(0x01050000)) .AddSimple("android:attr/bar", ResourceId(0x01040006)) .Build(); IdAssigner assigner; ASSERT_FALSE(assigner.Consume(context.get(), table.get())); } TEST_F(IdAssignerTests, FailWhenTypeHasTwoNonStagedIdsRegardlessOfStagedId) { auto table = test::ResourceTableBuilder() .AddSimple("android:attr/foo", ResourceId(0x01050000)) .AddSimple("android:attr/bar", ResourceId(0x01ff0006)) .Add(NewResourceBuilder("android:attr/staged_baz") .SetId(0x01ff0000) .SetVisibility({.staged_api = true, .level = Visibility::Level::kPublic}) .Build()) .Build(); IdAssigner assigner; ASSERT_FALSE(assigner.Consume(context.get(), table.get())); } TEST_F(IdAssignerTests, FailWhenTypeHaveBothStagedAndNonStagedIds) { auto table = test::ResourceTableBuilder() .AddSimple("android:attr/foo", ResourceId(0x01010000)) .Add(NewResourceBuilder("android:bool/staged_baz") .SetId(0x01010001) .SetVisibility({.staged_api = true, .level = Visibility::Level::kPublic}) .Build()) .Build(); IdAssigner assigner; ASSERT_FALSE(assigner.Consume(context.get(), table.get())); } TEST_F(IdAssignerTests, AssignIdsWithIdMap) { auto table = test::ResourceTableBuilder() .AddSimple("android:attr/foo") .AddSimple("android:attr/bar") .Build(); std::unordered_map id_map = { {test::ParseNameOrDie("android:attr/foo"), ResourceId(0x01010002)}}; IdAssigner assigner(&id_map); ASSERT_TRUE(assigner.Consume(context.get(), table.get())); ASSERT_TRUE(VerifyIds(table.get())); auto result = table->FindResource(test::ParseNameOrDie("android:attr/foo")); ASSERT_TRUE(result); const ResourceTable::SearchResult& search_result = result.value(); EXPECT_EQ(0x01010002, search_result.entry->id); } TEST_F(IdAssignerTests, UseAllEntryIds) { ResourceTable table; const size_t max_entry_id = std::numeric_limits::max(); for (size_t i = 0; i <= max_entry_id; i++) { ASSERT_TRUE( table.AddResource(NewResourceBuilder("android:attr/res" + std::to_string(i)).Build(), context->GetDiagnostics())); } IdAssigner assigner; ASSERT_TRUE(assigner.Consume(context.get(), &table)); } TEST_F(IdAssignerTests, ExaustEntryIds) { ResourceTable table; const size_t max_entry_id = std::numeric_limits::max() + 1u; for (size_t i = 0; i <= max_entry_id; i++) { ASSERT_TRUE( table.AddResource(NewResourceBuilder("android:attr/res" + std::to_string(i)).Build(), context->GetDiagnostics())); } IdAssigner assigner; ASSERT_FALSE(assigner.Consume(context.get(), &table)); } TEST_F(IdAssignerTests, ExaustEntryIdsLastIdIsPublic) { ResourceTable table; ASSERT_TRUE(table.AddResource(NewResourceBuilder("android:attr/res").SetId(0x0101ffff).Build(), context->GetDiagnostics())); const size_t max_entry_id = std::numeric_limits::max(); for (size_t i = 0; i <= max_entry_id; i++) { ASSERT_TRUE( table.AddResource(NewResourceBuilder("android:attr/res" + std::to_string(i)).Build(), context->GetDiagnostics())); } IdAssigner assigner; ASSERT_FALSE(assigner.Consume(context.get(), &table)); } ::testing::AssertionResult VerifyIds(ResourceTable* table) { std::set seen_ids; for (auto& package : table->packages) { for (auto& type : package->types) { for (auto& entry : type->entries) { if (!entry->id) { return ::testing::AssertionFailure() << "resource " << ResourceNameRef(package->name, type->named_type, entry->name) << " has no ID"; } if (!seen_ids.insert(entry->id.value()).second) { return ::testing::AssertionFailure() << "resource " << ResourceNameRef(package->name, type->named_type, entry->name) << " has a non-unique ID" << std::hex << entry->id.value() << std::dec; } } } } return ::testing::AssertionSuccess() << "all IDs are unique and assigned"; } } // namespace aapt