// Copyright (c) 2018, The Monero Project
//
// All rights reserved.
//
// Redistribution and use in source and binary forms, with or without modification, are
// permitted provided that the following conditions are met:
//
// 1. Redistributions of source code must retain the above copyright notice, this list of
//    conditions and the following disclaimer.
//
// 2. Redistributions in binary form must reproduce the above copyright notice, this list
//    of conditions and the following disclaimer in the documentation and/or other
//    materials provided with the distribution.
//
// 3. Neither the name of the copyright holder nor the names of its contributors may be
//    used to endorse or promote products derived from this software without specific
//    prior written permission.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY
// EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
// MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL
// THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
// PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
// INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
// STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
// THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.

#include <gtest/gtest.h>

#include <boost/algorithm/string/predicate.hpp>
#include <boost/utility/string_ref.hpp>
#include <string>
#include <system_error>
#include <type_traits>

#include "common/expect.h"

namespace
{
    struct move_only;
    struct throw_construct;
    struct throw_copies;
    struct throw_moves;

    struct move_only
    {
        move_only() = default;
        move_only(move_only const&) = delete;
        move_only(move_only&&) = default;
        ~move_only() = default;
        move_only& operator=(move_only const&) = delete;
        move_only& operator=(move_only&&) = default;
    };

    struct throw_construct
    {
        throw_construct() {}
        throw_construct(int) {}
        throw_construct(throw_construct const&) = default;
        throw_construct(throw_construct&&) = default;
        ~throw_construct() = default;
        throw_construct& operator=(throw_construct const&) = default;
        throw_construct& operator=(throw_construct&&) = default;
    };

    struct throw_copies
    {
        throw_copies() noexcept {}
        throw_copies(throw_copies const&) {}
        throw_copies(throw_copies&&) = default;
        ~throw_copies() = default;
        throw_copies& operator=(throw_copies const&) { return *this; }
        throw_copies& operator=(throw_copies&&) = default;
        bool operator==(throw_copies const&) noexcept { return true; }
        bool operator==(throw_moves const&) noexcept { return true; }
    };

    struct throw_moves
    {
        throw_moves() noexcept {}
        throw_moves(throw_moves const&) = default;
        throw_moves(throw_moves&&) {}
        ~throw_moves() = default;
        throw_moves& operator=(throw_moves const&) = default;
        throw_moves& operator=(throw_moves&&) { return *this; }
        bool operator==(throw_moves const&) { return true; }
        bool operator==(throw_copies const&) { return true; }
    };

    template<typename T>
    void construction_bench()
    {
        EXPECT_TRUE(std::is_copy_constructible<expect<T>>());
        EXPECT_TRUE(std::is_move_constructible<expect<T>>());
        EXPECT_TRUE(std::is_copy_assignable<expect<T>>());
        EXPECT_TRUE(std::is_move_assignable<expect<T>>());
        EXPECT_TRUE(std::is_destructible<expect<T>>());
    }

    template<typename T>
    void noexcept_bench()
    {
        EXPECT_TRUE(std::is_nothrow_copy_constructible<expect<T>>());
        EXPECT_TRUE(std::is_nothrow_move_constructible<expect<T>>());
        EXPECT_TRUE(std::is_nothrow_copy_assignable<expect<T>>());
        EXPECT_TRUE(std::is_nothrow_move_assignable<expect<T>>());
        EXPECT_TRUE(std::is_nothrow_destructible<expect<T>>());

        EXPECT_TRUE(noexcept(bool(std::declval<expect<T>>())));
        EXPECT_TRUE(noexcept(std::declval<expect<T>>().has_error()));
        EXPECT_TRUE(noexcept(std::declval<expect<T>>().error()));
        EXPECT_TRUE(noexcept(std::declval<expect<T>>().equal(std::declval<expect<T>>())));
        EXPECT_TRUE(noexcept(std::declval<expect<T>>() == std::declval<expect<T>>()));
        EXPECT_TRUE(noexcept(std::declval<expect<T>>() != std::declval<expect<T>>()));
    }

    template<typename T>
    void conversion_bench()
    {
        EXPECT_TRUE((std::is_convertible<std::error_code, expect<T>>()));
        EXPECT_TRUE((std::is_convertible<std::error_code&&, expect<T>>()));
        EXPECT_TRUE((std::is_convertible<std::error_code&, expect<T>>()));
        EXPECT_TRUE((std::is_convertible<std::error_code const&, expect<T>>()));

        EXPECT_TRUE((std::is_constructible<expect<T>, std::error_code>()));
        EXPECT_TRUE((std::is_constructible<expect<T>, std::error_code&&>()));
        EXPECT_TRUE((std::is_constructible<expect<T>, std::error_code&>()));
        EXPECT_TRUE((std::is_constructible<expect<T>, std::error_code const&>()));
    }
}


TEST(Expect, Constructions)
{
    construction_bench<void>();
    construction_bench<int>();

    EXPECT_TRUE(std::is_constructible<expect<void>>());

    EXPECT_TRUE((std::is_constructible<expect<throw_construct>, expect<int>>()));

    EXPECT_TRUE(std::is_move_constructible<expect<move_only>>());
    EXPECT_TRUE(std::is_move_assignable<expect<move_only>>());
}

TEST(Expect, Conversions)
{
    struct implicit { implicit(int) {} };
    struct explicit_only { explicit explicit_only(int) {} };

    conversion_bench<void>();
    conversion_bench<int>();

    EXPECT_TRUE((std::is_convertible<int, expect<int>>()));
    EXPECT_TRUE((std::is_convertible<int&&, expect<int>>()));
    EXPECT_TRUE((std::is_convertible<int&, expect<int>>()));
    EXPECT_TRUE((std::is_convertible<int const, expect<int>>()));
    EXPECT_TRUE((std::is_convertible<expect<unsigned>, expect<int>>()));
    EXPECT_TRUE((std::is_convertible<expect<unsigned>&&, expect<int>>()));
    EXPECT_TRUE((std::is_convertible<expect<unsigned>&, expect<int>>()));
    EXPECT_TRUE((std::is_convertible<expect<unsigned> const&, expect<int>>()));
    EXPECT_TRUE((std::is_convertible<expect<int>, expect<implicit>>()));
    EXPECT_TRUE((std::is_convertible<expect<int>&&, expect<implicit>>()));
    EXPECT_TRUE((std::is_convertible<expect<int>&, expect<implicit>>()));
    EXPECT_TRUE((std::is_convertible<expect<int> const&, expect<implicit>>()));
    EXPECT_TRUE(!(std::is_convertible<expect<int>, expect<explicit_only>>()));
    EXPECT_TRUE(!(std::is_convertible<expect<int>&&, expect<explicit_only>>()));
    EXPECT_TRUE(!(std::is_convertible<expect<int>&, expect<explicit_only>>()));
    EXPECT_TRUE(!(std::is_convertible<expect<int> const&, expect<explicit_only>>()));

    EXPECT_TRUE((std::is_constructible<expect<int>, int>()));
    EXPECT_TRUE((std::is_constructible<expect<int>, int&&>()));
    EXPECT_TRUE((std::is_constructible<expect<int>, int&>()));
    EXPECT_TRUE((std::is_constructible<expect<int>, int const&>()));
    EXPECT_TRUE((std::is_constructible<expect<int>, expect<unsigned>>()));
    EXPECT_TRUE((std::is_constructible<expect<int>, expect<unsigned>&&>()));
    EXPECT_TRUE((std::is_constructible<expect<int>, expect<unsigned>&>()));
    EXPECT_TRUE((std::is_constructible<expect<int>, expect<unsigned> const&>()));
    EXPECT_TRUE((std::is_constructible<expect<implicit>, expect<int>>()));
    EXPECT_TRUE((std::is_constructible<expect<implicit>, expect<int>&&>()));
    EXPECT_TRUE((std::is_constructible<expect<implicit>, expect<int>&>()));
    EXPECT_TRUE((std::is_constructible<expect<implicit>, expect<int> const&>()));
    EXPECT_TRUE(!(std::is_constructible<expect<explicit_only>, expect<int>>()));
    EXPECT_TRUE(!(std::is_constructible<expect<explicit_only>, expect<int>&&>()));
    EXPECT_TRUE(!(std::is_constructible<expect<explicit_only>, expect<int>&>()));
    EXPECT_TRUE(!(std::is_constructible<expect<explicit_only>, expect<int> const&>()));

    EXPECT_EQ(expect<int>{expect<short>{100}}.value(), 100);

    expect<std::string> val1{std::string{}};
    expect<const char*> val2{"foo"};

    EXPECT_EQ(val1.value(), std::string{});
    EXPECT_EQ(val2.value(), std::string{"foo"});

    const expect<std::string> val3{val2};

    EXPECT_EQ(val1.value(), std::string{});
    EXPECT_EQ(val2.value(), std::string{"foo"});
    EXPECT_EQ(val3.value(), std::string{"foo"});

    val1 = val2;

    EXPECT_EQ(val1.value(), "foo");
    EXPECT_EQ(val2.value(), std::string{"foo"});
    EXPECT_EQ(val3.value(), "foo");
}

TEST(Expect, NoExcept)
{
    noexcept_bench<void>();
    noexcept_bench<int>();

    EXPECT_TRUE(std::is_nothrow_constructible<expect<void>>());

    EXPECT_TRUE((std::is_nothrow_constructible<expect<int>, int>()));
    EXPECT_TRUE((std::is_nothrow_constructible<expect<int>, expect<unsigned>>()));
    EXPECT_TRUE((std::is_nothrow_constructible<expect<int>, expect<unsigned>&&>()));
    EXPECT_TRUE((std::is_nothrow_constructible<expect<int>, expect<unsigned>&>()));
    EXPECT_TRUE((std::is_nothrow_constructible<expect<int>, expect<unsigned> const&>()));

    EXPECT_TRUE(noexcept(expect<int>{std::declval<expect<unsigned>&&>()}));
    EXPECT_TRUE(noexcept(expect<int>{std::declval<expect<unsigned> const&>()}));
    EXPECT_TRUE(noexcept(std::declval<expect<int>>().has_value()));
    EXPECT_TRUE(noexcept(*std::declval<expect<int>>()));
    EXPECT_TRUE(noexcept(std::declval<expect<int>>().equal(std::declval<expect<unsigned>>())));
    EXPECT_TRUE(noexcept(std::declval<expect<unsigned>>().equal(std::declval<expect<int>>())));
    EXPECT_TRUE(noexcept(std::declval<expect<int>>().equal(0)));
    EXPECT_TRUE(noexcept(std::declval<expect<int>>() == std::declval<expect<unsigned>>()));
    EXPECT_TRUE(noexcept(std::declval<expect<unsigned>>() == std::declval<expect<int>>()));
    EXPECT_TRUE(noexcept(std::declval<expect<int>>() == 0));
    EXPECT_TRUE(noexcept(0 == std::declval<expect<int>>()));
    EXPECT_TRUE(noexcept(std::declval<expect<int>>() != std::declval<expect<unsigned>>()));
    EXPECT_TRUE(noexcept(std::declval<expect<unsigned>>() != std::declval<expect<int>>()));
    EXPECT_TRUE(noexcept(std::declval<expect<int>>() != 0));
    EXPECT_TRUE(noexcept(0 != std::declval<expect<int>>()));

    EXPECT_TRUE((std::is_nothrow_constructible<expect<throw_construct>, std::error_code>()));
    EXPECT_TRUE((std::is_nothrow_constructible<expect<throw_construct>, std::error_code&&>()));
    EXPECT_TRUE((std::is_nothrow_constructible<expect<throw_construct>, std::error_code&>()));
    EXPECT_TRUE((std::is_nothrow_constructible<expect<throw_construct>, std::error_code const&>()));
    EXPECT_TRUE((std::is_nothrow_constructible<expect<throw_construct>, throw_construct>()));
    EXPECT_TRUE((std::is_nothrow_constructible<expect<throw_construct>, throw_construct&&>()));
    EXPECT_TRUE((std::is_nothrow_constructible<expect<throw_construct>, throw_construct&>()));
    EXPECT_TRUE((std::is_nothrow_constructible<expect<throw_construct>, throw_construct const&>()));
    EXPECT_TRUE(!(std::is_nothrow_constructible<expect<throw_construct>, expect<int>>()));
    EXPECT_TRUE(!(std::is_nothrow_constructible<expect<throw_construct>, expect<int>&&>()));
    EXPECT_TRUE(!(std::is_nothrow_constructible<expect<throw_construct>, expect<int>&>()));
    EXPECT_TRUE(!(std::is_nothrow_constructible<expect<throw_construct>, expect<int> const&>()));
    EXPECT_TRUE(std::is_nothrow_copy_constructible<expect<throw_construct>>());
    EXPECT_TRUE(std::is_nothrow_move_constructible<expect<throw_construct>>());
    EXPECT_TRUE(std::is_nothrow_copy_assignable<expect<throw_construct>>());
    EXPECT_TRUE(std::is_nothrow_move_assignable<expect<throw_construct>>());
    EXPECT_TRUE(std::is_nothrow_destructible<expect<throw_construct>>());

    EXPECT_TRUE((std::is_nothrow_constructible<expect<throw_copies>, std::error_code>()));
    EXPECT_TRUE((std::is_nothrow_constructible<expect<throw_copies>, std::error_code&&>()));
    EXPECT_TRUE((std::is_nothrow_constructible<expect<throw_copies>, std::error_code&>()));
    EXPECT_TRUE((std::is_nothrow_constructible<expect<throw_copies>, std::error_code const&>()));
    EXPECT_TRUE((std::is_nothrow_constructible<expect<throw_copies>, throw_copies>()));
    EXPECT_TRUE((std::is_nothrow_constructible<expect<throw_copies>, throw_copies&&>()));
    EXPECT_TRUE(!(std::is_nothrow_constructible<expect<throw_copies>, throw_copies&>()));
    EXPECT_TRUE(!(std::is_nothrow_constructible<expect<throw_copies>, throw_copies const&>()));
    EXPECT_TRUE(!std::is_nothrow_copy_constructible<expect<throw_copies>>());
    EXPECT_TRUE(std::is_nothrow_move_constructible<expect<throw_copies>>());
    EXPECT_TRUE(!std::is_nothrow_copy_assignable<expect<throw_copies>>());
    EXPECT_TRUE(std::is_nothrow_move_assignable<expect<throw_copies>>());
    EXPECT_TRUE(std::is_nothrow_destructible<expect<throw_copies>>());
    EXPECT_TRUE(noexcept(std::declval<expect<throw_copies>>().equal(std::declval<expect<throw_copies>>())));
    EXPECT_TRUE(noexcept(std::declval<expect<throw_copies>>().equal(std::declval<throw_copies>())));
    EXPECT_TRUE(noexcept(std::declval<expect<throw_copies>>() == std::declval<expect<throw_copies>>()));
    EXPECT_TRUE(noexcept(std::declval<expect<throw_copies>>() == std::declval<throw_copies>()));
    EXPECT_TRUE(noexcept(std::declval<throw_copies>() == std::declval<expect<throw_copies>>()));
    EXPECT_TRUE(noexcept(std::declval<expect<throw_copies>>() != std::declval<expect<throw_copies>>()));
    EXPECT_TRUE(noexcept(std::declval<expect<throw_copies>>() != std::declval<throw_copies>()));
    EXPECT_TRUE(noexcept(std::declval<throw_copies>() != std::declval<expect<throw_copies>>()));
    EXPECT_TRUE(noexcept(std::declval<expect<throw_copies>>().equal(std::declval<expect<throw_moves>>())));
    EXPECT_TRUE(noexcept(std::declval<expect<throw_copies>>().equal(std::declval<throw_moves>())));
    EXPECT_TRUE(noexcept(std::declval<expect<throw_copies>>() == std::declval<expect<throw_moves>>()));
    EXPECT_TRUE(noexcept(std::declval<expect<throw_copies>>() == std::declval<throw_moves>()));
    EXPECT_TRUE(noexcept(std::declval<throw_moves>() == std::declval<expect<throw_copies>>()));
    EXPECT_TRUE(noexcept(std::declval<expect<throw_copies>>() != std::declval<expect<throw_moves>>()));
    EXPECT_TRUE(noexcept(std::declval<expect<throw_copies>>() != std::declval<throw_moves>()));
    EXPECT_TRUE(noexcept(std::declval<throw_moves>() != std::declval<expect<throw_copies>>()));

    EXPECT_TRUE((std::is_nothrow_constructible<expect<throw_moves>, std::error_code>()));
    EXPECT_TRUE((std::is_nothrow_constructible<expect<throw_moves>, std::error_code&&>()));
    EXPECT_TRUE((std::is_nothrow_constructible<expect<throw_moves>, std::error_code&>()));
    EXPECT_TRUE((std::is_nothrow_constructible<expect<throw_moves>, std::error_code const&>()));
    EXPECT_TRUE(!(std::is_nothrow_constructible<expect<throw_moves>, throw_moves>()));
    EXPECT_TRUE(!(std::is_nothrow_constructible<expect<throw_moves>, throw_moves&&>()));
    EXPECT_TRUE(!(std::is_nothrow_constructible<expect<throw_moves>, throw_moves&>()));
    EXPECT_TRUE(!(std::is_nothrow_constructible<expect<throw_moves>, throw_moves const&>()));
    EXPECT_TRUE(std::is_nothrow_copy_constructible<expect<throw_moves>>());
    EXPECT_TRUE(!std::is_nothrow_move_constructible<expect<throw_moves>>());
    EXPECT_TRUE(std::is_nothrow_copy_assignable<expect<throw_moves>>());
    EXPECT_TRUE(!std::is_nothrow_move_assignable<expect<throw_moves>>());
    EXPECT_TRUE(std::is_nothrow_destructible<expect<throw_copies>>());
    EXPECT_TRUE(!noexcept(std::declval<expect<throw_moves>>().equal(std::declval<expect<throw_moves>>())));
    EXPECT_TRUE(!noexcept(std::declval<expect<throw_moves>>().equal(std::declval<throw_moves>())));
    EXPECT_TRUE(!noexcept(std::declval<expect<throw_moves>>() == std::declval<expect<throw_moves>>()));
    EXPECT_TRUE(!noexcept(std::declval<expect<throw_moves>>() == std::declval<throw_moves>()));
    EXPECT_TRUE(!noexcept(std::declval<throw_moves>() == std::declval<expect<throw_moves>>()));
    EXPECT_TRUE(!noexcept(std::declval<expect<throw_moves>>() != std::declval<expect<throw_moves>>()));
    EXPECT_TRUE(!noexcept(std::declval<expect<throw_moves>>() != std::declval<throw_moves>()));
    EXPECT_TRUE(!noexcept(std::declval<throw_moves>() != std::declval<expect<throw_moves>>()));
    EXPECT_TRUE(!noexcept(std::declval<expect<throw_moves>>().equal(std::declval<expect<throw_copies>>())));
    EXPECT_TRUE(!noexcept(std::declval<expect<throw_moves>>().equal(std::declval<throw_copies>())));
    EXPECT_TRUE(!noexcept(std::declval<expect<throw_moves>>() == std::declval<expect<throw_copies>>()));
    EXPECT_TRUE(!noexcept(std::declval<expect<throw_moves>>() == std::declval<throw_copies>()));
    EXPECT_TRUE(!noexcept(std::declval<throw_copies>() == std::declval<expect<throw_moves>>()));
    EXPECT_TRUE(!noexcept(std::declval<expect<throw_moves>>() != std::declval<expect<throw_copies>>()));
    EXPECT_TRUE(!noexcept(std::declval<expect<throw_moves>>() != std::declval<throw_copies>()));
    EXPECT_TRUE(!noexcept(std::declval<throw_copies>() != std::declval<expect<throw_moves>>()));
}

TEST(Expect, Trivial)
{
    EXPECT_TRUE(std::is_trivially_copy_constructible<expect<void>>());
    EXPECT_TRUE(std::is_trivially_move_constructible<expect<void>>());
    EXPECT_TRUE(std::is_trivially_destructible<expect<void>>());
}

TEST(Expect, Assignment)
{ 
    expect<std::string> val1{std::string{}};
    expect<std::string> val2{"foobar"};

    ASSERT_TRUE(val1.has_value());
    ASSERT_TRUE(val2.has_value());
    EXPECT_TRUE(bool(val1));
    EXPECT_TRUE(bool(val2));
    EXPECT_TRUE(!val1.has_error());
    EXPECT_TRUE(!val2.has_error());
    EXPECT_EQ(val1.value(), std::string{});
    EXPECT_TRUE(*val1 == std::string{});
    EXPECT_TRUE(boost::equals(val1->c_str(), ""));
    EXPECT_TRUE(val2.value() == "foobar");
    EXPECT_TRUE(*val2 == "foobar");
    EXPECT_TRUE(boost::equals(val2->c_str(), "foobar"));
    EXPECT_EQ(val1.error(), std::error_code{});
    EXPECT_EQ(val2.error(), std::error_code{});
    EXPECT_TRUE(!val1.equal(std::error_code{}));
    EXPECT_TRUE(!val2.equal(std::error_code{}));
    EXPECT_TRUE(!(val1 == std::error_code{}));
    EXPECT_TRUE(!(std::error_code{} == val1));
    EXPECT_TRUE(!(val2 == std::error_code{}));
    EXPECT_TRUE(!(std::error_code{} == val2));
    EXPECT_TRUE(val1 != std::error_code{});
    EXPECT_TRUE(std::error_code{} != val1);
    EXPECT_TRUE(val2 != std::error_code{});
    EXPECT_TRUE(std::error_code{} != val2);
    EXPECT_TRUE(!val1.matches(std::error_condition{}));
    EXPECT_TRUE(!val2.matches(std::error_condition{}));

    val1 = std::move(val2);

    ASSERT_TRUE(val1.has_value());
    ASSERT_TRUE(val2.has_value());
    EXPECT_TRUE(bool(val1));
    EXPECT_TRUE(bool(val2));
    EXPECT_TRUE(!val1.has_error());
    EXPECT_TRUE(!val2.has_error());
    EXPECT_EQ(val1.value(), "foobar");
    EXPECT_TRUE(*val1 == "foobar");
    EXPECT_TRUE(boost::equals(val1->c_str(), "foobar"));
    EXPECT_EQ(val2.value(), std::string{});
    EXPECT_TRUE(*val2 == std::string{});
    EXPECT_TRUE(boost::equals(val2->c_str(), ""));
    EXPECT_EQ(val1.error(), std::error_code{});
    EXPECT_EQ(val2.error(), std::error_code{});
    EXPECT_TRUE(!val1.equal(std::error_code{}));
    EXPECT_TRUE(!val2.equal(std::error_code{}));
    EXPECT_TRUE(!(val1 == std::error_code{}));
    EXPECT_TRUE(!(std::error_code{} == val1));
    EXPECT_TRUE(!(val2 == std::error_code{}));
    EXPECT_TRUE(!(std::error_code{} == val2));
    EXPECT_TRUE(val1 != std::error_code{});
    EXPECT_TRUE(std::error_code{} != val1);
    EXPECT_TRUE(val2 != std::error_code{});
    EXPECT_TRUE(std::error_code{} != val2);
    EXPECT_TRUE(!val1.matches(std::error_condition{}));
    EXPECT_TRUE(!val2.matches(std::error_condition{}));

    val2 = val1;

    ASSERT_TRUE(val1.has_value());
    ASSERT_TRUE(val2.has_value());
    EXPECT_TRUE(bool(val1));
    EXPECT_TRUE(bool(val2));
    EXPECT_TRUE(!val1.has_error());
    EXPECT_TRUE(!val2.has_error());
    EXPECT_EQ(val1.value(), "foobar");
    EXPECT_TRUE(*val1 == "foobar");
    EXPECT_TRUE(boost::equals(val1->c_str(), "foobar"));
    EXPECT_EQ(val2.value(), "foobar");
    EXPECT_TRUE(*val2 == "foobar");
    EXPECT_TRUE(boost::equals(val2->c_str(), "foobar"));
    EXPECT_EQ(val1.error(), std::error_code{});
    EXPECT_EQ(val2.error(), std::error_code{});
    EXPECT_TRUE(!val1.equal(std::error_code{}));
    EXPECT_TRUE(!val2.equal(std::error_code{}));
    EXPECT_TRUE(!(val1 == std::error_code{}));
    EXPECT_TRUE(!(std::error_code{} == val1));
    EXPECT_TRUE(!(val2 == std::error_code{}));
    EXPECT_TRUE(!(std::error_code{} == val2));
    EXPECT_TRUE(val1 != std::error_code{});
    EXPECT_TRUE(std::error_code{} != val1);
    EXPECT_TRUE(val2 != std::error_code{});
    EXPECT_TRUE(std::error_code{} != val2);
    EXPECT_TRUE(!val1.matches(std::error_condition{}));
    EXPECT_TRUE(!val2.matches(std::error_condition{}));

    val1 = make_error_code(common_error::kInvalidArgument);

    ASSERT_TRUE(val1.has_error());
    ASSERT_TRUE(val2.has_value());
    EXPECT_TRUE(!val1);
    EXPECT_TRUE(bool(val2));
    EXPECT_TRUE(!val1.has_value());
    EXPECT_TRUE(!val2.has_error());
    EXPECT_EQ(val1.error(), common_error::kInvalidArgument);
    EXPECT_TRUE(val1 == common_error::kInvalidArgument);
    EXPECT_TRUE(common_error::kInvalidArgument == val1);
    EXPECT_STREQ(val2.value().c_str(), "foobar");
    EXPECT_TRUE(*val2 == "foobar");
    EXPECT_TRUE(boost::equals(val2->c_str(), "foobar"));
    EXPECT_NE(val1.error(), std::error_code{});
    EXPECT_EQ(val2.error(), std::error_code{});
    EXPECT_TRUE(val1.equal(common_error::kInvalidArgument));
    EXPECT_TRUE(!val2.equal(std::error_code{}));
    EXPECT_TRUE(!(val1 == std::error_code{}));
    EXPECT_TRUE(!(std::error_code{} == val1));
    EXPECT_TRUE(!(val2 == std::error_code{}));
    EXPECT_TRUE(!(std::error_code{} == val2));
    EXPECT_TRUE(val1 != std::error_code{});
    EXPECT_TRUE(std::error_code{} != val1);
    EXPECT_TRUE(val2 != std::error_code{});
    EXPECT_TRUE(std::error_code{} != val2);
    EXPECT_TRUE(val1.matches(std::errc::invalid_argument));
    EXPECT_TRUE(!val1.matches(std::error_condition{}));
    EXPECT_TRUE(!val2.matches(std::error_condition{}));

    val2 = val1;

    ASSERT_TRUE(val1.has_error());
    ASSERT_TRUE(val2.has_error());
    EXPECT_TRUE(!val1);
    EXPECT_TRUE(!val2);
    EXPECT_TRUE(!val1.has_value());
    EXPECT_TRUE(!val2.has_value());
    EXPECT_EQ(val1.error(), common_error::kInvalidArgument);
    EXPECT_TRUE(val1 == common_error::kInvalidArgument);
    EXPECT_TRUE(common_error::kInvalidArgument == val1);
    EXPECT_EQ(val2.error(), common_error::kInvalidArgument);
    EXPECT_TRUE(val2 == common_error::kInvalidArgument);
    EXPECT_TRUE(common_error::kInvalidArgument == val2);
    EXPECT_NE(val1.error(), std::error_code{});
    EXPECT_NE(val2.error(), std::error_code{});
    EXPECT_TRUE(val1.equal(common_error::kInvalidArgument));
    EXPECT_TRUE(val2.equal(common_error::kInvalidArgument));
    EXPECT_TRUE(!(val1 == std::error_code{}));
    EXPECT_TRUE(!(std::error_code{} == val1));
    EXPECT_TRUE(!(val2 == std::error_code{}));
    EXPECT_TRUE(!(std::error_code{} == val2));
    EXPECT_TRUE(val1 != std::error_code{});
    EXPECT_TRUE(std::error_code{} != val1);
    EXPECT_TRUE(val2 != std::error_code{});
    EXPECT_TRUE(std::error_code{} != val2);
    EXPECT_TRUE(val1.matches(std::errc::invalid_argument));
    EXPECT_TRUE(val2.matches(std::errc::invalid_argument));
    EXPECT_TRUE(!val1.matches(std::error_condition{}));
    EXPECT_TRUE(!val2.matches(std::error_condition{}));

    val1 = std::string{"barfoo"};

    ASSERT_TRUE(val1.has_value());
    ASSERT_TRUE(val2.has_error());
    EXPECT_TRUE(bool(val1));
    EXPECT_TRUE(!val2);
    EXPECT_TRUE(!val1.has_error());
    EXPECT_TRUE(!val2.has_value());
    EXPECT_STREQ(val1.value().c_str(), "barfoo");
    EXPECT_TRUE(*val1 == "barfoo");
    EXPECT_TRUE(boost::equals(val1->c_str(), "barfoo"));
    EXPECT_EQ(val2.error(), common_error::kInvalidArgument);
    EXPECT_TRUE(val2 == common_error::kInvalidArgument);
    EXPECT_TRUE(common_error::kInvalidArgument == val2);
    EXPECT_EQ(val1.error(), std::error_code{});
    EXPECT_NE(val2.error(), std::error_code{});
    EXPECT_TRUE(!val1.equal(std::error_code{}));
    EXPECT_TRUE(val2.equal(common_error::kInvalidArgument));
    EXPECT_TRUE(!(val1 == std::error_code{}));
    EXPECT_TRUE(!(std::error_code{} == val1));
    EXPECT_TRUE(!(val2 == std::error_code{}));
    EXPECT_TRUE(!(std::error_code{} == val2));
    EXPECT_TRUE(val1 != std::error_code{});
    EXPECT_TRUE(std::error_code{} != val1);
    EXPECT_TRUE(val2 != std::error_code{});
    EXPECT_TRUE(std::error_code{} != val2);
    EXPECT_TRUE(val2.matches(std::errc::invalid_argument));
    EXPECT_TRUE(!val1.matches(std::error_condition{}));
    EXPECT_TRUE(!val2.matches(std::error_condition{}));

    val2 = val1;

    ASSERT_TRUE(val1.has_value());
    ASSERT_TRUE(val2.has_value());
    EXPECT_TRUE(bool(val1));
    EXPECT_TRUE(bool(val2));
    EXPECT_TRUE(!val1.has_error());
    EXPECT_TRUE(!val2.has_error());
    EXPECT_EQ(val1.value(), "barfoo");
    EXPECT_TRUE(*val1 == "barfoo");
    EXPECT_TRUE(boost::equals(val1->c_str(), "barfoo"));
    EXPECT_EQ(val2.value(), "barfoo");
    EXPECT_TRUE(*val2 == "barfoo");
    EXPECT_TRUE(boost::equals(val2->c_str(), "barfoo"));
    EXPECT_EQ(val1.error(), std::error_code{});
    EXPECT_EQ(val2.error(), std::error_code{});
    EXPECT_TRUE(!val1.equal(std::error_code{}));
    EXPECT_TRUE(!val2.equal(std::error_code{}));
    EXPECT_TRUE(!(val1 == std::error_code{}));
    EXPECT_TRUE(!(std::error_code{} == val1));
    EXPECT_TRUE(!(val2 == std::error_code{}));
    EXPECT_TRUE(!(std::error_code{} == val2));
    EXPECT_TRUE(val1 != std::error_code{});
    EXPECT_TRUE(std::error_code{} != val1);
    EXPECT_TRUE(val2 != std::error_code{});
    EXPECT_TRUE(std::error_code{} != val2);
    EXPECT_TRUE(!val1.matches(std::error_condition{}));
    EXPECT_TRUE(!val2.matches(std::error_condition{}));
}

TEST(Expect, AssignmentThrowsOnMove)
{
    struct construct_error {};
    struct assignment_error {};

    struct throw_on_move {
        std::string msg;

        throw_on_move(const char* msg) : msg(msg) {}
        throw_on_move(throw_on_move&&) {
            throw construct_error{};
        }
        throw_on_move(throw_on_move const&) = default;
        ~throw_on_move() = default;
        throw_on_move& operator=(throw_on_move&&) {
            throw assignment_error{};
        }
        throw_on_move& operator=(throw_on_move const&) = default;
    };

    expect<throw_on_move> val1{expect<const char*>{"foobar"}};
    expect<throw_on_move> val2{common_error::kInvalidArgument};

    ASSERT_TRUE(val1.has_value());
    ASSERT_TRUE(val2.has_error());
    EXPECT_TRUE(!val1.has_error());
    EXPECT_TRUE(!val2.has_value());
    EXPECT_STREQ(val1->msg.c_str(), "foobar");
    EXPECT_EQ(val2.error(), common_error::kInvalidArgument);

    EXPECT_THROW(val2 = std::move(val1), construct_error);

    ASSERT_TRUE(val1.has_value());
    ASSERT_TRUE(val2.has_error());
    EXPECT_TRUE(!val1.has_error());
    EXPECT_TRUE(!val2.has_value());
    EXPECT_STREQ(val1->msg.c_str(), "foobar");
    EXPECT_EQ(val2.error(), common_error::kInvalidArgument);

    EXPECT_THROW(val1 = expect<const char*>{"barfoo"}, assignment_error);

    ASSERT_TRUE(val1.has_value());
    ASSERT_TRUE(val2.has_error());
    EXPECT_TRUE(!val1.has_error());
    EXPECT_TRUE(!val2.has_value());
    EXPECT_STREQ(val1->msg.c_str(), "foobar");
    EXPECT_EQ(val2.error(), common_error::kInvalidArgument);

    EXPECT_NO_THROW(val2 = val1);

    ASSERT_TRUE(val1.has_value());
    ASSERT_TRUE(val2.has_value());
    EXPECT_TRUE(!val1.has_error());
    EXPECT_TRUE(!val2.has_error());
    EXPECT_STREQ(val1->msg.c_str(), "foobar");
    EXPECT_STREQ(val2->msg.c_str(), "foobar");
}

TEST(Expect, EqualWithStrings)
{
    expect<std::string> val1{std::string{}};
    expect<std::string> val2{"barfoo"};
    expect<boost::string_ref> val3{boost::string_ref{}};

    EXPECT_TRUE(!val1.equal(val2));
    EXPECT_TRUE(val1.equal(val3));
    EXPECT_TRUE(!val2.equal(val1));
    EXPECT_TRUE(!val2.equal(val3));
    EXPECT_TRUE(val3.equal(val1));
    EXPECT_TRUE(!val3.equal(val2));
    EXPECT_TRUE(!(val1 == val2));
    EXPECT_TRUE(!(val2 == val1));
    EXPECT_TRUE(val1 == val3);
    EXPECT_TRUE(val3 == val1);
    EXPECT_TRUE(!(val2 == val3));
    EXPECT_TRUE(!(val3 == val2));
    EXPECT_TRUE(val1 != val2);
    EXPECT_TRUE(val2 != val1);
    EXPECT_TRUE(!(val1 != val3));
    EXPECT_TRUE(!(val3 != val1));
    EXPECT_TRUE(val2 != val3);
    EXPECT_TRUE(val3 != val2);

    EXPECT_TRUE(val1.equal(""));
    EXPECT_TRUE(val2.equal("barfoo"));
    EXPECT_TRUE(val3.equal(""));
    EXPECT_TRUE(!val1.equal(std::error_code{}));
    EXPECT_TRUE(!val2.equal(std::error_code{}));
    EXPECT_TRUE(!val3.equal(std::error_code{}));
    EXPECT_TRUE(val1 == "");
    EXPECT_TRUE("" == val1);
    EXPECT_TRUE(val2 == "barfoo");
    EXPECT_TRUE("barfoo" == val2);
    EXPECT_TRUE(val3 == "");
    EXPECT_TRUE("" == val3);
    EXPECT_TRUE(!(val1 != ""));
    EXPECT_TRUE(!("" != val1));
    EXPECT_TRUE(!(val2 != "barfoo"));
    EXPECT_TRUE(!("barfoo" != val2));
    EXPECT_TRUE(!(val3 != ""));
    EXPECT_TRUE(!("" != val3));
    EXPECT_TRUE(!(val1 == std::error_code{}));
    EXPECT_TRUE(!(std::error_code{} == val1));
    EXPECT_TRUE(!(val2 == std::error_code{}));
    EXPECT_TRUE(!(std::error_code{} == val2));
    EXPECT_TRUE(!(val3 == std::error_code{}));
    EXPECT_TRUE(!(std::error_code{} == val3));
    EXPECT_TRUE(val1 != std::error_code{});
    EXPECT_TRUE(std::error_code{} != val1);
    EXPECT_TRUE(val2 != std::error_code{});
    EXPECT_TRUE(std::error_code{} != val2);
    EXPECT_TRUE(val3 != std::error_code{});
    EXPECT_TRUE(std::error_code{} != val3);
    EXPECT_TRUE(!val1.matches(std::error_condition{}));
    EXPECT_TRUE(!val2.matches(std::error_condition{}));
    EXPECT_TRUE(!val3.matches(std::error_condition{}));

    val2 = make_error_code(common_error::kInvalidArgument);

    EXPECT_TRUE(!val1.equal(val2));
    EXPECT_TRUE(val1.equal(val3));
    EXPECT_TRUE(!val2.equal(val1));
    EXPECT_TRUE(!val2.equal(val3));
    EXPECT_TRUE(val3.equal(val1));
    EXPECT_TRUE(!val3.equal(val2));
    EXPECT_TRUE(!(val1 == val2));
    EXPECT_TRUE(!(val2 == val1));
    EXPECT_TRUE(val1 == val3);
    EXPECT_TRUE(val3 == val1);
    EXPECT_TRUE(!(val2 == val3));
    EXPECT_TRUE(!(val3 == val2));
    EXPECT_TRUE(val1 != val2);
    EXPECT_TRUE(val2 != val1);
    EXPECT_TRUE(!(val1 != val3));
    EXPECT_TRUE(!(val3 != val1));
    EXPECT_TRUE(val2 != val3);
    EXPECT_TRUE(val3 != val2);

    EXPECT_TRUE(!val1.equal(common_error::kInvalidArgument));
    EXPECT_TRUE(val2.equal(common_error::kInvalidArgument));
    EXPECT_TRUE(!val3.equal(common_error::kInvalidArgument));
    EXPECT_TRUE(val2 == common_error::kInvalidArgument);
    EXPECT_TRUE(common_error::kInvalidArgument == val2);
    EXPECT_TRUE(!(val2 != common_error::kInvalidArgument));
    EXPECT_TRUE(!(common_error::kInvalidArgument != val2));
    EXPECT_TRUE(val2.matches(std::errc::invalid_argument));
    EXPECT_TRUE(!val2.matches(std::error_condition{}));

    val1 = expect<std::string>{"barfoo"};

    EXPECT_TRUE(!val1.equal(val2));
    EXPECT_TRUE(!val1.equal(val3));
    EXPECT_TRUE(!val2.equal(val1));
    EXPECT_TRUE(!val2.equal(val3));
    EXPECT_TRUE(!val3.equal(val1));
    EXPECT_TRUE(!val3.equal(val2));
    EXPECT_TRUE(!(val1 == val2));
    EXPECT_TRUE(!(val2 == val1));
    EXPECT_TRUE(!(val1 == val3));
    EXPECT_TRUE(!(val3 == val1));
    EXPECT_TRUE(!(val2 == val3));
    EXPECT_TRUE(!(val3 == val2));
    EXPECT_TRUE(val1 != val2);
    EXPECT_TRUE(val2 != val1);
    EXPECT_TRUE(val1 != val3);
    EXPECT_TRUE(val3 != val1);
    EXPECT_TRUE(val2 != val3);
    EXPECT_TRUE(val3 != val2);

    EXPECT_TRUE(val1.equal("barfoo"));
    EXPECT_TRUE(val1 == "barfoo");
    EXPECT_TRUE("barfoo" == val1);
    EXPECT_TRUE(!(val1 != "barfoo"));
    EXPECT_TRUE(!("barfoo" != val1));
    EXPECT_TRUE(!val1.equal(common_error::kInvalidArgument));
    EXPECT_TRUE(!(val1 == common_error::kInvalidArgument));
    EXPECT_TRUE(!(common_error::kInvalidArgument == val1));
    EXPECT_TRUE(!(val1 == std::error_code{}));
    EXPECT_TRUE(!(std::error_code{} == val1));
    EXPECT_TRUE(!val1.matches(std::error_condition{}));
    EXPECT_TRUE(!val1.matches(std::errc::invalid_argument));
}

TEST(Expect, EqualWithVoid)
{
    const expect<void> val1;
    expect<void> val2;

    EXPECT_TRUE(val1.equal(val2));
    EXPECT_TRUE(val2.equal(val1));
    EXPECT_TRUE(!val1.equal(std::error_code{}));
    EXPECT_TRUE(!val2.equal(std::error_code{}));
    EXPECT_TRUE(val1 == val2);
    EXPECT_TRUE(val2 == val1);
    EXPECT_TRUE(!(val1 == std::error_code{}));
    EXPECT_TRUE(!(std::error_code{} == val1));
    EXPECT_TRUE(!(val2 == std::error_code{}));
    EXPECT_TRUE(!(std::error_code{} == val2));
    EXPECT_TRUE(!(val1 != val2));
    EXPECT_TRUE(!(val2 != val1));
    EXPECT_TRUE(val1 != std::error_code{});
    EXPECT_TRUE(std::error_code{} != val1);
    EXPECT_TRUE(!(val2 == std::error_code{}));
    EXPECT_TRUE(!(std::error_code{} == val2));

    val2 = make_error_code(common_error::kInvalidArgument);

    EXPECT_TRUE(!val1.equal(val2));
    EXPECT_TRUE(!val2.equal(val1));
    EXPECT_TRUE(!val1.equal(common_error::kInvalidArgument));
    EXPECT_TRUE(val2.equal(common_error::kInvalidArgument));
    EXPECT_TRUE(!val2.equal(std::error_code{}));
    EXPECT_TRUE(!(val1 == val2));
    EXPECT_TRUE(!(val2 == val1));
    EXPECT_TRUE(val2 == common_error::kInvalidArgument);
    EXPECT_TRUE(common_error::kInvalidArgument == val2);
    EXPECT_TRUE(!(val2 == std::error_code{}));
    EXPECT_TRUE(!(std::error_code{} == val2));
    EXPECT_TRUE(val1 != val2);
    EXPECT_TRUE(val2 != val1);
    EXPECT_TRUE(!(val2 != common_error::kInvalidArgument));
    EXPECT_TRUE(!(common_error::kInvalidArgument != val2));
    EXPECT_TRUE(val2 != std::error_code{});
    EXPECT_TRUE(std::error_code{} != val2);
}

TEST(Expect, EqualNoCopies)
{
    struct copy_error {};

    struct throw_on_copy {
        throw_on_copy() = default;
        throw_on_copy(int) noexcept {}
        throw_on_copy(throw_on_copy const&) {
            throw copy_error{};
        }
        ~throw_on_copy() = default;
        throw_on_copy& operator=(throw_on_copy const&) {
            throw copy_error{};
        }

        bool operator==(throw_on_copy const&) const noexcept { return true; }
    };

    expect<throw_on_copy> val1{expect<int>{0}};
    expect<throw_on_copy> val2{expect<int>{0}};

    EXPECT_TRUE(val1.equal(val2));
    EXPECT_TRUE(val2.equal(val1));
    EXPECT_TRUE(val1 == val2);
    EXPECT_TRUE(val2 == val1);
    EXPECT_TRUE(!(val1 != val2));
    EXPECT_TRUE(!(val2 != val1));

    EXPECT_TRUE(val1.equal(throw_on_copy{}));
    EXPECT_TRUE(val1 == throw_on_copy{});
    EXPECT_TRUE(throw_on_copy{} == val1);
    EXPECT_TRUE(!(val1 != throw_on_copy{}));
    EXPECT_TRUE(!(throw_on_copy{} != val1));

    throw_on_copy val3;

    EXPECT_TRUE(val1.equal(val3));
    EXPECT_TRUE(val1 == val3);
    EXPECT_TRUE(val3 == val1);
    EXPECT_TRUE(!(val1 != val3));
    EXPECT_TRUE(!(val3 != val1));

    expect<throw_on_copy> val4{common_error::kInvalidArgument};

    EXPECT_TRUE(!val4.equal(throw_on_copy{}));
    EXPECT_TRUE(!(val4 == throw_on_copy{}));
    EXPECT_TRUE(!(throw_on_copy{} == val4));
    EXPECT_TRUE(val4 != throw_on_copy{});
    EXPECT_TRUE(throw_on_copy{} != val4);
    EXPECT_TRUE(!val4.equal(val3));
    EXPECT_TRUE(!(val4 == val3));
    EXPECT_TRUE(!(val3 == val4));
    EXPECT_TRUE(val4 != val3);
    EXPECT_TRUE(val3 != val4);
}

TEST(Expect, Macros) {
    EXPECT_TRUE(
        [] () -> ::common_error {
            MONERO_PRECOND(true);
            return {common_error::kInvalidErrorCode};
        } () == common_error::kInvalidErrorCode
    );
    EXPECT_TRUE(
        [] () -> ::common_error {
            MONERO_PRECOND(false);
            return {common_error::kInvalidErrorCode};
        } () == common_error::kInvalidArgument
    );
    EXPECT_TRUE(
        [] () -> std::error_code {
            MONERO_PRECOND(true);
            return {common_error::kInvalidErrorCode};
        } () == common_error::kInvalidErrorCode
    );
    EXPECT_TRUE(
        [] () -> std::error_code {
            MONERO_PRECOND(false);
            return {common_error::kInvalidErrorCode};
        } () == common_error::kInvalidArgument
    );
    EXPECT_TRUE(
        [] () -> expect<void> {
            MONERO_PRECOND(true);
            return {common_error::kInvalidErrorCode};
        } () == common_error::kInvalidErrorCode
    );
    EXPECT_TRUE(
        [] () -> expect<void> {
            MONERO_PRECOND(false);
            return {common_error::kInvalidErrorCode};
        } () == common_error::kInvalidArgument
    );
    EXPECT_TRUE(
        [] () -> expect<int> {
            MONERO_PRECOND(true);
            return {common_error::kInvalidErrorCode};
        } () == common_error::kInvalidErrorCode
    );
    EXPECT_TRUE(
        [] () -> expect<int> {
            MONERO_PRECOND(false);
            return {common_error::kInvalidErrorCode};
        } () == common_error::kInvalidArgument
    );

    EXPECT_TRUE(
        [] () -> std::error_code {
            MONERO_CHECK(expect<void>{});
            return {common_error::kInvalidErrorCode};
        } () == common_error::kInvalidErrorCode
    );
    EXPECT_TRUE(
        [] () -> std::error_code {
            MONERO_CHECK(expect<void>{common_error::kInvalidArgument});
            return {common_error::kInvalidErrorCode};
        } () == common_error::kInvalidArgument
    );
    EXPECT_TRUE(
        [] () -> expect<void> {
            MONERO_CHECK(expect<void>{});
            return {common_error::kInvalidErrorCode};
        } () == common_error::kInvalidErrorCode
    );
    EXPECT_TRUE(
        [] () -> expect<void> {
            MONERO_CHECK(expect<void>{common_error::kInvalidArgument});
            return {common_error::kInvalidErrorCode};
        } () == common_error::kInvalidArgument
    );
    EXPECT_TRUE(
        [] () -> expect<int> {
            MONERO_CHECK(expect<void>{});
            return {common_error::kInvalidErrorCode};
        } () == common_error::kInvalidErrorCode
    );
    EXPECT_TRUE(
        [] () -> expect<int> {
            MONERO_CHECK(expect<void>{common_error::kInvalidArgument});
            return {common_error::kInvalidErrorCode};
        } () == common_error::kInvalidArgument
    );

    EXPECT_NO_THROW(MONERO_UNWRAP(success()));
    EXPECT_NO_THROW(MONERO_UNWRAP(expect<void>{}));
    EXPECT_NO_THROW(MONERO_UNWRAP(expect<int>{0}));
    EXPECT_THROW(
        MONERO_UNWRAP(expect<void>{common_error::kInvalidArgument}), std::system_error
    );
    EXPECT_THROW(
        MONERO_UNWRAP(expect<int>{common_error::kInvalidArgument}), std::system_error
    );
}