/*
* Copyright (C) 2019 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 <memory>
#include <type_traits>
#include <utility>
#include "gmock/gmock.h"
#include "gtest/gtest.h"
#include "idmap2/Result.h"
namespace android::idmap2 {
struct Container {
uint32_t value; // NOLINT(misc-non-private-member-variables-in-classes)
};
// Tests: Error
TEST(ResultTests, ErrorTraits) {
ASSERT_TRUE(std::is_move_constructible<Error>::value);
ASSERT_TRUE(std::is_move_assignable<Error>::value);
ASSERT_TRUE(std::is_copy_constructible<Error>::value);
ASSERT_TRUE(std::is_copy_assignable<Error>::value);
}
TEST(ResultTests, ErrorCtorFormat) {
Error e("%s=0x%08x", "resid", 0x7f010002);
ASSERT_EQ(e.GetMessage(), "resid=0x7f010002");
}
TEST(ResultTests, ErrorPropagateParent) {
Error e1("foo");
ASSERT_EQ(e1.GetMessage(), "foo");
Error e2(e1, "bar");
ASSERT_EQ(e2.GetMessage(), "foo -> bar");
Error e3(e2); // NOLINT(performance-unnecessary-copy-initialization)
ASSERT_EQ(e3.GetMessage(), "foo -> bar");
Error e4(e3, "%02d", 1);
ASSERT_EQ(e4.GetMessage(), "foo -> bar -> 01");
}
// Tests: Result<T> member functions
// Result(const Result&)
TEST(ResultTests, CopyConstructor) {
Result<uint32_t> r1(42U);
Result<uint32_t> r2(r1);
ASSERT_TRUE(r2);
ASSERT_EQ(*r2, 42U);
Result<uint32_t> r3 = r2;
ASSERT_TRUE(r3);
ASSERT_EQ(*r3, 42U);
}
// Result(const T&)
TEST(ResultTests, Constructor) {
uint32_t v = 42U;
Result<uint32_t> r1(v);
ASSERT_TRUE(r1);
ASSERT_EQ(*r1, 42U);
Error e("foo");
Result<uint32_t> r2(e);
ASSERT_FALSE(r2);
ASSERT_EQ(r2.GetErrorMessage(), "foo");
}
// Result(const T&&)
TEST(ResultTests, MoveConstructor) {
Result<uint32_t> r1(42U);
ASSERT_TRUE(r1);
ASSERT_EQ(*r1, 42U);
Result<uint32_t> r2(Error("foo"));
ASSERT_FALSE(r2);
ASSERT_EQ(r2.GetErrorMessage(), "foo");
}
// operator=
TEST(ResultTests, CopyAssignmentOperator) {
// note: 'Result<...> r2 = r1;' calls the copy ctor
Result<uint32_t> r1(42U);
Result<uint32_t> r2(0U);
r2 = r1;
ASSERT_TRUE(r2);
ASSERT_EQ(*r2, 42U);
Result<uint32_t> r3(Error("foo"));
r2 = r3;
ASSERT_FALSE(r2);
ASSERT_EQ(r2.GetErrorMessage(), "foo");
}
TEST(ResultTests, MoveAssignmentOperator) {
Result<uint32_t> r(0U);
r = Result<uint32_t>(42U);
ASSERT_TRUE(r);
ASSERT_EQ(*r, 42U);
r = Result<uint32_t>(Error("foo"));
ASSERT_FALSE(r);
ASSERT_EQ(r.GetErrorMessage(), "foo");
}
// operator bool()
TEST(ResultTests, BoolOperator) {
Result<uint32_t> r1(42U);
ASSERT_TRUE(r1);
ASSERT_EQ(*r1, 42U);
Result<uint32_t> r2(Error("foo"));
ASSERT_FALSE(r2);
ASSERT_EQ(r2.GetErrorMessage(), "foo");
}
// operator*
TEST(ResultTests, IndirectionOperator) {
const Result<uint32_t> r1(42U);
ASSERT_TRUE(r1);
ASSERT_EQ(*r1, 42U);
const Result<Container> r2(Container{42U});
ASSERT_TRUE(r2);
const Container& c = *r2;
ASSERT_EQ(c.value, 42U);
Result<Container> r3(Container{42U});
ASSERT_TRUE(r3);
ASSERT_EQ((*r3).value, 42U);
(*r3).value = 0U;
ASSERT_EQ((*r3).value, 0U);
}
// operator->
TEST(ResultTests, DereferenceOperator) {
const Result<Container> r1(Container{42U});
ASSERT_TRUE(r1);
ASSERT_EQ(r1->value, 42U);
Result<Container> r2(Container{42U});
ASSERT_TRUE(r2);
ASSERT_EQ(r2->value, 42U);
r2->value = 0U;
ASSERT_EQ(r2->value, 0U);
}
// Tests: intended use of Result<T>
TEST(ResultTests, ResultTraits) {
ASSERT_TRUE(std::is_move_constructible<Result<uint32_t>>::value);
ASSERT_TRUE(std::is_move_assignable<Result<uint32_t>>::value);
ASSERT_TRUE(std::is_copy_constructible<Result<uint32_t>>::value);
ASSERT_TRUE(std::is_copy_assignable<Result<uint32_t>>::value);
}
TEST(ResultTests, UnitTypeResult) {
Result<Unit> r(Unit{});
ASSERT_TRUE(r);
}
struct RefCountData {
int ctor; // NOLINT(misc-non-private-member-variables-in-classes)
int copy_ctor; // NOLINT(misc-non-private-member-variables-in-classes)
int dtor; // NOLINT(misc-non-private-member-variables-in-classes)
int move; // NOLINT(misc-non-private-member-variables-in-classes)
};
class RefCountContainer {
public:
explicit RefCountContainer(RefCountData& data) : data_(data) {
++data_.ctor;
}
RefCountContainer(RefCountContainer const&) = delete;
RefCountContainer(RefCountContainer&& rhs) noexcept : data_(rhs.data_) {
++data_.copy_ctor;
}
RefCountContainer& operator=(RefCountContainer const&) = delete;
RefCountContainer& operator=(RefCountContainer&& rhs) noexcept {
data_ = rhs.data_;
++data_.move;
return *this;
}
~RefCountContainer() {
++data_.dtor;
}
private:
RefCountData& data_;
};
TEST(ResultTests, ReferenceCount) {
ASSERT_TRUE(std::is_move_constructible<RefCountContainer>::value);
ASSERT_TRUE(std::is_move_assignable<RefCountContainer>::value);
ASSERT_FALSE(std::is_copy_constructible<RefCountContainer>::value);
ASSERT_FALSE(std::is_copy_assignable<RefCountContainer>::value);
RefCountData rc{0, 0, 0, 0};
{ Result<RefCountContainer> r(RefCountContainer{rc}); }
ASSERT_EQ(rc.ctor, 1);
ASSERT_EQ(rc.copy_ctor, 1);
ASSERT_EQ(rc.move, 0);
ASSERT_EQ(rc.dtor, 2);
}
Result<Container> CreateContainer(bool succeed) {
if (!succeed) {
return Error("foo");
}
return Container{42U};
}
TEST(ResultTests, FunctionReturn) {
auto r1 = CreateContainer(true);
ASSERT_TRUE(r1);
ASSERT_EQ(r1->value, 42U);
auto r2 = CreateContainer(false);
ASSERT_FALSE(r2);
ASSERT_EQ(r2.GetErrorMessage(), "foo");
ASSERT_EQ(r2.GetError().GetMessage(), "foo");
}
Result<Container> FailToCreateContainer() {
auto container = CreateContainer(false);
if (!container) {
return Error(container.GetError(), "bar");
}
return container;
}
TEST(ResultTests, CascadeError) {
auto container = FailToCreateContainer();
ASSERT_FALSE(container);
ASSERT_EQ(container.GetErrorMessage(), "foo -> bar");
}
struct NoCopyContainer {
uint32_t value; // NOLINT(misc-non-private-member-variables-in-classes)
DISALLOW_COPY_AND_ASSIGN(NoCopyContainer);
};
Result<std::unique_ptr<NoCopyContainer>> CreateNoCopyContainer(bool succeed) {
if (!succeed) {
return Error("foo");
}
std::unique_ptr<NoCopyContainer> p(new NoCopyContainer{0U});
p->value = 42U;
return std::move(p);
}
TEST(ResultTests, UniquePtr) {
auto r1 = CreateNoCopyContainer(true);
ASSERT_TRUE(r1);
ASSERT_EQ((*r1)->value, 42U);
(*r1)->value = 0U;
ASSERT_EQ((*r1)->value, 0U);
auto r2 = CreateNoCopyContainer(false);
ASSERT_FALSE(r2);
ASSERT_EQ(r2.GetErrorMessage(), "foo");
}
} // namespace android::idmap2