blob: 3061e80adf826169fef308206bc808964cf48c13 [file] [log] [blame]
/*
* Copyright (C) 2020 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_session.h"
#include <android-base/file.h>
#include <android-base/result-gmock.h>
#include <android-base/result.h>
#include <android-base/scopeguard.h>
#include <android-base/stringprintf.h>
#include <android-base/strings.h>
#include <errno.h>
#include <gmock/gmock.h>
#include <gtest/gtest.h>
#include <algorithm>
#include <filesystem>
#include <fstream>
#include <string>
#include "apexd_test_utils.h"
#include "apexd_utils.h"
#include "session_state.pb.h"
namespace android {
namespace apex {
namespace {
using android::base::Join;
using android::base::make_scope_guard;
using android::base::testing::Ok;
using ::apex::proto::SessionState;
using ::testing::Not;
using ::testing::UnorderedElementsAre;
// TODO(b/170329726): add unit tests for apexd_sessions.h
TEST(ApexdSessionTest, GetSessionsDirSessionsStoredInMetadata) {
if (access("/metadata", F_OK) != 0) {
GTEST_SKIP() << "Device doesn't have /metadata partition";
}
std::string result = GetSessionsDir();
ASSERT_EQ(result, "/metadata/apex/sessions");
}
TEST(ApexdSessionTest, GetSessionsDirNoMetadataPartitionFallbackToData) {
if (access("/metadata", F_OK) == 0) {
GTEST_SKIP() << "Device has /metadata partition";
}
std::string result = GetSessionsDir();
ASSERT_EQ(result, "/data/apex/sessions");
}
TEST(ApexdSessionTest, MigrateToMetadataSessionsDir) {
namespace fs = std::filesystem;
if (access("/metadata", F_OK) != 0) {
GTEST_SKIP() << "Device doesn't have /metadata partition";
}
// This is ugly, but does the job. To have a truly hermetic unit tests we
// need to refactor ApexSession class.
for (const auto& entry : fs::directory_iterator("/metadata/apex/sessions")) {
fs::remove_all(entry.path());
}
// This is a very ugly test set up, but to have something better we need to
// refactor ApexSession class.
class TestApexSession {
public:
TestApexSession(int id, const SessionState::State& state) {
path_ = "/data/apex/sessions/" + std::to_string(id);
if (auto status = CreateDirIfNeeded(path_, 0700); !status.ok()) {
ADD_FAILURE() << "Failed to create " << path_ << " : "
<< status.error();
}
SessionState session;
session.set_id(id);
session.set_state(state);
std::fstream state_file(
path_ + "/state", std::ios::out | std::ios::trunc | std::ios::binary);
if (!session.SerializeToOstream(&state_file)) {
ADD_FAILURE() << "Failed to write to " << path_;
}
}
~TestApexSession() { fs::remove_all(path_); }
private:
std::string path_;
};
auto deleter = make_scope_guard([&]() {
fs::remove_all("/metadata/apex/sessions/239");
fs::remove_all("/metadata/apex/sessions/1543");
});
TestApexSession session1(239, SessionState::SUCCESS);
TestApexSession session2(1543, SessionState::ACTIVATION_FAILED);
ASSERT_RESULT_OK(ApexSession::MigrateToMetadataSessionsDir());
auto sessions = ApexSession::GetSessions();
ASSERT_EQ(2u, sessions.size()) << Join(sessions, ',');
auto migrated_session_1 = ApexSession::GetSession(239);
ASSERT_RESULT_OK(migrated_session_1);
ASSERT_EQ(SessionState::SUCCESS, migrated_session_1->GetState());
auto migrated_session_2 = ApexSession::GetSession(1543);
ASSERT_RESULT_OK(migrated_session_2);
ASSERT_EQ(SessionState::ACTIVATION_FAILED, migrated_session_2->GetState());
}
TEST(ApexSessionManagerTest, CreateSession) {
TemporaryDir td;
auto manager = ApexSessionManager::Create(std::string(td.path));
auto session = manager->CreateSession(239);
ASSERT_RESULT_OK(session);
ASSERT_EQ(239, session->GetId());
std::string session_dir = std::string(td.path) + "/239";
ASSERT_EQ(session_dir, session->GetSessionDir());
}
TEST(ApexSessionManagerTest, GetSessionsNoSessionReturnsError) {
TemporaryDir td;
auto manager = ApexSessionManager::Create(std::string(td.path));
ASSERT_THAT(manager->GetSession(37), Not(Ok()));
}
TEST(ApexSessionManagerTest, GetSessionsReturnsErrorSessionNotCommitted) {
TemporaryDir td;
auto manager = ApexSessionManager::Create(std::string(td.path));
auto session = manager->CreateSession(73);
ASSERT_RESULT_OK(session);
ASSERT_THAT(manager->GetSession(73), Not(Ok()));
}
TEST(ApexSessionManagerTest, CreateCommitGetSession) {
TemporaryDir td;
auto manager = ApexSessionManager::Create(std::string(td.path));
auto session = manager->CreateSession(23);
ASSERT_RESULT_OK(session);
session->SetErrorMessage("error");
ASSERT_RESULT_OK(session->UpdateStateAndCommit(SessionState::STAGED));
auto same_session = manager->GetSession(23);
ASSERT_RESULT_OK(same_session);
ASSERT_EQ(23, same_session->GetId());
ASSERT_EQ("error", same_session->GetErrorMessage());
ASSERT_EQ(SessionState::STAGED, same_session->GetState());
}
TEST(ApexSessionManagerTest, GetSessionsNoSessionsCommitted) {
TemporaryDir td;
auto manager = ApexSessionManager::Create(std::string(td.path));
ASSERT_RESULT_OK(manager->CreateSession(3));
auto sessions = manager->GetSessions();
ASSERT_EQ(0u, sessions.size());
}
TEST(ApexSessionManager, GetSessionsCommittedSessions) {
TemporaryDir td;
auto manager = ApexSessionManager::Create(std::string(td.path));
auto session1 = manager->CreateSession(1543);
ASSERT_RESULT_OK(session1);
ASSERT_RESULT_OK(session1->UpdateStateAndCommit(SessionState::ACTIVATED));
auto session2 = manager->CreateSession(179);
ASSERT_RESULT_OK(session2);
ASSERT_RESULT_OK(session2->UpdateStateAndCommit(SessionState::SUCCESS));
// This sessions is not committed, it won't be returned in GetSessions.
ASSERT_RESULT_OK(manager->CreateSession(101));
auto sessions = manager->GetSessions();
std::sort(
sessions.begin(), sessions.end(),
[](const auto& s1, const auto& s2) { return s1.GetId() < s2.GetId(); });
ASSERT_EQ(2u, sessions.size());
ASSERT_EQ(179, sessions[0].GetId());
ASSERT_EQ(SessionState::SUCCESS, sessions[0].GetState());
ASSERT_EQ(1543, sessions[1].GetId());
ASSERT_EQ(SessionState::ACTIVATED, sessions[1].GetState());
}
TEST(ApexSessionManager, GetSessionsInState) {
TemporaryDir td;
auto manager = ApexSessionManager::Create(std::string(td.path));
auto session1 = manager->CreateSession(43);
ASSERT_RESULT_OK(session1);
ASSERT_RESULT_OK(session1->UpdateStateAndCommit(SessionState::ACTIVATED));
auto session2 = manager->CreateSession(41);
ASSERT_RESULT_OK(session2);
ASSERT_RESULT_OK(session2->UpdateStateAndCommit(SessionState::SUCCESS));
auto session3 = manager->CreateSession(23);
ASSERT_RESULT_OK(session3);
ASSERT_RESULT_OK(session3->UpdateStateAndCommit(SessionState::SUCCESS));
auto sessions = manager->GetSessionsInState(SessionState::SUCCESS);
std::sort(
sessions.begin(), sessions.end(),
[](const auto& s1, const auto& s2) { return s1.GetId() < s2.GetId(); });
ASSERT_EQ(2u, sessions.size());
ASSERT_EQ(23, sessions[0].GetId());
ASSERT_EQ(SessionState::SUCCESS, sessions[0].GetState());
ASSERT_EQ(41, sessions[1].GetId());
ASSERT_EQ(SessionState::SUCCESS, sessions[1].GetState());
}
TEST(ApexSessionManager, MigrateFromOldSessionsDir) {
TemporaryDir td;
auto old_manager = ApexSessionManager::Create(std::string(td.path));
auto session1 = old_manager->CreateSession(239);
ASSERT_RESULT_OK(session1);
ASSERT_RESULT_OK(session1->UpdateStateAndCommit(SessionState::STAGED));
auto session2 = old_manager->CreateSession(13);
ASSERT_RESULT_OK(session2);
ASSERT_RESULT_OK(session2->UpdateStateAndCommit(SessionState::SUCCESS));
auto session3 = old_manager->CreateSession(31);
ASSERT_RESULT_OK(session3);
ASSERT_RESULT_OK(session3->UpdateStateAndCommit(SessionState::ACTIVATED));
TemporaryDir td2;
auto new_manager = ApexSessionManager::Create(std::string(td2.path));
ASSERT_RESULT_OK(
new_manager->MigrateFromOldSessionsDir(std::string(td.path)));
auto sessions = new_manager->GetSessions();
std::sort(
sessions.begin(), sessions.end(),
[](const auto& s1, const auto& s2) { return s1.GetId() < s2.GetId(); });
ASSERT_EQ(3u, sessions.size());
ASSERT_EQ(13, sessions[0].GetId());
ASSERT_EQ(SessionState::SUCCESS, sessions[0].GetState());
ASSERT_EQ(31, sessions[1].GetId());
ASSERT_EQ(SessionState::ACTIVATED, sessions[1].GetState());
ASSERT_EQ(239, sessions[2].GetId());
ASSERT_EQ(SessionState::STAGED, sessions[2].GetState());
// Check that old manager directory doesn't have anything
auto old_sessions = old_manager->GetSessions();
ASSERT_TRUE(old_sessions.empty());
}
TEST(ApexSessionManager, MigrateFromOldSessionsDirSameDir) {
TemporaryDir td;
auto old_manager = ApexSessionManager::Create(std::string(td.path));
auto session1 = old_manager->CreateSession(239);
ASSERT_RESULT_OK(session1);
ASSERT_RESULT_OK(session1->UpdateStateAndCommit(SessionState::STAGED));
auto session2 = old_manager->CreateSession(13);
ASSERT_RESULT_OK(session2);
ASSERT_RESULT_OK(session2->UpdateStateAndCommit(SessionState::SUCCESS));
auto session3 = old_manager->CreateSession(31);
ASSERT_RESULT_OK(session3);
ASSERT_RESULT_OK(session3->UpdateStateAndCommit(SessionState::ACTIVATED));
auto new_manager = ApexSessionManager::Create(std::string(td.path));
ASSERT_RESULT_OK(
new_manager->MigrateFromOldSessionsDir(std::string(td.path)));
auto sessions = new_manager->GetSessions();
std::sort(
sessions.begin(), sessions.end(),
[](const auto& s1, const auto& s2) { return s1.GetId() < s2.GetId(); });
ASSERT_EQ(3u, sessions.size());
ASSERT_EQ(13, sessions[0].GetId());
ASSERT_EQ(SessionState::SUCCESS, sessions[0].GetState());
ASSERT_EQ(31, sessions[1].GetId());
ASSERT_EQ(SessionState::ACTIVATED, sessions[1].GetState());
ASSERT_EQ(239, sessions[2].GetId());
ASSERT_EQ(SessionState::STAGED, sessions[2].GetState());
// Directory is the same, so using old_manager should also work.
auto old_sessions = old_manager->GetSessions();
std::sort(
old_sessions.begin(), old_sessions.end(),
[](const auto& s1, const auto& s2) { return s1.GetId() < s2.GetId(); });
ASSERT_EQ(3u, old_sessions.size());
ASSERT_EQ(13, old_sessions[0].GetId());
ASSERT_EQ(SessionState::SUCCESS, old_sessions[0].GetState());
ASSERT_EQ(31, old_sessions[1].GetId());
ASSERT_EQ(SessionState::ACTIVATED, old_sessions[1].GetState());
ASSERT_EQ(239, old_sessions[2].GetId());
ASSERT_EQ(SessionState::STAGED, old_sessions[2].GetState());
}
TEST(ApexSessionManagerTest, GetStagedApexDirsSelf) {
TemporaryDir td;
auto manager = ApexSessionManager::Create(std::string(td.path));
auto session = manager->CreateSession(239);
ASSERT_RESULT_OK(session);
ASSERT_THAT(session->GetStagedApexDirs("/path/to/staged_session_dir"),
UnorderedElementsAre("/path/to/staged_session_dir/session_239"));
}
TEST(ApexSessionManagerTest, GetStagedApexDirsChildren) {
TemporaryDir td;
auto manager = ApexSessionManager::Create(std::string(td.path));
auto session = manager->CreateSession(239);
ASSERT_RESULT_OK(session);
auto child_session_1 = manager->CreateSession(240);
ASSERT_RESULT_OK(child_session_1);
auto child_session_2 = manager->CreateSession(241);
ASSERT_RESULT_OK(child_session_2);
session->SetChildSessionIds({240, 241});
ASSERT_THAT(session->GetStagedApexDirs("/path/to/staged_session_dir"),
UnorderedElementsAre("/path/to/staged_session_dir/session_240",
"/path/to/staged_session_dir/session_241"));
}
} // namespace
} // namespace apex
} // namespace android