blob: 5aa0da2faf4150ab70d687b7e4120a2252a4304e [file] [log] [blame]
// Copyright 2021 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "chrome/browser/ash/crosapi/browser_manager.h"
#include "ash/constants/ash_features.h"
#include "ash/public/cpp/shelf_model.h"
#include "base/test/scoped_feature_list.h"
#include "chrome/browser/ash/crosapi/browser_loader.h"
#include "chrome/browser/ash/login/users/fake_chrome_user_manager.h"
#include "chrome/browser/ash/profiles/profile_helper.h"
#include "chrome/browser/browser_process.h"
#include "chrome/browser/component_updater/fake_cros_component_manager.h"
#include "chrome/browser/lacros/browser_service_lacros.h"
#include "chrome/browser/profiles/profile_manager.h"
#include "chrome/browser/ui/ash/shelf/chrome_shelf_controller.h"
#include "chrome/test/base/scoped_testing_local_state.h"
#include "chrome/test/base/testing_browser_process.h"
#include "chrome/test/base/testing_profile.h"
#include "chromeos/crosapi/mojom/crosapi.mojom-test-utils.h"
#include "chromeos/crosapi/mojom/crosapi.mojom.h"
#include "components/account_id/account_id.h"
#include "components/component_updater/mock_component_updater_service.h"
#include "components/session_manager/core/session_manager.h"
#include "components/user_manager/scoped_user_manager.h"
#include "content/public/test/browser_task_environment.h"
#include "testing/gmock/include/gmock/gmock.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "ui/display/test/test_screen.h"
using ::component_updater::FakeCrOSComponentManager;
using ::component_updater::MockComponentUpdateService;
using testing::_;
using update_client::UpdateClient;
using user_manager::User;
namespace crosapi {
namespace {
constexpr char kSampleLacrosPath[] =
"/run/imageloader-lacros-dogfood-dev/97.0.4676/";
class MockBrowserService : public mojom::BrowserServiceInterceptorForTesting {
public:
BrowserService* GetForwardingInterface() override {
NOTREACHED();
return nullptr;
}
MOCK_METHOD(void,
NewWindow,
(bool incognito,
bool should_trigger_session_restore,
int64_t target_display_id,
NewWindowCallback callback),
(override));
MOCK_METHOD(void, OpenForFullRestore, (bool skip_crash_restore), (override));
MOCK_METHOD(void, UpdateKeepAlive, (bool enabled), (override));
};
class BrowserManagerFake : public BrowserManager {
public:
BrowserManagerFake(std::unique_ptr<BrowserLoader> browser_loader,
component_updater::ComponentUpdateService* update_service)
: BrowserManager(std::move(browser_loader), update_service) {}
~BrowserManagerFake() override = default;
// BrowserManager:
void Start(bool launching_at_login_screen = false) override {
++start_count_;
BrowserManager::Start(launching_at_login_screen);
}
int start_count() const { return start_count_; }
void SetStatePublic(State state) { SetState(state); }
void SimulateLacrosTermination() {
SetStatePublic(State::TERMINATING);
if (browser_service_.has_value())
OnBrowserServiceDisconnected(*crosapi_id_, browser_service_->mojo_id);
OnLacrosChromeTerminated();
}
void SimulateLacrosStart(mojom::BrowserService* browser_service) {
crosapi_id_ = CrosapiId::FromUnsafeValue(42); // Dummy value.
SetStatePublic(State::STARTING);
OnBrowserServiceConnected(*crosapi_id_,
mojo::RemoteSetElementId::FromUnsafeValue(42),
browser_service, 42);
}
// Make the State enum publicly available.
using BrowserManager::State;
int start_count_ = 0;
};
} // namespace
class MockBrowserLoader : public BrowserLoader {
public:
explicit MockBrowserLoader(
scoped_refptr<component_updater::CrOSComponentManager> manager)
: BrowserLoader(manager) {}
MockBrowserLoader(const MockBrowserLoader&) = delete;
MockBrowserLoader& operator=(const MockBrowserLoader&) = delete;
~MockBrowserLoader() override = default;
MOCK_METHOD1(Load, void(LoadCompletionCallback));
MOCK_METHOD0(Unload, void());
};
class BrowserManagerTest : public testing::Test {
public:
BrowserManagerTest() : local_state_(TestingBrowserProcess::GetGlobal()) {}
~BrowserManagerTest() override = default;
void SetUp() override {
// Enable Lacros by setting the appropriate flag.
feature_list_.InitAndEnableFeature(chromeos::features::kLacrosSupport);
fake_user_manager_ = new ash::FakeChromeUserManager;
scoped_user_manager_ = std::make_unique<user_manager::ScopedUserManager>(
base::WrapUnique(fake_user_manager_));
auto fake_cros_component_manager =
base::MakeRefCounted<FakeCrOSComponentManager>();
std::unique_ptr<MockBrowserLoader> browser_loader =
std::make_unique<testing::StrictMock<MockBrowserLoader>>(
fake_cros_component_manager);
browser_loader_ = browser_loader.get();
component_update_service_ =
std::make_unique<testing::NiceMock<MockComponentUpdateService>>();
fake_browser_manager_ = std::make_unique<BrowserManagerFake>(
std::move(browser_loader), component_update_service_.get());
shelf_model_ = std::make_unique<ash::ShelfModel>();
shelf_controller_ = std::make_unique<ChromeShelfController>(
&testing_profile_, shelf_model_.get(), /*shelf_item_factory=*/nullptr);
shelf_controller_->Init();
// We need to avoid a DCHECK which happens when the policies have not yet
// been loaded. As such we claim that the Lacros availability is allowed
// to be set by the user.
crosapi::browser_util::SetLacrosLaunchSwitchSourceForTest(
crosapi::browser_util::LacrosAvailability::kUserChoice);
EXPECT_CALL(mock_browser_service_, NewWindow(_, _, _, _)).Times(0);
EXPECT_CALL(mock_browser_service_, OpenForFullRestore(_)).Times(0);
}
void TearDown() override {
// Need to reverse the state back to non set.
crosapi::browser_util::ClearLacrosAvailabilityCacheForTest();
}
void AddRegularUser(const std::string& email) {
AccountId account_id = AccountId::FromUserEmail(email);
const User* user = fake_user_manager_->AddUser(account_id);
fake_user_manager_->UserLoggedIn(account_id, user->username_hash(),
/*browser_restart=*/false,
/*is_child=*/false);
fake_user_manager_->SimulateUserProfileLoad(account_id);
ash::ProfileHelper::Get()->SetUserToProfileMappingForTesting(
user, &testing_profile_);
}
protected:
// The order of these members is relevant for both construction and
// destruction timing.
content::BrowserTaskEnvironment task_environment_;
session_manager::SessionManager session_manager_;
TestingProfile testing_profile_;
ash::FakeChromeUserManager* fake_user_manager_ = nullptr;
std::unique_ptr<user_manager::ScopedUserManager> scoped_user_manager_;
MockBrowserLoader* browser_loader_ = nullptr;
std::unique_ptr<MockComponentUpdateService> component_update_service_;
std::unique_ptr<BrowserManagerFake> fake_browser_manager_;
ScopedTestingLocalState local_state_;
std::unique_ptr<ash::ShelfModel> shelf_model_;
std::unique_ptr<ChromeShelfController> shelf_controller_;
MockBrowserService mock_browser_service_;
display::test::TestScreen test_screen_{/*create_display=*/true,
/*register_screen=*/true};
private:
base::test::ScopedFeatureList feature_list_;
};
TEST_F(BrowserManagerTest, LacrosKeepAlive) {
AddRegularUser("user@test.com");
browser_util::SetProfileMigrationCompletedForUser(
local_state_.Get(),
ash::ProfileHelper::Get()
->GetUserByProfile(&testing_profile_)
->username_hash(),
browser_util::MigrationMode::kCopy);
EXPECT_TRUE(browser_util::IsLacrosEnabled());
EXPECT_TRUE(browser_util::IsLacrosAllowedToLaunch());
using State = BrowserManagerFake::State;
EXPECT_EQ(fake_browser_manager_->start_count(), 0);
// Attempt to mount the Lacros image. Will not start as it does not meet the
// automatic start criteria.
EXPECT_CALL(*browser_loader_, Load(_))
.WillOnce([](BrowserLoader::LoadCompletionCallback callback) {
std::move(callback).Run(base::FilePath("/run/lacros"),
browser_util::LacrosSelection::kRootfs,
base::Version());
});
fake_browser_manager_->InitializeAndStartIfNeeded();
EXPECT_EQ(fake_browser_manager_->start_count(), 0);
fake_browser_manager_->SetStatePublic(State::UNAVAILABLE);
EXPECT_EQ(fake_browser_manager_->start_count(), 0);
// Creating a ScopedKeepAlive does not start Lacros.
std::unique_ptr<BrowserManager::ScopedKeepAlive> keep_alive =
fake_browser_manager_->KeepAlive(BrowserManager::Feature::kTestOnly);
EXPECT_EQ(fake_browser_manager_->start_count(), 0);
// On termination, KeepAlive should start Lacros.
fake_browser_manager_->SimulateLacrosTermination();
EXPECT_EQ(fake_browser_manager_->start_count(), 1);
// Repeating the process starts Lacros again.
fake_browser_manager_->SimulateLacrosTermination();
EXPECT_EQ(fake_browser_manager_->start_count(), 2);
// Once the ScopedKeepAlive is destroyed, this should no longer happen.
keep_alive.reset();
fake_browser_manager_->SimulateLacrosTermination();
EXPECT_EQ(fake_browser_manager_->start_count(), 2);
}
TEST_F(BrowserManagerTest, LacrosKeepAliveReloadsWhenUpdateAvailable) {
AddRegularUser("user@test.com");
browser_util::SetProfileMigrationCompletedForUser(
local_state_.Get(),
ash::ProfileHelper::Get()
->GetUserByProfile(&testing_profile_)
->username_hash(),
browser_util::MigrationMode::kCopy);
EXPECT_TRUE(browser_util::IsLacrosEnabled());
EXPECT_TRUE(browser_util::IsLacrosAllowedToLaunch());
EXPECT_CALL(*browser_loader_, Load(_))
.WillOnce([](BrowserLoader::LoadCompletionCallback callback) {
std::move(callback).Run(base::FilePath("/run/lacros"),
browser_util::LacrosSelection::kRootfs,
base::Version());
});
fake_browser_manager_->InitializeAndStartIfNeeded();
using State = BrowserManagerFake::State;
EXPECT_EQ(fake_browser_manager_->start_count(), 0);
fake_browser_manager_->SetStatePublic(State::UNAVAILABLE);
EXPECT_EQ(fake_browser_manager_->start_count(), 0);
// Simulate an update event by the component update service.
const std::string lacros_component_id =
browser_util::kLacrosDogfoodDevInfo.crx_id;
static_cast<component_updater::ComponentUpdateService::Observer*>(
fake_browser_manager_.get())
->OnEvent(UpdateClient::Observer::Events::COMPONENT_UPDATED,
lacros_component_id);
std::unique_ptr<BrowserManager::ScopedKeepAlive> keep_alive =
fake_browser_manager_->KeepAlive(BrowserManager::Feature::kTestOnly);
EXPECT_CALL(*browser_loader_, Load(_))
.WillOnce([](BrowserLoader::LoadCompletionCallback callback) {
std::move(callback).Run(base::FilePath(kSampleLacrosPath),
browser_util::LacrosSelection::kStateful,
base::Version());
});
// On simulated termination, KeepAlive restarts Lacros. Since there is an
// update, it should first load the updated image.
EXPECT_EQ(fake_browser_manager_->start_count(), 0);
fake_browser_manager_->SimulateLacrosTermination();
EXPECT_GE(fake_browser_manager_->start_count(), 1);
}
TEST_F(BrowserManagerTest, NewWindowReloadsWhenUpdateAvailable) {
AddRegularUser("user@test.com");
browser_util::SetProfileMigrationCompletedForUser(
local_state_.Get(),
ash::ProfileHelper::Get()
->GetUserByProfile(&testing_profile_)
->username_hash(),
browser_util::MigrationMode::kCopy);
EXPECT_TRUE(browser_util::IsLacrosEnabled());
EXPECT_TRUE(browser_util::IsLacrosAllowedToLaunch());
EXPECT_CALL(*browser_loader_, Load(_))
.WillOnce([](BrowserLoader::LoadCompletionCallback callback) {
std::move(callback).Run(base::FilePath("/run/lacros"),
browser_util::LacrosSelection::kRootfs,
base::Version());
});
fake_browser_manager_->InitializeAndStartIfNeeded();
// Set the state of the browser manager as stopped, which would match the
// state after the browser mounted an image, ran, and was terminated.
using State = BrowserManagerFake::State;
fake_browser_manager_->SetStatePublic(State::STOPPED);
const std::string lacros_component_id =
browser_util::kLacrosDogfoodDevInfo.crx_id;
static_cast<component_updater::ComponentUpdateService::Observer*>(
fake_browser_manager_.get())
->OnEvent(UpdateClient::Observer::Events::COMPONENT_UPDATED,
lacros_component_id);
EXPECT_EQ(fake_browser_manager_->start_count(), 0);
EXPECT_CALL(*browser_loader_, Load(_));
EXPECT_CALL(mock_browser_service_, NewWindow(_, _, _, _))
.Times(1)
.RetiresOnSaturation();
fake_browser_manager_->NewWindow(/*incognito=*/false,
/*should_trigger_session_restore=*/false);
EXPECT_EQ(fake_browser_manager_->start_count(), 1);
fake_browser_manager_->SimulateLacrosStart(&mock_browser_service_);
}
TEST_F(BrowserManagerTest, LacrosKeepAliveDoesNotBlockRestart) {
EXPECT_CALL(mock_browser_service_, UpdateKeepAlive(_)).Times(0);
AddRegularUser("user@test.com");
browser_util::SetProfileMigrationCompletedForUser(
local_state_.Get(),
ash::ProfileHelper::Get()
->GetUserByProfile(&testing_profile_)
->username_hash(),
browser_util::MigrationMode::kCopy);
EXPECT_TRUE(browser_util::IsLacrosEnabled());
EXPECT_TRUE(browser_util::IsLacrosAllowedToLaunch());
using State = BrowserManagerFake::State;
EXPECT_EQ(fake_browser_manager_->start_count(), 0);
// Attempt to mount the Lacros image. Will not start as it does not meet the
// automatic start criteria.
EXPECT_CALL(*browser_loader_, Load(_))
.WillOnce([](BrowserLoader::LoadCompletionCallback callback) {
std::move(callback).Run(base::FilePath("/run/lacros"),
browser_util::LacrosSelection::kRootfs,
base::Version());
});
fake_browser_manager_->InitializeAndStartIfNeeded();
EXPECT_EQ(fake_browser_manager_->start_count(), 0);
fake_browser_manager_->SetStatePublic(State::UNAVAILABLE);
EXPECT_EQ(fake_browser_manager_->start_count(), 0);
// Creating a ScopedKeepAlive does not start Lacros.
std::unique_ptr<BrowserManager::ScopedKeepAlive> keep_alive =
fake_browser_manager_->KeepAlive(BrowserManager::Feature::kTestOnly);
EXPECT_EQ(fake_browser_manager_->start_count(), 0);
// Simulate a Lacros termination, keep alive should launch Lacros in a
// windowless state.
fake_browser_manager_->SimulateLacrosTermination();
EXPECT_EQ(fake_browser_manager_->start_count(), 1);
EXPECT_CALL(mock_browser_service_, UpdateKeepAlive(_))
.Times(1)
.RetiresOnSaturation();
fake_browser_manager_->SimulateLacrosStart(&mock_browser_service_);
// Terminating again causes keep alive to again start Lacros in a windowless
// state.
fake_browser_manager_->SimulateLacrosTermination();
EXPECT_EQ(fake_browser_manager_->start_count(), 2);
EXPECT_CALL(mock_browser_service_, UpdateKeepAlive(_))
.Times(1)
.RetiresOnSaturation();
fake_browser_manager_->SimulateLacrosStart(&mock_browser_service_);
// Request a relaunch. Keep alive should not start Lacros in a windowless
// state but Lacros should instead start with the kRestoreLastSession action.
fake_browser_manager_->set_relaunch_requested_for_testing(true);
fake_browser_manager_->SimulateLacrosTermination();
EXPECT_EQ(fake_browser_manager_->start_count(), 3);
EXPECT_CALL(mock_browser_service_, UpdateKeepAlive(_))
.Times(1)
.RetiresOnSaturation();
EXPECT_CALL(mock_browser_service_, OpenForFullRestore(_))
.Times(1)
.RetiresOnSaturation();
fake_browser_manager_->SimulateLacrosStart(&mock_browser_service_);
// Resetting the relaunch requested bit should cause keep alive to start
// Lacros in a windowless state.
fake_browser_manager_->set_relaunch_requested_for_testing(false);
fake_browser_manager_->SimulateLacrosTermination();
EXPECT_EQ(fake_browser_manager_->start_count(), 4);
}
} // namespace crosapi