| /* |
| * Copyright (C) 2016 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 "androidfw/ApkAssets.h" |
| |
| #include "android-base/file.h" |
| #include "android-base/test_utils.h" |
| #include "android-base/unique_fd.h" |
| #include "androidfw/Util.h" |
| |
| #include "TestHelpers.h" |
| #include "data/basic/R.h" |
| |
| using ::android::base::unique_fd; |
| using ::com::android::basic::R; |
| using ::testing::Eq; |
| using ::testing::Ge; |
| using ::testing::NotNull; |
| using ::testing::SizeIs; |
| using ::testing::StrEq; |
| |
| namespace android { |
| |
| TEST(ApkAssetsTest, LoadApk) { |
| std::unique_ptr<const ApkAssets> loaded_apk = |
| ApkAssets::Load(GetTestDataPath() + "/basic/basic.apk"); |
| ASSERT_THAT(loaded_apk, NotNull()); |
| |
| const LoadedArsc* loaded_arsc = loaded_apk->GetLoadedArsc(); |
| ASSERT_THAT(loaded_arsc, NotNull()); |
| ASSERT_THAT(loaded_arsc->GetPackageById(0x7fu), NotNull()); |
| ASSERT_THAT(loaded_apk->Open("res/layout/main.xml"), NotNull()); |
| } |
| |
| TEST(ApkAssetsTest, LoadApkFromFd) { |
| const std::string path = GetTestDataPath() + "/basic/basic.apk"; |
| unique_fd fd(::open(path.c_str(), O_RDONLY | O_BINARY)); |
| ASSERT_THAT(fd.get(), Ge(0)); |
| |
| std::unique_ptr<const ApkAssets> loaded_apk = |
| ApkAssets::LoadFromFd(std::move(fd), path, false /*system*/, false /*force_shared_lib*/); |
| ASSERT_THAT(loaded_apk, NotNull()); |
| |
| const LoadedArsc* loaded_arsc = loaded_apk->GetLoadedArsc(); |
| ASSERT_THAT(loaded_arsc, NotNull()); |
| ASSERT_THAT(loaded_arsc->GetPackageById(0x7fu), NotNull()); |
| ASSERT_THAT(loaded_apk->Open("res/layout/main.xml"), NotNull()); |
| } |
| |
| TEST(ApkAssetsTest, LoadApkAsSharedLibrary) { |
| std::unique_ptr<const ApkAssets> loaded_apk = |
| ApkAssets::Load(GetTestDataPath() + "/appaslib/appaslib.apk"); |
| ASSERT_THAT(loaded_apk, NotNull()); |
| |
| const LoadedArsc* loaded_arsc = loaded_apk->GetLoadedArsc(); |
| ASSERT_THAT(loaded_arsc, NotNull()); |
| ASSERT_THAT(loaded_arsc->GetPackages(), SizeIs(1u)); |
| EXPECT_FALSE(loaded_arsc->GetPackages()[0]->IsDynamic()); |
| |
| loaded_apk = ApkAssets::LoadAsSharedLibrary(GetTestDataPath() + "/appaslib/appaslib.apk"); |
| ASSERT_THAT(loaded_apk, NotNull()); |
| |
| loaded_arsc = loaded_apk->GetLoadedArsc(); |
| ASSERT_THAT(loaded_arsc, NotNull()); |
| ASSERT_THAT(loaded_arsc->GetPackages(), SizeIs(1u)); |
| EXPECT_TRUE(loaded_arsc->GetPackages()[0]->IsDynamic()); |
| } |
| |
| TEST(ApkAssetsTest, LoadApkWithIdmap) { |
| std::string contents; |
| ResTable target_table; |
| const std::string target_path = GetTestDataPath() + "/basic/basic.apk"; |
| ASSERT_TRUE(ReadFileFromZipToString(target_path, "resources.arsc", &contents)); |
| ASSERT_THAT(target_table.add(contents.data(), contents.size(), 0, true /*copyData*/), |
| Eq(NO_ERROR)); |
| |
| ResTable overlay_table; |
| const std::string overlay_path = GetTestDataPath() + "/overlay/overlay.apk"; |
| ASSERT_TRUE(ReadFileFromZipToString(overlay_path, "resources.arsc", &contents)); |
| ASSERT_THAT(overlay_table.add(contents.data(), contents.size(), 0, true /*copyData*/), |
| Eq(NO_ERROR)); |
| |
| util::unique_cptr<void> idmap_data; |
| void* temp_data; |
| size_t idmap_len; |
| |
| ASSERT_THAT(target_table.createIdmap(overlay_table, 0u, 0u, target_path.c_str(), |
| overlay_path.c_str(), &temp_data, &idmap_len), |
| Eq(NO_ERROR)); |
| idmap_data.reset(temp_data); |
| |
| TemporaryFile tf; |
| ASSERT_TRUE(base::WriteFully(tf.fd, idmap_data.get(), idmap_len)); |
| close(tf.fd); |
| |
| // Open something so that the destructor of TemporaryFile closes a valid fd. |
| tf.fd = open("/dev/null", O_WRONLY); |
| |
| ASSERT_THAT(ApkAssets::LoadOverlay(tf.path), NotNull()); |
| } |
| |
| TEST(ApkAssetsTest, CreateAndDestroyAssetKeepsApkAssetsOpen) { |
| std::unique_ptr<const ApkAssets> loaded_apk = |
| ApkAssets::Load(GetTestDataPath() + "/basic/basic.apk"); |
| ASSERT_THAT(loaded_apk, NotNull()); |
| |
| { ASSERT_THAT(loaded_apk->Open("res/layout/main.xml", Asset::ACCESS_BUFFER), NotNull()); } |
| |
| { ASSERT_THAT(loaded_apk->Open("res/layout/main.xml", Asset::ACCESS_BUFFER), NotNull()); } |
| } |
| |
| TEST(ApkAssetsTest, OpenUncompressedAssetFd) { |
| std::unique_ptr<const ApkAssets> loaded_apk = |
| ApkAssets::Load(GetTestDataPath() + "/basic/basic.apk"); |
| ASSERT_THAT(loaded_apk, NotNull()); |
| |
| auto asset = loaded_apk->Open("assets/uncompressed.txt", Asset::ACCESS_UNKNOWN); |
| ASSERT_THAT(asset, NotNull()); |
| |
| off64_t start, length; |
| unique_fd fd(asset->openFileDescriptor(&start, &length)); |
| ASSERT_THAT(fd.get(), Ge(0)); |
| |
| lseek64(fd.get(), start, SEEK_SET); |
| |
| std::string buffer; |
| buffer.resize(length); |
| ASSERT_TRUE(base::ReadFully(fd.get(), &*buffer.begin(), length)); |
| |
| EXPECT_THAT(buffer, StrEq("This should be uncompressed.\n\n")); |
| } |
| |
| } // namespace android |