chromium/chrome/browser/ash/crosapi/rootfs_lacros_loader_unittest.cc

// Copyright 2023 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/rootfs_lacros_loader.h"

#include <memory>
#include <string>

#include "base/files/file_path.h"
#include "base/files/file_util.h"
#include "base/files/scoped_temp_dir.h"
#include "base/test/bind.h"
#include "base/test/test_future.h"
#include "base/version.h"
#include "chromeos/ash/components/dbus/upstart/fake_upstart_client.h"
#include "components/user_manager/fake_user_manager.h"
#include "components/user_manager/scoped_user_manager.h"
#include "content/public/test/browser_task_environment.h"
#include "testing/gtest/include/gtest/gtest.h"

namespace crosapi {
namespace {

// Copied from rootfs_lacros_loader.cc
constexpr char kLacrosMounterUpstartJob[] = "lacros_2dmounter";

class RootfsLacrosLoaderTest : public testing::Test {
 public:
  RootfsLacrosLoaderTest() {
    CHECK(temp_dir_.CreateUniqueTempDir());
    metadata_path_ = temp_dir_.GetPath().Append("metadata");
    base::WriteFile(metadata_path_,
                    "{\"content\":{\"version\":\"" + version_str + "\"}}");
    rootfs_lacros_loader_ = std::make_unique<RootfsLacrosLoader>(
        &fake_upstart_client_, metadata_path_);
  }

 protected:
  content::BrowserTaskEnvironment task_environment_;

  const std::string version_str = "1.0.0";
  base::ScopedTempDir temp_dir_;
  base::FilePath metadata_path_;

  user_manager::TypedScopedUserManager<user_manager::FakeUserManager>
      scoped_user_manager_{std::make_unique<user_manager::FakeUserManager>()};
  ash::FakeUpstartClient fake_upstart_client_;
  std::unique_ptr<RootfsLacrosLoader> rootfs_lacros_loader_;
};

TEST_F(RootfsLacrosLoaderTest, LoadRootfsLacrosSelectedByCompatibilityCheck) {
  bool callback_called = false;
  fake_upstart_client_.set_start_job_cb(base::BindRepeating(
      [](bool* callback_called, const std::string& job,
         const std::vector<std::string>& upstart_env) {
        EXPECT_EQ(job, kLacrosMounterUpstartJob);
        *callback_called = true;
        return ash::FakeUpstartClient::StartJobResult(true /* success */);
      },
      &callback_called));

  EXPECT_EQ(RootfsLacrosLoader::State::kNotLoaded,
            rootfs_lacros_loader_->GetState());

  // If rootfs is selected by compatibility check, it first calls GetVersion to
  // read the version, and then Load is requested. Inside GetVersion, Load won't
  // complete.
  base::test::TestFuture<const base::Version&> future1;
  rootfs_lacros_loader_->GetVersion(
      future1.GetCallback<const base::Version&>());
  EXPECT_EQ(base::Version(version_str), future1.Get<0>());
  EXPECT_EQ(RootfsLacrosLoader::State::kVersionReadyButNotLoaded,
            rootfs_lacros_loader_->GetState());
  EXPECT_FALSE(callback_called);

  // Load is called after version is calculated.
  base::test::TestFuture<base::Version, const base::FilePath&> future2;
  rootfs_lacros_loader_->Load(
      future2.GetCallback<base::Version, const base::FilePath&>(),
      /*forced=*/false);
  EXPECT_EQ(base::Version(version_str), future2.Get<0>());
  EXPECT_TRUE(callback_called);

  EXPECT_EQ(RootfsLacrosLoader::State::kLoaded,
            rootfs_lacros_loader_->GetState());
}

TEST_F(RootfsLacrosLoaderTest, LoadRootfsLacrosSelectedByPolicy) {
  bool callback_called = false;
  fake_upstart_client_.set_start_job_cb(base::BindRepeating(
      [](bool* callback_called, const std::string& job,
         const std::vector<std::string>& upstart_env) {
        EXPECT_EQ(job, kLacrosMounterUpstartJob);
        *callback_called = true;
        return ash::FakeUpstartClient::StartJobResult(true /* success */);
      },
      &callback_called));

  EXPECT_EQ(RootfsLacrosLoader::State::kNotLoaded,
            rootfs_lacros_loader_->GetState());

  // If rootfs is selected by policy, it does not call GetVersion. Instead, it
  // calls Load directly and compute read the version inside Load together.
  base::test::TestFuture<base::Version, const base::FilePath&> future;
  rootfs_lacros_loader_->Load(
      future.GetCallback<base::Version, const base::FilePath&>(),
      /*forced=*/false);
  EXPECT_EQ(base::Version(version_str), future.Get<0>());
  EXPECT_TRUE(callback_called);

  EXPECT_EQ(RootfsLacrosLoader::State::kLoaded,
            rootfs_lacros_loader_->GetState());
}

TEST_F(RootfsLacrosLoaderTest, UnloadRequestedOnVersionReady) {
  EXPECT_EQ(RootfsLacrosLoader::State::kNotLoaded,
            rootfs_lacros_loader_->GetState());

  // First, request loader to get version and stops at
  // `kVersionReadyButNotLoaded`.
  base::test::TestFuture<const base::Version&> future1;
  rootfs_lacros_loader_->GetVersion(
      future1.GetCallback<const base::Version&>());
  EXPECT_EQ(base::Version(version_str), future1.Get<0>());
  EXPECT_EQ(RootfsLacrosLoader::State::kVersionReadyButNotLoaded,
            rootfs_lacros_loader_->GetState());

  // Simulate the case that stateful is selected by compatibility check so that
  // it requests rootfs lacros loader to unload.
  base::test::TestFuture<void> future2;
  rootfs_lacros_loader_->Unload(future2.GetCallback());
  EXPECT_EQ(RootfsLacrosLoader::State::kUnloaded,
            rootfs_lacros_loader_->GetState());
}

}  // namespace
}  // namespace crosapi