blob: 6ce9e1edde683d800e41d66281989ddc80f72a3e [file] [log] [blame] [edit]
// Copyright 2021 The ChromiumOS Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "missive/util/status_macros.h"
#include <memory>
#include <utility>
#include <base/functional/bind.h>
#include <base/functional/callback.h>
#include <gmock/gmock.h>
#include <gtest/gtest.h>
#include <gtest/gtest-spi.h>
#include "missive/util/status.h"
#include "missive/util/statusor.h"
namespace reporting {
namespace {
Status StatusTestFunction(bool fail) {
if (fail) {
return Status(error::INVALID_ARGUMENT, "Fail was true.");
}
return Status::StatusOK();
}
Status ReturnIfErrorStatusWrapperFunction(bool fail) {
RETURN_IF_ERROR_STATUS(StatusTestFunction(fail));
// Return error here to make sure that we aren't just returning the OK from
// StatusTestFunction.
return Status(error::INTERNAL, "Returning Internal Error");
}
// RETURN_IF_ERROR_STATUS macro actually returns on a non-OK status.
TEST(StatusMacros, ReturnsOnErrorStatus) {
Status test_status = ReturnIfErrorStatusWrapperFunction(/*fail=*/true);
EXPECT_FALSE(test_status.ok());
EXPECT_EQ(test_status.code(), error::INVALID_ARGUMENT);
}
// RETURN_IF_ERROR_STATUS macro continues on an OK status.
TEST(StatusMacros, ReturnIfErrorStatusContinuesOnOk) {
Status test_status = ReturnIfErrorStatusWrapperFunction(/*fail=*/false);
EXPECT_FALSE(test_status.ok());
EXPECT_EQ(test_status.code(), error::INTERNAL);
}
base::unexpected<Status> UnexpectedStatusTestFunction(bool fail) {
if (fail) {
return base::unexpected(Status(error::INVALID_ARGUMENT, "Fail was true."));
}
return base::unexpected(Status::StatusOK());
}
StatusOr<int> ReturnIfErrorUnexpectedStatusWrapperFunction(bool fail) {
RETURN_IF_ERROR_STATUS(UnexpectedStatusTestFunction(fail));
// Return error here to make sure that we aren't just returning the OK from
// UnexpectedStatusTestFunction.
return base::unexpected(Status(error::INTERNAL, "Returning Internal Error"));
}
// RETURN_IF_ERROR_STATUS macro actually returns on a non-OK status.
TEST(StatusMacros, ReturnsOnErrorUnexpectedStatus) {
StatusOr<int> test_status =
ReturnIfErrorUnexpectedStatusWrapperFunction(/*fail=*/true);
EXPECT_FALSE(test_status.has_value());
EXPECT_EQ(test_status.error().code(), error::INVALID_ARGUMENT);
}
// RETURN_IF_ERROR_STATUS macro continues on an OK status.
TEST(StatusMacros, ReturnIfErrorUnexpectedStatusContinuesOnOk) {
StatusOr<int> test_status =
ReturnIfErrorUnexpectedStatusWrapperFunction(/*fail=*/false);
EXPECT_FALSE(test_status.has_value());
EXPECT_EQ(test_status.error().code(), error::INTERNAL);
}
// Function to test StatusOr macros.
template <typename T>
StatusOr<T> AssignOrOnceCallbackAndReturnTestFunction(bool fail,
T return_value) {
if (fail) {
return base::unexpected(
Status(error::INVALID_ARGUMENT, "Test failure requested."));
}
return std::forward<T>(return_value);
}
// ASSIGN_OR_ONCE_CALLBACK_AND_RETURN testing
void AssignOrOnceCallbackWrapperFunction(
bool fail, base::OnceCallback<void(Status)> callback) {
constexpr int kReturnValue = 42;
int value;
ASSIGN_OR_ONCE_CALLBACK_AND_RETURN(
value, callback,
AssignOrOnceCallbackAndReturnTestFunction(fail, kReturnValue));
ASSERT_EQ(value, kReturnValue);
}
class CallbackTestClass {
public:
explicit CallbackTestClass(Status test_status) : test_status_(test_status) {}
void AssignInCallback(Status status) {
num_callback_invocations_++;
test_status_ = status;
}
int num_callback_invocations() { return num_callback_invocations_; }
Status status() { return test_status_; }
private:
Status test_status_;
int num_callback_invocations_ = 0;
};
// ASSIGN_OR_ONCE_CALLBACK_AND_RETURN assigns on OK error.
TEST(StatusMacros, OnceCallbackAssignOnOk) {
CallbackTestClass callback_test_class(Status::StatusOK());
base::OnceCallback<void(Status)> callback =
base::BindOnce(&CallbackTestClass::AssignInCallback,
base::Unretained(&callback_test_class));
AssignOrOnceCallbackWrapperFunction(/*fail=*/false, std::move(callback));
constexpr int kExpectedNumberOfCallbackInvocations = 0;
EXPECT_EQ(callback_test_class.num_callback_invocations(),
kExpectedNumberOfCallbackInvocations);
EXPECT_EQ(callback_test_class.status(), Status::StatusOK());
}
// ASSIGN_OR_ONCE_CALLBACK_AND_RETURN calls the callback and returns on non-OK
// status.
TEST(StatusMacros, OnceCallbackAndReturnOnError) {
CallbackTestClass callback_test_class(Status::StatusOK());
base::OnceCallback<void(Status)> callback =
base::BindOnce(&CallbackTestClass::AssignInCallback,
base::Unretained(&callback_test_class));
AssignOrOnceCallbackWrapperFunction(/*fail=*/true, std::move(callback));
constexpr int kExpectedNumberOfCallbackInvocations = 1;
EXPECT_EQ(callback_test_class.num_callback_invocations(),
kExpectedNumberOfCallbackInvocations);
EXPECT_EQ(callback_test_class.status().code(), error::INVALID_ARGUMENT);
}
void MultipleAssignOrOnceCallbackWrapperFunction(
base::OnceCallback<void(Status)> callback) {
constexpr int kReturnValue = 42;
constexpr bool kFail = false;
int value;
ASSIGN_OR_ONCE_CALLBACK_AND_RETURN(
value, callback,
AssignOrOnceCallbackAndReturnTestFunction(kFail, kReturnValue));
ASSIGN_OR_ONCE_CALLBACK_AND_RETURN(
value, callback,
AssignOrOnceCallbackAndReturnTestFunction(kFail, kReturnValue));
ASSIGN_OR_ONCE_CALLBACK_AND_RETURN(
value, callback,
AssignOrOnceCallbackAndReturnTestFunction(kFail, kReturnValue));
ASSERT_EQ(value, kReturnValue);
}
// ASSIGN_OR_ONCE_CALLBACK_AND_RETURN can be used multiple times in a function.
TEST(StatusMacros, MultipleAssignOrOnceCallbackCompletes) {
CallbackTestClass callback_test_class(Status::StatusOK());
base::OnceCallback<void(Status)> callback =
base::BindOnce(&CallbackTestClass::AssignInCallback,
base::Unretained(&callback_test_class));
MultipleAssignOrOnceCallbackWrapperFunction(std::move(callback));
constexpr int kExpectedNumberOfCallbackInvocations = 0;
EXPECT_EQ(callback_test_class.num_callback_invocations(),
kExpectedNumberOfCallbackInvocations);
EXPECT_EQ(callback_test_class.status(), Status::StatusOK());
}
TEST(StatusMacros, CheckOKOnStatus) {
const auto ok_status = Status::StatusOK();
CHECK_OK(ok_status);
CHECK_OK(ok_status) << "error message";
// rvalue
CHECK_OK(Status::StatusOK());
// Can't check on error status here because CHECK does not use gtest
// utilities.
}
TEST(StatusMacros, DCheckOKOnStatus) {
const auto ok_status = Status::StatusOK();
DCHECK_OK(ok_status);
DCHECK_OK(ok_status) << "error message";
// rvalue
DCHECK_OK(Status::StatusOK());
// Can't check on error status here because DCHECK does not use gtest
// utilities.
}
void AssertOKErrorStatus() {
ASSERT_OK(Status(error::INTERNAL, ""));
}
TEST(StatusMacros, AssertOKOnStatus) {
const auto ok_status = Status::StatusOK();
ASSERT_OK(ok_status);
ASSERT_OK(ok_status) << "error message";
// rvalue
ASSERT_OK(Status::StatusOK());
EXPECT_FATAL_FAILURE(AssertOKErrorStatus(), "error::INTERNAL");
}
void ExpectOKErrorStatus() {
EXPECT_OK(Status(error::INTERNAL, ""));
}
TEST(StatusMacros, ExpectOKOnStatus) {
EXPECT_OK(Status::StatusOK());
EXPECT_OK(Status::StatusOK()) << "error message";
EXPECT_NONFATAL_FAILURE(ExpectOKErrorStatus(), "error::INTERNAL");
}
TEST(StatusMacros, CheckOKOnStatusOr) {
StatusOr<int> status_or(2);
CHECK_OK(status_or);
CHECK_OK(status_or) << "error message";
// rvalue
CHECK_OK(StatusOr<int>(2));
// Can't check on error status here because CHECK does not use gtest
// utilities.
}
TEST(StatusMacros, DCheckOKOnStatusOr) {
StatusOr<int> status_or(2);
DCHECK_OK(status_or);
DCHECK_OK(status_or) << "error message";
// rvalue
DCHECK_OK(StatusOr<int>(2));
// Can't check on error status here because DCHECK does not use gtest
// utilities.
}
void AssertOKErrorStatusOr() {
ASSERT_OK(StatusOr<int>(base::unexpected(Status(error::INTERNAL, ""))));
}
TEST(StatusMacros, AssertOKOnStatusOr) {
StatusOr<int> status_or(2);
ASSERT_OK(status_or);
ASSERT_OK(status_or) << "error message";
// rvalue
ASSERT_OK(StatusOr<int>(2));
EXPECT_FATAL_FAILURE(AssertOKErrorStatusOr(), "error::INTERNAL");
}
void ExpectOKErrorStatusOr() {
EXPECT_OK(StatusOr<int>(base::unexpected(Status(error::INTERNAL, ""))));
}
TEST(StatusMacros, ExpectOKOnStatusOr) {
StatusOr<int> status_or(2);
EXPECT_OK(status_or);
EXPECT_OK(status_or) << "error message";
// rvalue
EXPECT_OK(StatusOr<int>(2));
EXPECT_NONFATAL_FAILURE(ExpectOKErrorStatusOr(), "error::INTERNAL");
}
} // namespace
} // namespace reporting