/*
* 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 "filter/ConfigFilter.h"
#include "io/FileSystem.h"
#include "link/TableMerger.h"
#include "test/Builders.h"
#include "test/Context.h"
#include <gtest/gtest.h>
namespace aapt {
struct TableMergerTest : public ::testing::Test {
std::unique_ptr<IAaptContext> mContext;
void SetUp() override {
mContext = test::ContextBuilder()
// We are compiling this package.
.setCompilationPackage(u"com.app.a")
// Merge all packages that have this package ID.
.setPackageId(0x7f)
// Mangle all packages that do not have this package name.
.setNameManglerPolicy(NameManglerPolicy{ u"com.app.a", { u"com.app.b" } })
.build();
}
};
TEST_F(TableMergerTest, SimpleMerge) {
std::unique_ptr<ResourceTable> tableA = test::ResourceTableBuilder()
.setPackageId(u"com.app.a", 0x7f)
.addReference(u"@com.app.a:id/foo", u"@com.app.a:id/bar")
.addReference(u"@com.app.a:id/bar", u"@com.app.b:id/foo")
.addValue(u"@com.app.a:styleable/view", test::StyleableBuilder()
.addItem(u"@com.app.b:id/foo")
.build())
.build();
std::unique_ptr<ResourceTable> tableB = test::ResourceTableBuilder()
.setPackageId(u"com.app.b", 0x7f)
.addSimple(u"@com.app.b:id/foo")
.build();
ResourceTable finalTable;
TableMerger merger(mContext.get(), &finalTable, TableMergerOptions{});
io::FileCollection collection;
ASSERT_TRUE(merger.merge({}, tableA.get()));
ASSERT_TRUE(merger.mergeAndMangle({}, u"com.app.b", tableB.get(), &collection));
EXPECT_TRUE(merger.getMergedPackages().count(u"com.app.b") != 0);
// Entries from com.app.a should not be mangled.
AAPT_EXPECT_TRUE(finalTable.findResource(test::parseNameOrDie(u"@com.app.a:id/foo")));
AAPT_EXPECT_TRUE(finalTable.findResource(test::parseNameOrDie(u"@com.app.a:id/bar")));
AAPT_EXPECT_TRUE(finalTable.findResource(test::parseNameOrDie(u"@com.app.a:styleable/view")));
// The unmangled name should not be present.
AAPT_EXPECT_FALSE(finalTable.findResource(test::parseNameOrDie(u"@com.app.b:id/foo")));
// Look for the mangled name.
AAPT_EXPECT_TRUE(finalTable.findResource(test::parseNameOrDie(u"@com.app.a:id/com.app.b$foo")));
}
TEST_F(TableMergerTest, MergeFile) {
ResourceTable finalTable;
TableMergerOptions options;
options.autoAddOverlay = false;
TableMerger merger(mContext.get(), &finalTable, options);
ResourceFile fileDesc;
fileDesc.config = test::parseConfigOrDie("hdpi-v4");
fileDesc.name = test::parseNameOrDie(u"@layout/main");
fileDesc.source = Source("res/layout-hdpi/main.xml");
test::TestFile testFile("path/to/res/layout-hdpi/main.xml.flat");
ASSERT_TRUE(merger.mergeFile(fileDesc, &testFile));
FileReference* file = test::getValueForConfig<FileReference>(&finalTable,
u"@com.app.a:layout/main",
test::parseConfigOrDie("hdpi-v4"));
ASSERT_NE(nullptr, file);
EXPECT_EQ(std::u16string(u"res/layout-hdpi-v4/main.xml"), *file->path);
}
TEST_F(TableMergerTest, MergeFileOverlay) {
ResourceTable finalTable;
TableMergerOptions tableMergerOptions;
tableMergerOptions.autoAddOverlay = false;
TableMerger merger(mContext.get(), &finalTable, tableMergerOptions);
ResourceFile fileDesc;
fileDesc.name = test::parseNameOrDie(u"@xml/foo");
test::TestFile fileA("path/to/fileA.xml.flat");
test::TestFile fileB("path/to/fileB.xml.flat");
ASSERT_TRUE(merger.mergeFile(fileDesc, &fileA));
ASSERT_TRUE(merger.mergeFileOverlay(fileDesc, &fileB));
}
TEST_F(TableMergerTest, MergeFileReferences) {
std::unique_ptr<ResourceTable> tableA = test::ResourceTableBuilder()
.setPackageId(u"com.app.a", 0x7f)
.addFileReference(u"@com.app.a:xml/file", u"res/xml/file.xml")
.build();
std::unique_ptr<ResourceTable> tableB = test::ResourceTableBuilder()
.setPackageId(u"com.app.b", 0x7f)
.addFileReference(u"@com.app.b:xml/file", u"res/xml/file.xml")
.build();
ResourceTable finalTable;
TableMerger merger(mContext.get(), &finalTable, TableMergerOptions{});
io::FileCollection collection;
collection.insertFile("res/xml/file.xml");
ASSERT_TRUE(merger.merge({}, tableA.get()));
ASSERT_TRUE(merger.mergeAndMangle({}, u"com.app.b", tableB.get(), &collection));
FileReference* f = test::getValue<FileReference>(&finalTable, u"@com.app.a:xml/file");
ASSERT_NE(f, nullptr);
EXPECT_EQ(std::u16string(u"res/xml/file.xml"), *f->path);
f = test::getValue<FileReference>(&finalTable, u"@com.app.a:xml/com.app.b$file");
ASSERT_NE(f, nullptr);
EXPECT_EQ(std::u16string(u"res/xml/com.app.b$file.xml"), *f->path);
}
TEST_F(TableMergerTest, OverrideResourceWithOverlay) {
std::unique_ptr<ResourceTable> base = test::ResourceTableBuilder()
.setPackageId(u"", 0x00)
.addValue(u"@bool/foo", ResourceUtils::tryParseBool(u"true"))
.build();
std::unique_ptr<ResourceTable> overlay = test::ResourceTableBuilder()
.setPackageId(u"", 0x00)
.addValue(u"@bool/foo", ResourceUtils::tryParseBool(u"false"))
.build();
ResourceTable finalTable;
TableMergerOptions tableMergerOptions;
tableMergerOptions.autoAddOverlay = false;
TableMerger merger(mContext.get(), &finalTable, tableMergerOptions);
ASSERT_TRUE(merger.merge({}, base.get()));
ASSERT_TRUE(merger.mergeOverlay({}, overlay.get()));
BinaryPrimitive* foo = test::getValue<BinaryPrimitive>(&finalTable, u"@com.app.a:bool/foo");
ASSERT_NE(nullptr, foo);
EXPECT_EQ(0x0u, foo->value.data);
}
TEST_F(TableMergerTest, MergeAddResourceFromOverlay) {
std::unique_ptr<ResourceTable> tableA = test::ResourceTableBuilder()
.setPackageId(u"", 0x7f)
.setSymbolState(u"@bool/foo", {}, SymbolState::kUndefined)
.build();
std::unique_ptr<ResourceTable> tableB = test::ResourceTableBuilder()
.setPackageId(u"", 0x7f)
.addValue(u"@bool/foo", ResourceUtils::tryParseBool(u"true"))
.build();
ResourceTable finalTable;
TableMerger merger(mContext.get(), &finalTable, TableMergerOptions{});
ASSERT_TRUE(merger.merge({}, tableA.get()));
ASSERT_TRUE(merger.mergeOverlay({}, tableB.get()));
}
TEST_F(TableMergerTest, MergeAddResourceFromOverlayWithAutoAddOverlay) {
std::unique_ptr<ResourceTable> tableA = test::ResourceTableBuilder()
.setPackageId(u"", 0x7f)
.build();
std::unique_ptr<ResourceTable> tableB = test::ResourceTableBuilder()
.setPackageId(u"", 0x7f)
.addValue(u"@bool/foo", ResourceUtils::tryParseBool(u"true"))
.build();
ResourceTable finalTable;
TableMergerOptions options;
options.autoAddOverlay = true;
TableMerger merger(mContext.get(), &finalTable, options);
ASSERT_TRUE(merger.merge({}, tableA.get()));
ASSERT_TRUE(merger.mergeOverlay({}, tableB.get()));
}
TEST_F(TableMergerTest, FailToMergeNewResourceWithoutAutoAddOverlay) {
std::unique_ptr<ResourceTable> tableA = test::ResourceTableBuilder()
.setPackageId(u"", 0x7f)
.build();
std::unique_ptr<ResourceTable> tableB = test::ResourceTableBuilder()
.setPackageId(u"", 0x7f)
.addValue(u"@bool/foo", ResourceUtils::tryParseBool(u"true"))
.build();
ResourceTable finalTable;
TableMergerOptions options;
options.autoAddOverlay = false;
TableMerger merger(mContext.get(), &finalTable, options);
ASSERT_TRUE(merger.merge({}, tableA.get()));
ASSERT_FALSE(merger.mergeOverlay({}, tableB.get()));
}
} // namespace aapt