| // Copyright 2021 The Chromium OS Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include <memory> |
| #include <utility> |
| |
| #include <base/files/file_path.h> |
| #include <base/files/file_util.h> |
| #include <base/memory/scoped_refptr.h> |
| #include <base/test/task_environment.h> |
| #include <brillo/file_utils.h> |
| #include <gmock/gmock.h> |
| #include <gtest/gtest.h> |
| |
| #include "rmad/constants.h" |
| #include "rmad/metrics/mock_metrics_utils.h" |
| #include "rmad/state_handler/repair_complete_state_handler.h" |
| #include "rmad/state_handler/state_handler_test_common.h" |
| #include "rmad/system/mock_power_manager_client.h" |
| #include "rmad/utils/mock_sys_utils.h" |
| |
| using testing::_; |
| using testing::Assign; |
| using testing::DoAll; |
| using testing::NiceMock; |
| using testing::Return; |
| |
| namespace { |
| |
| constexpr char kPowerwashCountFilePath[] = "powerwash_count"; |
| |
| } // namespace |
| |
| namespace rmad { |
| |
| class RepairCompleteStateHandlerTest : public StateHandlerTest { |
| public: |
| // Helper class to mock the callback function to send signal. |
| class SignalSender { |
| public: |
| MOCK_METHOD(void, SendPowerCableStateSignal, (bool), (const)); |
| }; |
| |
| scoped_refptr<RepairCompleteStateHandler> CreateStateHandler( |
| bool* reboot_called = nullptr, |
| bool* shutdown_called = nullptr, |
| bool* metrics_called = nullptr, |
| bool record_metrics_success = true) { |
| auto mock_power_manager_client = |
| std::make_unique<NiceMock<MockPowerManagerClient>>(); |
| if (reboot_called) { |
| ON_CALL(*mock_power_manager_client, Restart()) |
| .WillByDefault(DoAll(Assign(reboot_called, true), Return(true))); |
| } else { |
| ON_CALL(*mock_power_manager_client, Restart()) |
| .WillByDefault(Return(true)); |
| } |
| if (shutdown_called) { |
| ON_CALL(*mock_power_manager_client, Shutdown()) |
| .WillByDefault(DoAll(Assign(shutdown_called, true), Return(true))); |
| } else { |
| ON_CALL(*mock_power_manager_client, Shutdown()) |
| .WillByDefault(Return(true)); |
| } |
| auto mock_sys_utils = std::make_unique<NiceMock<MockSysUtils>>(); |
| ON_CALL(*mock_sys_utils, IsPowerSourcePresent()) |
| .WillByDefault(Return(true)); |
| auto mock_metrics_utils = std::make_unique<NiceMock<MockMetricsUtils>>(); |
| ON_CALL(*mock_metrics_utils, Record(_, _)) |
| .WillByDefault(DoAll(Assign(metrics_called, true), |
| Return(record_metrics_success))); |
| auto handler = base::MakeRefCounted<RepairCompleteStateHandler>( |
| json_store_, GetTempDirPath(), GetTempDirPath(), |
| std::move(mock_power_manager_client), std::move(mock_sys_utils), |
| std::move(mock_metrics_utils)); |
| auto callback = |
| base::BindRepeating(&SignalSender::SendPowerCableStateSignal, |
| base::Unretained(&signal_sender_)); |
| handler->RegisterSignalSender(std::move(callback)); |
| |
| ON_CALL(signal_sender_, SendPowerCableStateSignal(_)) |
| .WillByDefault(Return()); |
| return handler; |
| } |
| |
| base::FilePath GetPowerwashCountFilePath() const { |
| return GetTempDirPath().AppendASCII(kPowerwashCountFilePath); |
| } |
| |
| base::FilePath GetPowerwashRequestFilePath() const { |
| return GetTempDirPath().AppendASCII(kPowerwashRequestFilePath); |
| } |
| |
| base::FilePath GetCutoffRequestFilePath() const { |
| return GetTempDirPath().AppendASCII(kCutoffRequestFilePath); |
| } |
| |
| base::FilePath GetDisablePowerwashFilePath() const { |
| return GetTempDirPath().AppendASCII(kDisablePowerwashFilePath); |
| } |
| |
| base::FilePath GetTestDirPath() const { |
| return GetTempDirPath().AppendASCII(kTestDirPath); |
| } |
| |
| protected: |
| NiceMock<SignalSender> signal_sender_; |
| |
| // Variables for TaskRunner. |
| base::test::SingleThreadTaskEnvironment task_environment_{ |
| base::test::TaskEnvironment::TimeSource::MOCK_TIME}; |
| }; |
| |
| TEST_F(RepairCompleteStateHandlerTest, |
| InitializeState_PowerwashRequired_Success) { |
| EXPECT_TRUE(json_store_->SetValue(kWipeDevice, true)); |
| base::WriteFile(GetPowerwashCountFilePath(), "1\n"); |
| auto handler = CreateStateHandler(); |
| EXPECT_EQ(handler->InitializeState(), RMAD_ERROR_OK); |
| EXPECT_TRUE(handler->GetState().repair_complete().powerwash_required()); |
| |
| int powerwash_count; |
| EXPECT_TRUE(json_store_->GetValue(kPowerwashCount, &powerwash_count)); |
| EXPECT_EQ(powerwash_count, 1); |
| |
| handler->RunState(); |
| |
| // Override signal sender mock. |
| EXPECT_CALL(signal_sender_, SendPowerCableStateSignal(_)) |
| .WillOnce([](bool is_connected) { |
| EXPECT_TRUE(is_connected); |
| }); |
| task_environment_.FastForwardBy( |
| RepairCompleteStateHandler::kReportPowerCableInterval); |
| |
| // Should not send signal after cleanup. |
| handler->CleanUpState(); |
| task_environment_.FastForwardBy( |
| RepairCompleteStateHandler::kReportPowerCableInterval); |
| } |
| |
| TEST_F(RepairCompleteStateHandlerTest, |
| InitializeState_PowerwashNotRequired_NoPowerwashCountFile) { |
| // powerwash_count not set. |
| EXPECT_TRUE(json_store_->SetValue(kWipeDevice, false)); |
| auto handler = CreateStateHandler(); |
| EXPECT_EQ(handler->InitializeState(), RMAD_ERROR_OK); |
| EXPECT_FALSE(handler->GetState().repair_complete().powerwash_required()); |
| |
| int powerwash_count; |
| EXPECT_TRUE(json_store_->GetValue(kPowerwashCount, &powerwash_count)); |
| EXPECT_EQ(powerwash_count, 0); |
| } |
| |
| TEST_F(RepairCompleteStateHandlerTest, InitializeState_Fail) { |
| // |kWipeDevice| not set. |
| auto handler = CreateStateHandler(); |
| EXPECT_EQ(handler->InitializeState(), |
| RMAD_ERROR_STATE_HANDLER_INITIALIZATION_FAILED); |
| } |
| |
| TEST_F(RepairCompleteStateHandlerTest, GetNextStateCase_PowerwashRequired) { |
| EXPECT_TRUE(json_store_->SetValue(kWipeDevice, true)); |
| base::WriteFile(GetPowerwashCountFilePath(), "1\n"); |
| bool reboot_called = false; |
| auto handler = CreateStateHandler(&reboot_called); |
| EXPECT_EQ(handler->InitializeState(), RMAD_ERROR_OK); |
| EXPECT_FALSE(base::PathExists(GetPowerwashRequestFilePath())); |
| |
| handler->RunState(); |
| |
| RmadState state; |
| state.mutable_repair_complete()->set_shutdown( |
| RepairCompleteState::RMAD_REPAIR_COMPLETE_BATTERY_CUTOFF); |
| state.mutable_repair_complete()->set_powerwash_required(true); |
| |
| { |
| auto [error, state_case] = handler->GetNextStateCase(state); |
| EXPECT_EQ(error, RMAD_ERROR_EXPECT_REBOOT); |
| EXPECT_EQ(state_case, RmadState::StateCase::kRepairComplete); |
| EXPECT_FALSE(reboot_called); |
| EXPECT_TRUE(base::PathExists(GetPowerwashRequestFilePath())); |
| } |
| |
| // A second call to |GetNextStateCase before rebooting is fine. |
| { |
| auto [error, state_case] = handler->GetNextStateCase(state); |
| EXPECT_EQ(error, RMAD_ERROR_EXPECT_REBOOT); |
| EXPECT_EQ(state_case, RmadState::StateCase::kRepairComplete); |
| EXPECT_FALSE(reboot_called); |
| EXPECT_TRUE(base::PathExists(GetPowerwashRequestFilePath())); |
| } |
| |
| // Reboot is called after a delay. |
| task_environment_.FastForwardBy(RepairCompleteStateHandler::kShutdownDelay); |
| EXPECT_TRUE(reboot_called); |
| } |
| |
| TEST_F(RepairCompleteStateHandlerTest, |
| GetNextStateCase_SkipPowerwash_PowerwashNotRequired_Reboot) { |
| EXPECT_TRUE(json_store_->SetValue(kWipeDevice, false)); |
| base::WriteFile(GetPowerwashCountFilePath(), "1\n"); |
| bool reboot_called = false, shutdown_called = false, metrics_called = false; |
| auto handler = |
| CreateStateHandler(&reboot_called, &shutdown_called, &metrics_called); |
| EXPECT_EQ(handler->InitializeState(), RMAD_ERROR_OK); |
| EXPECT_FALSE(base::PathExists(GetPowerwashRequestFilePath())); |
| |
| // Check that the state file exists now. |
| EXPECT_TRUE(base::PathExists(GetStateFilePath())); |
| |
| handler->RunState(); |
| |
| RmadState state; |
| state.mutable_repair_complete()->set_shutdown( |
| RepairCompleteState::RMAD_REPAIR_COMPLETE_REBOOT); |
| state.mutable_repair_complete()->set_powerwash_required(false); |
| |
| { |
| auto [error, state_case] = handler->GetNextStateCase(state); |
| EXPECT_EQ(error, RMAD_ERROR_EXPECT_REBOOT); |
| EXPECT_EQ(state_case, RmadState::StateCase::kRepairComplete); |
| EXPECT_FALSE(reboot_called); |
| EXPECT_FALSE(shutdown_called); |
| EXPECT_TRUE(metrics_called); |
| EXPECT_FALSE(base::PathExists(GetPowerwashRequestFilePath())); |
| EXPECT_FALSE(base::PathExists(GetCutoffRequestFilePath())); |
| EXPECT_FALSE(base::PathExists(GetStateFilePath())); |
| } |
| |
| // A second call to |GetNextStateCase| before rebooting is fine. |
| { |
| auto [error, state_case] = handler->GetNextStateCase(state); |
| EXPECT_EQ(error, RMAD_ERROR_EXPECT_REBOOT); |
| EXPECT_EQ(state_case, RmadState::StateCase::kRepairComplete); |
| EXPECT_FALSE(reboot_called); |
| EXPECT_FALSE(shutdown_called); |
| EXPECT_TRUE(metrics_called); |
| EXPECT_FALSE(base::PathExists(GetPowerwashRequestFilePath())); |
| EXPECT_FALSE(base::PathExists(GetCutoffRequestFilePath())); |
| EXPECT_FALSE(base::PathExists(GetStateFilePath())); |
| } |
| |
| // Reboot is called after a delay. |
| task_environment_.FastForwardBy(RepairCompleteStateHandler::kShutdownDelay); |
| EXPECT_TRUE(reboot_called); |
| EXPECT_FALSE(shutdown_called); |
| EXPECT_FALSE(base::PathExists(GetCutoffRequestFilePath())); |
| } |
| |
| TEST_F(RepairCompleteStateHandlerTest, |
| GetNextStateCase_SkipPowerwash_PowerwashNotRequired_Shutdown) { |
| EXPECT_TRUE(json_store_->SetValue(kWipeDevice, false)); |
| base::WriteFile(GetPowerwashCountFilePath(), "1\n"); |
| bool reboot_called = false, shutdown_called = false, metrics_called = false; |
| auto handler = |
| CreateStateHandler(&reboot_called, &shutdown_called, &metrics_called); |
| EXPECT_EQ(handler->InitializeState(), RMAD_ERROR_OK); |
| EXPECT_FALSE(base::PathExists(GetPowerwashRequestFilePath())); |
| |
| // Check that the state file exists now. |
| EXPECT_TRUE(base::PathExists(GetStateFilePath())); |
| |
| handler->RunState(); |
| |
| RmadState state; |
| state.mutable_repair_complete()->set_shutdown( |
| RepairCompleteState::RMAD_REPAIR_COMPLETE_SHUTDOWN); |
| state.mutable_repair_complete()->set_powerwash_required(false); |
| |
| { |
| auto [error, state_case] = handler->GetNextStateCase(state); |
| EXPECT_EQ(error, RMAD_ERROR_EXPECT_SHUTDOWN); |
| EXPECT_EQ(state_case, RmadState::StateCase::kRepairComplete); |
| EXPECT_FALSE(reboot_called); |
| EXPECT_FALSE(shutdown_called); |
| EXPECT_TRUE(metrics_called); |
| EXPECT_FALSE(base::PathExists(GetPowerwashRequestFilePath())); |
| EXPECT_FALSE(base::PathExists(GetCutoffRequestFilePath())); |
| EXPECT_FALSE(base::PathExists(GetStateFilePath())); |
| } |
| |
| // A second call to |GetNextStateCase| before shutting down is fine. |
| { |
| auto [error, state_case] = handler->GetNextStateCase(state); |
| EXPECT_EQ(error, RMAD_ERROR_EXPECT_SHUTDOWN); |
| EXPECT_EQ(state_case, RmadState::StateCase::kRepairComplete); |
| EXPECT_FALSE(reboot_called); |
| EXPECT_FALSE(shutdown_called); |
| EXPECT_TRUE(metrics_called); |
| EXPECT_FALSE(base::PathExists(GetPowerwashRequestFilePath())); |
| EXPECT_FALSE(base::PathExists(GetCutoffRequestFilePath())); |
| EXPECT_FALSE(base::PathExists(GetStateFilePath())); |
| } |
| |
| // Shutdown is called after a delay. |
| task_environment_.FastForwardBy(RepairCompleteStateHandler::kShutdownDelay); |
| EXPECT_FALSE(reboot_called); |
| EXPECT_TRUE(shutdown_called); |
| EXPECT_FALSE(base::PathExists(GetCutoffRequestFilePath())); |
| } |
| |
| TEST_F(RepairCompleteStateHandlerTest, |
| GetNextStateCase_SkipPowerwash_PowerwashNotRequired_Cutoff) { |
| EXPECT_TRUE(json_store_->SetValue(kWipeDevice, false)); |
| base::WriteFile(GetPowerwashCountFilePath(), "1\n"); |
| bool reboot_called = false, shutdown_called = false, metrics_called = false; |
| auto handler = |
| CreateStateHandler(&reboot_called, &shutdown_called, &metrics_called); |
| EXPECT_EQ(handler->InitializeState(), RMAD_ERROR_OK); |
| EXPECT_FALSE(base::PathExists(GetPowerwashRequestFilePath())); |
| |
| // Check that the state file exists now. |
| EXPECT_TRUE(base::PathExists(GetStateFilePath())); |
| |
| handler->RunState(); |
| |
| RmadState state; |
| state.mutable_repair_complete()->set_shutdown( |
| RepairCompleteState::RMAD_REPAIR_COMPLETE_BATTERY_CUTOFF); |
| state.mutable_repair_complete()->set_powerwash_required(false); |
| |
| { |
| auto [error, state_case] = handler->GetNextStateCase(state); |
| EXPECT_EQ(error, RMAD_ERROR_EXPECT_SHUTDOWN); |
| EXPECT_EQ(state_case, RmadState::StateCase::kRepairComplete); |
| EXPECT_FALSE(reboot_called); |
| EXPECT_FALSE(shutdown_called); |
| EXPECT_TRUE(metrics_called); |
| EXPECT_FALSE(base::PathExists(GetPowerwashRequestFilePath())); |
| EXPECT_FALSE(base::PathExists(GetCutoffRequestFilePath())); |
| EXPECT_FALSE(base::PathExists(GetStateFilePath())); |
| } |
| |
| // A second call to |GetNextStateCase| before rebooting is fine. |
| { |
| auto [error, state_case] = handler->GetNextStateCase(state); |
| EXPECT_EQ(error, RMAD_ERROR_EXPECT_SHUTDOWN); |
| EXPECT_EQ(state_case, RmadState::StateCase::kRepairComplete); |
| EXPECT_FALSE(reboot_called); |
| EXPECT_FALSE(shutdown_called); |
| EXPECT_TRUE(metrics_called); |
| EXPECT_FALSE(base::PathExists(GetPowerwashRequestFilePath())); |
| EXPECT_FALSE(base::PathExists(GetCutoffRequestFilePath())); |
| EXPECT_FALSE(base::PathExists(GetStateFilePath())); |
| } |
| |
| // Reboot and cutoff are called after a delay. |
| task_environment_.FastForwardBy(RepairCompleteStateHandler::kShutdownDelay); |
| EXPECT_TRUE(reboot_called); |
| EXPECT_FALSE(shutdown_called); |
| EXPECT_TRUE(base::PathExists(GetCutoffRequestFilePath())); |
| } |
| |
| TEST_F(RepairCompleteStateHandlerTest, |
| GetNextStateCase_SkipPowerwash_PowerwashComplete) { |
| EXPECT_TRUE(json_store_->SetValue(kWipeDevice, true)); |
| base::WriteFile(GetPowerwashCountFilePath(), "1\n"); |
| bool reboot_called = false, shutdown_called = false, metrics_called = false; |
| auto handler = |
| CreateStateHandler(&reboot_called, &shutdown_called, &metrics_called); |
| EXPECT_EQ(handler->InitializeState(), RMAD_ERROR_OK); |
| EXPECT_FALSE(base::PathExists(GetPowerwashRequestFilePath())); |
| |
| // Powerwash is complete. |
| base::WriteFile(GetPowerwashCountFilePath(), "2\n"); |
| |
| // Check that the state file exists now. |
| EXPECT_TRUE(base::PathExists(GetStateFilePath())); |
| |
| handler->RunState(); |
| |
| RmadState state; |
| state.mutable_repair_complete()->set_shutdown( |
| RepairCompleteState::RMAD_REPAIR_COMPLETE_REBOOT); |
| state.mutable_repair_complete()->set_powerwash_required(true); |
| |
| { |
| auto [error, state_case] = handler->GetNextStateCase(state); |
| EXPECT_EQ(error, RMAD_ERROR_EXPECT_REBOOT); |
| EXPECT_EQ(state_case, RmadState::StateCase::kRepairComplete); |
| EXPECT_FALSE(reboot_called); |
| EXPECT_FALSE(shutdown_called); |
| EXPECT_TRUE(metrics_called); |
| EXPECT_FALSE(base::PathExists(GetPowerwashRequestFilePath())); |
| EXPECT_FALSE(base::PathExists(GetCutoffRequestFilePath())); |
| EXPECT_FALSE(base::PathExists(GetStateFilePath())); |
| } |
| |
| // A second call to |GetNextStateCase| before rebooting is fine. |
| { |
| auto [error, state_case] = handler->GetNextStateCase(state); |
| EXPECT_EQ(error, RMAD_ERROR_EXPECT_REBOOT); |
| EXPECT_EQ(state_case, RmadState::StateCase::kRepairComplete); |
| EXPECT_FALSE(reboot_called); |
| EXPECT_FALSE(shutdown_called); |
| EXPECT_TRUE(metrics_called); |
| EXPECT_FALSE(base::PathExists(GetPowerwashRequestFilePath())); |
| EXPECT_FALSE(base::PathExists(GetCutoffRequestFilePath())); |
| EXPECT_FALSE(base::PathExists(GetStateFilePath())); |
| } |
| |
| // Reboot is called after a delay. |
| task_environment_.FastForwardBy(RepairCompleteStateHandler::kShutdownDelay); |
| EXPECT_TRUE(reboot_called); |
| EXPECT_FALSE(shutdown_called); |
| EXPECT_FALSE(base::PathExists(GetCutoffRequestFilePath())); |
| } |
| |
| TEST_F(RepairCompleteStateHandlerTest, |
| GetNextStateCase_SkipPowerwash_PowerwashDisabledManually) { |
| EXPECT_TRUE(json_store_->SetValue(kWipeDevice, true)); |
| base::WriteFile(GetPowerwashCountFilePath(), "1\n"); |
| bool reboot_called = false, shutdown_called = false, metrics_called = false; |
| auto handler = |
| CreateStateHandler(&reboot_called, &shutdown_called, &metrics_called); |
| EXPECT_EQ(handler->InitializeState(), RMAD_ERROR_OK); |
| EXPECT_FALSE(base::PathExists(GetPowerwashRequestFilePath())); |
| |
| // Powerwash is not done yet, but disabled manually. |
| brillo::TouchFile(GetDisablePowerwashFilePath()); |
| |
| // Check that the state file exists now. |
| EXPECT_TRUE(base::PathExists(GetStateFilePath())); |
| |
| handler->RunState(); |
| |
| RmadState state; |
| state.mutable_repair_complete()->set_shutdown( |
| RepairCompleteState::RMAD_REPAIR_COMPLETE_REBOOT); |
| state.mutable_repair_complete()->set_powerwash_required(true); |
| |
| { |
| auto [error, state_case] = handler->GetNextStateCase(state); |
| EXPECT_EQ(error, RMAD_ERROR_EXPECT_REBOOT); |
| EXPECT_EQ(state_case, RmadState::StateCase::kRepairComplete); |
| EXPECT_FALSE(reboot_called); |
| EXPECT_FALSE(shutdown_called); |
| EXPECT_TRUE(metrics_called); |
| EXPECT_FALSE(base::PathExists(GetPowerwashRequestFilePath())); |
| EXPECT_FALSE(base::PathExists(GetCutoffRequestFilePath())); |
| EXPECT_FALSE(base::PathExists(GetStateFilePath())); |
| } |
| |
| // A second call to |GetNextStateCase| before rebooting is fine. |
| { |
| auto [error, state_case] = handler->GetNextStateCase(state); |
| EXPECT_EQ(error, RMAD_ERROR_EXPECT_REBOOT); |
| EXPECT_EQ(state_case, RmadState::StateCase::kRepairComplete); |
| EXPECT_FALSE(reboot_called); |
| EXPECT_FALSE(shutdown_called); |
| EXPECT_TRUE(metrics_called); |
| EXPECT_FALSE(base::PathExists(GetPowerwashRequestFilePath())); |
| EXPECT_FALSE(base::PathExists(GetCutoffRequestFilePath())); |
| EXPECT_FALSE(base::PathExists(GetStateFilePath())); |
| } |
| |
| // Reboot is called after a delay. |
| task_environment_.FastForwardBy(RepairCompleteStateHandler::kShutdownDelay); |
| EXPECT_TRUE(reboot_called); |
| EXPECT_FALSE(shutdown_called); |
| EXPECT_FALSE(base::PathExists(GetCutoffRequestFilePath())); |
| } |
| |
| TEST_F(RepairCompleteStateHandlerTest, |
| GetNextStateCase_SkipPowerwash_PowerwashDisabledInTestMode) { |
| EXPECT_TRUE(json_store_->SetValue(kWipeDevice, true)); |
| base::WriteFile(GetPowerwashCountFilePath(), "1\n"); |
| bool reboot_called = false, shutdown_called = false, metrics_called = false; |
| auto handler = |
| CreateStateHandler(&reboot_called, &shutdown_called, &metrics_called); |
| EXPECT_EQ(handler->InitializeState(), RMAD_ERROR_OK); |
| EXPECT_FALSE(base::PathExists(GetPowerwashRequestFilePath())); |
| |
| // Powerwash is not done yet, but disabled in test mode. |
| brillo::TouchFile(GetTestDirPath()); |
| |
| // Check that the state file exists now. |
| EXPECT_TRUE(base::PathExists(GetStateFilePath())); |
| |
| handler->RunState(); |
| |
| RmadState state; |
| state.mutable_repair_complete()->set_shutdown( |
| RepairCompleteState::RMAD_REPAIR_COMPLETE_REBOOT); |
| state.mutable_repair_complete()->set_powerwash_required(true); |
| |
| { |
| auto [error, state_case] = handler->GetNextStateCase(state); |
| EXPECT_EQ(error, RMAD_ERROR_EXPECT_REBOOT); |
| EXPECT_EQ(state_case, RmadState::StateCase::kRepairComplete); |
| EXPECT_FALSE(reboot_called); |
| EXPECT_FALSE(shutdown_called); |
| EXPECT_TRUE(metrics_called); |
| EXPECT_FALSE(base::PathExists(GetPowerwashRequestFilePath())); |
| EXPECT_FALSE(base::PathExists(GetCutoffRequestFilePath())); |
| } |
| |
| // A second call to |GetNextStateCase| before rebooting is fine. |
| { |
| auto [error, state_case] = handler->GetNextStateCase(state); |
| EXPECT_EQ(error, RMAD_ERROR_EXPECT_REBOOT); |
| EXPECT_EQ(state_case, RmadState::StateCase::kRepairComplete); |
| EXPECT_FALSE(reboot_called); |
| EXPECT_FALSE(shutdown_called); |
| EXPECT_TRUE(metrics_called); |
| EXPECT_FALSE(base::PathExists(GetPowerwashRequestFilePath())); |
| EXPECT_FALSE(base::PathExists(GetCutoffRequestFilePath())); |
| } |
| |
| // Check that the state file is cleared. |
| EXPECT_FALSE(base::PathExists(GetStateFilePath())); |
| |
| // Reboot is called after a delay. |
| task_environment_.FastForwardBy(RepairCompleteStateHandler::kShutdownDelay); |
| EXPECT_TRUE(reboot_called); |
| EXPECT_FALSE(shutdown_called); |
| EXPECT_FALSE(base::PathExists(GetCutoffRequestFilePath())); |
| } |
| |
| TEST_F(RepairCompleteStateHandlerTest, GetNextStateCase_MissingState) { |
| EXPECT_TRUE(json_store_->SetValue(kWipeDevice, true)); |
| base::WriteFile(GetPowerwashCountFilePath(), "1\n"); |
| auto handler = CreateStateHandler(); |
| EXPECT_EQ(handler->InitializeState(), RMAD_ERROR_OK); |
| |
| // No RepairCompleteState. |
| RmadState state; |
| |
| // Check that the state file exists now. |
| EXPECT_TRUE(base::PathExists(GetStateFilePath())); |
| |
| auto [error, state_case] = handler->GetNextStateCase(state); |
| EXPECT_EQ(error, RMAD_ERROR_REQUEST_INVALID); |
| EXPECT_EQ(state_case, RmadState::StateCase::kRepairComplete); |
| |
| // Check that the state file still exists. |
| EXPECT_TRUE(base::PathExists(GetStateFilePath())); |
| } |
| |
| TEST_F(RepairCompleteStateHandlerTest, GetNextStateCase_MissingArgs) { |
| EXPECT_TRUE(json_store_->SetValue(kWipeDevice, true)); |
| base::WriteFile(GetPowerwashCountFilePath(), "1\n"); |
| auto handler = CreateStateHandler(); |
| EXPECT_EQ(handler->InitializeState(), RMAD_ERROR_OK); |
| |
| RmadState state; |
| state.mutable_repair_complete()->set_shutdown( |
| RepairCompleteState::RMAD_REPAIR_COMPLETE_UNKNOWN); |
| state.mutable_repair_complete()->set_powerwash_required(true); |
| |
| // Check that the state file exists now. |
| EXPECT_TRUE(base::PathExists(GetStateFilePath())); |
| |
| auto [error, state_case] = handler->GetNextStateCase(state); |
| EXPECT_EQ(error, RMAD_ERROR_REQUEST_ARGS_MISSING); |
| EXPECT_EQ(state_case, RmadState::StateCase::kRepairComplete); |
| |
| // Check that the state file still exists. |
| EXPECT_TRUE(base::PathExists(GetStateFilePath())); |
| } |
| |
| TEST_F(RepairCompleteStateHandlerTest, GetNextStateCase_ArgsViolation) { |
| EXPECT_TRUE(json_store_->SetValue(kWipeDevice, true)); |
| base::WriteFile(GetPowerwashCountFilePath(), "1\n"); |
| auto handler = CreateStateHandler(); |
| EXPECT_EQ(handler->InitializeState(), RMAD_ERROR_OK); |
| |
| RmadState state; |
| state.mutable_repair_complete()->set_shutdown( |
| RepairCompleteState::RMAD_REPAIR_COMPLETE_REBOOT); |
| state.mutable_repair_complete()->set_powerwash_required(false); |
| |
| // Check that the state file exists now. |
| EXPECT_TRUE(base::PathExists(GetStateFilePath())); |
| |
| auto [error, state_case] = handler->GetNextStateCase(state); |
| EXPECT_EQ(error, RMAD_ERROR_REQUEST_ARGS_VIOLATION); |
| EXPECT_EQ(state_case, RmadState::StateCase::kRepairComplete); |
| |
| // Check that the state file still exists. |
| EXPECT_TRUE(base::PathExists(GetStateFilePath())); |
| } |
| |
| TEST_F(RepairCompleteStateHandlerTest, GetNextStateCase_MetricsFailed) { |
| EXPECT_TRUE(json_store_->SetValue(kWipeDevice, false)); |
| base::WriteFile(GetPowerwashCountFilePath(), "1\n"); |
| bool reboot_called = false, shutdown_called = false, metrics_called = false; |
| auto handler = CreateStateHandler(&reboot_called, &shutdown_called, |
| &metrics_called, false); |
| EXPECT_EQ(handler->InitializeState(), RMAD_ERROR_OK); |
| |
| // Check that the state file exists now. |
| EXPECT_TRUE(base::PathExists(GetStateFilePath())); |
| |
| handler->RunState(); |
| |
| RmadState state; |
| state.mutable_repair_complete()->set_shutdown( |
| RepairCompleteState::RMAD_REPAIR_COMPLETE_BATTERY_CUTOFF); |
| |
| auto [error, state_case] = handler->GetNextStateCase(state); |
| // Structured metrics recording is expected to fail as current library does |
| // not support recording locally without user consent. We shouldn't let it |
| // block the flow until the library actually supports it. |
| EXPECT_EQ(error, RMAD_ERROR_EXPECT_SHUTDOWN); |
| EXPECT_EQ(state_case, RmadState::StateCase::kRepairComplete); |
| EXPECT_FALSE(reboot_called); |
| EXPECT_FALSE(shutdown_called); |
| EXPECT_TRUE(metrics_called); |
| EXPECT_FALSE(base::PathExists(GetPowerwashRequestFilePath())); |
| EXPECT_FALSE(base::PathExists(GetCutoffRequestFilePath())); |
| |
| // Check that the state file is cleared. |
| EXPECT_FALSE(base::PathExists(GetStateFilePath())); |
| |
| // Cutoff and reboot are called after a delay. |
| task_environment_.FastForwardBy(RepairCompleteStateHandler::kShutdownDelay); |
| EXPECT_TRUE(reboot_called); |
| EXPECT_FALSE(shutdown_called); |
| EXPECT_TRUE(base::PathExists(GetCutoffRequestFilePath())); |
| } |
| |
| TEST_F(RepairCompleteStateHandlerTest, GetNextStateCase_JsonFailed) { |
| EXPECT_TRUE(json_store_->SetValue(kWipeDevice, false)); |
| base::WriteFile(GetPowerwashCountFilePath(), "1\n"); |
| bool reboot_called = false, shutdown_called = false, metrics_called = false; |
| auto handler = |
| CreateStateHandler(&reboot_called, &shutdown_called, &metrics_called); |
| EXPECT_EQ(handler->InitializeState(), RMAD_ERROR_OK); |
| |
| // Check that the state file exists now. |
| EXPECT_TRUE(base::PathExists(GetStateFilePath())); |
| |
| handler->RunState(); |
| |
| RmadState state; |
| state.mutable_repair_complete()->set_shutdown( |
| RepairCompleteState::RMAD_REPAIR_COMPLETE_BATTERY_CUTOFF); |
| |
| // Make |json_store_| read-only. |
| base::SetPosixFilePermissions(GetStateFilePath(), 0444); |
| |
| auto [error, state_case] = handler->GetNextStateCase(state); |
| EXPECT_EQ(error, RMAD_ERROR_CANNOT_WRITE); |
| EXPECT_EQ(state_case, RmadState::StateCase::kRepairComplete); |
| EXPECT_FALSE(reboot_called); |
| EXPECT_FALSE(shutdown_called); |
| EXPECT_TRUE(metrics_called); |
| EXPECT_FALSE(base::PathExists(GetPowerwashRequestFilePath())); |
| EXPECT_FALSE(base::PathExists(GetCutoffRequestFilePath())); |
| |
| // Check that the shutdown action won't be called if the state file cannot be |
| // cleared. |
| task_environment_.FastForwardBy(RepairCompleteStateHandler::kShutdownDelay); |
| EXPECT_FALSE(reboot_called); |
| EXPECT_FALSE(shutdown_called); |
| EXPECT_FALSE(base::PathExists(GetCutoffRequestFilePath())); |
| } |
| |
| } // namespace rmad |