The Error Module

To use this module, include ccbase/error.hpp.

Overview

The expected<T> class is a polymorphic type that is either in a valid or an invalid state. If an instance is in an invalid state, then it stores an exception; if it is in a valid state, then it stores an instance of T. This allows us to return arbitrarily-rich error codes from a given function, without forcing the programmer to respond to any errors immediately.

Here’s an example where the expected<T> class is used to provide an error-checked wrapper over a low-level POSIX function:

std::system_error current_system_error() noexcept
{ return std::system_error{errno, std::system_category()}; }

cc::expected<int>
safe_open(const char* path, int flags)
noexcept
{
        auto r = int{};
        do r = ::open(path, flags, S_IRUSR | S_IWUSR);
        while (r == -1 && errno == EINTR);

        if (r == -1) return current_system_error();
        return r;
}

int main()
{
        try {
                // Deferences the returned `expected` object, which
                // returns the stored value or throws the stored
                // exception.
                auto fd = *safe_open("test.txt", O_RDONLY);
        catch (const std::system_error& e) {
                // Respond to the exception here.
        }

        // This time, we defer responding to the exception.
        auto exp = safe_open("test.txt", O_RDONLY);

        // Do other stuff.

        if (!exp) {
                // Respond to the error.
        }
}

Sometimes it makes sense to return an expected<T> object from a function, even if function does not really return anything. Consider the following example:

cc::expected<void>
safe_close(int fd)
{
        auto r = ::close(fd);
        if (r == 0) { return cc::no_error; }
        if (errno != EINTR) { return current_system_error(); }

        for (;;) {
                r = ::close(fd);
                if (errno == EBADF) { return cc::no_error; }
                if (errno != EINTR) { current_system_error(); }
        }
}

This module’s implementation of expected<T> is based on the one Andrei Alexandrescu discusses in his talk at C++ Next 2012, called “Systematic Error Handling” [5]. Further improvements were made based on the implementation found in MNMLSTC Core. CCBase’s implementation differs in the following ways:

  • It works for reference types.
  • One can move the std::exception_ptr from an expected<T> object without rethrowing the referred exception. Attempting to do this with current implementations of expected<T> may cause the program to crash because of a double-free.
  • If CC_DONT_EXPECTED_ENFORCE_DISMISSAL is not defined, the destructor of expected<T> will throw if the state of the object is not read at least once prior to destruction. Otherwise, no code is generated to perform this bookkeeping.
  • If the expected type T is copy assignable or move assignable, then the implementation will use these operations instead of redundantly destroying and copy or move constructing the expected object. This can potentially accelerate copy- or move-assignment of expected<T>, when T has a non-trivial destructor.
  • Accessors are overloaded for lvalue and rvalues to expected<T>. Accessors overloaded for lvalues return lvalue references, while those overloaded for rvalues return rvalue references.
  • Relational operators are not implemented. I am not convinced relational operators can be defined intuitively for algebraic data types like boost::optional and expected<T>.
[5]http://channel9.msdn.com/Shows/Going+Deep/C-and-Beyond-2012-Andrei-Alexandrescu-Systematic-Error-Handling-in-C
[6]http://anto-nonco.blogspot.com/2013/03/extending-expected-to-deal-with.html

Reference

class in_place_t

This class is used as a tag to disambiguate overloaded member functions that use variadic templates to construct values in-place.

An instance of this class is available in the :ns`cc` namespace by the name in_place.

class bad_expected_type
Inherits:std::logic_error

Thrown when an operation that requires an expected<T> object to be in a valid state is performed on an expected<T> object in an invalid state, or vice versa.

class expected<T>

An expected<T> object is either in a valid or an invalid state. If CC_EXPECTED_DONT_ENFORCE_DISMISSAL is not defined, then the expected<T> object is also associated with a binary dismissal state. The expected<T> object is put into the dismissed state when a member function is invoked that reveals whether the object is in a valid or invalid state.

The expected<T> object can also be put into the dismissed state manually using the dismiss() member function. If the expected<T> object is not in the dismissed state when its destructor is called, then an exception is thrown. This feature is intended to help enforce that all potential errors produced by functions returning expected<T> are checked.

Note that this class has distinct specializations for references and void. The specialization to references has slightly different semantics, and the void specialization lacks the member functions that have only to do with the expected type.

type value_type

This is the type managed by an expected<T> object that is in a valid state.

Warning

The value_type must satify the following requirements. These requirements are enforced using static assertions within the class definition.

  • Its decayed type must not be std::exception_ptr. [1]
  • Its decayed type must not be derived from std::exception. [1]
  • It must be either an object or an lvalue reference. [2]
  • It must be copy constructible. [2] [3]
  • It must be nothrow destructible. [4]
[1](1, 2) This would ambiguate calls to certain member functions.
[2](1, 2) This would make the definition of expected<T> ill-formed.
[3]Note that copy constructibility implies move constructibility.
[4]This would violate exception safety.
expected()
Noexcept:If T is nothrow default constructible.

Disabled if T is not default constructible. Default-constructs an expected<T> object in a valid state.

expected(const T& rhs)
Noexcept:If T is nothrow copy constructible.

Constructs an expected<T> object in a valid state by copying rhs.

expected(T&& rhs)
Noexcept:If T is nothrow move constructible.

Constructs an expected<T> object in a valid state by moving rhs.

expected(const in_place_t&, Args)
Noexcept:If T is nothrow constructible from Args.

Constructs an expected<T> object in a valid state by initializing the expected type in-place using Args.

expected(const expected& rhs)

Constructs an expected<T> object in the same state as rhs, and copies either the value stored by rhs or its std::exception_ptr.

Noexcept:If T is nothrow copy constructible.
expected(expected&& rhs)

Constructs an expected<T> object in the same state as rhs, and moves either the value stored by rhs or its std::exception_ptr.

Noexcept:If T is nothrow move constructible.
expected(const std::exception_ptr& ptr) noexcept

Constructs an expected<T> object in an invalid state by copying ptr.

expected(exception_ptr&& ptr) noexcept

Constructs an expected<T> object in an invalid state by moving ptr.

expected(const Exception& e)
Requires:Exception to be derived from std::exception.
Noexcept:If CC_NO_DEBUG is defined.

Constructs an expected<T> object in an invalid state from e. If CC_NO_DEBUG is not defined, then this function throws if typeid(e) != typeid(Exception). If this happens, then slicing has occurred.

expected& operator=(const expected& rhs)
Noexcept:If T is copy assignable, then the function is noexcept if T is nothrow copy assignable. Otherwise, the function is noexcept if T is nothrow copy constructible. In addition, CC_EXPECTED_DONT_ENFORCE_DISMISSAL must be defined.

Assigns this expected<T> object to the same state as rhs, by copying either the value stored by rhs or its std::exception_ptr. If CC_EXPECTED_DONT_ENFORCE_DISMISSAL is not defined, then an exception is thrown if the object’s state was not read prior to assignment.

expected& operator=(expected&& rhs)
Noexcept:If T is move assignable, then the function is noexcept if T is nothrow move assignable. Otherwise, the function is noexcept if T is nothrow move constructible. In addition, CC_EXPECTED_DONT_ENFORCE_DISMISSAL must be defined.

Assigns this expected<T> object to the same state as rhs, by moving either the value stored by rhs or its std::exception_ptr. If CC_EXPECTED_DONT_ENFORCE_DISMISSAL is not defined, then an exception is thrown if the object’s state was not read prior to assignment.

expected& operator=(const T& rhs)
Noexcept:If T is copy assignable, then the function is noexcept if T is nothrow copy assignable. Otherwise, the function is noexcept if T is nothrow copy constructible. In addition, CC_EXPECTED_DONT_ENFORCE_DISMISSAL must be defined.

Assigns this expected<T> object to a valid state, and copies the value stored by rhs. If CC_EXPECTED_DONT_ENFORCE_DISMISSAL is not defined, then an exception is thrown if the object’s state was not read prior to assignment.

expected& operator=(T&& rhs)
Noexcept:If T is move assignable, then the function is noexcept if T is nothrow move assignable. Otherwise, the function is noexcept if T is nothrow move constructible. In addition, CC_EXPECTED_DONT_ENFORCE_DISMISSAL must be defined.

Assigns this expected<T> object to a valid state, and moves the value stored by rhs. If CC_EXPECTED_DONT_ENFORCE_DISMISSAL is not defined, then an exception is thrown if the object’s state was not read prior to assignment.

expected& operator=(const std::exception_ptr& ptr)
Noexcept:If CC_EXPECTED_DONT_ENFORCE_DISMISSAL is defined.

Assigns this expected<T> object to an invalid state by copying ptr. If CC_EXPECTED_DONT_ENFORCE_DISMISSAL is not defined, then an exception is thrown if the object’s state was not read prior to assignment.

expected& operator=(std::exception_ptr&& ptr)
Noexcept:If CC_EXPECTED_DONT_ENFORCE_DISMISSAL is defined.

Assigns this expected<T> object to an invalid state by moving ptr. If CC_EXPECTED_DONT_ENFORCE_DISMISSAL is not defined, then an exception is thrown if the object’s state was not read prior to assignment.

~expected()
Noexcept:If CC_EXPECTED_DONT_ENFORCE_DISMISSAL is defined.

Destroys this expected<T> object by calling either the destructor of the expected type, or the destructor of the std::exception_ptr. If CC_DONT_EXPECTED_ENFORCE_DISMISSAL is not defined, an exception is thrown after destroying the managed object if the expected<T> object is not in the dismissed state.

void raise() const
Noexcept:false
Attributes:[[noreturn]]

If the expected<T> object is invalid, throws the exception referred to by the managed std::exception_ptr. Otherwise, throws bad_expected_type. If CC_DONT_EXPECTED_ENFORCE_DISMISSAL is not defined, this function also puts the expected<T> object in the dismissed state.

expected& dismiss() noexcept
const expected& dismiss() const noexcept

If CC_EXPECTED_DONT_ENFORCE_DISMISSAL is defined, puts the expected<T> object in the dismissed state. Otherwise, this function is a no-op.

operator bool() const noexcept

Returns whether this expected<T> object is valid. If CC_EXPECTED_DONT_ENFORCE_DISMISSAL is defined, puts the expected<T> object in the dismissed state.

T& value()
T&& value()
const T& value() const
T& operator*()
T&& operator*()
const T& operator*() const

The functions returning lvalue references are overloaded for lvalues to expected<T>, while those returning rvalue references are overloaded for rvalues to expected<T>.

If this expected<T> object is valid, returns a reference to the managed value. Otherwise, throws bad_expected_type. If CC_EXPECTED_DONT_ENFORCE_DISMISSAL is defined, puts the expected<T> object in the dismissed state.

T* operator->()
const T* operator->() const

These functions are overloaded for lvalues to expected<T>. If the expected<T> object is valid, returns a pointer to the managed value. Otherwise, throws bad_expected_type. If CC_EXPECTED_DONT_ENFORCE_DISMISSAL is defined, puts the expected<T> object in the dismissed state.

const std::exception_ptr& exception() const
std::exception_ptr&& exception()

This first function is overloaded for lvalues to expected<T>, while the second is overloaded for rvalues to expected<T>. If the expected<T> object is valid, returns an reference to the managed std::exception_ptr. Otherwise, throws bad_expected_type.

Warning

If transferring ownership of the std::exception_ptr to another expected<T> object, then the rvalue overload of exception() must be used. Otherwise, the program may crash due to double-free issues.

void swap(expected& rhs)
Noexcept:If T is nothrow move constructible.

Exchanges the state and contents of this expected<T> object with those of rhs. If CC_DONT_EXPECTED_ENFORCE_DISMISSAL is defined, then the dismissal states of both expected<T> objects are also exchanged.

expected& emplace(Args)
Noexcept:If T is nothrow constructible from Args and CC_EXPECTED_DONT_ENFORCE_DISMISSAL is defined.

Assigns this expected<T> object to a valid state by first destroying the managed object (either value_type or std::exception_ptr), and then constructing value_type in-place from Args.

Exception expect() const

Returns an exception of type Exception by rethrowing the std::exception_ptr managed by this expected<T> object. If the expected<T> object is valid, bad_expected_type is thrown. If the managed std::exception_ptr does not refer to an exception of type Exception, then a nested_exception is thrown. This nested_exception is used to nest the exception referred to by the std::exception_ptr within an instance of bad_expected_type.

Functions

void swap(expected& lhs, expected& rhs)

This global version of swap is defined for ADL.

expected<ReturnValue> attempt(const NullaryFunction& f)

Invokes f and returns the resulting value (if not void) or thrown exception wrapped in an expected<T> object.

Specializations

class expected<T&>

This is the specialization of expected<T> to reference types. Consult the class reference for expected<T> to read the complete descriptions.

expected() noexcept
expected(T& rhs) noexcept
expected(const expected& rhs) noexcept
expected(expected&& rhs) noexcept
expected(const std::exception_ptr& ptr) noexcept
expected(exception_ptr&& ptr) noexcept
expected(const Exception& e)
Requires:Exception to be derived from std::exception.
Noexcept:If CC_NO_DEBUG is defined.
expected& operator=(const expected& rhs)
Noexcept:If CC_EXPECTED_DONT_ENFORCE_DISMISSAL is defined.
expected& operator=(expected&& rhs)
Noexcept:If CC_EXPECTED_DONT_ENFORCE_DISMISSAL is defined.
expected& operator=(T& rhs)
Noexcept:If CC_EXPECTED_DONT_ENFORCE_DISMISSAL is defined.
expected& operator=(const std::exception_ptr& ptr)
Noexcept:If CC_EXPECTED_DONT_ENFORCE_DISMISSAL is defined.
expected& operator=(std::exception_ptr&& ptr)
Noexcept:If CC_EXPECTED_DONT_ENFORCE_DISMISSAL is defined.
~expected()
Noexcept:If CC_EXPECTED_DONT_ENFORCE_DISMISSAL is defined.
void raise() const
Noexcept:false
Attributes:[[noreturn]]
expected& dismiss() noexcept
const expected& dismiss() const noexcept
operator bool() const noexcept
T& value()
T&& value()
const T& value() const
T& operator*()
T&& operator*()
const T& operator*() const
T* operator->()
const T* operator->() const
const std::exception_ptr& exception() const
std::exception_ptr&& exception()
void swap(expected& rhs) noexcept
Exception expect() const
class expected<void>

This is the specialization of expected<T> to void. Consult the class reference for expected<T> to read the complete descriptions.

expected() noexcept
expected(const std::exception_ptr& ptr) noexcept
expected(exception_ptr&& ptr) noexcept
expected(const Exception& e)
Requires:Exception to be derived from std::exception.
Noexcept:If CC_NO_DEBUG is defined.
expected& operator=(const expected& rhs)
Noexcept:If CC_EXPECTED_DONT_ENFORCE_DISMISSAL is defined.
expected& operator=(expected&& rhs)
Noexcept:If CC_EXPECTED_DONT_ENFORCE_DISMISSAL is defined.
expected& operator=(const std::exception_ptr& ptr)
Noexcept:If CC_EXPECTED_DONT_ENFORCE_DISMISSAL is defined.
expected& operator=(std::exception_ptr&& ptr)
Noexcept:If CC_EXPECTED_DONT_ENFORCE_DISMISSAL is defined.
~expected()
Noexcept:If CC_EXPECTED_DONT_ENFORCE_DISMISSAL is defined.
void raise() const
Noexcept:false
Attributes:[[noreturn]]
expected& dismiss() noexcept
const expected& dismiss() const noexcept
operator bool() const noexcept
const std::exception_ptr& exception() const
std::exception_ptr&& exception()
void swap(expected& rhs) noexcept
Exception expect() const
class hash<expected<T>>

This is a specialization of std::hash<U> to expected<T>.

type result_type

Aliased to typename hash<T>::result_type.

result_type operator()(const expected& val) const noexcept

If val is valid, returns the hash value of the managed value type. Otherwise, returns a default-constructed result_type.