/* * Copyright (C) 2021 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 "apexd.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "apex_database.h" #include "apex_file.h" #include "apex_file_repository.h" #include "apex_manifest.pb.h" #include "apexd_checkpoint.h" #include "apexd_loop.h" #include "apexd_session.h" #include "apexd_test_utils.h" #include "apexd_utils.h" #include "com_android_apex.h" #include "gmock/gmock-matchers.h" namespace android { namespace apex { namespace fs = std::filesystem; using MountedApexData = MountedApexDatabase::MountedApexData; using android::apex::testing::ApexFileEq; using android::base::GetExecutableDirectory; using android::base::GetProperty; using android::base::Join; using android::base::make_scope_guard; using android::base::ReadFileToString; using android::base::ReadFully; using android::base::RemoveFileIfExists; using android::base::Result; using android::base::Split; using android::base::StringPrintf; using android::base::unique_fd; using android::base::WriteStringToFile; using android::base::testing::HasError; using android::base::testing::HasValue; using android::base::testing::Ok; using android::base::testing::WithMessage; using android::dm::DeviceMapper; using ::apex::proto::SessionState; using com::android::apex::testing::ApexInfoXmlEq; using ::testing::ByRef; using ::testing::Contains; using ::testing::ElementsAre; using ::testing::HasSubstr; using ::testing::IsEmpty; using ::testing::Not; using ::testing::StartsWith; using ::testing::UnorderedElementsAre; using ::testing::UnorderedElementsAreArray; using ::testing::internal::CaptureStderr; using ::testing::internal::GetCapturedStderr; static std::string GetTestDataDir() { return GetExecutableDirectory(); } static std::string GetTestFile(const std::string& name) { return GetTestDataDir() + "/" + name; } static int64_t GetMTime(const std::string& path) { struct stat st_buf; if (stat(path.c_str(), &st_buf) != 0) { PLOG(ERROR) << "Failed to stat " << path; return 0; } return st_buf.st_mtime; } // A very basic mock of CheckpointInterface. class MockCheckpointInterface : public CheckpointInterface { public: Result SupportsFsCheckpoints() override { return supports_fs_checkpoint_; } Result NeedsCheckpoint() override { return needs_checkpoint_; } Result NeedsRollback() override { return needs_rollback_; } Result StartCheckpoint(int32_t num_retries) override { return {}; } Result AbortChanges(const std::string& msg, bool retry) override { return {}; } void SetSupportsCheckpoint(bool value) { supports_fs_checkpoint_ = value; } void SetNeedsCheckpoint(bool value) { needs_checkpoint_ = value; } void SetNeedsRollback(bool value) { needs_rollback_ = value; } private: bool supports_fs_checkpoint_, needs_checkpoint_, needs_rollback_; }; static constexpr const char* kTestApexdStatusSysprop = "apexd.status.test"; static constexpr const char* kTestVmPayloadMetadataPartitionProp = "apexd.vm.payload_metadata_partition.test"; static constexpr const char* kTestActiveApexSelinuxCtx = "u:object_r:shell_data_file:s0"; // A test fixture that provides frequently required temp directories for tests class ApexdUnitTest : public ::testing::Test { public: ApexdUnitTest() { built_in_dir_ = StringPrintf("%s/pre-installed-apex", td_.path); data_dir_ = StringPrintf("%s/data-apex", td_.path); decompression_dir_ = StringPrintf("%s/decompressed-apex", td_.path); ota_reserved_dir_ = StringPrintf("%s/ota-reserved", td_.path); hash_tree_dir_ = StringPrintf("%s/apex-hash-tree", td_.path); staged_session_dir_ = StringPrintf("%s/staged-session-dir", td_.path); metadata_sepolicy_staged_dir_ = StringPrintf("%s/metadata-sepolicy-staged-dir", td_.path); vm_payload_disk_ = StringPrintf("%s/vm-payload", td_.path); config_ = {kTestApexdStatusSysprop, {built_in_dir_}, data_dir_.c_str(), decompression_dir_.c_str(), ota_reserved_dir_.c_str(), hash_tree_dir_.c_str(), staged_session_dir_.c_str(), metadata_sepolicy_staged_dir_.c_str(), kTestVmPayloadMetadataPartitionProp, kTestActiveApexSelinuxCtx}; } const std::string& GetBuiltInDir() { return built_in_dir_; } const std::string& GetDataDir() { return data_dir_; } const std::string& GetDecompressionDir() { return decompression_dir_; } const std::string& GetOtaReservedDir() { return ota_reserved_dir_; } const std::string& GetHashTreeDir() { return hash_tree_dir_; } const std::string GetStagedDir(int session_id) { return StringPrintf("%s/session_%d", staged_session_dir_.c_str(), session_id); } const std::string& GetMetadataSepolicyStagedDir() { return metadata_sepolicy_staged_dir_; } std::string GetRootDigest(const ApexFile& apex) { if (apex.IsCompressed()) { return ""; } auto digest = apex.VerifyApexVerity(apex.GetBundledPublicKey()); if (!digest.ok()) { return ""; } return digest->root_digest; } std::string AddPreInstalledApex(const std::string& apex_name) { fs::copy(GetTestFile(apex_name), built_in_dir_); return StringPrintf("%s/%s", built_in_dir_.c_str(), apex_name.c_str()); } std::string AddDataApex(const std::string& apex_name) { fs::copy(GetTestFile(apex_name), data_dir_); return StringPrintf("%s/%s", data_dir_.c_str(), apex_name.c_str()); } std::string AddDataApex(const std::string& apex_name, const std::string& target_name) { fs::copy(GetTestFile(apex_name), data_dir_ + "/" + target_name); return StringPrintf("%s/%s", data_dir_.c_str(), target_name.c_str()); } std::string AddDecompressedApex(const std::string& apex_name) { auto apex_file = ApexFile::Open(GetTestFile(apex_name)); CHECK(apex_file.ok()); std::string target_name = apex_file->GetManifest().name() + "@" + std::to_string(apex_file->GetManifest().version()) + std::string(kDecompressedApexPackageSuffix); fs::copy(GetTestFile(apex_name), decompression_dir_ + "/" + target_name); return StringPrintf("%s/%s", decompression_dir_.c_str(), target_name.c_str()); } std::string AddBlockApex(const std::string& apex_name, const std::string& public_key = "", const std::string& root_digest = "", bool is_factory = true) { auto apex_path = vm_payload_disk_ + std::to_string(block_device_index_++); auto apex_file = GetTestFile(apex_name); AddToMetadata(apex_name, public_key, root_digest, is_factory); // loop_devices_ will be disposed after each test loop_devices_.push_back(*WriteBlockApex(apex_file, apex_path)); return apex_path; } // Copies the compressed apex to |built_in_dir| and decompresses it to // |decompressed_dir| and then hard links to |target_dir| std::string PrepareCompressedApex(const std::string& name, const std::string& built_in_dir) { fs::copy(GetTestFile(name), built_in_dir); auto compressed_apex = ApexFile::Open( StringPrintf("%s/%s", built_in_dir.c_str(), name.c_str())); std::vector compressed_apex_list; compressed_apex_list.emplace_back(std::cref(*compressed_apex)); auto return_value = ProcessCompressedApex(compressed_apex_list, /*is_ota_chroot*/ false); return StringPrintf("%s/%s", built_in_dir.c_str(), name.c_str()); } std::string PrepareCompressedApex(const std::string& name) { return PrepareCompressedApex(name, built_in_dir_); } Result CreateStagedSession(const std::string& apex_name, int session_id) { CreateDirIfNeeded(GetStagedDir(session_id), 0755); fs::copy(GetTestFile(apex_name), GetStagedDir(session_id)); auto result = ApexSession::CreateSession(session_id); result->SetBuildFingerprint(GetProperty("ro.build.fingerprint", "")); return result; } void SetBlockApexEnabled(bool enabled) { // The first partition(1) is "metadata" partition base::SetProperty(kTestVmPayloadMetadataPartitionProp, enabled ? (vm_payload_disk_ + "1") : ""); } protected: void SetUp() override { SetConfig(config_); ApexFileRepository::GetInstance().Reset(decompression_dir_); ASSERT_EQ(mkdir(built_in_dir_.c_str(), 0755), 0); ASSERT_EQ(mkdir(data_dir_.c_str(), 0755), 0); ASSERT_EQ(mkdir(decompression_dir_.c_str(), 0755), 0); ASSERT_EQ(mkdir(ota_reserved_dir_.c_str(), 0755), 0); ASSERT_EQ(mkdir(hash_tree_dir_.c_str(), 0755), 0); ASSERT_EQ(mkdir(staged_session_dir_.c_str(), 0755), 0); ASSERT_EQ(mkdir(metadata_sepolicy_staged_dir_.c_str(), 0755), 0); DeleteDirContent(ApexSession::GetSessionsDir()); } void AddToMetadata(const std::string& apex_name, const std::string& public_key, const std::string& root_digest, bool is_factory) { android::microdroid::Metadata metadata; // The first partition is metadata partition auto metadata_partition = vm_payload_disk_ + "1"; if (access(metadata_partition.c_str(), F_OK) == 0) { metadata = *android::microdroid::ReadMetadata(metadata_partition); } auto apex = metadata.add_apexes(); apex->set_name(apex_name); apex->set_public_key(public_key); apex->set_root_digest(root_digest); apex->set_is_factory(is_factory); std::ofstream out(metadata_partition); android::microdroid::WriteMetadata(metadata, out); } void TearDown() override { DeleteDirContent(ApexSession::GetSessionsDir()); SetBlockApexEnabled(false); } private: TemporaryDir td_; std::string built_in_dir_; std::string data_dir_; std::string decompression_dir_; std::string ota_reserved_dir_; std::string hash_tree_dir_; std::string vm_payload_disk_; std::string vm_payload_metadata_path_; std::string staged_session_dir_; std::string metadata_sepolicy_staged_dir_; ApexdConfig config_; std::vector loop_devices_; // to be cleaned up int block_device_index_ = 2; // "1" is reserved for metadata; }; // Apex that does not have pre-installed version, does not get selected TEST_F(ApexdUnitTest, ApexMustHavePreInstalledVersionForSelection) { AddPreInstalledApex("apex.apexd_test.apex"); AddPreInstalledApex("com.android.apex.cts.shim.apex"); auto shared_lib_1 = ApexFile::Open(AddPreInstalledApex( "com.android.apex.test.sharedlibs_generated.v1.libvX.apex")); auto& instance = ApexFileRepository::GetInstance(); // Pre-installed data needs to be present so that we can add data apex ASSERT_THAT(instance.AddPreInstalledApex({GetBuiltInDir()}), Ok()); auto apexd_test_file = ApexFile::Open(AddDataApex("apex.apexd_test.apex")); auto shim_v1 = ApexFile::Open(AddDataApex("com.android.apex.cts.shim.apex")); // Normally both pre-installed and data apex would be activated for a shared // libs apex, but if they are the same version only the data apex will be. auto shared_lib_2 = ApexFile::Open( AddDataApex("com.android.apex.test.sharedlibs_generated.v1.libvX.apex")); ASSERT_THAT(instance.AddDataApex(GetDataDir()), Ok()); const auto all_apex = instance.AllApexFilesByName(); // Pass a blank instance so that the data apex files are not considered // pre-installed const ApexFileRepository instance_blank; auto result = SelectApexForActivation(all_apex, instance_blank); ASSERT_EQ(result.size(), 0u); // When passed proper instance they should get selected result = SelectApexForActivation(all_apex, instance); ASSERT_EQ(result.size(), 3u); ASSERT_THAT(result, UnorderedElementsAre(ApexFileEq(ByRef(*apexd_test_file)), ApexFileEq(ByRef(*shim_v1)), ApexFileEq(ByRef(*shared_lib_2)))); } // Higher version gets priority when selecting for activation TEST_F(ApexdUnitTest, HigherVersionOfApexIsSelected) { auto apexd_test_file_v2 = ApexFile::Open(AddPreInstalledApex("apex.apexd_test_v2.apex")); AddPreInstalledApex("com.android.apex.cts.shim.apex"); auto& instance = ApexFileRepository::GetInstance(); ASSERT_THAT(instance.AddPreInstalledApex({GetBuiltInDir()}), Ok()); TemporaryDir data_dir; AddDataApex("apex.apexd_test.apex"); auto shim_v2 = ApexFile::Open(AddDataApex("com.android.apex.cts.shim.v2.apex")); ASSERT_THAT(instance.AddDataApex(GetDataDir()), Ok()); auto all_apex = instance.AllApexFilesByName(); auto result = SelectApexForActivation(all_apex, instance); ASSERT_EQ(result.size(), 2u); ASSERT_THAT(result, UnorderedElementsAre(ApexFileEq(ByRef(*apexd_test_file_v2)), ApexFileEq(ByRef(*shim_v2)))); } // When versions are equal, non-pre-installed version gets priority TEST_F(ApexdUnitTest, DataApexGetsPriorityForSameVersions) { AddPreInstalledApex("apex.apexd_test.apex"); AddPreInstalledApex("com.android.apex.cts.shim.apex"); // Initialize pre-installed APEX information auto& instance = ApexFileRepository::GetInstance(); ASSERT_THAT(instance.AddPreInstalledApex({GetBuiltInDir()}), Ok()); auto apexd_test_file = ApexFile::Open(AddDataApex("apex.apexd_test.apex")); auto shim_v1 = ApexFile::Open(AddDataApex("com.android.apex.cts.shim.apex")); // Initialize ApexFile repo ASSERT_THAT(instance.AddDataApex(GetDataDir()), Ok()); auto all_apex = instance.AllApexFilesByName(); auto result = SelectApexForActivation(all_apex, instance); ASSERT_EQ(result.size(), 2u); ASSERT_THAT(result, UnorderedElementsAre(ApexFileEq(ByRef(*apexd_test_file)), ApexFileEq(ByRef(*shim_v1)))); } // Both versions of shared libs can be selected when preinstalled version is // lower than data version TEST_F(ApexdUnitTest, SharedLibsCanHaveBothVersionSelected) { auto shared_lib_v1 = ApexFile::Open(AddPreInstalledApex( "com.android.apex.test.sharedlibs_generated.v1.libvX.apex")); // Initialize pre-installed APEX information auto& instance = ApexFileRepository::GetInstance(); ASSERT_THAT(instance.AddPreInstalledApex({GetBuiltInDir()}), Ok()); auto shared_lib_v2 = ApexFile::Open( AddDataApex("com.android.apex.test.sharedlibs_generated.v2.libvY.apex")); // Initialize data APEX information ASSERT_THAT(instance.AddDataApex(GetDataDir()), Ok()); auto all_apex = instance.AllApexFilesByName(); auto result = SelectApexForActivation(all_apex, instance); ASSERT_EQ(result.size(), 2u); ASSERT_THAT(result, UnorderedElementsAre(ApexFileEq(ByRef(*shared_lib_v1)), ApexFileEq(ByRef(*shared_lib_v2)))); } // Data version of shared libs should not be selected if lower than // preinstalled version TEST_F(ApexdUnitTest, SharedLibsDataVersionDeletedIfLower) { auto shared_lib_v2 = ApexFile::Open(AddPreInstalledApex( "com.android.apex.test.sharedlibs_generated.v2.libvY.apex")); // Initialize pre-installed APEX information auto& instance = ApexFileRepository::GetInstance(); ASSERT_THAT(instance.AddPreInstalledApex({GetBuiltInDir()}), Ok()); auto shared_lib_v1 = ApexFile::Open( AddDataApex("com.android.apex.test.sharedlibs_generated.v1.libvX.apex")); // Initialize data APEX information ASSERT_THAT(instance.AddDataApex(GetDataDir()), Ok()); auto all_apex = instance.AllApexFilesByName(); auto result = SelectApexForActivation(all_apex, instance); ASSERT_EQ(result.size(), 1u); ASSERT_THAT(result, UnorderedElementsAre(ApexFileEq(ByRef(*shared_lib_v2)))); } TEST_F(ApexdUnitTest, ProcessCompressedApex) { auto compressed_apex = ApexFile::Open( AddPreInstalledApex("com.android.apex.compressed.v1.capex")); std::vector compressed_apex_list; compressed_apex_list.emplace_back(std::cref(*compressed_apex)); auto return_value = ProcessCompressedApex(compressed_apex_list, /* is_ota_chroot= */ false); std::string decompressed_file_path = StringPrintf( "%s/com.android.apex.compressed@1%s", GetDecompressionDir().c_str(), kDecompressedApexPackageSuffix); // Assert output path is not empty auto exists = PathExists(decompressed_file_path); ASSERT_THAT(exists, HasValue(true)); // Assert that return value contains decompressed APEX auto decompressed_apex = ApexFile::Open(decompressed_file_path); ASSERT_THAT(return_value, UnorderedElementsAre(ApexFileEq(ByRef(*decompressed_apex)))); } TEST_F(ApexdUnitTest, ProcessCompressedApexRunsVerification) { auto compressed_apex_mismatch_key = ApexFile::Open(AddPreInstalledApex( "com.android.apex.compressed_key_mismatch_with_original.capex")); auto compressed_apex_version_mismatch = ApexFile::Open( AddPreInstalledApex("com.android.apex.compressed.v1_with_v2_apex.capex")); std::vector compressed_apex_list; compressed_apex_list.emplace_back(std::cref(*compressed_apex_mismatch_key)); compressed_apex_list.emplace_back( std::cref(*compressed_apex_version_mismatch)); auto return_value = ProcessCompressedApex(compressed_apex_list, /* is_ota_chroot= */ false); ASSERT_EQ(return_value.size(), 0u); } TEST_F(ApexdUnitTest, ValidateDecompressedApex) { auto capex = ApexFile::Open( AddPreInstalledApex("com.android.apex.compressed.v1.capex")); auto decompressed_v1 = ApexFile::Open( AddDataApex("com.android.apex.compressed.v1_original.apex")); auto result = ValidateDecompressedApex(std::cref(*capex), std::cref(*decompressed_v1)); ASSERT_THAT(result, Ok()); // Validation checks version auto decompressed_v2 = ApexFile::Open( AddDataApex("com.android.apex.compressed.v2_original.apex")); result = ValidateDecompressedApex(std::cref(*capex), std::cref(*decompressed_v2)); ASSERT_THAT( result, HasError(WithMessage(HasSubstr( "Compressed APEX has different version than decompressed APEX")))); // Validation check root digest auto decompressed_v1_different_digest = ApexFile::Open(AddDataApex( "com.android.apex.compressed.v1_different_digest_original.apex")); result = ValidateDecompressedApex( std::cref(*capex), std::cref(*decompressed_v1_different_digest)); ASSERT_THAT(result, HasError(WithMessage(HasSubstr( "does not match with expected root digest")))); // Validation checks key auto capex_different_key = ApexFile::Open( AddDataApex("com.android.apex.compressed_different_key.capex")); result = ValidateDecompressedApex(std::cref(*capex_different_key), std::cref(*decompressed_v1)); ASSERT_THAT( result, HasError(WithMessage(HasSubstr( "Public key of compressed APEX is different than original")))); } TEST_F(ApexdUnitTest, ProcessCompressedApexCanBeCalledMultipleTimes) { auto compressed_apex = ApexFile::Open( AddPreInstalledApex("com.android.apex.compressed.v1.capex")); std::vector compressed_apex_list; compressed_apex_list.emplace_back(std::cref(*compressed_apex)); auto return_value = ProcessCompressedApex(compressed_apex_list, /* is_ota_chroot= */ false); ASSERT_EQ(return_value.size(), 1u); // Capture the creation time of the decompressed APEX std::error_code ec; auto decompressed_apex_path = StringPrintf( "%s/com.android.apex.compressed@1%s", GetDecompressionDir().c_str(), kDecompressedApexPackageSuffix); auto last_write_time_1 = fs::last_write_time(decompressed_apex_path, ec); ASSERT_FALSE(ec) << "Failed to capture last write time of " << decompressed_apex_path; // Now try to decompress the same capex again. It should not fail. return_value = ProcessCompressedApex(compressed_apex_list, /* is_ota_chroot= */ false); ASSERT_EQ(return_value.size(), 1u); // Ensure the decompressed APEX file did not change auto last_write_time_2 = fs::last_write_time(decompressed_apex_path, ec); ASSERT_FALSE(ec) << "Failed to capture last write time of " << decompressed_apex_path; ASSERT_EQ(last_write_time_1, last_write_time_2); } // Test behavior of ProcessCompressedApex when is_ota_chroot is true TEST_F(ApexdUnitTest, ProcessCompressedApexOnOtaChroot) { auto compressed_apex = ApexFile::Open( AddPreInstalledApex("com.android.apex.compressed.v1.capex")); std::vector compressed_apex_list; compressed_apex_list.emplace_back(std::cref(*compressed_apex)); auto return_value = ProcessCompressedApex(compressed_apex_list, /* is_ota_chroot= */ true); ASSERT_EQ(return_value.size(), 1u); // Decompressed APEX should be located in decompression_dir std::string decompressed_file_path = StringPrintf("%s/com.android.apex.compressed@1%s", GetDecompressionDir().c_str(), kOtaApexPackageSuffix); // Assert output path is not empty auto exists = PathExists(decompressed_file_path); ASSERT_THAT(exists, HasValue(true)) << decompressed_file_path << " does not exist"; // Assert that return value contains the decompressed APEX auto apex_file = ApexFile::Open(decompressed_file_path); ASSERT_THAT(return_value, UnorderedElementsAre(ApexFileEq(ByRef(*apex_file)))); } // When decompressing APEX, reuse existing OTA APEX TEST_F(ApexdUnitTest, ProcessCompressedApexReuseOtaApex) { // Push a compressed APEX that will fail to decompress auto compressed_apex = ApexFile::Open(AddPreInstalledApex( "com.android.apex.compressed.v1_not_decompressible.capex")); std::vector compressed_apex_list; compressed_apex_list.emplace_back(std::cref(*compressed_apex)); // If we try to decompress capex directly, it should fail since the capex // pushed is faulty and cannot be decompressed auto return_value = ProcessCompressedApex(compressed_apex_list, /* is_ota_chroot= */ false); ASSERT_EQ(return_value.size(), 0u); // But, if there is an ota_apex present for reuse, it should reuse that // and avoid decompressing the faulty capex // Push an OTA apex that should be reused to skip decompression auto ota_apex_path = StringPrintf("%s/com.android.apex.compressed@1%s", GetDecompressionDir().c_str(), kOtaApexPackageSuffix); fs::copy(GetTestFile("com.android.apex.compressed.v1_original.apex"), ota_apex_path); return_value = ProcessCompressedApex(compressed_apex_list, /* is_ota_chroot= */ false); ASSERT_EQ(return_value.size(), 1u); // Ota Apex should be cleaned up ASSERT_THAT(PathExists(ota_apex_path), HasValue(false)); ASSERT_EQ(return_value[0].GetPath(), StringPrintf("%s/com.android.apex.compressed@1%s", GetDecompressionDir().c_str(), kDecompressedApexPackageSuffix)); } TEST_F(ApexdUnitTest, ShouldAllocateSpaceForDecompressionNewApex) { auto& instance = ApexFileRepository::GetInstance(); ASSERT_THAT(instance.AddPreInstalledApex({GetBuiltInDir()}), Ok()); // A brand new compressed APEX is being introduced: selected bool result = ShouldAllocateSpaceForDecompression("com.android.brand.new", 1, instance); ASSERT_TRUE(result); } TEST_F(ApexdUnitTest, ShouldAllocateSpaceForDecompressionWasNotCompressedBefore) { // Prepare fake pre-installed apex AddPreInstalledApex("apex.apexd_test.apex"); auto& instance = ApexFileRepository::GetInstance(); ASSERT_THAT(instance.AddPreInstalledApex({GetBuiltInDir()}), Ok()); // An existing pre-installed APEX is now compressed in the OTA: selected { bool result = ShouldAllocateSpaceForDecompression( "com.android.apex.test_package", 1, instance); ASSERT_TRUE(result); } // Even if there is a data apex (lower version) // Include data apex within calculation now AddDataApex("apex.apexd_test_v2.apex"); ASSERT_THAT(instance.AddDataApex(GetDataDir()), Ok()); { bool result = ShouldAllocateSpaceForDecompression( "com.android.apex.test_package", 3, instance); ASSERT_TRUE(result); } // But not if data apex has equal or higher version { bool result = ShouldAllocateSpaceForDecompression( "com.android.apex.test_package", 2, instance); ASSERT_FALSE(result); } } TEST_F(ApexdUnitTest, ShouldAllocateSpaceForDecompressionVersionCompare) { // Prepare fake pre-installed apex PrepareCompressedApex("com.android.apex.compressed.v1.capex"); auto& instance = ApexFileRepository::GetInstance(); ASSERT_THAT(instance.AddPreInstalledApex({GetBuiltInDir()}), Ok()); ASSERT_THAT(instance.AddDataApex(GetDataDir()), Ok()); { // New Compressed apex has higher version than decompressed data apex: // selected bool result = ShouldAllocateSpaceForDecompression( "com.android.apex.compressed", 2, instance); ASSERT_TRUE(result) << "Higher version test with decompressed data returned false"; } // Compare against decompressed data apex { // New Compressed apex has same version as decompressed data apex: not // selected bool result = ShouldAllocateSpaceForDecompression( "com.android.apex.compressed", 1, instance); ASSERT_FALSE(result) << "Same version test with decompressed data returned true"; } { // New Compressed apex has lower version than decompressed data apex: // selected bool result = ShouldAllocateSpaceForDecompression( "com.android.apex.compressed", 0, instance); ASSERT_TRUE(result) << "lower version test with decompressed data returned false"; } // Replace decompressed data apex with a higher version ApexFileRepository instance_new(GetDecompressionDir()); ASSERT_THAT(instance_new.AddPreInstalledApex({GetBuiltInDir()}), Ok()); TemporaryDir data_dir_new; fs::copy(GetTestFile("com.android.apex.compressed.v2_original.apex"), data_dir_new.path); ASSERT_THAT(instance_new.AddDataApex(data_dir_new.path), Ok()); { // New Compressed apex has higher version as data apex: selected bool result = ShouldAllocateSpaceForDecompression( "com.android.apex.compressed", 3, instance_new); ASSERT_TRUE(result) << "Higher version test with new data returned false"; } { // New Compressed apex has same version as data apex: not selected bool result = ShouldAllocateSpaceForDecompression( "com.android.apex.compressed", 2, instance_new); ASSERT_FALSE(result) << "Same version test with new data returned true"; } { // New Compressed apex has lower version than data apex: not selected bool result = ShouldAllocateSpaceForDecompression( "com.android.apex.compressed", 1, instance_new); ASSERT_FALSE(result) << "lower version test with new data returned true"; } } TEST_F(ApexdUnitTest, CalculateSizeForCompressedApexEmptyList) { ApexFileRepository instance; int64_t result = CalculateSizeForCompressedApex({}, instance); ASSERT_EQ(0LL, result); } TEST_F(ApexdUnitTest, CalculateSizeForCompressedApex) { ApexFileRepository instance; AddPreInstalledApex("com.android.apex.compressed.v1.capex"); ASSERT_THAT(instance.AddPreInstalledApex({GetBuiltInDir()}), Ok()); std::vector> input = { std::make_tuple("new_apex", 1, 1), std::make_tuple("new_apex_2", 1, 2), std::make_tuple("com.android.apex.compressed", 1, 4), // will be ignored std::make_tuple("com.android.apex.compressed", 2, 8), }; int64_t result = CalculateSizeForCompressedApex(input, instance); ASSERT_EQ(1 + 2 + 8LL, result); } TEST_F(ApexdUnitTest, ReserveSpaceForCompressedApexCreatesSingleFile) { TemporaryDir dest_dir; // Reserving space should create a single file in dest_dir with exact size ASSERT_THAT(ReserveSpaceForCompressedApex(100, dest_dir.path), Ok()); auto files = ReadDir(dest_dir.path, [](auto _) { return true; }); ASSERT_THAT(files, Ok()); ASSERT_EQ(files->size(), 1u); EXPECT_EQ(fs::file_size((*files)[0]), 100u); } TEST_F(ApexdUnitTest, ReserveSpaceForCompressedApexSafeToCallMultipleTimes) { TemporaryDir dest_dir; // Calling ReserveSpaceForCompressedApex multiple times should still create // a single file ASSERT_THAT(ReserveSpaceForCompressedApex(100, dest_dir.path), Ok()); ASSERT_THAT(ReserveSpaceForCompressedApex(100, dest_dir.path), Ok()); auto files = ReadDir(dest_dir.path, [](auto _) { return true; }); ASSERT_THAT(files, Ok()); ASSERT_EQ(files->size(), 1u); EXPECT_EQ(fs::file_size((*files)[0]), 100u); } TEST_F(ApexdUnitTest, ReserveSpaceForCompressedApexShrinkAndGrow) { TemporaryDir dest_dir; // Create a 100 byte file ASSERT_THAT(ReserveSpaceForCompressedApex(100, dest_dir.path), Ok()); // Should be able to shrink and grow the reserved space ASSERT_THAT(ReserveSpaceForCompressedApex(1000, dest_dir.path), Ok()); auto files = ReadDir(dest_dir.path, [](auto _) { return true; }); ASSERT_THAT(files, Ok()); ASSERT_EQ(files->size(), 1u); EXPECT_EQ(fs::file_size((*files)[0]), 1000u); ASSERT_THAT(ReserveSpaceForCompressedApex(10, dest_dir.path), Ok()); files = ReadDir(dest_dir.path, [](auto _) { return true; }); ASSERT_THAT(files, Ok()); ASSERT_EQ(files->size(), 1u); EXPECT_EQ(fs::file_size((*files)[0]), 10u); } TEST_F(ApexdUnitTest, ReserveSpaceForCompressedApexDeallocateIfPassedZero) { TemporaryDir dest_dir; // Create a file first ASSERT_THAT(ReserveSpaceForCompressedApex(100, dest_dir.path), Ok()); auto files = ReadDir(dest_dir.path, [](auto _) { return true; }); ASSERT_THAT(files, Ok()); ASSERT_EQ(files->size(), 1u); // Should delete the reserved file if size passed is 0 ASSERT_THAT(ReserveSpaceForCompressedApex(0, dest_dir.path), Ok()); files = ReadDir(dest_dir.path, [](auto _) { return true; }); ASSERT_THAT(files, Ok()); ASSERT_EQ(files->size(), 0u); } TEST_F(ApexdUnitTest, ReserveSpaceForCapexCleansOtaApex) { TemporaryDir dest_dir; auto ota_apex_path = StringPrintf( "%s/ota_apex%s", GetDecompressionDir().c_str(), kOtaApexPackageSuffix); auto create_ota_apex = [&]() { // Create an ota_apex first fs::copy(GetTestFile("com.android.apex.compressed.v1_original.apex"), ota_apex_path); ASSERT_THAT(PathExists(ota_apex_path), HasValue(true)); }; create_ota_apex(); // Should not delete the reserved file if size passed is negative ASSERT_THAT(ReserveSpaceForCompressedApex(-1, dest_dir.path), Not(Ok())); ASSERT_THAT(PathExists(ota_apex_path), HasValue(true)); // Should delete the reserved file if size passed is 0 ASSERT_THAT(ReserveSpaceForCompressedApex(0, dest_dir.path), Ok()); ASSERT_THAT(PathExists(ota_apex_path), HasValue(false)); create_ota_apex(); // Should delete the reserved file if size passed is positive ASSERT_THAT(ReserveSpaceForCompressedApex(10, dest_dir.path), Ok()); ASSERT_THAT(PathExists(ota_apex_path), HasValue(false)); } TEST_F(ApexdUnitTest, ReserveSpaceForCompressedApexErrorForNegativeValue) { TemporaryDir dest_dir; // Should return error if negative value is passed ASSERT_THAT(ReserveSpaceForCompressedApex(-1, dest_dir.path), Not(Ok())); } TEST_F(ApexdUnitTest, GetStagedApexFilesNoChild) { // Create staged session auto apex_session = CreateStagedSession("apex.apexd_test.apex", 123); apex_session->UpdateStateAndCommit(SessionState::STAGED); // Query for its file auto result = GetStagedApexFiles(123, {}); auto apex_file = ApexFile::Open( StringPrintf("%s/apex.apexd_test.apex", GetStagedDir(123).c_str())); ASSERT_THAT(result, HasValue(UnorderedElementsAre(ApexFileEq(ByRef(*apex_file))))); } TEST_F(ApexdUnitTest, GetStagedApexFilesOnlyStaged) { // Create staged session auto apex_session = CreateStagedSession("apex.apexd_test.apex", 123); apex_session->UpdateStateAndCommit(SessionState::VERIFIED); // Query for its file auto result = GetStagedApexFiles(123, {}); ASSERT_THAT( result, HasError(WithMessage(HasSubstr("Session 123 is not in state STAGED")))); } TEST_F(ApexdUnitTest, GetStagedApexFilesChecksNumberOfApexFiles) { // Create staged session auto apex_session = CreateStagedSession("apex.apexd_test.apex", 123); apex_session->UpdateStateAndCommit(SessionState::STAGED); auto staged_dir = GetStagedDir(123); { // Delete the staged apex file DeleteDirContent(staged_dir); // Query for its file auto result = GetStagedApexFiles(123, {}); ASSERT_THAT(result, HasError(WithMessage(HasSubstr( "Expected exactly one APEX file in directory")))); ASSERT_THAT(result, HasError(WithMessage(HasSubstr("Found: 0")))); } { // Copy multiple files to staged dir fs::copy(GetTestFile("apex.apexd_test.apex"), staged_dir); fs::copy(GetTestFile("apex.apexd_test_v2.apex"), staged_dir); // Query for its file auto result = GetStagedApexFiles(123, {}); ASSERT_THAT(result, HasError(WithMessage(HasSubstr( "Expected exactly one APEX file in directory")))); ASSERT_THAT(result, HasError(WithMessage(HasSubstr("Found: 2")))); } } TEST_F(ApexdUnitTest, GetStagedApexFilesWithChildren) { // Create staged session auto parent_apex_session = CreateStagedSession("apex.apexd_test.apex", 123); parent_apex_session->UpdateStateAndCommit(SessionState::STAGED); auto child_session_1 = CreateStagedSession("apex.apexd_test.apex", 124); auto child_session_2 = CreateStagedSession("apex.apexd_test.apex", 125); // Query for its file auto result = GetStagedApexFiles(123, {124, 125}); ASSERT_THAT(result, Ok()); auto child_apex_file_1 = ApexFile::Open( StringPrintf("%s/apex.apexd_test.apex", GetStagedDir(124).c_str())); auto child_apex_file_2 = ApexFile::Open( StringPrintf("%s/apex.apexd_test.apex", GetStagedDir(125).c_str())); ASSERT_THAT(*result, UnorderedElementsAre(ApexFileEq(ByRef(*child_apex_file_1)), ApexFileEq(ByRef(*child_apex_file_2)))); } // A test fixture to use for tests that mount/unmount apexes. class ApexdMountTest : public ApexdUnitTest { public: void UnmountOnTearDown(const std::string& apex_file) { to_unmount_.push_back(apex_file); } protected: void SetUp() final { ApexdUnitTest::SetUp(); GetApexDatabaseForTesting().Reset(); GetChangedActiveApexesForTesting().clear(); ASSERT_THAT(SetUpApexTestEnvironment(), Ok()); } void TearDown() final { ApexdUnitTest::TearDown(); for (const auto& apex : to_unmount_) { if (auto status = DeactivatePackage(apex); !status.ok()) { LOG(ERROR) << "Failed to unmount " << apex << " : " << status.error(); } } } private: MountNamespaceRestorer restorer_; std::vector to_unmount_; }; // TODO(b/187864524): cover other negative scenarios. TEST_F(ApexdMountTest, InstallPackageRejectsApexWithoutRebootlessSupport) { std::string file_path = AddPreInstalledApex("apex.apexd_test.apex"); ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}); ASSERT_THAT(ActivatePackage(file_path), Ok()); UnmountOnTearDown(file_path); auto ret = InstallPackage(GetTestFile("apex.apexd_test.apex")); ASSERT_THAT( ret, HasError(WithMessage(HasSubstr("does not support non-staged update")))); } TEST_F(ApexdMountTest, InstallPackageRejectsNoPreInstalledApex) { auto ret = InstallPackage(GetTestFile("test.rebootless_apex_v1.apex")); ASSERT_THAT( ret, HasError(WithMessage(HasSubstr( "No active version found for package test.apex.rebootless")))); } TEST_F(ApexdMountTest, InstallPackageRejectsNoHashtree) { std::string file_path = AddPreInstalledApex("test.rebootless_apex_v1.apex"); ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}); ASSERT_THAT(ActivatePackage(file_path), Ok()); UnmountOnTearDown(file_path); auto ret = InstallPackage(GetTestFile("test.rebootless_apex_v2_no_hashtree.apex")); ASSERT_THAT( ret, HasError(WithMessage(HasSubstr(" does not have an embedded hash tree")))); } TEST_F(ApexdMountTest, InstallPackageRejectsNoActiveApex) { std::string file_path = AddPreInstalledApex("test.rebootless_apex_v1.apex"); ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}); auto ret = InstallPackage(GetTestFile("test.rebootless_apex_v2.apex")); ASSERT_THAT( ret, HasError(WithMessage(HasSubstr( "No active version found for package test.apex.rebootless")))); } TEST_F(ApexdMountTest, InstallPackageRejectsManifestMismatch) { std::string file_path = AddPreInstalledApex("test.rebootless_apex_v1.apex"); ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}); ASSERT_THAT(ActivatePackage(file_path), Ok()); UnmountOnTearDown(file_path); auto ret = InstallPackage( GetTestFile("test.rebootless_apex_manifest_mismatch.apex")); ASSERT_THAT( ret, HasError(WithMessage(HasSubstr( "Manifest inside filesystem does not match manifest outside it")))); } TEST_F(ApexdMountTest, InstallPackageRejectsCorrupted) { std::string file_path = AddPreInstalledApex("test.rebootless_apex_v1.apex"); ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}); ASSERT_THAT(ActivatePackage(file_path), Ok()); UnmountOnTearDown(file_path); auto ret = InstallPackage(GetTestFile("test.rebootless_apex_corrupted.apex")); ASSERT_THAT(ret, HasError(WithMessage(HasSubstr("Can't verify /dev/block/dm-")))); } TEST_F(ApexdMountTest, InstallPackageRejectsProvidesSharedLibs) { std::string file_path = AddPreInstalledApex("test.rebootless_apex_v1.apex"); ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}); ASSERT_THAT(ActivatePackage(file_path), Ok()); UnmountOnTearDown(file_path); auto ret = InstallPackage( GetTestFile("test.rebootless_apex_provides_sharedlibs.apex")); ASSERT_THAT(ret, HasError(WithMessage(HasSubstr(" is a shared libs APEX")))); } TEST_F(ApexdMountTest, InstallPackageRejectsProvidesNativeLibs) { std::string file_path = AddPreInstalledApex("test.rebootless_apex_v1.apex"); ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}); ASSERT_THAT(ActivatePackage(file_path), Ok()); UnmountOnTearDown(file_path); auto ret = InstallPackage( GetTestFile("test.rebootless_apex_provides_native_libs.apex")); ASSERT_THAT(ret, HasError(WithMessage(HasSubstr(" provides native libs")))); } TEST_F(ApexdMountTest, InstallPackageRejectsRequiresSharedApexLibs) { std::string file_path = AddPreInstalledApex("test.rebootless_apex_v1.apex"); ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}); ASSERT_THAT(ActivatePackage(file_path), Ok()); UnmountOnTearDown(file_path); auto ret = InstallPackage( GetTestFile("test.rebootless_apex_requires_shared_apex_libs.apex")); ASSERT_THAT(ret, HasError(WithMessage(HasSubstr(" requires shared apex libs")))); } TEST_F(ApexdMountTest, InstallPackageRejectsJniLibs) { std::string file_path = AddPreInstalledApex("test.rebootless_apex_v1.apex"); ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}); ASSERT_THAT(ActivatePackage(file_path), Ok()); UnmountOnTearDown(file_path); auto ret = InstallPackage(GetTestFile("test.rebootless_apex_jni_libs.apex")); ASSERT_THAT(ret, HasError(WithMessage(HasSubstr(" requires JNI libs")))); } TEST_F(ApexdMountTest, InstallPackageAcceptsAddRequiredNativeLib) { std::string file_path = AddPreInstalledApex("test.rebootless_apex_v1.apex"); ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}); ASSERT_THAT(ActivatePackage(file_path), Ok()); UnmountOnTearDown(file_path); auto ret = InstallPackage(GetTestFile("test.rebootless_apex_add_native_lib.apex")); ASSERT_THAT(ret, Ok()); UnmountOnTearDown(ret->GetPath()); } TEST_F(ApexdMountTest, InstallPackageAcceptsRemoveRequiredNativeLib) { std::string file_path = AddPreInstalledApex("test.rebootless_apex_v1.apex"); ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}); ASSERT_THAT(ActivatePackage(file_path), Ok()); UnmountOnTearDown(file_path); auto ret = InstallPackage( GetTestFile("test.rebootless_apex_remove_native_lib.apex")); ASSERT_THAT(ret, Ok()); UnmountOnTearDown(ret->GetPath()); } TEST_F(ApexdMountTest, InstallPackageRejectsAppInApex) { std::string file_path = AddPreInstalledApex("test.rebootless_apex_v1.apex"); ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}); ASSERT_THAT(ActivatePackage(file_path), Ok()); UnmountOnTearDown(file_path); auto ret = InstallPackage(GetTestFile("test.rebootless_apex_app_in_apex.apex")); ASSERT_THAT(ret, HasError(WithMessage(HasSubstr("contains app inside")))); } TEST_F(ApexdMountTest, InstallPackageRejectsPrivAppInApex) { std::string file_path = AddPreInstalledApex("test.rebootless_apex_v1.apex"); ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}); ASSERT_THAT(ActivatePackage(file_path), Ok()); UnmountOnTearDown(file_path); auto ret = InstallPackage(GetTestFile("test.rebootless_apex_priv_app_in_apex.apex")); ASSERT_THAT(ret, HasError(WithMessage(HasSubstr("contains priv-app inside")))); } TEST_F(ApexdMountTest, InstallPackagePreInstallVersionActive) { std::string file_path = AddPreInstalledApex("test.rebootless_apex_v1.apex"); ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}); ASSERT_THAT(ActivatePackage(file_path), Ok()); UnmountOnTearDown(file_path); { auto active_apex = GetActivePackage("test.apex.rebootless"); ASSERT_THAT(active_apex, Ok()); ASSERT_EQ(active_apex->GetPath(), file_path); } auto ret = InstallPackage(GetTestFile("test.rebootless_apex_v2.apex")); ASSERT_THAT(ret, Ok()); UnmountOnTearDown(ret->GetPath()); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/test.apex.rebootless", "/apex/test.apex.rebootless@2")); // Check that /apex/test.apex.rebootless is a bind mount of // /apex/test.apex.rebootless@2. auto manifest = ReadManifest("/apex/test.apex.rebootless/apex_manifest.pb"); ASSERT_THAT(manifest, Ok()); ASSERT_EQ(2u, manifest->version()); // Check that GetActivePackage correctly reports upgraded version. auto active_apex = GetActivePackage("test.apex.rebootless"); ASSERT_THAT(active_apex, Ok()); ASSERT_EQ(active_apex->GetPath(), ret->GetPath()); // Check that pre-installed APEX is still around ASSERT_EQ(0, access(file_path.c_str(), F_OK)) << "Can't access " << file_path << " : " << strerror(errno); auto& db = GetApexDatabaseForTesting(); // Check that upgraded APEX is mounted on top of dm-verity device. db.ForallMountedApexes( "test.apex.rebootless", [&](const MountedApexData& data, bool latest) { ASSERT_TRUE(latest); ASSERT_EQ(data.full_path, ret->GetPath()); ASSERT_EQ(data.device_name, "test.apex.rebootless@2_1"); }); } TEST_F(ApexdMountTest, InstallPackagePreInstallVersionActiveSamegrade) { std::string file_path = AddPreInstalledApex("test.rebootless_apex_v1.apex"); ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}); ASSERT_THAT(ActivatePackage(file_path), Ok()); UnmountOnTearDown(file_path); { auto active_apex = GetActivePackage("test.apex.rebootless"); ASSERT_THAT(active_apex, Ok()); ASSERT_EQ(active_apex->GetPath(), file_path); } auto ret = InstallPackage(GetTestFile("test.rebootless_apex_v1.apex")); ASSERT_THAT(ret, Ok()); UnmountOnTearDown(ret->GetPath()); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/test.apex.rebootless", "/apex/test.apex.rebootless@1")); // Check that GetActivePackage correctly reports upgraded version. auto active_apex = GetActivePackage("test.apex.rebootless"); ASSERT_THAT(active_apex, Ok()); ASSERT_EQ(active_apex->GetPath(), ret->GetPath()); // Check that pre-installed APEX is still around ASSERT_EQ(0, access(file_path.c_str(), F_OK)) << "Can't access " << file_path << " : " << strerror(errno); auto& db = GetApexDatabaseForTesting(); // Check that upgraded APEX is mounted on top of dm-verity device. db.ForallMountedApexes( "test.apex.rebootless", [&](const MountedApexData& data, bool latest) { ASSERT_TRUE(latest); ASSERT_EQ(data.full_path, ret->GetPath()); ASSERT_EQ(data.device_name, "test.apex.rebootless@1_1"); }); } TEST_F(ApexdMountTest, InstallPackageUnloadOldApex) { std::string file_path = AddPreInstalledApex("test.rebootless_apex_v1.apex"); ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}); bool unloaded = false; bool loaded = false; const std::string prop = "apex.test.apex.rebootless.ready"; std::thread monitor_apex_ready_prop([&]() { unloaded = base::WaitForProperty(prop, "false", 10s); loaded = base::WaitForProperty(prop, "true", 10s); }); ASSERT_THAT(ActivatePackage(file_path), Ok()); UnmountOnTearDown(file_path); auto ret = InstallPackage(GetTestFile("test.rebootless_apex_v2.apex")); ASSERT_THAT(ret, Ok()); UnmountOnTearDown(ret->GetPath()); monitor_apex_ready_prop.join(); ASSERT_TRUE(unloaded); ASSERT_TRUE(loaded); } TEST_F(ApexdMountTest, InstallPackageWithService) { std::string file_path = AddPreInstalledApex("test.rebootless_apex_service_v1.apex"); ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}); ASSERT_THAT(ActivatePackage(file_path), Ok()); UnmountOnTearDown(file_path); auto ret = InstallPackage(GetTestFile("test.rebootless_apex_service_v2.apex")); ASSERT_THAT(ret, Ok()); auto manifest = ReadManifest("/apex/test.apex.rebootless/apex_manifest.pb"); ASSERT_THAT(manifest, Ok()); ASSERT_EQ(2u, manifest->version()); UnmountOnTearDown(ret->GetPath()); } TEST_F(ApexdMountTest, InstallPackageDataVersionActive) { AddPreInstalledApex("test.rebootless_apex_v1.apex"); ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}); std::string file_path = AddDataApex("test.rebootless_apex_v1.apex"); ASSERT_THAT(ActivatePackage(file_path), Ok()); UnmountOnTearDown(file_path); { auto active_apex = GetActivePackage("test.apex.rebootless"); ASSERT_THAT(active_apex, Ok()); ASSERT_EQ(active_apex->GetPath(), file_path); } auto ret = InstallPackage(GetTestFile("test.rebootless_apex_v2.apex")); ASSERT_THAT(ret, Ok()); UnmountOnTearDown(ret->GetPath()); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/test.apex.rebootless", "/apex/test.apex.rebootless@2")); // Check that /apex/test.apex.rebootless is a bind mount of // /apex/test.apex.rebootless@2. auto manifest = ReadManifest("/apex/test.apex.rebootless/apex_manifest.pb"); ASSERT_THAT(manifest, Ok()); ASSERT_EQ(2u, manifest->version()); // Check that GetActivePackage correctly reports upgraded version. auto active_apex = GetActivePackage("test.apex.rebootless"); ASSERT_THAT(active_apex, Ok()); ASSERT_EQ(active_apex->GetPath(), ret->GetPath()); // Check that previously active APEX was deleted. ASSERT_EQ(-1, access(file_path.c_str(), F_OK)); ASSERT_EQ(ENOENT, errno); auto& db = GetApexDatabaseForTesting(); // Check that upgraded APEX is mounted on top of dm-verity device. db.ForallMountedApexes( "test.apex.rebootless", [&](const MountedApexData& data, bool latest) { ASSERT_TRUE(latest); ASSERT_EQ(data.full_path, ret->GetPath()); ASSERT_EQ(data.device_name, "test.apex.rebootless@2_1"); }); } TEST_F(ApexdMountTest, InstallPackageResolvesPathCollision) { AddPreInstalledApex("test.rebootless_apex_v1.apex"); ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}); std::string file_path = AddDataApex("test.rebootless_apex_v1.apex", "test.apex.rebootless@1_1.apex"); ASSERT_THAT(ActivatePackage(file_path), Ok()); UnmountOnTearDown(file_path); { auto active_apex = GetActivePackage("test.apex.rebootless"); ASSERT_THAT(active_apex, Ok()); ASSERT_EQ(active_apex->GetPath(), file_path); } auto ret = InstallPackage(GetTestFile("test.rebootless_apex_v1.apex")); ASSERT_THAT(ret, Ok()); UnmountOnTearDown(ret->GetPath()); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/test.apex.rebootless", "/apex/test.apex.rebootless@1")); // Check that /apex/test.apex.rebootless is a bind mount of // /apex/test.apex.rebootless@2. auto manifest = ReadManifest("/apex/test.apex.rebootless/apex_manifest.pb"); ASSERT_THAT(manifest, Ok()); ASSERT_EQ(1u, manifest->version()); // Check that GetActivePackage correctly reports upgraded version. auto active_apex = GetActivePackage("test.apex.rebootless"); ASSERT_THAT(active_apex, Ok()); ASSERT_EQ(active_apex->GetPath(), ret->GetPath()); // Check that we correctly resolved active apex path collision. ASSERT_EQ(active_apex->GetPath(), GetDataDir() + "/test.apex.rebootless@1_2.apex"); // Check that previously active APEX was deleted. ASSERT_EQ(-1, access(file_path.c_str(), F_OK)); ASSERT_EQ(ENOENT, errno); auto& db = GetApexDatabaseForTesting(); // Check that upgraded APEX is mounted on top of dm-verity device. db.ForallMountedApexes( "test.apex.rebootless", [&](const MountedApexData& data, bool latest) { ASSERT_TRUE(latest); ASSERT_EQ(data.full_path, ret->GetPath()); ASSERT_EQ(data.device_name, "test.apex.rebootless@1_2"); }); } TEST_F(ApexdMountTest, InstallPackageDataVersionActiveSamegrade) { AddPreInstalledApex("test.rebootless_apex_v1.apex"); ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}); std::string file_path = AddDataApex("test.rebootless_apex_v2.apex"); ASSERT_THAT(ActivatePackage(file_path), Ok()); UnmountOnTearDown(file_path); { auto active_apex = GetActivePackage("test.apex.rebootless"); ASSERT_THAT(active_apex, Ok()); ASSERT_EQ(active_apex->GetPath(), file_path); } auto ret = InstallPackage(GetTestFile("test.rebootless_apex_v2.apex")); ASSERT_THAT(ret, Ok()); UnmountOnTearDown(ret->GetPath()); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/test.apex.rebootless", "/apex/test.apex.rebootless@2")); // Check that /apex/test.apex.rebootless is a bind mount of // /apex/test.apex.rebootless@2. auto manifest = ReadManifest("/apex/test.apex.rebootless/apex_manifest.pb"); ASSERT_THAT(manifest, Ok()); ASSERT_EQ(2u, manifest->version()); // Check that GetActivePackage correctly reports upgraded version. auto active_apex = GetActivePackage("test.apex.rebootless"); ASSERT_THAT(active_apex, Ok()); ASSERT_EQ(active_apex->GetPath(), ret->GetPath()); // Check that previously active APEX was deleted. ASSERT_EQ(-1, access(file_path.c_str(), F_OK)); ASSERT_EQ(ENOENT, errno); auto& db = GetApexDatabaseForTesting(); // Check that upgraded APEX is mounted on top of dm-verity device. db.ForallMountedApexes( "test.apex.rebootless", [&](const MountedApexData& data, bool latest) { ASSERT_TRUE(latest); ASSERT_EQ(data.full_path, ret->GetPath()); ASSERT_EQ(data.device_name, "test.apex.rebootless@2_1"); }); } TEST_F(ApexdMountTest, InstallPackageUnmountFailsPreInstalledApexActive) { std::string file_path = AddPreInstalledApex("test.rebootless_apex_v1.apex"); ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}); ASSERT_THAT(ActivatePackage(file_path), Ok()); UnmountOnTearDown(file_path); { auto active_apex = GetActivePackage("test.apex.rebootless"); ASSERT_THAT(active_apex, Ok()); ASSERT_EQ(active_apex->GetPath(), file_path); } unique_fd fd(open("/apex/test.apex.rebootless/apex_manifest.pb", O_RDONLY | O_CLOEXEC)); ASSERT_NE(-1, fd.get()); auto ret = InstallPackage(GetTestFile("test.rebootless_apex_v2.apex")); ASSERT_THAT(ret, Not(Ok())); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/test.apex.rebootless", "/apex/test.apex.rebootless@1")); // Check that GetActivePackage correctly reports upgraded version. auto active_apex = GetActivePackage("test.apex.rebootless"); ASSERT_THAT(active_apex, Ok()); ASSERT_EQ(active_apex->GetPath(), file_path); // Check that old APEX is still around ASSERT_EQ(0, access(file_path.c_str(), F_OK)) << "Can't access " << file_path << " : " << strerror(errno); auto& db = GetApexDatabaseForTesting(); // Check that upgraded APEX is mounted on top of dm-verity device. db.ForallMountedApexes("test.apex.rebootless", [&](const MountedApexData& data, bool latest) { ASSERT_TRUE(latest); ASSERT_EQ(data.full_path, file_path); }); } TEST_F(ApexdMountTest, InstallPackageUnmountFailedUpdatedApexActive) { AddPreInstalledApex("test.rebootless_apex_v1.apex"); ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}); std::string file_path = AddDataApex("test.rebootless_apex_v1.apex"); ASSERT_THAT(ActivatePackage(file_path), Ok()); UnmountOnTearDown(file_path); { auto active_apex = GetActivePackage("test.apex.rebootless"); ASSERT_THAT(active_apex, Ok()); ASSERT_EQ(active_apex->GetPath(), file_path); } unique_fd fd(open("/apex/test.apex.rebootless/apex_manifest.pb", O_RDONLY | O_CLOEXEC)); ASSERT_NE(-1, fd.get()); auto ret = InstallPackage(GetTestFile("test.rebootless_apex_v2.apex")); ASSERT_THAT(ret, Not(Ok())); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/test.apex.rebootless", "/apex/test.apex.rebootless@1")); // Check that GetActivePackage correctly reports old apex. auto active_apex = GetActivePackage("test.apex.rebootless"); ASSERT_THAT(active_apex, Ok()); ASSERT_EQ(active_apex->GetPath(), file_path); // Check that old APEX is still around ASSERT_EQ(0, access(file_path.c_str(), F_OK)) << "Can't access " << file_path << " : " << strerror(errno); auto& db = GetApexDatabaseForTesting(); db.ForallMountedApexes( "test.apex.rebootless", [&](const MountedApexData& data, bool latest) { ASSERT_TRUE(latest); ASSERT_EQ(data.full_path, file_path); ASSERT_EQ(data.device_name, "test.apex.rebootless@1"); }); } TEST_F(ApexdMountTest, InstallPackageUpdatesApexInfoList) { auto apex_1 = AddPreInstalledApex("test.rebootless_apex_v1.apex"); auto apex_2 = AddPreInstalledApex("apex.apexd_test.apex"); ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}); UnmountOnTearDown(apex_1); UnmountOnTearDown(apex_2); ASSERT_THAT(ActivatePackage(apex_1), Ok()); ASSERT_THAT(ActivatePackage(apex_2), Ok()); // Call OnAllPackagesActivated to create /apex/apex-info-list.xml. OnAllPackagesActivated(/* is_bootstrap= */ false); // Check /apex/apex-info-list.xml was created. ASSERT_EQ(0, access("/apex/apex-info-list.xml", F_OK)); auto ret = InstallPackage(GetTestFile("test.rebootless_apex_v2.apex")); ASSERT_THAT(ret, Ok()); UnmountOnTearDown(ret->GetPath()); ASSERT_EQ(access("/apex/apex-info-list.xml", F_OK), 0); auto info_list = com::android::apex::readApexInfoList("/apex/apex-info-list.xml"); ASSERT_TRUE(info_list.has_value()); auto apex_info_xml_1 = com::android::apex::ApexInfo( /* moduleName= */ "test.apex.rebootless", /* modulePath= */ apex_1, /* preinstalledModulePath= */ apex_1, /* versionCode= */ 1, /* versionName= */ "1", /* isFactory= */ true, /* isActive= */ false, GetMTime(apex_1), /* provideSharedApexLibs= */ false); auto apex_info_xml_2 = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.test_package", /* modulePath= */ apex_2, /* preinstalledModulePath= */ apex_2, /* versionCode= */ 1, /* versionName= */ "1", /* isFactory= */ true, /* isActive= */ true, GetMTime(apex_2), /* provideSharedApexLibs= */ false); auto apex_info_xml_3 = com::android::apex::ApexInfo( /* moduleName= */ "test.apex.rebootless", /* modulePath= */ ret->GetPath(), /* preinstalledModulePath= */ apex_1, /* versionCode= */ 2, /* versionName= */ "2", /* isFactory= */ false, /* isActive= */ true, GetMTime(ret->GetPath()), /* provideSharedApexLibs= */ false); ASSERT_THAT(info_list->getApexInfo(), UnorderedElementsAre(ApexInfoXmlEq(apex_info_xml_1), ApexInfoXmlEq(apex_info_xml_2), ApexInfoXmlEq(apex_info_xml_3))); } TEST_F(ApexdMountTest, ActivatePackageBannedName) { auto status = ActivatePackage(GetTestFile("sharedlibs.apex")); ASSERT_THAT(status, HasError(WithMessage("Package name sharedlibs is not allowed."))); } TEST_F(ApexdMountTest, ActivatePackageNoCode) { std::string file_path = AddPreInstalledApex("apex.apexd_test_nocode.apex"); ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}); ASSERT_THAT(ActivatePackage(file_path), Ok()); UnmountOnTearDown(file_path); std::string mountinfo; ASSERT_TRUE(ReadFileToString("/proc/self/mountinfo", &mountinfo)); bool found_apex_mountpoint = false; for (const auto& line : Split(mountinfo, "\n")) { std::vector tokens = Split(line, " "); // line format: // mnt_id parent_mnt_id major:minor source target option propagation_type // ex) 33 260:19 / /apex rw,nosuid,nodev - if (tokens.size() >= 7 && tokens[4] == "/apex/com.android.apex.test_package@1") { found_apex_mountpoint = true; // Make sure that option contains noexec std::vector options = Split(tokens[5], ","); EXPECT_THAT(options, Contains("noexec")); break; } } EXPECT_TRUE(found_apex_mountpoint); } TEST_F(ApexdMountTest, ActivatePackageManifestMissmatch) { std::string file_path = AddPreInstalledApex("apex.apexd_test_manifest_mismatch.apex"); ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}); auto status = ActivatePackage(file_path); ASSERT_THAT( status, HasError(WithMessage(HasSubstr( "Manifest inside filesystem does not match manifest outside it")))); } TEST_F(ApexdMountTest, ActivatePackage) { std::string file_path = AddPreInstalledApex("apex.apexd_test.apex"); ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}); ASSERT_THAT(ActivatePackage(file_path), Ok()); UnmountOnTearDown(file_path); auto active_apex = GetActivePackage("com.android.apex.test_package"); ASSERT_THAT(active_apex, Ok()); ASSERT_EQ(active_apex->GetPath(), file_path); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.test_package", "/apex/com.android.apex.test_package@1")); ASSERT_THAT(DeactivatePackage(file_path), Ok()); ASSERT_THAT(GetActivePackage("com.android.apex.test_package"), Not(Ok())); auto new_apex_mounts = GetApexMounts(); ASSERT_EQ(new_apex_mounts.size(), 0u); } TEST_F(ApexdMountTest, ActivatePackageShowsUpInMountedApexDatabase) { std::string file_path = AddPreInstalledApex("apex.apexd_test.apex"); ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}); ASSERT_THAT(ActivatePackage(file_path), Ok()); UnmountOnTearDown(file_path); auto active_apex = GetActivePackage("com.android.apex.test_package"); ASSERT_THAT(active_apex, Ok()); ASSERT_EQ(active_apex->GetPath(), file_path); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.test_package", "/apex/com.android.apex.test_package@1")); // Check that mounted apex database contains information about our APEX. auto& db = GetApexDatabaseForTesting(); std::optional mounted_apex; db.ForallMountedApexes("com.android.apex.test_package", [&](const MountedApexData& d, bool active) { if (active) { mounted_apex.emplace(d); } }); ASSERT_TRUE(mounted_apex) << "Haven't found com.android.apex.test_package in the database of " << "mounted apexes"; } TEST_F(ApexdMountTest, ActivatePackageNoHashtree) { AddPreInstalledApex("apex.apexd_test.apex"); ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}); std::string file_path = AddDataApex("apex.apexd_test_no_hashtree.apex"); ASSERT_THAT(ActivatePackage(file_path), Ok()); UnmountOnTearDown(file_path); // Check that hashtree was generated std::string hashtree_path = GetHashTreeDir() + "/com.android.apex.test_package@1"; ASSERT_EQ(0, access(hashtree_path.c_str(), F_OK)); // Check that block device can be read. auto block_device = GetBlockDeviceForApex("com.android.apex.test_package@1"); ASSERT_THAT(block_device, Ok()); ASSERT_THAT(ReadDevice(*block_device), Ok()); } TEST_F(ApexdMountTest, ActivatePackageNoHashtreeShowsUpInMountedDatabase) { AddPreInstalledApex("apex.apexd_test.apex"); ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}); std::string file_path = AddDataApex("apex.apexd_test_no_hashtree.apex"); ASSERT_THAT(ActivatePackage(file_path), Ok()); UnmountOnTearDown(file_path); // Get loop devices that were used to mount APEX. auto children = ListChildLoopDevices("com.android.apex.test_package@1"); ASSERT_THAT(children, Ok()); ASSERT_EQ(2u, children->size()) << "Unexpected number of children: " << Join(*children, ","); auto& db = GetApexDatabaseForTesting(); std::optional mounted_apex; db.ForallMountedApexes("com.android.apex.test_package", [&](const MountedApexData& d, bool active) { if (active) { mounted_apex.emplace(d); } }); ASSERT_TRUE(mounted_apex) << "Haven't found com.android.apex.test_package@1 in the database of " << "mounted apexes"; ASSERT_EQ(file_path, mounted_apex->full_path); ASSERT_EQ("/apex/com.android.apex.test_package@1", mounted_apex->mount_point); ASSERT_EQ("com.android.apex.test_package@1", mounted_apex->device_name); // For loops we only check that both loop_name and hashtree_loop_name are // children of the top device mapper device. ASSERT_THAT(*children, Contains(mounted_apex->loop_name)); ASSERT_THAT(*children, Contains(mounted_apex->hashtree_loop_name)); ASSERT_NE(mounted_apex->loop_name, mounted_apex->hashtree_loop_name); } TEST_F(ApexdMountTest, DeactivePackageFreesLoopDevices) { AddPreInstalledApex("apex.apexd_test.apex"); ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}); std::string file_path = AddDataApex("apex.apexd_test_no_hashtree.apex"); ASSERT_THAT(ActivatePackage(file_path), Ok()); UnmountOnTearDown(file_path); // Get loop devices that were used to mount APEX. auto children = ListChildLoopDevices("com.android.apex.test_package@1"); ASSERT_THAT(children, Ok()); ASSERT_EQ(2u, children->size()) << "Unexpected number of children: " << Join(*children, ","); ASSERT_THAT(DeactivatePackage(file_path), Ok()); for (const auto& loop : *children) { struct loop_info li; unique_fd fd(TEMP_FAILURE_RETRY(open(loop.c_str(), O_RDWR | O_CLOEXEC))); EXPECT_NE(-1, fd.get()) << "Failed to open " << loop << " : " << strerror(errno); EXPECT_EQ(-1, ioctl(fd.get(), LOOP_GET_STATUS, &li)) << loop << " is still alive"; EXPECT_EQ(ENXIO, errno) << "Unexpected errno : " << strerror(errno); } } TEST_F(ApexdMountTest, NoHashtreeApexNewSessionDoesNotImpactActivePackage) { MockCheckpointInterface checkpoint_interface; checkpoint_interface.SetSupportsCheckpoint(true); InitializeVold(&checkpoint_interface); AddPreInstalledApex("apex.apexd_test_no_hashtree.apex"); ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}); std::string file_path = AddDataApex("apex.apexd_test_no_hashtree.apex"); ASSERT_THAT(ActivatePackage(file_path), Ok()); UnmountOnTearDown(file_path); ASSERT_THAT(CreateStagedSession("apex.apexd_test_no_hashtree_2.apex", 239), Ok()); auto status = SubmitStagedSession(239, {}, /* has_rollback_enabled= */ false, /* is_rollback= */ false, /* rollback_id= */ -1); ASSERT_THAT(status, Ok()); // Check that new hashtree file was created. { std::string hashtree_path = GetHashTreeDir() + "/com.android.apex.test_package@1.new"; ASSERT_THAT(PathExists(hashtree_path), HasValue(true)) << hashtree_path << " does not exist"; } // Check that active hashtree is still there. { std::string hashtree_path = GetHashTreeDir() + "/com.android.apex.test_package@1"; ASSERT_THAT(PathExists(hashtree_path), HasValue(true)) << hashtree_path << " does not exist"; } // Check that block device of active APEX can still be read. auto block_device = GetBlockDeviceForApex("com.android.apex.test_package@1"); ASSERT_THAT(block_device, Ok()); ASSERT_THAT(ReadDevice(*block_device), Ok()); } TEST_F(ApexdMountTest, NoHashtreeApexStagePackagesMovesHashtree) { MockCheckpointInterface checkpoint_interface; checkpoint_interface.SetSupportsCheckpoint(true); InitializeVold(&checkpoint_interface); AddPreInstalledApex("apex.apexd_test_no_hashtree.apex"); ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}); auto read_fn = [](const std::string& path) -> std::vector { static constexpr size_t kBufSize = 4096; std::vector buffer(kBufSize); unique_fd fd(TEMP_FAILURE_RETRY(open(path.c_str(), O_RDONLY | O_CLOEXEC))); if (fd.get() == -1) { PLOG(ERROR) << "Failed to open " << path; ADD_FAILURE(); return buffer; } if (!ReadFully(fd.get(), buffer.data(), kBufSize)) { PLOG(ERROR) << "Failed to read " << path; ADD_FAILURE(); } return buffer; }; ASSERT_THAT(CreateStagedSession("apex.apexd_test_no_hashtree_2.apex", 37), Ok()); auto status = SubmitStagedSession(37, {}, /* has_rollback_enabled= */ false, /* is_rollback= */ false, /* rollback_id= */ -1); ASSERT_THAT(status, Ok()); auto staged_apex = std::move((*status)[0]); // Check that new hashtree file was created. std::vector original_hashtree_data; { std::string hashtree_path = GetHashTreeDir() + "/com.android.apex.test_package@1.new"; ASSERT_THAT(PathExists(hashtree_path), HasValue(true)); original_hashtree_data = read_fn(hashtree_path); } ASSERT_THAT(StagePackages({staged_apex.GetPath()}), Ok()); // Check that hashtree file was moved. { std::string hashtree_path = GetHashTreeDir() + "/com.android.apex.test_package@1.new"; ASSERT_THAT(PathExists(hashtree_path), HasValue(false)); } { std::string hashtree_path = GetHashTreeDir() + "/com.android.apex.test_package@1"; ASSERT_THAT(PathExists(hashtree_path), HasValue(true)); std::vector moved_hashtree_data = read_fn(hashtree_path); ASSERT_EQ(moved_hashtree_data, original_hashtree_data); } } TEST_F(ApexdMountTest, DeactivePackageTearsDownVerityDevice) { AddPreInstalledApex("apex.apexd_test.apex"); ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}); std::string file_path = AddDataApex("apex.apexd_test_v2.apex"); ASSERT_THAT(ActivatePackage(file_path), Ok()); UnmountOnTearDown(file_path); ASSERT_THAT(DeactivatePackage(file_path), Ok()); auto& dm = DeviceMapper::Instance(); ASSERT_EQ(dm::DmDeviceState::INVALID, dm.GetState("com.android.apex.test_package@2")); } TEST_F(ApexdMountTest, ActivateDeactivateSharedLibsApex) { ASSERT_EQ(mkdir("/apex/sharedlibs", 0755), 0); ASSERT_EQ(mkdir("/apex/sharedlibs/lib", 0755), 0); ASSERT_EQ(mkdir("/apex/sharedlibs/lib64", 0755), 0); auto deleter = make_scope_guard([]() { std::error_code ec; fs::remove_all("/apex/sharedlibs", ec); if (ec) { LOG(ERROR) << "Failed to delete /apex/sharedlibs : " << ec; } }); std::string file_path = AddPreInstalledApex( "com.android.apex.test.sharedlibs_generated.v1.libvX.apex"); ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}); UnmountOnTearDown(file_path); ASSERT_THAT(ActivatePackage(file_path), Ok()); auto active_apex = GetActivePackage("com.android.apex.test.sharedlibs"); ASSERT_THAT(active_apex, Ok()); ASSERT_EQ(active_apex->GetPath(), file_path); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.test.sharedlibs@1")); ASSERT_THAT(DeactivatePackage(file_path), Ok()); ASSERT_THAT(GetActivePackage("com.android.apex.test.sharedlibs"), Not(Ok())); auto new_apex_mounts = GetApexMounts(); ASSERT_EQ(new_apex_mounts.size(), 0u); } TEST_F(ApexdMountTest, RemoveInactiveDataApex) { AddPreInstalledApex("com.android.apex.compressed.v2.capex"); // Add a decompressed apex that will not be mounted, so should be removed auto decompressed_apex = StringPrintf("%s/com.android.apex.compressed@1%s", GetDecompressionDir().c_str(), kDecompressedApexPackageSuffix); fs::copy(GetTestFile("com.android.apex.compressed.v1_original.apex"), decompressed_apex); // Add a decompressed apex that will be mounted, so should be not be removed auto active_decompressed_apex = StringPrintf( "%s/com.android.apex.compressed@2%s", GetDecompressionDir().c_str(), kDecompressedApexPackageSuffix); fs::copy(GetTestFile("com.android.apex.compressed.v2_original.apex"), active_decompressed_apex); // Apex that do not have kDecompressedApexPackageSuffix, should not be removed // from decompression_dir auto decompressed_different_suffix = StringPrintf("%s/com.android.apex.compressed@2%s", GetDecompressionDir().c_str(), kApexPackageSuffix); fs::copy(GetTestFile("com.android.apex.compressed.v2_original.apex"), decompressed_different_suffix); AddPreInstalledApex("apex.apexd_test.apex"); auto data_apex = AddDataApex("apex.apexd_test.apex"); auto active_data_apex = AddDataApex("apex.apexd_test_v2.apex"); // Activate some of the apex ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}); UnmountOnTearDown(active_decompressed_apex); UnmountOnTearDown(active_data_apex); ASSERT_THAT(ActivatePackage(active_decompressed_apex), Ok()); ASSERT_THAT(ActivatePackage(active_data_apex), Ok()); // Clean up inactive apex packages RemoveInactiveDataApex(); // Verify inactive apex packages have been deleted ASSERT_TRUE(*PathExists(active_decompressed_apex)); ASSERT_TRUE(*PathExists(active_data_apex)); ASSERT_TRUE(*PathExists(decompressed_different_suffix)); ASSERT_FALSE(*PathExists(decompressed_apex)); ASSERT_FALSE(*PathExists(data_apex)); } TEST_F(ApexdMountTest, OnOtaChrootBootstrapOnlyPreInstalledApexes) { std::string apex_path_1 = AddPreInstalledApex("apex.apexd_test.apex"); std::string apex_path_2 = AddPreInstalledApex("apex.apexd_test_different_app.apex"); ASSERT_EQ(OnOtaChrootBootstrap(), 0); UnmountOnTearDown(apex_path_1); UnmountOnTearDown(apex_path_2); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.test_package", "/apex/com.android.apex.test_package@1", "/apex/com.android.apex.test_package_2", "/apex/com.android.apex.test_package_2@1")); ASSERT_EQ(access("/apex/apex-info-list.xml", F_OK), 0); auto info_list = com::android::apex::readApexInfoList("/apex/apex-info-list.xml"); ASSERT_TRUE(info_list.has_value()); auto apex_info_xml_1 = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.test_package", /* modulePath= */ apex_path_1, /* preinstalledModulePath= */ apex_path_1, /* versionCode= */ 1, /* versionName= */ "1", /* isFactory= */ true, /* isActive= */ true, GetMTime(apex_path_1), /* provideSharedApexLibs= */ false); auto apex_info_xml_2 = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.test_package_2", /* modulePath= */ apex_path_2, /* preinstalledModulePath= */ apex_path_2, /* versionCode= */ 1, /* versionName= */ "1", /* isFactory= */ true, /* isActive= */ true, GetMTime(apex_path_2), /* provideSharedApexLibs= */ false); ASSERT_THAT(info_list->getApexInfo(), UnorderedElementsAre(ApexInfoXmlEq(apex_info_xml_1), ApexInfoXmlEq(apex_info_xml_2))); } TEST_F(ApexdMountTest, OnOtaChrootBootstrapFailsToScanPreInstalledApexes) { AddPreInstalledApex("apex.apexd_test.apex"); AddPreInstalledApex("apex.apexd_test_corrupt_superblock_apex.apex"); ASSERT_EQ(OnOtaChrootBootstrap(), 1); } TEST_F(ApexdMountTest, OnOtaChrootBootstrapDataHasHigherVersion) { std::string apex_path_1 = AddPreInstalledApex("apex.apexd_test.apex"); std::string apex_path_2 = AddPreInstalledApex("apex.apexd_test_different_app.apex"); std::string apex_path_3 = AddDataApex("apex.apexd_test_v2.apex"); ASSERT_EQ(OnOtaChrootBootstrap(), 0); UnmountOnTearDown(apex_path_2); UnmountOnTearDown(apex_path_3); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.test_package", "/apex/com.android.apex.test_package@2", "/apex/com.android.apex.test_package_2", "/apex/com.android.apex.test_package_2@1")); ASSERT_EQ(access("/apex/apex-info-list.xml", F_OK), 0); auto info_list = com::android::apex::readApexInfoList("/apex/apex-info-list.xml"); ASSERT_TRUE(info_list.has_value()); auto apex_info_xml_1 = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.test_package", /* modulePath= */ apex_path_1, /* preinstalledModulePath= */ apex_path_1, /* versionCode= */ 1, /* versionName= */ "1", /* isFactory= */ true, /* isActive= */ false, GetMTime(apex_path_1), /* provideSharedApexLibs= */ false); auto apex_info_xml_2 = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.test_package_2", /* modulePath= */ apex_path_2, /* preinstalledModulePath= */ apex_path_2, /* versionCode= */ 1, /* versionName= */ "1", /* isFactory= */ true, /* isActive= */ true, GetMTime(apex_path_2), /* provideSharedApexLibs= */ false); auto apex_info_xml_3 = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.test_package", /* modulePath= */ apex_path_3, /* preinstalledModulePath= */ apex_path_1, /* versionCode= */ 2, /* versionName= */ "2", /* isFactory= */ false, /* isActive= */ true, GetMTime(apex_path_3), /* provideSharedApexLibs= */ false); ASSERT_THAT(info_list->getApexInfo(), UnorderedElementsAre(ApexInfoXmlEq(apex_info_xml_1), ApexInfoXmlEq(apex_info_xml_2), ApexInfoXmlEq(apex_info_xml_3))); } TEST_F(ApexdMountTest, OnOtaChrootBootstrapDataHasSameVersion) { std::string apex_path_1 = AddPreInstalledApex("apex.apexd_test.apex"); std::string apex_path_2 = AddPreInstalledApex("apex.apexd_test_different_app.apex"); std::string apex_path_3 = AddDataApex("apex.apexd_test.apex"); ASSERT_EQ(OnOtaChrootBootstrap(), 0); UnmountOnTearDown(apex_path_2); UnmountOnTearDown(apex_path_3); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.test_package", "/apex/com.android.apex.test_package@1", "/apex/com.android.apex.test_package_2", "/apex/com.android.apex.test_package_2@1")); ASSERT_EQ(access("/apex/apex-info-list.xml", F_OK), 0); auto info_list = com::android::apex::readApexInfoList("/apex/apex-info-list.xml"); ASSERT_TRUE(info_list.has_value()); auto apex_info_xml_1 = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.test_package", /* modulePath= */ apex_path_1, /* preinstalledModulePath= */ apex_path_1, /* versionCode= */ 1, /* versionName= */ "1", /* isFactory= */ true, /* isActive= */ false, GetMTime(apex_path_1), /* provideSharedApexLibs= */ false); auto apex_info_xml_2 = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.test_package_2", /* modulePath= */ apex_path_2, /* preinstalledModulePath= */ apex_path_2, /* versionCode= */ 1, /* versionName= */ "1", /* isFactory= */ true, /* isActive= */ true, GetMTime(apex_path_2), /* provideSharedApexLibs= */ false); auto apex_info_xml_3 = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.test_package", /* modulePath= */ apex_path_3, /* preinstalledModulePath= */ apex_path_1, /* versionCode= */ 1, /* versionName= */ "1", /* isFactory= */ false, /* isActive= */ true, GetMTime(apex_path_3), /* provideSharedApexLibs= */ false); ASSERT_THAT(info_list->getApexInfo(), UnorderedElementsAre(ApexInfoXmlEq(apex_info_xml_1), ApexInfoXmlEq(apex_info_xml_2), ApexInfoXmlEq(apex_info_xml_3))); } TEST_F(ApexdMountTest, OnOtaChrootBootstrapSystemHasHigherVersion) { std::string apex_path_1 = AddPreInstalledApex("apex.apexd_test_v2.apex"); std::string apex_path_2 = AddPreInstalledApex("apex.apexd_test_different_app.apex"); AddDataApex("apex.apexd_test.apex"); ASSERT_EQ(OnOtaChrootBootstrap(), 0); UnmountOnTearDown(apex_path_1); UnmountOnTearDown(apex_path_2); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.test_package", "/apex/com.android.apex.test_package@2", "/apex/com.android.apex.test_package_2", "/apex/com.android.apex.test_package_2@1")); ASSERT_EQ(access("/apex/apex-info-list.xml", F_OK), 0); auto info_list = com::android::apex::readApexInfoList("/apex/apex-info-list.xml"); ASSERT_TRUE(info_list.has_value()); auto apex_info_xml_1 = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.test_package", /* modulePath= */ apex_path_1, /* preinstalledModulePath= */ apex_path_1, /* versionCode= */ 2, /* versionName= */ "2", /* isFactory= */ true, /* isActive= */ true, GetMTime(apex_path_1), /* provideSharedApexLibs= */ false); auto apex_info_xml_2 = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.test_package_2", /* modulePath= */ apex_path_2, /* preinstalledModulePath= */ apex_path_2, /* versionCode= */ 1, /* versionName= */ "1", /* isFactory= */ true, /* isActive= */ true, GetMTime(apex_path_2), /* provideSharedApexLibs= */ false); ASSERT_THAT(info_list->getApexInfo(), UnorderedElementsAre(ApexInfoXmlEq(apex_info_xml_1), ApexInfoXmlEq(apex_info_xml_2))); } TEST_F(ApexdMountTest, OnOtaChrootBootstrapDataHasSameVersionButDifferentKey) { std::string apex_path_1 = AddPreInstalledApex("apex.apexd_test.apex"); std::string apex_path_2 = AddPreInstalledApex("apex.apexd_test_different_app.apex"); AddDataApex("apex.apexd_test_different_key.apex"); ASSERT_EQ(OnOtaChrootBootstrap(), 0); UnmountOnTearDown(apex_path_1); UnmountOnTearDown(apex_path_2); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.test_package", "/apex/com.android.apex.test_package@1", "/apex/com.android.apex.test_package_2", "/apex/com.android.apex.test_package_2@1")); ASSERT_EQ(access("/apex/apex-info-list.xml", F_OK), 0); auto info_list = com::android::apex::readApexInfoList("/apex/apex-info-list.xml"); ASSERT_TRUE(info_list.has_value()); auto apex_info_xml_1 = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.test_package", /* modulePath= */ apex_path_1, /* preinstalledModulePath= */ apex_path_1, /* versionCode= */ 1, /* versionName= */ "1", /* isFactory= */ true, /* isActive= */ true, GetMTime(apex_path_1), /* provideSharedApexLibs= */ false); auto apex_info_xml_2 = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.test_package_2", /* modulePath= */ apex_path_2, /* preinstalledModulePath= */ apex_path_2, /* versionCode= */ 1, /* versionName= */ "1", /* isFactory= */ true, /* isActive= */ true, GetMTime(apex_path_2), /* provideSharedApexLibs= */ false); ASSERT_THAT(info_list->getApexInfo(), UnorderedElementsAre(ApexInfoXmlEq(apex_info_xml_1), ApexInfoXmlEq(apex_info_xml_2))); } TEST_F(ApexdMountTest, OnOtaChrootBootstrapDataHasHigherVersionButDifferentKey) { std::string apex_path_1 = AddPreInstalledApex("apex.apexd_test.apex"); std::string apex_path_2 = AddPreInstalledApex("apex.apexd_test_different_app.apex"); std::string apex_path_3 = AddDataApex("apex.apexd_test_different_key_v2.apex"); { auto apex = ApexFile::Open(apex_path_3); ASSERT_THAT(apex, Ok()); ASSERT_EQ(static_cast(apex->GetManifest().version()), 2ULL); } ASSERT_EQ(OnOtaChrootBootstrap(), 0); UnmountOnTearDown(apex_path_1); UnmountOnTearDown(apex_path_2); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.test_package", "/apex/com.android.apex.test_package@1", "/apex/com.android.apex.test_package_2", "/apex/com.android.apex.test_package_2@1")); ASSERT_EQ(access("/apex/apex-info-list.xml", F_OK), 0); auto info_list = com::android::apex::readApexInfoList("/apex/apex-info-list.xml"); ASSERT_TRUE(info_list.has_value()); auto apex_info_xml_1 = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.test_package", /* modulePath= */ apex_path_1, /* preinstalledModulePath= */ apex_path_1, /* versionCode= */ 1, /* versionName= */ "1", /* isFactory= */ true, /* isActive= */ true, GetMTime(apex_path_1), /* provideSharedApexLibs= */ false); auto apex_info_xml_2 = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.test_package_2", /* modulePath= */ apex_path_2, /* preinstalledModulePath= */ apex_path_2, /* versionCode= */ 1, /* versionName= */ "1", /* isFactory= */ true, /* isActive= */ true, GetMTime(apex_path_2), /* provideSharedApexLibs= */ false); ASSERT_THAT(info_list->getApexInfo(), UnorderedElementsAre(ApexInfoXmlEq(apex_info_xml_1), ApexInfoXmlEq(apex_info_xml_2))); } TEST_F(ApexdMountTest, OnOtaChrootBootstrapDataApexWithoutPreInstalledApex) { std::string apex_path_1 = AddPreInstalledApex("apex.apexd_test.apex"); AddDataApex("apex.apexd_test_different_app.apex"); ASSERT_EQ(OnOtaChrootBootstrap(), 0); UnmountOnTearDown(apex_path_1); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.test_package", "/apex/com.android.apex.test_package@1")); ASSERT_EQ(access("/apex/apex-info-list.xml", F_OK), 0); auto info_list = com::android::apex::readApexInfoList("/apex/apex-info-list.xml"); ASSERT_TRUE(info_list.has_value()); auto apex_info_xml_1 = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.test_package", /* modulePath= */ apex_path_1, /* preinstalledModulePath= */ apex_path_1, /* versionCode= */ 1, /* versionName= */ "1", /* isFactory= */ true, /* isActive= */ true, GetMTime(apex_path_1), /* provideSharedApexLibs= */ false); ASSERT_THAT(info_list->getApexInfo(), UnorderedElementsAre(ApexInfoXmlEq(apex_info_xml_1))); } TEST_F(ApexdMountTest, OnOtaChrootBootstrapPreInstalledSharedLibsApex) { std::string apex_path_1 = AddPreInstalledApex("apex.apexd_test.apex"); std::string apex_path_2 = AddPreInstalledApex( "com.android.apex.test.sharedlibs_generated.v1.libvX.apex"); std::string apex_path_3 = AddDataApex("apex.apexd_test_v2.apex"); ASSERT_EQ(OnOtaChrootBootstrap(), 0); UnmountOnTearDown(apex_path_2); UnmountOnTearDown(apex_path_3); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.test_package", "/apex/com.android.apex.test_package@2", "/apex/com.android.apex.test.sharedlibs@1")); ASSERT_EQ(access("/apex/apex-info-list.xml", F_OK), 0); auto info_list = com::android::apex::readApexInfoList("/apex/apex-info-list.xml"); ASSERT_TRUE(info_list.has_value()); auto apex_info_xml_1 = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.test_package", /* modulePath= */ apex_path_1, /* preinstalledModulePath= */ apex_path_1, /* versionCode= */ 1, /* versionName= */ "1", /* isFactory= */ true, /* isActive= */ false, GetMTime(apex_path_1), /* provideSharedApexLibs= */ false); auto apex_info_xml_2 = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.test.sharedlibs", /* modulePath= */ apex_path_2, /* preinstalledModulePath= */ apex_path_2, /* versionCode= */ 1, /* versionName= */ "1", /* isFactory= */ true, /* isActive= */ true, GetMTime(apex_path_2), /* provideSharedApexLibs= */ false); auto apex_info_xml_3 = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.test_package", /* modulePath= */ apex_path_3, /* preinstalledModulePath= */ apex_path_1, /* versionCode= */ 2, /* versionName= */ "2", /* isFactory= */ false, /* isActive= */ true, GetMTime(apex_path_3), /* provideSharedApexLibs= */ false); ASSERT_THAT(info_list->getApexInfo(), UnorderedElementsAre(ApexInfoXmlEq(apex_info_xml_1), ApexInfoXmlEq(apex_info_xml_2), ApexInfoXmlEq(apex_info_xml_3))); ASSERT_EQ(access("/apex/sharedlibs", F_OK), 0); // Check /apex/sharedlibs is populated properly. std::vector sharedlibs; for (const auto& p : fs::recursive_directory_iterator("/apex/sharedlibs")) { if (fs::is_symlink(p)) { auto src = fs::read_symlink(p.path()); ASSERT_EQ(p.path().filename(), src.filename()); sharedlibs.push_back(p.path().parent_path().string() + "->" + src.parent_path().string()); } } std::vector expected = { "/apex/sharedlibs/lib/libsharedlibtest.so->" "/apex/com.android.apex.test.sharedlibs@1/lib/libsharedlibtest.so", "/apex/sharedlibs/lib/libc++.so->" "/apex/com.android.apex.test.sharedlibs@1/lib/libc++.so", }; // On 64bit devices we also have lib64. if (!GetProperty("ro.product.cpu.abilist64", "").empty()) { expected.push_back( "/apex/sharedlibs/lib64/libsharedlibtest.so->" "/apex/com.android.apex.test.sharedlibs@1/lib64/libsharedlibtest.so"); expected.push_back( "/apex/sharedlibs/lib64/libc++.so->" "/apex/com.android.apex.test.sharedlibs@1/lib64/libc++.so"); } ASSERT_THAT(sharedlibs, UnorderedElementsAreArray(expected)); } TEST_F(ApexdMountTest, OnOtaChrootBootstrapSharedLibsApexBothVersions) { std::string apex_path_1 = AddPreInstalledApex("apex.apexd_test.apex"); std::string apex_path_2 = AddPreInstalledApex( "com.android.apex.test.sharedlibs_generated.v1.libvX.apex"); std::string apex_path_3 = AddDataApex("apex.apexd_test_v2.apex"); std::string apex_path_4 = AddDataApex("com.android.apex.test.sharedlibs_generated.v2.libvY.apex"); ASSERT_EQ(OnOtaChrootBootstrap(), 0); UnmountOnTearDown(apex_path_2); UnmountOnTearDown(apex_path_3); UnmountOnTearDown(apex_path_4); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.test_package", "/apex/com.android.apex.test_package@2", "/apex/com.android.apex.test.sharedlibs@1", "/apex/com.android.apex.test.sharedlibs@2")); ASSERT_EQ(access("/apex/apex-info-list.xml", F_OK), 0); auto info_list = com::android::apex::readApexInfoList("/apex/apex-info-list.xml"); ASSERT_TRUE(info_list.has_value()); auto apex_info_xml_1 = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.test_package", /* modulePath= */ apex_path_1, /* preinstalledModulePath= */ apex_path_1, /* versionCode= */ 1, /* versionName= */ "1", /* isFactory= */ true, /* isActive= */ false, GetMTime(apex_path_1), /* provideSharedApexLibs= */ false); auto apex_info_xml_2 = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.test.sharedlibs", /* modulePath= */ apex_path_2, /* preinstalledModulePath= */ apex_path_2, /* versionCode= */ 1, /* versionName= */ "1", /* isFactory= */ true, /* isActive= */ false, GetMTime(apex_path_2), /* provideSharedApexLibs= */ false); auto apex_info_xml_3 = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.test_package", /* modulePath= */ apex_path_3, /* preinstalledModulePath= */ apex_path_1, /* versionCode= */ 2, /* versionName= */ "2", /* isFactory= */ false, /* isActive= */ true, GetMTime(apex_path_3), /* provideSharedApexLibs= */ false); auto apex_info_xml_4 = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.test.sharedlibs", /* modulePath= */ apex_path_4, /* preinstalledModulePath= */ apex_path_2, /* versionCode= */ 2, /* versionName= */ "2", /* isFactory= */ false, /* isActive= */ true, GetMTime(apex_path_4), /* provideSharedApexLibs= */ false); ASSERT_THAT(info_list->getApexInfo(), UnorderedElementsAre(ApexInfoXmlEq(apex_info_xml_1), ApexInfoXmlEq(apex_info_xml_2), ApexInfoXmlEq(apex_info_xml_3), ApexInfoXmlEq(apex_info_xml_4))); ASSERT_EQ(access("/apex/sharedlibs", F_OK), 0); // Check /apex/sharedlibs is populated properly. // Because we don't want to hardcode full paths (they are pretty long and have // a hash in them which might change if new prebuilts are dropped in), the // assertion logic is a little bit clunky. std::vector sharedlibs; for (const auto& p : fs::recursive_directory_iterator("/apex/sharedlibs")) { if (fs::is_symlink(p)) { auto src = fs::read_symlink(p.path()); ASSERT_EQ(p.path().filename(), src.filename()); sharedlibs.push_back(p.path().parent_path().string() + "->" + src.parent_path().string()); } } std::vector expected = { "/apex/sharedlibs/lib/libsharedlibtest.so->" "/apex/com.android.apex.test.sharedlibs@2/lib/libsharedlibtest.so", "/apex/sharedlibs/lib/libsharedlibtest.so->" "/apex/com.android.apex.test.sharedlibs@1/lib/libsharedlibtest.so", "/apex/sharedlibs/lib/libc++.so->" "/apex/com.android.apex.test.sharedlibs@2/lib/libc++.so", }; // On 64bit devices we also have lib64. if (!GetProperty("ro.product.cpu.abilist64", "").empty()) { expected.push_back( "/apex/sharedlibs/lib64/libsharedlibtest.so->" "/apex/com.android.apex.test.sharedlibs@2/lib64/libsharedlibtest.so"); expected.push_back( "/apex/sharedlibs/lib64/libsharedlibtest.so->" "/apex/com.android.apex.test.sharedlibs@1/lib64/libsharedlibtest.so"); expected.push_back( "/apex/sharedlibs/lib64/libc++.so->" "/apex/com.android.apex.test.sharedlibs@2/lib64/libc++.so"); } ASSERT_THAT(sharedlibs, UnorderedElementsAreArray(expected)); } // Test when we move from uncompressed APEX to CAPEX via ota TEST_F(ApexdMountTest, OnOtaChrootBootstrapOnlyCompressedApexes) { std::string apex_path = AddPreInstalledApex("com.android.apex.compressed.v1.capex"); ASSERT_EQ(OnOtaChrootBootstrap(), 0); // Decompressed APEX should be mounted from decompression_dir std::string decompressed_apex = StringPrintf("%s/com.android.apex.compressed@1%s", GetDecompressionDir().c_str(), kOtaApexPackageSuffix); UnmountOnTearDown(decompressed_apex); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.compressed", "/apex/com.android.apex.compressed@1")); ASSERT_EQ(access("/apex/apex-info-list.xml", F_OK), 0); auto info_list = com::android::apex::readApexInfoList("/apex/apex-info-list.xml"); ASSERT_TRUE(info_list.has_value()); auto apex_info_xml_decompressed = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.compressed", /* modulePath= */ decompressed_apex, /* preinstalledModulePath= */ apex_path, /* versionCode= */ 1, /* versionName= */ "1", /* isFactory= */ true, /* isActive= */ true, GetMTime(decompressed_apex), /* provideSharedApexLibs= */ false); ASSERT_THAT(info_list->getApexInfo(), UnorderedElementsAre(ApexInfoXmlEq(apex_info_xml_decompressed))); auto& db = GetApexDatabaseForTesting(); // Check that it was mounted from decompressed apex. It should also be mounted // on dm-verity device. db.ForallMountedApexes("com.android.apex.compressed", [&](const MountedApexData& data, bool latest) { ASSERT_TRUE(latest); ASSERT_EQ(data.full_path, decompressed_apex); ASSERT_EQ(data.device_name, "com.android.apex.compressed@1.chroot"); }); } // Test we decompress only once even if OnOtaChrootBootstrap is called multiple // times TEST_F(ApexdMountTest, OnOtaChrootBootstrapDecompressOnlyOnceMultipleCalls) { std::string apex_path = AddPreInstalledApex("com.android.apex.compressed.v1.capex"); ASSERT_EQ(OnOtaChrootBootstrap(), 0); // Decompressed OTA APEX should be mounted std::string decompressed_ota_apex = StringPrintf("%s/com.android.apex.compressed@1%s", GetDecompressionDir().c_str(), kOtaApexPackageSuffix); UnmountOnTearDown(decompressed_ota_apex); // Capture the creation time of the OTA APEX std::error_code ec; auto last_write_time_1 = fs::last_write_time(decompressed_ota_apex, ec); ASSERT_FALSE(ec) << "Failed to capture last write time of " << decompressed_ota_apex; // Call OnOtaChrootBootstrap again. Since we do not hardlink decompressed APEX // to /data/apex/active directory when in chroot, when selecting apex for // activation, we will end up selecting compressed APEX again. ASSERT_EQ(OnOtaChrootBootstrap(), 0); // Compare write time to ensure we did not decompress again auto last_write_time_2 = fs::last_write_time(decompressed_ota_apex, ec); ASSERT_FALSE(ec) << "Failed to capture last write time of " << decompressed_ota_apex << ec.message(); ASSERT_EQ(last_write_time_1, last_write_time_2); } // Test when we upgrade existing CAPEX to higher version via OTA TEST_F(ApexdMountTest, OnOtaChrootBootstrapUpgradeCapex) { TemporaryDir previous_built_in_dir; PrepareCompressedApex("com.android.apex.compressed.v1.capex", previous_built_in_dir.path); // Place a higher version capex in current built_in_dir std::string apex_path = AddPreInstalledApex("com.android.apex.compressed.v2.capex"); ASSERT_EQ(OnOtaChrootBootstrap(), 0); // Upgraded decompressed APEX should be mounted from decompression dir std::string decompressed_active_apex = StringPrintf("%s/com.android.apex.compressed@2%s", GetDecompressionDir().c_str(), kOtaApexPackageSuffix); UnmountOnTearDown(decompressed_active_apex); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.compressed", "/apex/com.android.apex.compressed@2")); ASSERT_EQ(access("/apex/apex-info-list.xml", F_OK), 0); auto info_list = com::android::apex::readApexInfoList("/apex/apex-info-list.xml"); ASSERT_TRUE(info_list.has_value()); auto apex_info_xml_decompressed = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.compressed", /* modulePath= */ decompressed_active_apex, /* preinstalledModulePath= */ apex_path, /* versionCode= */ 2, /* versionName= */ "2", /* isFactory= */ true, /* isActive= */ true, GetMTime(decompressed_active_apex), /* provideSharedApexLibs= */ false); ASSERT_THAT(info_list->getApexInfo(), UnorderedElementsAre(ApexInfoXmlEq(apex_info_xml_decompressed))); auto& db = GetApexDatabaseForTesting(); // Check that it was mounted from decompressed apex. It should also be mounted // on dm-verity device. db.ForallMountedApexes("com.android.apex.compressed", [&](const MountedApexData& data, bool latest) { ASSERT_TRUE(latest); ASSERT_EQ(data.full_path, decompressed_active_apex); ASSERT_EQ(data.device_name, "com.android.apex.compressed@2.chroot"); }); } // Test when we update existing CAPEX to same version via OTA TEST_F(ApexdMountTest, OnOtaChrootBootstrapSamegradeCapex) { TemporaryDir previous_built_in_dir; PrepareCompressedApex("com.android.apex.compressed.v1.capex", previous_built_in_dir.path); // Place a same version capex in current built_in_dir, under a different name auto apex_path = StringPrintf("%s/different-name.capex", GetBuiltInDir().c_str()); fs::copy(GetTestFile("com.android.apex.compressed.v1.capex"), apex_path); ASSERT_EQ(OnOtaChrootBootstrap(), 0); // Previously decompressed APEX should be mounted from decompression_dir std::string decompressed_active_apex = StringPrintf( "%s/com.android.apex.compressed@1%s", GetDecompressionDir().c_str(), kDecompressedApexPackageSuffix); UnmountOnTearDown(decompressed_active_apex); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.compressed", "/apex/com.android.apex.compressed@1")); ASSERT_EQ(access("/apex/apex-info-list.xml", F_OK), 0); auto info_list = com::android::apex::readApexInfoList("/apex/apex-info-list.xml"); ASSERT_TRUE(info_list.has_value()); auto apex_info_xml_decompressed = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.compressed", /* modulePath= */ decompressed_active_apex, /* preinstalledModulePath= */ apex_path, /* versionCode= */ 1, /* versionName= */ "1", /* isFactory= */ true, /* isActive= */ true, GetMTime(decompressed_active_apex), /* provideSharedApexLibs= */ false); ASSERT_THAT(info_list->getApexInfo(), UnorderedElementsAre(ApexInfoXmlEq(apex_info_xml_decompressed))); auto& db = GetApexDatabaseForTesting(); // Check that it was mounted from decompressed apex. It should also be mounted // on dm-verity device. db.ForallMountedApexes("com.android.apex.compressed", [&](const MountedApexData& data, bool latest) { ASSERT_TRUE(latest); ASSERT_EQ(data.full_path, decompressed_active_apex); ASSERT_EQ(data.device_name, "com.android.apex.compressed@1.chroot"); }); } // Test when we update existing CAPEX to same version, but different digest TEST_F(ApexdMountTest, OnOtaChrootBootstrapSamegradeCapexDifferentDigest) { TemporaryDir previous_built_in_dir; auto different_digest_apex_path = PrepareCompressedApex( "com.android.apex.compressed.v1_different_digest.capex", previous_built_in_dir.path); // Place a same version capex in current built_in_dir, which has different // digest auto apex_path = AddPreInstalledApex("com.android.apex.compressed.v1.capex"); ASSERT_EQ(OnOtaChrootBootstrap(), 0); // New decompressed ota APEX should be mounted with kOtaApexPackageSuffix std::string decompressed_ota_apex = StringPrintf("%s/com.android.apex.compressed@1%s", GetDecompressionDir().c_str(), kOtaApexPackageSuffix); UnmountOnTearDown(decompressed_ota_apex); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.compressed", "/apex/com.android.apex.compressed@1")); ASSERT_EQ(access("/apex/apex-info-list.xml", F_OK), 0); auto info_list = com::android::apex::readApexInfoList("/apex/apex-info-list.xml"); ASSERT_TRUE(info_list.has_value()); auto apex_info_xml_decompressed = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.compressed", /* modulePath= */ decompressed_ota_apex, /* preinstalledModulePath= */ apex_path, /* versionCode= */ 1, /* versionName= */ "1", /* isFactory= */ true, /* isActive= */ true, GetMTime(decompressed_ota_apex), /* provideSharedApexLibs= */ false); ASSERT_THAT(info_list->getApexInfo(), UnorderedElementsAre(ApexInfoXmlEq(apex_info_xml_decompressed))); auto& db = GetApexDatabaseForTesting(); // Check that it was mounted from decompressed apex. It should also be mounted // on dm-verity device. db.ForallMountedApexes("com.android.apex.compressed", [&](const MountedApexData& data, bool latest) { ASSERT_TRUE(latest); ASSERT_EQ(data.full_path, decompressed_ota_apex); ASSERT_EQ(data.device_name, "com.android.apex.compressed@1.chroot"); }); // Ensure decompressed apex has same digest as pre-installed auto pre_installed_apex = ApexFile::Open(apex_path); auto decompressed_apex = ApexFile::Open(decompressed_ota_apex); auto different_digest_apex = ApexFile::Open(different_digest_apex_path); ASSERT_EQ( pre_installed_apex->GetManifest().capexmetadata().originalapexdigest(), GetRootDigest(*decompressed_apex)); ASSERT_NE( pre_installed_apex->GetManifest().capexmetadata().originalapexdigest(), GetRootDigest(*different_digest_apex)); // Ensure we didn't remove previous decompressed APEX std::string previous_decompressed_apex = StringPrintf( "%s/com.android.apex.compressed@1%s", GetDecompressionDir().c_str(), kDecompressedApexPackageSuffix); auto path_exists = PathExists(previous_decompressed_apex); ASSERT_TRUE(*path_exists); } // Test when we update existing CAPEX to same version, but different key via OTA TEST_F(ApexdMountTest, OnOtaChrootBootstrapSamegradeCapexDifferentKey) { TemporaryDir previous_built_in_dir; PrepareCompressedApex("com.android.apex.compressed_different_key.capex", previous_built_in_dir.path); // Place a same version capex in current built_in_dir, which has different key auto apex_path = AddPreInstalledApex("com.android.apex.compressed.v1.capex"); ASSERT_EQ(OnOtaChrootBootstrap(), 0); // New decompressed APEX should be mounted from ota_reserved directory std::string decompressed_active_apex = StringPrintf("%s/com.android.apex.compressed@1%s", GetDecompressionDir().c_str(), kOtaApexPackageSuffix); UnmountOnTearDown(decompressed_active_apex); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.compressed", "/apex/com.android.apex.compressed@1")); ASSERT_EQ(access("/apex/apex-info-list.xml", F_OK), 0); auto info_list = com::android::apex::readApexInfoList("/apex/apex-info-list.xml"); ASSERT_TRUE(info_list.has_value()); auto apex_info_xml_decompressed = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.compressed", /* modulePath= */ decompressed_active_apex, /* preinstalledModulePath= */ apex_path, /* versionCode= */ 1, /* versionName= */ "1", /* isFactory= */ true, /* isActive= */ true, GetMTime(decompressed_active_apex), /* provideSharedApexLibs= */ false); ASSERT_THAT(info_list->getApexInfo(), UnorderedElementsAre(ApexInfoXmlEq(apex_info_xml_decompressed))); auto& db = GetApexDatabaseForTesting(); // Check that it was mounted from decompressed apex. It should also be mounted // on dm-verity device. db.ForallMountedApexes("com.android.apex.compressed", [&](const MountedApexData& data, bool latest) { ASSERT_TRUE(latest); ASSERT_EQ(data.full_path, decompressed_active_apex); ASSERT_EQ(data.device_name, "com.android.apex.compressed@1.chroot"); }); } // Test when we remove CAPEX via OTA TEST_F(ApexdMountTest, OnOtaChrootBootstrapCapexToApex) { TemporaryDir previous_built_in_dir; PrepareCompressedApex("com.android.apex.compressed.v1.capex", previous_built_in_dir.path); // Place a uncompressed version apex in current built_in_dir std::string apex_path = AddPreInstalledApex("com.android.apex.compressed.v1_original.apex"); ASSERT_EQ(OnOtaChrootBootstrap(), 0); // New uncompressed APEX should be mounted UnmountOnTearDown(apex_path); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.compressed", "/apex/com.android.apex.compressed@1")); ASSERT_EQ(access("/apex/apex-info-list.xml", F_OK), 0); auto info_list = com::android::apex::readApexInfoList("/apex/apex-info-list.xml"); ASSERT_TRUE(info_list.has_value()); auto apex_info_xml_uncompressed = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.compressed", /* modulePath= */ apex_path, /* preinstalledModulePath= */ apex_path, /* versionCode= */ 1, /* versionName= */ "1", /* isFactory= */ true, /* isActive= */ true, GetMTime(apex_path), /* provideSharedApexLibs= */ false); ASSERT_THAT(info_list->getApexInfo(), UnorderedElementsAre(ApexInfoXmlEq(apex_info_xml_uncompressed))); } TEST_F(ApexdMountTest, OnOtaChrootBootstrapDecompressedApexVersionDifferentThanCapex) { TemporaryDir previous_built_in_dir; PrepareCompressedApex("com.android.apex.compressed.v2.capex", previous_built_in_dir.path); // Place a lower version capex in current built_in_dir, so that previously // decompressed APEX has higher version but still doesn't get picked during // selection. std::string apex_path = AddPreInstalledApex("com.android.apex.compressed.v1.capex"); ASSERT_EQ(OnOtaChrootBootstrap(), 0); // Pre-installed CAPEX should be decompressed again and mounted from // decompression_dir std::string decompressed_active_apex = StringPrintf("%s/com.android.apex.compressed@1%s", GetDecompressionDir().c_str(), kOtaApexPackageSuffix); UnmountOnTearDown(decompressed_active_apex); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.compressed", "/apex/com.android.apex.compressed@1")); ASSERT_EQ(access("/apex/apex-info-list.xml", F_OK), 0); auto info_list = com::android::apex::readApexInfoList("/apex/apex-info-list.xml"); ASSERT_TRUE(info_list.has_value()); auto apex_info_xml_decompressed = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.compressed", /* modulePath= */ decompressed_active_apex, /* preinstalledModulePath= */ apex_path, /* versionCode= */ 1, /* versionName= */ "1", /* isFactory= */ true, /* isActive= */ true, GetMTime(decompressed_active_apex), /* provideSharedApexLibs= */ false); ASSERT_THAT(info_list->getApexInfo(), UnorderedElementsAre(ApexInfoXmlEq(apex_info_xml_decompressed))); } // Test when we update CAPEX and there is a higher version present in data TEST_F(ApexdMountTest, OnOtaChrootBootstrapDataHigherThanCapex) { auto system_apex_path = PrepareCompressedApex("com.android.apex.compressed.v1.capex"); auto data_apex_path = AddDataApex("com.android.apex.compressed.v2_original.apex"); ASSERT_EQ(OnOtaChrootBootstrap(), 0); // Data APEX should be mounted UnmountOnTearDown(data_apex_path); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.compressed", "/apex/com.android.apex.compressed@2")); ASSERT_EQ(access("/apex/apex-info-list.xml", F_OK), 0); auto info_list = com::android::apex::readApexInfoList("/apex/apex-info-list.xml"); ASSERT_TRUE(info_list.has_value()); auto apex_info_xml_data = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.compressed", /* modulePath= */ data_apex_path, /* preinstalledModulePath= */ system_apex_path, /* versionCode= */ 2, /* versionName= */ "2", /* isFactory= */ false, /* isActive= */ true, GetMTime(data_apex_path), /* provideSharedApexLibs= */ false); auto apex_info_xml_system = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.compressed", /* modulePath= */ system_apex_path, /* preinstalledModulePath= */ system_apex_path, /* versionCode= */ 1, /* versionName= */ "1", /* isFactory= */ true, /* isActive= */ false, GetMTime(system_apex_path), /* provideSharedApexLibs= */ false); ASSERT_THAT(info_list->getApexInfo(), UnorderedElementsAre(ApexInfoXmlEq(apex_info_xml_data), ApexInfoXmlEq(apex_info_xml_system))); auto& db = GetApexDatabaseForTesting(); // Check that it was mounted from decompressed apex. It should also be mounted // on dm-verity device. db.ForallMountedApexes("com.android.apex.compressed", [&](const MountedApexData& data, bool latest) { ASSERT_TRUE(latest); ASSERT_EQ(data.full_path, data_apex_path); ASSERT_EQ(data.device_name, "com.android.apex.compressed@2.chroot"); }); } // Test when we update CAPEX and there is a lower version present in data TEST_F(ApexdMountTest, OnOtaChrootBootstrapDataLowerThanCapex) { auto apex_path = AddPreInstalledApex("com.android.apex.compressed.v2.capex"); AddDataApex("com.android.apex.compressed.v1_original.apex"); ASSERT_EQ(OnOtaChrootBootstrap(), 0); // Decompressed APEX should be mounted from reserved dir std::string decompressed_active_apex = StringPrintf("%s/com.android.apex.compressed@2%s", GetDecompressionDir().c_str(), kOtaApexPackageSuffix); UnmountOnTearDown(decompressed_active_apex); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.compressed", "/apex/com.android.apex.compressed@2")); ASSERT_EQ(access("/apex/apex-info-list.xml", F_OK), 0); auto info_list = com::android::apex::readApexInfoList("/apex/apex-info-list.xml"); ASSERT_TRUE(info_list.has_value()); auto apex_info_xml = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.compressed", /* modulePath= */ decompressed_active_apex, /* preinstalledModulePath= */ apex_path, /* versionCode= */ 2, /* versionName= */ "2", /* isFactory= */ true, /* isActive= */ true, GetMTime(decompressed_active_apex), /* provideSharedApexLibs= */ false); ASSERT_THAT(info_list->getApexInfo(), UnorderedElementsAre(ApexInfoXmlEq(apex_info_xml))); auto& db = GetApexDatabaseForTesting(); // Check that it was mounted from decompressed apex. It should also be mounted // on dm-verity device. db.ForallMountedApexes("com.android.apex.compressed", [&](const MountedApexData& data, bool latest) { ASSERT_TRUE(latest); ASSERT_EQ(data.full_path, decompressed_active_apex); ASSERT_EQ(data.device_name, "com.android.apex.compressed@2.chroot"); }); } // Test when we update CAPEX and there is a same version present in data TEST_F(ApexdMountTest, OnOtaChrootBootstrapDataSameAsCapex) { auto system_apex_path = PrepareCompressedApex("com.android.apex.compressed.v1.capex"); auto data_apex_path = AddDataApex("com.android.apex.compressed.v1_original.apex"); ASSERT_EQ(OnOtaChrootBootstrap(), 0); // Data APEX should be mounted UnmountOnTearDown(data_apex_path); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.compressed", "/apex/com.android.apex.compressed@1")); ASSERT_EQ(access("/apex/apex-info-list.xml", F_OK), 0); auto info_list = com::android::apex::readApexInfoList("/apex/apex-info-list.xml"); ASSERT_TRUE(info_list.has_value()); auto apex_info_xml_data = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.compressed", /* modulePath= */ data_apex_path, /* preinstalledModulePath= */ system_apex_path, /* versionCode= */ 1, /* versionName= */ "1", /* isFactory= */ false, /* isActive= */ true, GetMTime(data_apex_path), /* provideSharedApexLibs= */ false); auto apex_info_xml_system = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.compressed", /* modulePath= */ system_apex_path, /* preinstalledModulePath= */ system_apex_path, /* versionCode= */ 1, /* versionName= */ "1", /* isFactory= */ true, /* isActive= */ false, GetMTime(system_apex_path), /* provideSharedApexLibs= */ false); ASSERT_THAT(info_list->getApexInfo(), UnorderedElementsAre(ApexInfoXmlEq(apex_info_xml_data), ApexInfoXmlEq(apex_info_xml_system))); auto& db = GetApexDatabaseForTesting(); // Check that it was mounted from decompressed apex. It should also be mounted // on dm-verity device. db.ForallMountedApexes("com.android.apex.compressed", [&](const MountedApexData& data, bool latest) { ASSERT_TRUE(latest); ASSERT_EQ(data.full_path, data_apex_path); ASSERT_EQ(data.device_name, "com.android.apex.compressed@1.chroot"); }); } TEST_F(ApexdMountTest, OnOtaChrootBootstrapDataHasDifferentKeyThanCapex) { AddDataApex("com.android.apex.compressed_different_key.capex"); // Place a same version capex in current built_in_dir, which has different key auto apex_path = AddPreInstalledApex("com.android.apex.compressed.v1.capex"); ASSERT_EQ(OnOtaChrootBootstrap(), 0); // New decompressed APEX should be mounted from ota_reserved directory std::string decompressed_active_apex = StringPrintf("%s/com.android.apex.compressed@1%s", GetDecompressionDir().c_str(), kOtaApexPackageSuffix); UnmountOnTearDown(decompressed_active_apex); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.compressed", "/apex/com.android.apex.compressed@1")); ASSERT_EQ(access("/apex/apex-info-list.xml", F_OK), 0); auto info_list = com::android::apex::readApexInfoList("/apex/apex-info-list.xml"); ASSERT_TRUE(info_list.has_value()); auto apex_info_xml_decompressed = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.compressed", /* modulePath= */ decompressed_active_apex, /* preinstalledModulePath= */ apex_path, /* versionCode= */ 1, /* versionName= */ "1", /* isFactory= */ true, /* isActive= */ true, GetMTime(decompressed_active_apex), /* provideSharedApexLibs= */ false); ASSERT_THAT(info_list->getApexInfo(), UnorderedElementsAre(ApexInfoXmlEq(apex_info_xml_decompressed))); auto& db = GetApexDatabaseForTesting(); // Check that it was mounted from decompressed apex. It should also be mounted // on dm-verity device. db.ForallMountedApexes("com.android.apex.compressed", [&](const MountedApexData& data, bool latest) { ASSERT_TRUE(latest); ASSERT_EQ(data.full_path, decompressed_active_apex); ASSERT_EQ(data.device_name, "com.android.apex.compressed@1.chroot"); }); } static std::string GetSelinuxContext(const std::string& file) { char* ctx; if (getfilecon(file.c_str(), &ctx) < 0) { PLOG(ERROR) << "Failed to getfilecon " << file; return ""; } std::string result(ctx); freecon(ctx); return result; } TEST_F(ApexdMountTest, OnOtaChrootBootstrapSelinuxLabelsAreCorrect) { std::string apex_path_1 = AddPreInstalledApex("apex.apexd_test.apex"); std::string apex_path_2 = AddPreInstalledApex( "com.android.apex.test.sharedlibs_generated.v1.libvX.apex"); std::string apex_path_3 = AddDataApex("apex.apexd_test_v2.apex"); UnmountOnTearDown(apex_path_2); UnmountOnTearDown(apex_path_3); ASSERT_EQ(OnOtaChrootBootstrap(), 0); EXPECT_EQ(GetSelinuxContext("/apex/apex-info-list.xml"), "u:object_r:apex_info_file:s0"); EXPECT_EQ(GetSelinuxContext("/apex/sharedlibs"), "u:object_r:apex_mnt_dir:s0"); EXPECT_EQ(GetSelinuxContext("/apex/com.android.apex.test_package"), "u:object_r:system_file:s0"); EXPECT_EQ(GetSelinuxContext("/apex/com.android.apex.test_package@2"), "u:object_r:system_file:s0"); } TEST_F(ApexdMountTest, OnOtaChrootBootstrapDmDevicesHaveCorrectName) { std::string apex_path_1 = AddPreInstalledApex("apex.apexd_test.apex"); std::string apex_path_2 = AddPreInstalledApex("apex.apexd_test_different_app.apex"); std::string apex_path_3 = AddDataApex("apex.apexd_test_v2.apex"); ASSERT_EQ(OnOtaChrootBootstrap(), 0); UnmountOnTearDown(apex_path_2); UnmountOnTearDown(apex_path_3); MountedApexDatabase& db = GetApexDatabaseForTesting(); // com.android.apex.test_package_2 should be mounted directly on top of loop // device. db.ForallMountedApexes("com.android.apex.test_package_2", [&](const MountedApexData& data, bool latest) { ASSERT_TRUE(latest); ASSERT_THAT(data.device_name, IsEmpty()); ASSERT_THAT(data.loop_name, StartsWith("/dev")); }); // com.android.apex.test_package should be mounted on top of dm-verity device. db.ForallMountedApexes("com.android.apex.test_package", [&](const MountedApexData& data, bool latest) { ASSERT_TRUE(latest); ASSERT_EQ(data.device_name, "com.android.apex.test_package@2.chroot"); ASSERT_THAT(data.loop_name, StartsWith("/dev")); }); } TEST_F(ApexdMountTest, OnOtaChrootBootstrapFailsToActivatePreInstalledApexKeepsGoing) { std::string apex_path_1 = AddPreInstalledApex("apex.apexd_test_manifest_mismatch.apex"); std::string apex_path_2 = AddPreInstalledApex("apex.apexd_test_different_app.apex"); ASSERT_EQ(OnOtaChrootBootstrap(), 0); UnmountOnTearDown(apex_path_2); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.test_package_2", "/apex/com.android.apex.test_package_2@1")); ASSERT_EQ(access("/apex/apex-info-list.xml", F_OK), 0); auto info_list = com::android::apex::readApexInfoList("/apex/apex-info-list.xml"); ASSERT_TRUE(info_list.has_value()); auto apex_info_xml_1 = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.test_package", /* modulePath= */ apex_path_1, /* preinstalledModulePath= */ apex_path_1, /* versionCode= */ 137, /* versionName= */ "1", /* isFactory= */ true, /* isActive= */ false, GetMTime(apex_path_1), /* provideSharedApexLibs= */ false); auto apex_info_xml_2 = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.test_package_2", /* modulePath= */ apex_path_2, /* preinstalledModulePath= */ apex_path_2, /* versionCode= */ 1, /* versionName= */ "1", /* isFactory= */ true, /* isActive= */ true, GetMTime(apex_path_2), /* provideSharedApexLibs= */ false); ASSERT_THAT(info_list->getApexInfo(), UnorderedElementsAre(ApexInfoXmlEq(apex_info_xml_1), ApexInfoXmlEq(apex_info_xml_2))); } TEST_F(ApexdMountTest, OnOtaChrootBootstrapFailsToActivateDataApexFallsBackToPreInstalled) { std::string apex_path_1 = AddPreInstalledApex("apex.apexd_test.apex"); std::string apex_path_2 = AddPreInstalledApex("apex.apexd_test_different_app.apex"); std::string apex_path_3 = AddDataApex("apex.apexd_test_manifest_mismatch.apex"); ASSERT_EQ(OnOtaChrootBootstrap(), 0); UnmountOnTearDown(apex_path_1); UnmountOnTearDown(apex_path_2); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.test_package", "/apex/com.android.apex.test_package@1", "/apex/com.android.apex.test_package_2", "/apex/com.android.apex.test_package_2@1")); ASSERT_EQ(access("/apex/apex-info-list.xml", F_OK), 0); auto info_list = com::android::apex::readApexInfoList("/apex/apex-info-list.xml"); ASSERT_TRUE(info_list.has_value()); auto apex_info_xml_1 = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.test_package", /* modulePath= */ apex_path_1, /* preinstalledModulePath= */ apex_path_1, /* versionCode= */ 1, /* versionName= */ "1", /* isFactory= */ true, /* isActive= */ true, GetMTime(apex_path_1), /* provideSharedApexLibs= */ false); auto apex_info_xml_2 = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.test_package_2", /* modulePath= */ apex_path_2, /* preinstalledModulePath= */ apex_path_2, /* versionCode= */ 1, /* versionName= */ "1", /* isFactory= */ true, /* isActive= */ true, GetMTime(apex_path_2), /* provideSharedApexLibs= */ false); ASSERT_THAT(info_list->getApexInfo(), UnorderedElementsAre(ApexInfoXmlEq(apex_info_xml_1), ApexInfoXmlEq(apex_info_xml_2))); } void PrepareFlattenedApex(const std::string& apex_dir, const std::string& apex_name, int version) { ASSERT_EQ(mkdir(apex_dir.c_str(), 0755), 0); ::apex::proto::ApexManifest manifest; manifest.set_name(apex_name); manifest.set_version(version); manifest.set_versionname(std::to_string(version)); std::string out; manifest.SerializeToString(&out); ASSERT_TRUE(WriteStringToFile(out, apex_dir + "/apex_manifest.pb")); } TEST_F(ApexdMountTest, ActivateFlattenedApex) { std::string apex_dir_1 = GetBuiltInDir() + "/com.android.apex.test_package"; std::string apex_dir_2 = GetBuiltInDir() + "/com.android.apex.test_package_2"; PrepareFlattenedApex(apex_dir_1, "com.android.apex.test_package", 2); PrepareFlattenedApex(apex_dir_2, "com.android.apex.test_package_2", 1); ASSERT_EQ(ActivateFlattenedApex(), 0); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.test_package", "/apex/com.android.apex.test_package_2")); ASSERT_EQ(access("/apex/apex-info-list.xml", F_OK), 0); ASSERT_EQ(GetSelinuxContext("/apex/apex-info-list.xml"), "u:object_r:apex_info_file:s0"); auto info_list = com::android::apex::readApexInfoList("/apex/apex-info-list.xml"); ASSERT_TRUE(info_list.has_value()); auto apex_info_xml_1 = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.test_package", /* modulePath= */ apex_dir_1, /* preinstalledModulePath= */ apex_dir_1, /* versionCode= */ 2, /* versionName= */ "2", /* isFactory= */ true, /* isActive= */ true, /* lastUpdateMillis= */ 0, /* provideSharedApexLibs= */ false); auto apex_info_xml_2 = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.test_package_2", /* modulePath= */ apex_dir_2, /* preinstalledModulePath= */ apex_dir_2, /* versionCode= */ 1, /* versionName= */ "1", /* isFactory= */ true, /* isActive= */ true, /* lastUpdateMillis= */ 0, /* provideSharedApexLibs= */ false); ASSERT_THAT(info_list->getApexInfo(), UnorderedElementsAre(ApexInfoXmlEq(apex_info_xml_1), ApexInfoXmlEq(apex_info_xml_2))); } TEST_F(ApexdMountTest, ActivateFlattenedApexShouldFailWithDuplicate) { // Two flattened APEXes with the same name PrepareFlattenedApex(GetBuiltInDir() + "/com.android.apex.test_package", "com.android.apex.test_package", 1); PrepareFlattenedApex(GetBuiltInDir() + "/com.android.apex.test_package_2", "com.android.apex.test_package", 1); CaptureStderr(); ASSERT_EQ(ActivateFlattenedApex(), 1); std::string error = GetCapturedStderr(); ASSERT_THAT(error, HasSubstr("duplicate of com.android.apex.test_package found")); } TEST_F(ApexdMountTest, ActivateFlattenedApexSupportsMultiApex) { auto apex_dir = GetBuiltInDir() + "/com.android.apex.test_package"; // Two flattened APEXes with the same name PrepareFlattenedApex(apex_dir, "com.android.apex.test_package", 1); PrepareFlattenedApex(apex_dir + "_something_else", "com.android.apex.test_package", 1); // With sysprop indicating multi-apex std::string property_prefix = "debug.apexd.test.persistprefix."; android::base::SetProperty(property_prefix + "com.android.apex.test_package", "com.android.apex.test_package"); ASSERT_EQ(ActivateFlattenedApex({property_prefix}), 0); // Succeeds // apex-info-list.xml should have original paths (realpaths) not symlinks auto info_list = com::android::apex::readApexInfoList("/apex/apex-info-list.xml"); ASSERT_TRUE(info_list.has_value()); auto apex_info = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.test_package", /* modulePath= */ apex_dir, /* preinstalledModulePath= */ apex_dir, /* versionCode= */ 1, /* versionName= */ "1", /* isFactory= */ true, /* isActive= */ true, /* lastUpdateMillis= */ 0, /* provideSharedApexLibs= */ false); ASSERT_THAT(info_list->getApexInfo(), ElementsAre(ApexInfoXmlEq(apex_info))); android::base::SetProperty(property_prefix + "com.android.apex.test_package", ""); } TEST_F(ApexdMountTest, ActivateFlattenedApexShouldHaveRealPaths) { // Prepare flattened apexes somewhere else TemporaryDir dir; auto apex_dir_1 = fmt::format("{}/apex1", dir.path); auto apex_dir_2 = fmt::format("{}/apex2", dir.path); PrepareFlattenedApex(apex_dir_1, "com.android.apex.test_package", 2); PrepareFlattenedApex(apex_dir_2, "com.android.apex.test_package_2", 1); // Symlink flattened apexes under builtin dir. auto symlink_apex_dir1 = fmt::format("{}/apex1", GetBuiltInDir()); auto symlink_apex_dir2 = fmt::format("{}/apex2", GetBuiltInDir()); ASSERT_EQ(0, symlink(apex_dir_1.c_str(), symlink_apex_dir1.c_str())); ASSERT_EQ(0, symlink(apex_dir_2.c_str(), symlink_apex_dir2.c_str())); ASSERT_EQ(ActivateFlattenedApex(), 0); // apex-info-list.xml should have original paths (realpaths) not symlinks auto info_list = com::android::apex::readApexInfoList("/apex/apex-info-list.xml"); ASSERT_TRUE(info_list.has_value()); auto apex_info_xml_1 = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.test_package", /* modulePath= */ apex_dir_1, /* preinstalledModulePath= */ apex_dir_1, /* versionCode= */ 2, /* versionName= */ "2", /* isFactory= */ true, /* isActive= */ true, /* lastUpdateMillis= */ 0, /* provideSharedApexLibs= */ false); auto apex_info_xml_2 = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.test_package_2", /* modulePath= */ apex_dir_2, /* preinstalledModulePath= */ apex_dir_2, /* versionCode= */ 1, /* versionName= */ "1", /* isFactory= */ true, /* isActive= */ true, /* lastUpdateMillis= */ 0, /* provideSharedApexLibs= */ false); ASSERT_THAT(info_list->getApexInfo(), UnorderedElementsAre(ApexInfoXmlEq(apex_info_xml_1), ApexInfoXmlEq(apex_info_xml_2))); } TEST_F(ApexdMountTest, OnStartOnlyPreInstalledApexes) { MockCheckpointInterface checkpoint_interface; // Need to call InitializeVold before calling OnStart InitializeVold(&checkpoint_interface); std::string apex_path_1 = AddPreInstalledApex("apex.apexd_test.apex"); std::string apex_path_2 = AddPreInstalledApex("apex.apexd_test_different_app.apex"); ASSERT_THAT( ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}), Ok()); OnStart(); UnmountOnTearDown(apex_path_1); UnmountOnTearDown(apex_path_2); ASSERT_EQ(GetProperty(kTestApexdStatusSysprop, ""), "starting"); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.test_package", "/apex/com.android.apex.test_package@1", "/apex/com.android.apex.test_package_2", "/apex/com.android.apex.test_package_2@1")); } TEST_F(ApexdMountTest, OnStartDataHasHigherVersion) { MockCheckpointInterface checkpoint_interface; // Need to call InitializeVold before calling OnStart InitializeVold(&checkpoint_interface); AddPreInstalledApex("apex.apexd_test.apex"); std::string apex_path_2 = AddPreInstalledApex("apex.apexd_test_different_app.apex"); std::string apex_path_3 = AddDataApex("apex.apexd_test_v2.apex"); ASSERT_THAT( ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}), Ok()); OnStart(); UnmountOnTearDown(apex_path_2); UnmountOnTearDown(apex_path_3); ASSERT_EQ(GetProperty(kTestApexdStatusSysprop, ""), "starting"); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.test_package", "/apex/com.android.apex.test_package@2", "/apex/com.android.apex.test_package_2", "/apex/com.android.apex.test_package_2@1")); } TEST_F(ApexdMountTest, OnStartDataHasWrongSHA) { MockCheckpointInterface checkpoint_interface; // Need to call InitializeVold before calling OnStart InitializeVold(&checkpoint_interface); std::string apex_path = AddPreInstalledApex("com.android.apex.cts.shim.apex"); AddDataApex("com.android.apex.cts.shim.v2_wrong_sha.apex"); ASSERT_THAT( ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}), Ok()); UnmountOnTearDown(apex_path); OnStart(); // Check system shim apex is activated instead of the data one. auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.cts.shim", "/apex/com.android.apex.cts.shim@1")); } TEST_F(ApexdMountTest, OnStartDataHasSameVersion) { MockCheckpointInterface checkpoint_interface; // Need to call InitializeVold before calling OnStart InitializeVold(&checkpoint_interface); AddPreInstalledApex("apex.apexd_test.apex"); std::string apex_path_2 = AddPreInstalledApex("apex.apexd_test_different_app.apex"); std::string apex_path_3 = AddDataApex("apex.apexd_test.apex"); ASSERT_THAT( ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}), Ok()); OnStart(); UnmountOnTearDown(apex_path_2); UnmountOnTearDown(apex_path_3); ASSERT_EQ(GetProperty(kTestApexdStatusSysprop, ""), "starting"); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.test_package", "/apex/com.android.apex.test_package@1", "/apex/com.android.apex.test_package_2", "/apex/com.android.apex.test_package_2@1")); auto& db = GetApexDatabaseForTesting(); // Check that it was mounted from data apex, not pre-installed one. db.ForallMountedApexes("com.android.apex.test_package", [&](const MountedApexData& data, bool latest) { ASSERT_TRUE(latest); ASSERT_EQ(data.full_path, apex_path_3); }); } TEST_F(ApexdMountTest, OnStartSystemHasHigherVersion) { MockCheckpointInterface checkpoint_interface; // Need to call InitializeVold before calling OnStart InitializeVold(&checkpoint_interface); std::string apex_path_1 = AddPreInstalledApex("apex.apexd_test_v2.apex"); std::string apex_path_2 = AddPreInstalledApex("apex.apexd_test_different_app.apex"); AddDataApex("apex.apexd_test.apex"); ASSERT_THAT( ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}), Ok()); OnStart(); UnmountOnTearDown(apex_path_1); UnmountOnTearDown(apex_path_2); ASSERT_EQ(GetProperty(kTestApexdStatusSysprop, ""), "starting"); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.test_package", "/apex/com.android.apex.test_package@2", "/apex/com.android.apex.test_package_2", "/apex/com.android.apex.test_package_2@1")); auto& db = GetApexDatabaseForTesting(); // Check that it was mounted from pre-installed one. db.ForallMountedApexes("com.android.apex.test_package", [&](const MountedApexData& data, bool latest) { ASSERT_TRUE(latest); ASSERT_EQ(data.full_path, apex_path_1); }); } TEST_F(ApexdMountTest, OnStartFailsToActivateApexOnDataFallsBackToBuiltIn) { MockCheckpointInterface checkpoint_interface; // Need to call InitializeVold before calling OnStart InitializeVold(&checkpoint_interface); std::string apex_path_1 = AddPreInstalledApex("apex.apexd_test.apex"); std::string apex_path_2 = AddPreInstalledApex("apex.apexd_test_different_app.apex"); AddDataApex("apex.apexd_test_manifest_mismatch.apex"); ASSERT_THAT( ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}), Ok()); OnStart(); UnmountOnTearDown(apex_path_1); UnmountOnTearDown(apex_path_2); ASSERT_EQ(GetProperty(kTestApexdStatusSysprop, ""), "starting"); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.test_package", "/apex/com.android.apex.test_package@1", "/apex/com.android.apex.test_package_2", "/apex/com.android.apex.test_package_2@1")); auto& db = GetApexDatabaseForTesting(); // Check that it was mounted from pre-installed apex. db.ForallMountedApexes("com.android.apex.test_package", [&](const MountedApexData& data, bool latest) { ASSERT_TRUE(latest); ASSERT_EQ(data.full_path, apex_path_1); }); } TEST_F(ApexdMountTest, OnStartApexOnDataHasWrongKeyFallsBackToBuiltIn) { MockCheckpointInterface checkpoint_interface; // Need to call InitializeVold before calling OnStart InitializeVold(&checkpoint_interface); std::string apex_path_1 = AddPreInstalledApex("apex.apexd_test.apex"); std::string apex_path_2 = AddPreInstalledApex("apex.apexd_test_different_app.apex"); std::string apex_path_3 = AddDataApex("apex.apexd_test_different_key_v2.apex"); { auto apex = ApexFile::Open(apex_path_3); ASSERT_THAT(apex, Ok()); ASSERT_EQ(static_cast(apex->GetManifest().version()), 2ULL); } ASSERT_THAT( ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}), Ok()); OnStart(); UnmountOnTearDown(apex_path_1); UnmountOnTearDown(apex_path_2); ASSERT_EQ(GetProperty(kTestApexdStatusSysprop, ""), "starting"); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.test_package", "/apex/com.android.apex.test_package@1", "/apex/com.android.apex.test_package_2", "/apex/com.android.apex.test_package_2@1")); auto& db = GetApexDatabaseForTesting(); // Check that it was mounted from pre-installed apex. db.ForallMountedApexes("com.android.apex.test_package", [&](const MountedApexData& data, bool latest) { ASSERT_TRUE(latest); ASSERT_EQ(data.full_path, apex_path_1); }); } TEST_F(ApexdMountTest, OnStartOnlyPreInstalledCapexes) { MockCheckpointInterface checkpoint_interface; // Need to call InitializeVold before calling OnStart InitializeVold(&checkpoint_interface); std::string apex_path_1 = AddPreInstalledApex("com.android.apex.compressed.v1.capex"); ASSERT_THAT( ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}), Ok()); OnStart(); // Decompressed APEX should be mounted std::string decompressed_active_apex = StringPrintf( "%s/com.android.apex.compressed@1%s", GetDecompressionDir().c_str(), kDecompressedApexPackageSuffix); UnmountOnTearDown(decompressed_active_apex); ASSERT_EQ(GetProperty(kTestApexdStatusSysprop, ""), "starting"); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.compressed", "/apex/com.android.apex.compressed@1")); auto& db = GetApexDatabaseForTesting(); // Check that it was mounted from decompressed apex. db.ForallMountedApexes("com.android.apex.compressed", [&](const MountedApexData& data, bool latest) { ASSERT_TRUE(latest); ASSERT_EQ(data.full_path, decompressed_active_apex); ASSERT_EQ(data.device_name, "com.android.apex.compressed"); }); } TEST_F(ApexdMountTest, OnStartDataHasHigherVersionThanCapex) { MockCheckpointInterface checkpoint_interface; // Need to call InitializeVold before calling OnStart InitializeVold(&checkpoint_interface); AddPreInstalledApex("com.android.apex.compressed.v1.capex"); std::string apex_path_2 = AddDataApex("com.android.apex.compressed.v2_original.apex"); ASSERT_THAT( ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}), Ok()); OnStart(); UnmountOnTearDown(apex_path_2); ASSERT_EQ(GetProperty(kTestApexdStatusSysprop, ""), "starting"); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.compressed", "/apex/com.android.apex.compressed@2")); auto& db = GetApexDatabaseForTesting(); // Check that it was mounted from data apex. db.ForallMountedApexes("com.android.apex.compressed", [&](const MountedApexData& data, bool latest) { ASSERT_TRUE(latest); ASSERT_EQ(data.full_path, apex_path_2); ASSERT_EQ(data.device_name, "com.android.apex.compressed"); }); } TEST_F(ApexdMountTest, OnStartDataHasSameVersionAsCapex) { MockCheckpointInterface checkpoint_interface; // Need to call InitializeVold before calling OnStart InitializeVold(&checkpoint_interface); AddPreInstalledApex("com.android.apex.compressed.v1.capex"); std::string apex_path_2 = AddDataApex("com.android.apex.compressed.v1_original.apex"); ASSERT_THAT( ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}), Ok()); OnStart(); // Data APEX should be mounted UnmountOnTearDown(apex_path_2); ASSERT_EQ(GetProperty(kTestApexdStatusSysprop, ""), "starting"); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.compressed", "/apex/com.android.apex.compressed@1")); auto& db = GetApexDatabaseForTesting(); // Check that it was mounted from data apex, not pre-installed one. db.ForallMountedApexes("com.android.apex.compressed", [&](const MountedApexData& data, bool latest) { ASSERT_TRUE(latest); ASSERT_EQ(data.full_path, apex_path_2); ASSERT_EQ(data.device_name, "com.android.apex.compressed"); }); } TEST_F(ApexdMountTest, OnStartSystemHasHigherVersionCapexThanData) { MockCheckpointInterface checkpoint_interface; // Need to call InitializeVold before calling OnStart InitializeVold(&checkpoint_interface); std::string apex_path_1 = AddPreInstalledApex("com.android.apex.compressed.v2.capex"); AddDataApex("com.android.apex.compressed.v1_original.apex"); ASSERT_THAT( ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}), Ok()); OnStart(); // Decompressed APEX should be mounted std::string decompressed_active_apex = StringPrintf( "%s/com.android.apex.compressed@2%s", GetDecompressionDir().c_str(), kDecompressedApexPackageSuffix); UnmountOnTearDown(decompressed_active_apex); ASSERT_EQ(GetProperty(kTestApexdStatusSysprop, ""), "starting"); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.compressed", "/apex/com.android.apex.compressed@2")); auto& db = GetApexDatabaseForTesting(); // Check that it was mounted from compressed apex db.ForallMountedApexes("com.android.apex.compressed", [&](const MountedApexData& data, bool latest) { ASSERT_TRUE(latest); ASSERT_EQ(data.full_path, decompressed_active_apex); ASSERT_EQ(data.device_name, "com.android.apex.compressed"); }); } TEST_F(ApexdMountTest, OnStartFailsToActivateApexOnDataFallsBackToCapex) { MockCheckpointInterface checkpoint_interface; // Need to call InitializeVold before calling OnStart InitializeVold(&checkpoint_interface); AddPreInstalledApex("com.android.apex.compressed.v1.capex"); AddDataApex("com.android.apex.compressed.v2_manifest_mismatch.apex"); ASSERT_THAT( ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}), Ok()); OnStart(); // Decompressed APEX should be mounted std::string decompressed_active_apex = StringPrintf( "%s/com.android.apex.compressed@1%s", GetDecompressionDir().c_str(), kDecompressedApexPackageSuffix); UnmountOnTearDown(decompressed_active_apex); ASSERT_EQ(GetProperty(kTestApexdStatusSysprop, ""), "starting"); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.compressed", "/apex/com.android.apex.compressed@1")); auto& db = GetApexDatabaseForTesting(); // Check that it was mounted from decompressed apex. It should also be mounted // on dm-verity device. db.ForallMountedApexes("com.android.apex.compressed", [&](const MountedApexData& data, bool latest) { ASSERT_TRUE(latest); ASSERT_EQ(data.full_path, decompressed_active_apex); ASSERT_EQ(data.device_name, "com.android.apex.compressed"); }); } // Test scenario when we fallback to capex but it already has a decompressed // version on data TEST_F(ApexdMountTest, OnStartFallbackToAlreadyDecompressedCapex) { MockCheckpointInterface checkpoint_interface; // Need to call InitializeVold before calling OnStart InitializeVold(&checkpoint_interface); PrepareCompressedApex("com.android.apex.compressed.v1.capex"); AddDataApex("com.android.apex.compressed.v2_manifest_mismatch.apex"); ASSERT_THAT( ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}), Ok()); OnStart(); // Decompressed APEX should be mounted std::string decompressed_active_apex = StringPrintf( "%s/com.android.apex.compressed@1%s", GetDecompressionDir().c_str(), kDecompressedApexPackageSuffix); UnmountOnTearDown(decompressed_active_apex); ASSERT_EQ(GetProperty(kTestApexdStatusSysprop, ""), "starting"); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.compressed", "/apex/com.android.apex.compressed@1")); auto& db = GetApexDatabaseForTesting(); // Check that it was mounted from decompressed apex. db.ForallMountedApexes("com.android.apex.compressed", [&](const MountedApexData& data, bool latest) { ASSERT_TRUE(latest); ASSERT_EQ(data.full_path, decompressed_active_apex); ASSERT_EQ(data.device_name, "com.android.apex.compressed"); }); } // Test scenario when we fallback to capex but it has same version as corrupt // data apex TEST_F(ApexdMountTest, OnStartFallbackToCapexSameVersion) { MockCheckpointInterface checkpoint_interface; // Need to call InitializeVold before calling OnStart InitializeVold(&checkpoint_interface); AddPreInstalledApex("com.android.apex.compressed.v2.capex"); // Add data apex using the common naming convention for /data/apex/active // directory fs::copy(GetTestFile("com.android.apex.compressed.v2_manifest_mismatch.apex"), GetDataDir() + "/com.android.apex.compressed@2.apex"); ASSERT_THAT( ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}), Ok()); OnStart(); // Decompressed APEX should be mounted std::string decompressed_active_apex = StringPrintf( "%s/com.android.apex.compressed@2%s", GetDecompressionDir().c_str(), kDecompressedApexPackageSuffix); UnmountOnTearDown(decompressed_active_apex); ASSERT_EQ(GetProperty(kTestApexdStatusSysprop, ""), "starting"); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.compressed", "/apex/com.android.apex.compressed@2")); auto& db = GetApexDatabaseForTesting(); // Check that it was mounted from decompressed apex. db.ForallMountedApexes("com.android.apex.compressed", [&](const MountedApexData& data, bool latest) { ASSERT_TRUE(latest); ASSERT_EQ(data.full_path, decompressed_active_apex); ASSERT_EQ(data.device_name, "com.android.apex.compressed"); }); } TEST_F(ApexdMountTest, OnStartCapexToApex) { MockCheckpointInterface checkpoint_interface; // Need to call InitializeVold before calling OnStart InitializeVold(&checkpoint_interface); TemporaryDir previous_built_in_dir; PrepareCompressedApex("com.android.apex.compressed.v1.capex", previous_built_in_dir.path); auto apex_path = AddPreInstalledApex("com.android.apex.compressed.v1_original.apex"); ASSERT_THAT( ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}), Ok()); OnStart(); // Uncompressed APEX should be mounted UnmountOnTearDown(apex_path); ASSERT_EQ(GetProperty(kTestApexdStatusSysprop, ""), "starting"); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.compressed", "/apex/com.android.apex.compressed@1")); auto& db = GetApexDatabaseForTesting(); // Check that it was mounted from decompressed apex. db.ForallMountedApexes("com.android.apex.compressed", [&](const MountedApexData& data, bool latest) { ASSERT_TRUE(latest); ASSERT_EQ(data.full_path, apex_path); ASSERT_THAT(data.device_name, IsEmpty()); }); } // Test to ensure we do not mount decompressed APEX from /data/apex/active TEST_F(ApexdMountTest, OnStartOrphanedDecompressedApexInActiveDirectory) { MockCheckpointInterface checkpoint_interface; // Need to call InitializeVold before calling OnStart InitializeVold(&checkpoint_interface); // Place a decompressed APEX in /data/apex/active. This apex should not // be mounted since it's not in correct location. Instead, the // pre-installed APEX should be mounted. auto decompressed_apex_in_active_dir = StringPrintf("%s/com.android.apex.compressed@1%s", GetDataDir().c_str(), kDecompressedApexPackageSuffix); fs::copy(GetTestFile("com.android.apex.compressed.v1_original.apex"), decompressed_apex_in_active_dir); auto apex_path = AddPreInstalledApex("com.android.apex.compressed.v1_original.apex"); ASSERT_THAT( ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}), Ok()); OnStart(); // Pre-installed APEX should be mounted UnmountOnTearDown(apex_path); auto& db = GetApexDatabaseForTesting(); // Check that pre-installed APEX has been activated db.ForallMountedApexes("com.android.apex.compressed", [&](const MountedApexData& data, bool latest) { ASSERT_TRUE(latest); ASSERT_EQ(data.full_path, apex_path); ASSERT_THAT(data.device_name, IsEmpty()); }); } // Test scenario when decompressed version has different version than // pre-installed CAPEX TEST_F(ApexdMountTest, OnStartDecompressedApexVersionDifferentThanCapex) { MockCheckpointInterface checkpoint_interface; // Need to call InitializeVold before calling OnStart InitializeVold(&checkpoint_interface); TemporaryDir previous_built_in_dir; PrepareCompressedApex("com.android.apex.compressed.v2.capex", previous_built_in_dir.path); auto apex_path = AddPreInstalledApex("com.android.apex.compressed.v1.capex"); ASSERT_THAT( ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}), Ok()); OnStart(); // Existing higher version decompressed APEX should be ignored and new // pre-installed CAPEX should be decompressed and mounted std::string decompressed_active_apex = StringPrintf( "%s/com.android.apex.compressed@1%s", GetDecompressionDir().c_str(), kDecompressedApexPackageSuffix); UnmountOnTearDown(decompressed_active_apex); ASSERT_EQ(GetProperty(kTestApexdStatusSysprop, ""), "starting"); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.compressed", "/apex/com.android.apex.compressed@1")); auto& db = GetApexDatabaseForTesting(); // Check that it was mounted from newly decompressed apex. db.ForallMountedApexes("com.android.apex.compressed", [&](const MountedApexData& data, bool latest) { ASSERT_TRUE(latest); ASSERT_EQ(data.full_path, decompressed_active_apex); ASSERT_EQ(data.device_name, "com.android.apex.compressed"); }); } // Test that ota_apex is persisted until slot switch TEST_F(ApexdMountTest, OnStartOtaApexKeptUntilSlotSwitch) { MockCheckpointInterface checkpoint_interface; // Need to call InitializeVold before calling OnStart InitializeVold(&checkpoint_interface); // Imagine current system has v1 capex and we have v2 incoming via ota auto old_capex = AddPreInstalledApex("com.android.apex.compressed.v1.capex"); auto ota_apex_path = StringPrintf("%s/com.android.apex.compressed@2%s", GetDecompressionDir().c_str(), kOtaApexPackageSuffix); fs::copy(GetTestFile("com.android.apex.compressed.v2_original.apex"), ota_apex_path.c_str()); ASSERT_THAT( ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}), Ok()); // When we call OnStart for the first time, it will decompress v1 capex and // activate it, while after second call it will decompress v2 capex and // activate it. We need to make sure that activated APEXes are cleaned up // after test finishes. auto old_decompressed_apex = StringPrintf( "%s/com.android.apex.compressed@1%s", GetDecompressionDir().c_str(), kDecompressedApexPackageSuffix); auto new_decompressed_apex = StringPrintf( "%s/com.android.apex.compressed@2%s", GetDecompressionDir().c_str(), kDecompressedApexPackageSuffix); UnmountOnTearDown(old_decompressed_apex); UnmountOnTearDown(new_decompressed_apex); // First try starting without slot switch. Since we are booting with // old pre-installed capex, ota_apex should not be deleted OnStart(); auto path_exists = PathExists(ota_apex_path); ASSERT_TRUE(*path_exists); // When we switch slot, the pre-installed APEX will match ota_apex // and the ota_apex will end up getting renamed. RemoveFileIfExists(old_capex); AddPreInstalledApex("com.android.apex.compressed.v2.capex"); ApexFileRepository::GetInstance().Reset(GetDecompressionDir()); ASSERT_THAT( ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}), Ok()); OnStart(); path_exists = PathExists(ota_apex_path); ASSERT_FALSE(*path_exists); } // Test scenario when decompressed version has same version but different // digest TEST_F(ApexdMountTest, OnStartDecompressedApexVersionSameAsCapexDifferentDigest) { MockCheckpointInterface checkpoint_interface; // Need to call InitializeVold before calling OnStart InitializeVold(&checkpoint_interface); // Push a CAPEX to system without decompressing it auto apex_path = AddPreInstalledApex("com.android.apex.compressed.v1.capex"); auto pre_installed_apex = ApexFile::Open(apex_path); // Now push an APEX with different root digest as decompressed APEX auto decompressed_apex_path = StringPrintf( "%s/com.android.apex.compressed@1%s", GetDecompressionDir().c_str(), kDecompressedApexPackageSuffix); fs::copy(GetTestFile( "com.android.apex.compressed.v1_different_digest_original.apex"), decompressed_apex_path); auto different_digest_apex = ApexFile::Open(decompressed_apex_path); auto different_digest = GetRootDigest(*different_digest_apex); ASSERT_NE( pre_installed_apex->GetManifest().capexmetadata().originalapexdigest(), different_digest); ASSERT_THAT( ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}), Ok()); OnStart(); // Existing same version decompressed APEX with different root digest should // be ignored and the pre-installed CAPEX should be decompressed again. UnmountOnTearDown(decompressed_apex_path); // Ensure decompressed apex has same digest as pre-installed auto decompressed_apex = ApexFile::Open(decompressed_apex_path); ASSERT_EQ( pre_installed_apex->GetManifest().capexmetadata().originalapexdigest(), GetRootDigest(*decompressed_apex)); ASSERT_NE(GetRootDigest(*decompressed_apex), different_digest); } // Test when decompressed APEX has different key than CAPEX TEST_F(ApexdMountTest, OnStartDecompressedApexVersionSameAsCapexDifferentKey) { MockCheckpointInterface checkpoint_interface; // Need to call InitializeVold before calling OnStart InitializeVold(&checkpoint_interface); TemporaryDir previous_built_in_dir; auto different_key_apex_path = PrepareCompressedApex("com.android.apex.compressed_different_key.capex", previous_built_in_dir.path); // Place a same version capex in current built_in_dir, which has different key auto apex_path = AddPreInstalledApex("com.android.apex.compressed.v1.capex"); ASSERT_THAT( ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}), Ok()); OnStart(); // Existing same version decompressed APEX should be ignored and new // pre-installed CAPEX should be decompressed and mounted std::string decompressed_active_apex = StringPrintf( "%s/com.android.apex.compressed@1%s", GetDecompressionDir().c_str(), kDecompressedApexPackageSuffix); UnmountOnTearDown(decompressed_active_apex); // Ensure decompressed apex has same digest as pre-installed auto pre_installed_apex = ApexFile::Open(apex_path); auto decompressed_apex = ApexFile::Open(decompressed_active_apex); auto different_key_apex = ApexFile::Open(different_key_apex_path); ASSERT_EQ( pre_installed_apex->GetManifest().capexmetadata().originalapexdigest(), GetRootDigest(*decompressed_apex)); ASSERT_NE( pre_installed_apex->GetManifest().capexmetadata().originalapexdigest(), GetRootDigest(*different_key_apex)); } TEST_F(ApexdMountTest, PopulateFromMountsChecksPathPrefix) { AddPreInstalledApex("apex.apexd_test.apex"); std::string apex_path = AddDataApex("apex.apexd_test_v2.apex"); // Mount an apex from decomrpession_dir PrepareCompressedApex("com.android.apex.compressed.v1.capex"); std::string decompressed_apex = StringPrintf("%s/com.android.apex.compressed@1.decompressed.apex", GetDecompressionDir().c_str()); // Mount an apex from some other directory TemporaryDir td; AddPreInstalledApex("apex.apexd_test_different_app.apex"); fs::copy(GetTestFile("apex.apexd_test_different_app.apex"), td.path); std::string other_apex = StringPrintf("%s/apex.apexd_test_different_app.apex", td.path); auto& instance = ApexFileRepository::GetInstance(); ASSERT_THAT(instance.AddPreInstalledApex({GetBuiltInDir()}), Ok()); ASSERT_THAT(ActivatePackage(apex_path), Ok()); ASSERT_THAT(ActivatePackage(decompressed_apex), Ok()); ASSERT_THAT(ActivatePackage(other_apex), Ok()); auto& db = GetApexDatabaseForTesting(); // Remember mount information for |other_apex|, since it won't be available in // the database. We will need to tear it down manually. std::optional other_apex_mount_data; db.ForallMountedApexes( "com.android.apex.test_package_2", [&other_apex_mount_data](const MountedApexData& data, bool latest) { if (latest) { other_apex_mount_data.emplace(data); } }); UnmountOnTearDown(apex_path); UnmountOnTearDown(decompressed_apex); ASSERT_TRUE(other_apex_mount_data.has_value()); auto deleter = make_scope_guard([&other_apex_mount_data]() { if (!other_apex_mount_data.has_value()) { return; } if (umount2("/apex/com.android.apex.test_package_2", 0) != 0) { PLOG(ERROR) << "Failed to unmount /apex/com.android.apex.test_package_2"; } auto res = Unmount(*other_apex_mount_data, /* deferred= */ false); if (!res.ok()) { LOG(ERROR) << res.error(); } }); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.test_package", "/apex/com.android.apex.test_package@2", "/apex/com.android.apex.compressed", "/apex/com.android.apex.compressed@1", "/apex/com.android.apex.test_package_2", "/apex/com.android.apex.test_package_2@1")); // Clear the database before calling PopulateFromMounts db.Reset(); // Populate from mount db.PopulateFromMounts(GetDataDir(), GetDecompressionDir(), GetHashTreeDir()); // Count number of package and collect package names int package_count = 0; std::vector mounted_paths; db.ForallMountedApexes([&](const std::string& package, const MountedApexData& data, bool latest) { package_count++; mounted_paths.push_back(data.full_path); }); ASSERT_EQ(package_count, 2); ASSERT_THAT(mounted_paths, UnorderedElementsAre(apex_path, decompressed_apex)); } TEST_F(ApexdMountTest, UnmountAll) { AddPreInstalledApex("apex.apexd_test.apex"); std::string apex_path_2 = AddPreInstalledApex("apex.apexd_test_different_app.apex"); std::string apex_path_3 = AddDataApex("apex.apexd_test_v2.apex"); // Mount an apex from decomrpession_dir PrepareCompressedApex("com.android.apex.compressed.v1.capex"); std::string decompressed_apex = StringPrintf("%s/com.android.apex.compressed@1.decompressed.apex", GetDecompressionDir().c_str()); auto& instance = ApexFileRepository::GetInstance(); ASSERT_THAT(instance.AddPreInstalledApex({GetBuiltInDir()}), Ok()); ASSERT_THAT(ActivatePackage(apex_path_2), Ok()); ASSERT_THAT(ActivatePackage(apex_path_3), Ok()); ASSERT_THAT(ActivatePackage(decompressed_apex), Ok()); UnmountOnTearDown(apex_path_2); UnmountOnTearDown(apex_path_3); UnmountOnTearDown(decompressed_apex); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.test_package", "/apex/com.android.apex.test_package@2", "/apex/com.android.apex.compressed", "/apex/com.android.apex.compressed@1", "/apex/com.android.apex.test_package_2", "/apex/com.android.apex.test_package_2@1")); auto& db = GetApexDatabaseForTesting(); // UnmountAll expects apex database to empty, hence this reset. db.Reset(); ASSERT_EQ(0, UnmountAll()); auto new_apex_mounts = GetApexMounts(); ASSERT_EQ(new_apex_mounts.size(), 0u); } TEST_F(ApexdMountTest, UnmountAllSharedLibsApex) { ASSERT_EQ(mkdir("/apex/sharedlibs", 0755), 0); ASSERT_EQ(mkdir("/apex/sharedlibs/lib", 0755), 0); ASSERT_EQ(mkdir("/apex/sharedlibs/lib64", 0755), 0); auto deleter = make_scope_guard([]() { std::error_code ec; fs::remove_all("/apex/sharedlibs", ec); if (ec) { LOG(ERROR) << "Failed to delete /apex/sharedlibs : " << ec; } }); std::string apex_path_1 = AddPreInstalledApex( "com.android.apex.test.sharedlibs_generated.v1.libvX.apex"); std::string apex_path_2 = AddDataApex("com.android.apex.test.sharedlibs_generated.v2.libvY.apex"); auto& instance = ApexFileRepository::GetInstance(); ASSERT_THAT(instance.AddPreInstalledApex({GetBuiltInDir()}), Ok()); ASSERT_THAT(ActivatePackage(apex_path_1), Ok()); ASSERT_THAT(ActivatePackage(apex_path_2), Ok()); UnmountOnTearDown(apex_path_1); UnmountOnTearDown(apex_path_2); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.test.sharedlibs@1", "/apex/com.android.apex.test.sharedlibs@2")); auto& db = GetApexDatabaseForTesting(); // UnmountAll expects apex database to empty, hence this reset. db.Reset(); ASSERT_EQ(0, UnmountAll()); auto new_apex_mounts = GetApexMounts(); ASSERT_EQ(new_apex_mounts.size(), 0u); } TEST_F(ApexdMountTest, OnStartInVmModeActivatesPreInstalled) { MockCheckpointInterface checkpoint_interface; // Need to call InitializeVold before calling OnStart InitializeVold(&checkpoint_interface); auto path1 = AddPreInstalledApex("apex.apexd_test.apex"); auto path2 = AddPreInstalledApex("apex.apexd_test_different_app.apex"); // In VM mode, we don't scan /data/apex AddDataApex("apex.apexd_test_v2.apex"); ASSERT_EQ(0, OnStartInVmMode()); UnmountOnTearDown(path1); UnmountOnTearDown(path2); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.test_package", "/apex/com.android.apex.test_package@1", "/apex/com.android.apex.test_package_2", "/apex/com.android.apex.test_package_2@1", // Emits apex-info-list as well "/apex/apex-info-list.xml")); ASSERT_EQ(GetProperty(kTestApexdStatusSysprop, ""), "ready"); } TEST_F(ApexdMountTest, OnStartInVmModeFailsWithCapex) { MockCheckpointInterface checkpoint_interface; // Need to call InitializeVold before calling OnStart InitializeVold(&checkpoint_interface); AddPreInstalledApex("com.android.apex.compressed.v2.capex"); ASSERT_EQ(1, OnStartInVmMode()); } TEST_F(ApexdMountTest, OnStartInVmModeActivatesBlockDevicesAsWell) { MockCheckpointInterface checkpoint_interface; // Need to call InitializeVold before calling OnStart InitializeVold(&checkpoint_interface); // Set system property to enable block apexes SetBlockApexEnabled(true); auto path1 = AddBlockApex("apex.apexd_test.apex"); ASSERT_EQ(0, OnStartInVmMode()); UnmountOnTearDown(path1); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.test_package", "/apex/com.android.apex.test_package@1", // Emits apex-info-list as well "/apex/apex-info-list.xml")); ASSERT_EQ(access("/apex/apex-info-list.xml", F_OK), 0); auto info_list = com::android::apex::readApexInfoList("/apex/apex-info-list.xml"); ASSERT_TRUE(info_list.has_value()); auto apex_info_xml_1 = com::android::apex::ApexInfo( /* moduleName= */ "com.android.apex.test_package", /* modulePath= */ path1, /* preinstalledModulePath= */ path1, /* versionCode= */ 1, /* versionName= */ "1", /* isFactory= */ true, /* isActive= */ true, GetMTime(path1), /* provideSharedApexLibs= */ false); ASSERT_THAT(info_list->getApexInfo(), UnorderedElementsAre(ApexInfoXmlEq(apex_info_xml_1))); } TEST_F(ApexdMountTest, OnStartInVmModeFailsWithDuplicateNames) { MockCheckpointInterface checkpoint_interface; // Need to call InitializeVold before calling OnStart InitializeVold(&checkpoint_interface); // Set system property to enable block apexes SetBlockApexEnabled(true); AddPreInstalledApex("apex.apexd_test.apex"); AddBlockApex("apex.apexd_test_v2.apex"); ASSERT_EQ(1, OnStartInVmMode()); } TEST_F(ApexdMountTest, OnStartInVmSupportsMultipleSharedLibsApexes) { MockCheckpointInterface checkpoint_interface; InitializeVold(&checkpoint_interface); SetBlockApexEnabled(true); auto path1 = AddBlockApex("com.android.apex.test.sharedlibs_generated.v1.libvX.apex", /*public_key=*/"", /*root_digest=*/"", /*is_factory=*/true); auto path2 = AddBlockApex("com.android.apex.test.sharedlibs_generated.v2.libvY.apex", /*public_key=*/"", /*root_digest=*/"", /*is_factory=*/false); ASSERT_EQ(0, OnStartInVmMode()); UnmountOnTearDown(path1); UnmountOnTearDown(path2); // Btw, in case duplicates are sharedlibs apexes, both should be activated auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.test.sharedlibs@1", "/apex/com.android.apex.test.sharedlibs@2", // Emits apex-info-list as well "/apex/apex-info-list.xml")); } TEST_F(ApexdMountTest, OnStartInVmShouldRejectInDuplicateFactoryApexes) { MockCheckpointInterface checkpoint_interface; InitializeVold(&checkpoint_interface); SetBlockApexEnabled(true); auto path1 = AddBlockApex("com.android.apex.test.sharedlibs_generated.v1.libvX.apex", /*public_key=*/"", /*root_digest=*/"", /*is_factory=*/true); auto path2 = AddBlockApex("com.android.apex.test.sharedlibs_generated.v2.libvY.apex", /*public_key=*/"", /*root_digest=*/"", /*is_factory=*/true); ASSERT_EQ(1, OnStartInVmMode()); UnmountOnTearDown(path1); UnmountOnTearDown(path2); } TEST_F(ApexdMountTest, OnStartInVmShouldRejectInDuplicateNonFactoryApexes) { MockCheckpointInterface checkpoint_interface; InitializeVold(&checkpoint_interface); SetBlockApexEnabled(true); auto path1 = AddBlockApex("com.android.apex.test.sharedlibs_generated.v1.libvX.apex", /*public_key=*/"", /*root_digest=*/"", /*is_factory=*/false); auto path2 = AddBlockApex("com.android.apex.test.sharedlibs_generated.v2.libvY.apex", /*public_key=*/"", /*root_digest=*/"", /*is_factory=*/false); ASSERT_EQ(1, OnStartInVmMode()); UnmountOnTearDown(path1); UnmountOnTearDown(path2); } TEST_F(ApexdMountTest, OnStartInVmModeFailsWithWrongPubkey) { MockCheckpointInterface checkpoint_interface; // Need to call InitializeVold before calling OnStart InitializeVold(&checkpoint_interface); // Set system property to enable block apexes SetBlockApexEnabled(true); AddBlockApex("apex.apexd_test.apex", /*public_key=*/"wrong pubkey"); ASSERT_EQ(1, OnStartInVmMode()); } TEST_F(ApexdMountTest, GetActivePackagesReturningBlockApexesAsWell) { MockCheckpointInterface checkpoint_interface; // Need to call InitializeVold before calling OnStart InitializeVold(&checkpoint_interface); // Set system property to enable block apexes SetBlockApexEnabled(true); auto path1 = AddBlockApex("apex.apexd_test.apex"); ASSERT_EQ(0, OnStartInVmMode()); UnmountOnTearDown(path1); auto active_apexes = GetActivePackages(); ASSERT_EQ(1u, active_apexes.size()); ASSERT_EQ(path1, active_apexes[0].GetPath()); } TEST_F(ApexdMountTest, OnStartInVmModeFailsWithWrongRootDigest) { MockCheckpointInterface checkpoint_interface; // Need to call InitializeVold before calling OnStart InitializeVold(&checkpoint_interface); // Set system property to enable block apexes SetBlockApexEnabled(true); AddBlockApex("apex.apexd_test.apex", /*public_key=*/"", /*root_digest=*/"wrong root digest"); ASSERT_EQ(1, OnStartInVmMode()); } // Test that OnStart works with only block devices TEST_F(ApexdMountTest, OnStartOnlyBlockDevices) { MockCheckpointInterface checkpoint_interface; // Need to call InitializeVold before calling OnStart InitializeVold(&checkpoint_interface); // Set system property to enable block apexes SetBlockApexEnabled(true); auto path1 = AddBlockApex("apex.apexd_test.apex"); ASSERT_THAT(android::apex::AddBlockApex(ApexFileRepository::GetInstance()), Ok()); OnStart(); UnmountOnTearDown(path1); ASSERT_EQ(GetProperty(kTestApexdStatusSysprop, ""), "starting"); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.test_package", "/apex/com.android.apex.test_package@1")); } // Test that we can have a mix of both block and system apexes TEST_F(ApexdMountTest, OnStartBlockAndSystemInstalled) { MockCheckpointInterface checkpoint_interface; // Need to call InitializeVold before calling OnStart InitializeVold(&checkpoint_interface); // Set system property to enable block apexes SetBlockApexEnabled(true); auto path1 = AddPreInstalledApex("apex.apexd_test.apex"); auto path2 = AddBlockApex("apex.apexd_test_different_app.apex"); auto& instance = ApexFileRepository::GetInstance(); ASSERT_THAT(instance.AddPreInstalledApex({GetBuiltInDir()}), Ok()); ASSERT_THAT(android::apex::AddBlockApex(instance), Ok()); OnStart(); UnmountOnTearDown(path1); UnmountOnTearDown(path2); ASSERT_EQ(GetProperty(kTestApexdStatusSysprop, ""), "starting"); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.test_package", "/apex/com.android.apex.test_package@1", "/apex/com.android.apex.test_package_2", "/apex/com.android.apex.test_package_2@1")); } TEST_F(ApexdMountTest, OnStartBlockAndCompressedInstalled) { MockCheckpointInterface checkpoint_interface; // Need to call InitializeVold before calling OnStart InitializeVold(&checkpoint_interface); // Set system property to enable block apexes SetBlockApexEnabled(true); auto path1 = AddPreInstalledApex("com.android.apex.compressed.v1.capex"); auto path2 = AddBlockApex("apex.apexd_test.apex"); auto& instance = ApexFileRepository::GetInstance(); ASSERT_THAT(instance.AddPreInstalledApex({GetBuiltInDir()}), Ok()); ASSERT_THAT(android::apex::AddBlockApex(instance), Ok()); OnStart(); UnmountOnTearDown(path1); UnmountOnTearDown(path2); // Decompressed APEX should be mounted std::string decompressed_active_apex = StringPrintf( "%s/com.android.apex.compressed@1%s", GetDecompressionDir().c_str(), kDecompressedApexPackageSuffix); UnmountOnTearDown(decompressed_active_apex); ASSERT_EQ(GetProperty(kTestApexdStatusSysprop, ""), "starting"); auto apex_mounts = GetApexMounts(); ASSERT_THAT(apex_mounts, UnorderedElementsAre("/apex/com.android.apex.compressed", "/apex/com.android.apex.compressed@1", "/apex/com.android.apex.test_package", "/apex/com.android.apex.test_package@1")); } // Test that data version of apex is used if newer TEST_F(ApexdMountTest, BlockAndNewerData) { // MockCheckpointInterface checkpoint_interface; //// Need to call InitializeVold before calling OnStart // InitializeVold(&checkpoint_interface); // Set system property to enable block apexes SetBlockApexEnabled(true); auto& instance = ApexFileRepository::GetInstance(); AddBlockApex("apex.apexd_test.apex"); ASSERT_THAT(android::apex::AddBlockApex(instance), Ok()); TemporaryDir data_dir; auto apexd_test_file_v2 = ApexFile::Open(AddDataApex("apex.apexd_test_v2.apex")); ASSERT_THAT(instance.AddDataApex(GetDataDir()), Ok()); auto all_apex = instance.AllApexFilesByName(); auto result = SelectApexForActivation(all_apex, instance); ASSERT_EQ(result.size(), 1u); ASSERT_THAT(result, UnorderedElementsAre(ApexFileEq(ByRef(*apexd_test_file_v2)))); } // Test that data version of apex not is used if older TEST_F(ApexdMountTest, BlockApexAndOlderData) { MockCheckpointInterface checkpoint_interface; // Need to call InitializeVold before calling OnStart InitializeVold(&checkpoint_interface); // Set system property to enable block apexes SetBlockApexEnabled(true); auto& instance = ApexFileRepository::GetInstance(); auto apexd_test_file_v2 = ApexFile::Open(AddBlockApex("apex.apexd_test_v2.apex")); ASSERT_THAT(android::apex::AddBlockApex(instance), Ok()); TemporaryDir data_dir; AddDataApex("apex.apexd_test.apex"); ASSERT_THAT(instance.AddDataApex(GetDataDir()), Ok()); auto all_apex = instance.AllApexFilesByName(); auto result = SelectApexForActivation(all_apex, instance); ASSERT_EQ(result.size(), 1u); ASSERT_THAT(result, UnorderedElementsAre(ApexFileEq(ByRef(*apexd_test_file_v2)))); } // Test that AddBlockApex does nothing if system property not set. TEST_F(ApexdMountTest, AddBlockApexWithoutSystemProp) { MockCheckpointInterface checkpoint_interface; // Need to call InitializeVold before calling OnStart InitializeVold(&checkpoint_interface); auto& instance = ApexFileRepository::GetInstance(); AddBlockApex("apex.apexd_test.apex"); ASSERT_THAT(android::apex::AddBlockApex(instance), Ok()); ASSERT_EQ(instance.AllApexFilesByName().size(), 0ul); } // Test that adding block apex fails if preinstalled version exists TEST_F(ApexdMountTest, AddBlockApexFailsWithDuplicate) { MockCheckpointInterface checkpoint_interface; // Need to call InitializeVold before calling OnStart InitializeVold(&checkpoint_interface); // Set system property to enable block apexes SetBlockApexEnabled(true); AddPreInstalledApex("apex.apexd_test.apex"); AddBlockApex("apex.apexd_test_v2.apex"); auto& instance = ApexFileRepository::GetInstance(); ASSERT_THAT(instance.AddPreInstalledApex({GetBuiltInDir()}), Ok()); ASSERT_THAT(android::apex::AddBlockApex(instance), HasError(WithMessage(HasSubstr( "duplicate of com.android.apex.test_package found")))); } // Test that adding block apex fails if preinstalled compressed version exists TEST_F(ApexdMountTest, AddBlockApexFailsWithCompressedDuplicate) { MockCheckpointInterface checkpoint_interface; // Need to call InitializeVold before calling OnStart InitializeVold(&checkpoint_interface); // Set system property to enable block apexes SetBlockApexEnabled(true); auto path1 = AddPreInstalledApex("com.android.apex.compressed.v1.capex"); auto path2 = AddBlockApex("com.android.apex.compressed.v1_original.apex"); auto& instance = ApexFileRepository::GetInstance(); ASSERT_THAT(instance.AddPreInstalledApex({GetBuiltInDir()}), Ok()); ASSERT_THAT(android::apex::AddBlockApex(instance), HasError(WithMessage(HasSubstr( "duplicate of com.android.apex.compressed found")))); } TEST_F(ApexdMountTest, CopySepolicyToMetadata) { std::string file_path = AddPreInstalledApex("com.android.sepolicy.apex"); ASSERT_THAT( ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}), Ok()); ASSERT_THAT(ActivatePackage(file_path), Ok()); UnmountOnTearDown(file_path); ASSERT_THAT(CreateStagedSession("com.android.sepolicy.apex", 666), Ok()); ASSERT_THAT( SubmitStagedSession(666, {}, /* has_rollback_enabled= */ false, /* is_rollback= */ false, /* rollback_id= */ -1), Ok()); auto staged_dir = GetMetadataSepolicyStagedDir(); ASSERT_THAT(PathExists(staged_dir + "/SEPolicy.zip"), HasValue(true)); ASSERT_THAT(PathExists(staged_dir + "/SEPolicy.zip.sig"), HasValue(true)); } TEST_F(ApexdMountTest, AbortSepolicyApexInstall) { std::string file_path = AddPreInstalledApex("com.android.sepolicy.apex"); ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}); ASSERT_THAT(CreateStagedSession("com.android.sepolicy.apex", 666), Ok()); ASSERT_THAT( SubmitStagedSession(666, {}, /* has_rollback_enabled= */ false, /* is_rollback= */ false, /* rollback_id= */ -1), Ok()); auto staged_dir = GetMetadataSepolicyStagedDir(); ASSERT_THAT(PathExists(staged_dir), HasValue(true)); ASSERT_FALSE(IsEmptyDirectory(staged_dir)); ASSERT_THAT(AbortStagedSession(666), Ok()); ASSERT_THAT(PathExists(staged_dir), HasValue(false)); } class ApexActivationFailureTests : public ApexdMountTest {}; TEST_F(ApexActivationFailureTests, BuildFingerprintDifferent) { auto apex_session = CreateStagedSession("apex.apexd_test.apex", 123); apex_session->SetBuildFingerprint("wrong fingerprint"); apex_session->UpdateStateAndCommit(SessionState::STAGED); OnStart(); apex_session = ApexSession::GetSession(123); ASSERT_THAT(apex_session->GetErrorMessage(), HasSubstr("APEX build fingerprint has changed")); } TEST_F(ApexActivationFailureTests, ApexFileMissingInStagingDirectory) { auto apex_session = CreateStagedSession("apex.apexd_test.apex", 123); apex_session->UpdateStateAndCommit(SessionState::STAGED); // Delete the apex file in staging directory DeleteDirContent(GetStagedDir(123)); OnStart(); apex_session = ApexSession::GetSession(123); ASSERT_THAT(apex_session->GetErrorMessage(), HasSubstr("No APEX packages found")); } TEST_F(ApexActivationFailureTests, MultipleApexFileInStagingDirectory) { auto apex_session = CreateStagedSession("apex.apexd_test.apex", 123); CreateStagedSession("com.android.apex.compressed.v1_original.apex", 123); apex_session->UpdateStateAndCommit(SessionState::STAGED); OnStart(); apex_session = ApexSession::GetSession(123); ASSERT_THAT(apex_session->GetErrorMessage(), HasSubstr("More than one APEX package found")); } TEST_F(ApexActivationFailureTests, CorruptedSuperblockApexCannotBeStaged) { auto apex_session = CreateStagedSession("apex.apexd_test_corrupt_superblock_apex.apex", 123); apex_session->UpdateStateAndCommit(SessionState::STAGED); OnStart(); apex_session = ApexSession::GetSession(123); ASSERT_THAT(apex_session->GetErrorMessage(), HasSubstr("Couldn't find filesystem magic")); } TEST_F(ApexActivationFailureTests, CorruptedApexCannotBeStaged) { auto apex_session = CreateStagedSession("corrupted_b146895998.apex", 123); apex_session->UpdateStateAndCommit(SessionState::STAGED); OnStart(); apex_session = ApexSession::GetSession(123); ASSERT_THAT(apex_session->GetErrorMessage(), HasSubstr("Activation failed for packages")); } TEST_F(ApexActivationFailureTests, ActivatePackageImplFails) { auto shim_path = AddPreInstalledApex("com.android.apex.cts.shim.apex"); auto& instance = ApexFileRepository::GetInstance(); ASSERT_THAT(instance.AddPreInstalledApex({GetBuiltInDir()}), Ok()); auto apex_session = CreateStagedSession("com.android.apex.cts.shim.v2_wrong_sha.apex", 123); apex_session->UpdateStateAndCommit(SessionState::STAGED); UnmountOnTearDown(shim_path); OnStart(); apex_session = ApexSession::GetSession(123); ASSERT_THAT(apex_session->GetErrorMessage(), HasSubstr("Failed to activate packages")); ASSERT_THAT(apex_session->GetErrorMessage(), HasSubstr("has unexpected SHA512 hash")); } TEST_F(ApexActivationFailureTests, StagedSessionFailsWhenNotInFsCheckpointMode) { MockCheckpointInterface checkpoint_interface; checkpoint_interface.SetSupportsCheckpoint(true); // Need to call InitializeVold before calling OnStart InitializeVold(&checkpoint_interface); auto pre_installed_apex = AddPreInstalledApex("apex.apexd_test.apex"); auto& instance = ApexFileRepository::GetInstance(); ASSERT_THAT(instance.AddPreInstalledApex({GetBuiltInDir()}), Ok()); auto apex_session = CreateStagedSession("apex.apexd_test.apex", 123); apex_session->UpdateStateAndCommit(SessionState::STAGED); UnmountOnTearDown(pre_installed_apex); OnStart(); apex_session = ApexSession::GetSession(123); ASSERT_EQ(apex_session->GetState(), SessionState::ACTIVATION_FAILED); ASSERT_THAT( apex_session->GetErrorMessage(), HasSubstr("Cannot install apex session if not in fs-checkpoint mode")); } TEST_F(ApexActivationFailureTests, StagedSessionRevertsWhenInFsRollbackMode) { MockCheckpointInterface checkpoint_interface; checkpoint_interface.SetSupportsCheckpoint(true); checkpoint_interface.SetNeedsRollback(true); // Need to call InitializeVold before calling OnStart InitializeVold(&checkpoint_interface); auto pre_installed_apex = AddPreInstalledApex("apex.apexd_test.apex"); auto& instance = ApexFileRepository::GetInstance(); ASSERT_THAT(instance.AddPreInstalledApex({GetBuiltInDir()}), Ok()); auto apex_session = CreateStagedSession("apex.apexd_test.apex", 123); apex_session->UpdateStateAndCommit(SessionState::STAGED); UnmountOnTearDown(pre_installed_apex); OnStart(); apex_session = ApexSession::GetSession(123); ASSERT_EQ(apex_session->GetState(), SessionState::REVERTED); } TEST_F(ApexdMountTest, OnBootstrapCreatesEmptyDmDevices) { AddPreInstalledApex("apex.apexd_test.apex"); AddPreInstalledApex("com.android.apex.compressed.v1.capex"); DeviceMapper& dm = DeviceMapper::Instance(); auto cleaner = make_scope_guard([&]() { dm.DeleteDeviceIfExists("com.android.apex.test_package", 1s); dm.DeleteDeviceIfExists("com.android.apex.compressed", 1s); }); ASSERT_EQ(0, OnBootstrap()); ASSERT_EQ(dm::DmDeviceState::SUSPENDED, dm.GetState("com.android.apex.test_package")); ASSERT_EQ(dm::DmDeviceState::SUSPENDED, dm.GetState("com.android.apex.compressed")); } TEST_F(ApexdUnitTest, StagePackagesFailKey) { auto status = StagePackages({GetTestFile("apex.apexd_test_no_inst_key.apex")}); ASSERT_THAT( status, HasError(WithMessage(("No preinstalled apex found for package " "com.android.apex.test_package.no_inst_key")))); } TEST_F(ApexdUnitTest, StagePackagesSuccess) { AddPreInstalledApex("apex.apexd_test.apex"); auto& instance = ApexFileRepository::GetInstance(); ASSERT_THAT(instance.AddPreInstalledApex({GetBuiltInDir()}), Ok()); auto status = StagePackages({GetTestFile("apex.apexd_test.apex")}); ASSERT_THAT(status, Ok()); auto staged_path = StringPrintf("%s/com.android.apex.test_package@1.apex", GetDataDir().c_str()); ASSERT_EQ(0, access(staged_path.c_str(), F_OK)); } TEST_F(ApexdUnitTest, StagePackagesClearsPreviouslyActivePackage) { AddPreInstalledApex("apex.apexd_test.apex"); auto& instance = ApexFileRepository::GetInstance(); ASSERT_THAT(instance.AddPreInstalledApex({GetBuiltInDir()}), Ok()); auto current_apex = AddDataApex("apex.apexd_test.apex"); ASSERT_EQ(0, access(current_apex.c_str(), F_OK)); auto status = StagePackages({GetTestFile("apex.apexd_test_v2.apex")}); ASSERT_THAT(status, Ok()); auto staged_path = StringPrintf("%s/com.android.apex.test_package@2.apex", GetDataDir().c_str()); ASSERT_EQ(0, access(staged_path.c_str(), F_OK)); ASSERT_EQ(-1, access(current_apex.c_str(), F_OK)); ASSERT_EQ(ENOENT, errno); } TEST_F(ApexdUnitTest, StagePackagesClearsPreviouslyActivePackageDowngrade) { AddPreInstalledApex("apex.apexd_test.apex"); auto& instance = ApexFileRepository::GetInstance(); ASSERT_THAT(instance.AddPreInstalledApex({GetBuiltInDir()}), Ok()); auto current_apex = AddDataApex("apex.apexd_test_v2.apex"); ASSERT_EQ(0, access(current_apex.c_str(), F_OK)); auto status = StagePackages({GetTestFile("apex.apexd_test.apex")}); ASSERT_THAT(status, Ok()); auto staged_path = StringPrintf("%s/com.android.apex.test_package@1.apex", GetDataDir().c_str()); ASSERT_EQ(0, access(staged_path.c_str(), F_OK)); ASSERT_EQ(-1, access(current_apex.c_str(), F_OK)); ASSERT_EQ(ENOENT, errno); } TEST_F(ApexdUnitTest, StagePackagesAlreadyStagedPackage) { AddPreInstalledApex("apex.apexd_test.apex"); auto& instance = ApexFileRepository::GetInstance(); ASSERT_THAT(instance.AddPreInstalledApex({GetBuiltInDir()}), Ok()); auto status = StagePackages({GetTestFile("apex.apexd_test.apex")}); ASSERT_THAT(status, Ok()); auto staged_path = StringPrintf("%s/com.android.apex.test_package@1.apex", GetDataDir().c_str()); struct stat stat1; ASSERT_EQ(0, stat(staged_path.c_str(), &stat1)); ASSERT_TRUE(S_ISREG(stat1.st_mode)); { auto apex = ApexFile::Open(staged_path); ASSERT_THAT(apex, Ok()); ASSERT_FALSE(apex->GetManifest().nocode()); } auto status2 = StagePackages({GetTestFile("apex.apexd_test_nocode.apex")}); ASSERT_THAT(status2, Ok()); struct stat stat2; ASSERT_EQ(0, stat(staged_path.c_str(), &stat2)); ASSERT_TRUE(S_ISREG(stat2.st_mode)); ASSERT_NE(stat1.st_ino, stat2.st_ino); { auto apex = ApexFile::Open(staged_path); ASSERT_THAT(apex, Ok()); ASSERT_TRUE(apex->GetManifest().nocode()); } } TEST_F(ApexdUnitTest, StagePackagesMultiplePackages) { AddPreInstalledApex("apex.apexd_test.apex"); AddPreInstalledApex("apex.apexd_test_different_app.apex"); auto& instance = ApexFileRepository::GetInstance(); ASSERT_THAT(instance.AddPreInstalledApex({GetBuiltInDir()}), Ok()); auto status = StagePackages({GetTestFile("apex.apexd_test_v2.apex"), GetTestFile("apex.apexd_test_different_app.apex")}); ASSERT_THAT(status, Ok()); auto staged_path1 = StringPrintf("%s/com.android.apex.test_package@2.apex", GetDataDir().c_str()); auto staged_path2 = StringPrintf("%s/com.android.apex.test_package_2@1.apex", GetDataDir().c_str()); ASSERT_EQ(0, access(staged_path1.c_str(), F_OK)); ASSERT_EQ(0, access(staged_path2.c_str(), F_OK)); } TEST_F(ApexdUnitTest, UnstagePackages) { auto file_path1 = AddDataApex("apex.apexd_test.apex"); auto file_path2 = AddDataApex("apex.apexd_test_different_app.apex"); ASSERT_THAT(UnstagePackages({file_path1}), Ok()); ASSERT_EQ(-1, access(file_path1.c_str(), F_OK)); ASSERT_EQ(errno, ENOENT); ASSERT_EQ(0, access(file_path2.c_str(), F_OK)); } TEST_F(ApexdUnitTest, UnstagePackagesEmptyInput) { auto file_path1 = AddDataApex("apex.apexd_test.apex"); auto file_path2 = AddDataApex("apex.apexd_test_different_app.apex"); ASSERT_THAT(UnstagePackages({}), HasError(WithMessage("Empty set of inputs"))); ASSERT_EQ(0, access(file_path1.c_str(), F_OK)); ASSERT_EQ(0, access(file_path2.c_str(), F_OK)); } TEST_F(ApexdUnitTest, UnstagePackagesFail) { auto file_path1 = AddDataApex("apex.apexd_test.apex"); auto bad_path = GetDataDir() + "/missing.apex"; ASSERT_THAT(UnstagePackages({file_path1, bad_path}), Not(Ok())); ASSERT_EQ(0, access(file_path1.c_str(), F_OK)); } TEST_F(ApexdUnitTest, UnstagePackagesFailPreInstalledApex) { auto file_path1 = AddPreInstalledApex("apex.apexd_test.apex"); auto file_path2 = AddDataApex("apex.apexd_test_different_app.apex"); auto& instance = ApexFileRepository::GetInstance(); ASSERT_THAT(instance.AddPreInstalledApex({GetBuiltInDir()}), Ok()); ASSERT_THAT(UnstagePackages({file_path1, file_path2}), HasError(WithMessage("Can't uninstall pre-installed apex " + file_path1))); ASSERT_EQ(0, access(file_path1.c_str(), F_OK)); ASSERT_EQ(0, access(file_path2.c_str(), F_OK)); } TEST_F(ApexdUnitTest, RevertStoresCrashingNativeProcess) { MockCheckpointInterface checkpoint_interface; checkpoint_interface.SetSupportsCheckpoint(true); InitializeVold(&checkpoint_interface); auto apex_session = CreateStagedSession("apex.apexd_test.apex", 1543); ASSERT_THAT(apex_session, Ok()); ASSERT_THAT(apex_session->UpdateStateAndCommit(SessionState::ACTIVATED), Ok()); ASSERT_THAT(RevertActiveSessions("test_process", ""), Ok()); apex_session = ApexSession::GetSession(1543); ASSERT_THAT(apex_session, Ok()); ASSERT_EQ(apex_session->GetCrashingNativeProcess(), "test_process"); } TEST_F(ApexdUnitTest, MountAndDeriveClasspathNoJar) { AddPreInstalledApex("apex.apexd_test_classpath.apex"); ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}); // Call MountAndDeriveClassPath auto apex_file = ApexFile::Open(GetTestFile("apex.apexd_test.apex")); auto package_name = apex_file->GetManifest().name(); std::vector apex_files; apex_files.emplace_back(std::move(*apex_file)); auto class_path = MountAndDeriveClassPath(apex_files); ASSERT_THAT(class_path, Ok()); ASSERT_THAT(class_path->HasClassPathJars(package_name), false); } TEST_F(ApexdUnitTest, MountAndDeriveClassPathJarsPresent) { AddPreInstalledApex("apex.apexd_test_classpath.apex"); ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}); // Call MountAndDeriveClassPath auto apex_file = ApexFile::Open(GetTestFile("apex.apexd_test_classpath.apex")); auto package_name = apex_file->GetManifest().name(); std::vector apex_files; apex_files.emplace_back(std::move(*apex_file)); auto class_path = MountAndDeriveClassPath(apex_files); ASSERT_THAT(class_path, Ok()); ASSERT_THAT(class_path->HasClassPathJars(package_name), true); } TEST_F(ApexdUnitTest, ProcessCompressedApexWrongSELinuxContext) { auto compressed_apex = ApexFile::Open( AddPreInstalledApex("com.android.apex.compressed.v1.capex")); std::vector compressed_apex_list; compressed_apex_list.emplace_back(std::cref(*compressed_apex)); auto return_value = ProcessCompressedApex(compressed_apex_list, /* is_ota_chroot= */ false); ASSERT_EQ(return_value.size(), 1u); auto decompressed_apex_path = StringPrintf( "%s/com.android.apex.compressed@1%s", GetDecompressionDir().c_str(), kDecompressedApexPackageSuffix); // Verify that so far it has correct context. ASSERT_EQ(kTestActiveApexSelinuxCtx, GetSelinuxContext(decompressed_apex_path)); // Manually mess up the context ASSERT_EQ(0, setfilecon(decompressed_apex_path.c_str(), "u:object_r:apex_data_file:s0")); ASSERT_EQ("u:object_r:apex_data_file:s0", GetSelinuxContext(decompressed_apex_path)); auto attempt_2 = ProcessCompressedApex(compressed_apex_list, /* is_ota_chroot= */ false); ASSERT_EQ(attempt_2.size(), 1u); // Verify that it again has correct context. ASSERT_EQ(kTestActiveApexSelinuxCtx, GetSelinuxContext(decompressed_apex_path)); } TEST_F(ApexdMountTest, OnStartNoApexUpdated) { MockCheckpointInterface checkpoint_interface; // Need to call InitializeVold before calling OnStart InitializeVold(&checkpoint_interface); AddPreInstalledApex("com.android.apex.compressed.v1.capex"); std::string apex_path_1 = AddPreInstalledApex("apex.apexd_test.apex"); std::string apex_path_2 = AddPreInstalledApex("apex.apexd_test_different_app.apex"); std::string apex_path_3 = AddDataApex("apex.apexd_test_v2.apex"); std::string apex_path_4 = AddDecompressedApex("com.android.apex.compressed.v1_original.apex"); ASSERT_THAT( ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}), Ok()); OnStart(); UnmountOnTearDown(apex_path_2); UnmountOnTearDown(apex_path_3); UnmountOnTearDown(apex_path_4); auto updated_apexes = GetChangedActiveApexesForTesting(); ASSERT_EQ(updated_apexes.size(), 0u); // Quick check that all apexes were mounted auto apex_mounts = GetApexMounts(); ASSERT_EQ(apex_mounts.size(), 6u); } TEST_F(ApexdMountTest, OnStartDecompressingConsideredApexUpdate) { MockCheckpointInterface checkpoint_interface; // Need to call InitializeVold before calling OnStart InitializeVold(&checkpoint_interface); AddPreInstalledApex("com.android.apex.compressed.v1.capex"); std::string apex_path_1 = AddPreInstalledApex("apex.apexd_test.apex"); std::string decompressed_active_apex = StringPrintf( "%s/com.android.apex.compressed@1%s", GetDecompressionDir().c_str(), kDecompressedApexPackageSuffix); UnmountOnTearDown(decompressed_active_apex); ASSERT_THAT( ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}), Ok()); OnStart(); UnmountOnTearDown(apex_path_1); UnmountOnTearDown(decompressed_active_apex); auto updated_apexes = GetChangedActiveApexesForTesting(); ASSERT_EQ(updated_apexes.size(), 1u); auto apex_file = ApexFile::Open(decompressed_active_apex); ASSERT_THAT(apex_file, Ok()); ASSERT_TRUE(IsActiveApexChanged(*apex_file)); } TEST_F(ApexdMountTest, ActivatesStagedSession) { MockCheckpointInterface checkpoint_interface; // Need to call InitializeVold before calling OnStart InitializeVold(&checkpoint_interface); std::string preinstalled_apex = AddPreInstalledApex("apex.apexd_test.apex"); auto apex_session = CreateStagedSession("apex.apexd_test_v2.apex", 37); apex_session->UpdateStateAndCommit(SessionState::STAGED); ASSERT_THAT( ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}), Ok()); std::string active_apex = GetDataDir() + "/" + "com.android.apex.test_package@2.apex"; UnmountOnTearDown(preinstalled_apex); UnmountOnTearDown(active_apex); OnStart(); // Quick check that session was activated { auto session = ApexSession::GetSession(37); ASSERT_THAT(session, Ok()); ASSERT_EQ(session->GetState(), SessionState::ACTIVATED); } auto updated_apexes = GetChangedActiveApexesForTesting(); ASSERT_EQ(updated_apexes.size(), 1u); auto apex_file = ApexFile::Open(active_apex); ASSERT_THAT(apex_file, Ok()); ASSERT_TRUE(IsActiveApexChanged(*apex_file)); } TEST_F(ApexdMountTest, FailsToActivateStagedSession) { MockCheckpointInterface checkpoint_interface; // Need to call InitializeVold before calling OnStart InitializeVold(&checkpoint_interface); std::string preinstalled_apex = AddPreInstalledApex("apex.apexd_test.apex"); auto apex_session = CreateStagedSession("apex.apexd_test_manifest_mismatch.apex", 73); apex_session->UpdateStateAndCommit(SessionState::STAGED); ASSERT_THAT( ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}), Ok()); UnmountOnTearDown(preinstalled_apex); OnStart(); // Quick check that session was activated { auto session = ApexSession::GetSession(73); ASSERT_THAT(session, Ok()); ASSERT_NE(session->GetState(), SessionState::ACTIVATED); } auto updated_apexes = GetChangedActiveApexesForTesting(); ASSERT_EQ(updated_apexes.size(), 1u); auto apex_file = ApexFile::Open(preinstalled_apex); ASSERT_THAT(apex_file, Ok()); ASSERT_TRUE(IsActiveApexChanged(*apex_file)); } TEST_F(ApexdMountTest, FailsToActivateApexFallbacksToSystemOne) { MockCheckpointInterface checkpoint_interface; // Need to call InitializeVold before calling OnStart InitializeVold(&checkpoint_interface); std::string preinstalled_apex = AddPreInstalledApex("apex.apexd_test.apex"); AddDataApex("apex.apexd_test_manifest_mismatch.apex"); ASSERT_THAT( ApexFileRepository::GetInstance().AddPreInstalledApex({GetBuiltInDir()}), Ok()); UnmountOnTearDown(preinstalled_apex); OnStart(); auto updated_apexes = GetChangedActiveApexesForTesting(); ASSERT_EQ(updated_apexes.size(), 1u); auto apex_file = ApexFile::Open(preinstalled_apex); ASSERT_THAT(apex_file, Ok()); ASSERT_TRUE(IsActiveApexChanged(*apex_file)); } class LogTestToLogcat : public ::testing::EmptyTestEventListener { void OnTestStart(const ::testing::TestInfo& test_info) override { #ifdef __ANDROID__ using base::LogId; using base::LogSeverity; using base::StringPrintf; base::LogdLogger l; std::string msg = StringPrintf("=== %s::%s (%s:%d)", test_info.test_suite_name(), test_info.name(), test_info.file(), test_info.line()); l(LogId::MAIN, LogSeverity::INFO, "ApexTestCases", __FILE__, __LINE__, msg.c_str()); #else UNUSED(test_info); #endif } }; } // namespace apex } // namespace android int main(int argc, char** argv) { ::testing::InitGoogleTest(&argc, argv); android::base::InitLogging(argv, &android::base::StderrLogger); android::base::SetMinimumLogSeverity(android::base::VERBOSE); ::testing::UnitTest::GetInstance()->listeners().Append( new android::apex::LogTestToLogcat()); return RUN_ALL_TESTS(); }