123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167 |
- #ifndef BOOST_OUTCOME_EXPERIMENTAL_STATUS_OUTCOME_HPP
- #define BOOST_OUTCOME_EXPERIMENTAL_STATUS_OUTCOME_HPP
- #include "../basic_outcome.hpp"
- #include "../detail/trait_std_exception.hpp"
- #include "status_result.hpp"
- #include "boost/exception_ptr.hpp"
- BOOST_OUTCOME_SYSTEM_ERROR2_NAMESPACE_BEGIN
- template <class DomainType> inline std::exception_ptr basic_outcome_failure_exception_from_error(const status_code<DomainType> &sc)
- {
- (void) sc;
- #ifndef BOOST_NO_EXCEPTIONS
- try
- {
- sc.throw_exception();
- }
- catch(...)
- {
- return std::current_exception();
- }
- #endif
- return {};
- }
- BOOST_OUTCOME_SYSTEM_ERROR2_NAMESPACE_END
- BOOST_OUTCOME_V2_NAMESPACE_EXPORT_BEGIN
- namespace experimental
- {
- namespace policy
- {
- template <class T, class EC, class E>
- using default_status_outcome_policy = std::conditional_t< //
- std::is_void<EC>::value && std::is_void<E>::value,
- BOOST_OUTCOME_V2_NAMESPACE::policy::terminate,
- std::conditional_t<(is_status_code<EC>::value || is_errored_status_code<EC>::value) &&
- (std::is_void<E>::value || BOOST_OUTCOME_V2_NAMESPACE::trait::is_exception_ptr_available<E>::value),
- status_code_throw<T, EC, E>,
- BOOST_OUTCOME_V2_NAMESPACE::policy::fail_to_compile_observers
- >>;
- }
-
- template <class R, class S = erased_errored_status_code<typename system_code::value_type>, class P = std::exception_ptr,
- class NoValuePolicy = policy::default_status_outcome_policy<R, S, P>>
- using status_outcome = basic_outcome<R, S, P, NoValuePolicy>;
-
- BOOST_OUTCOME_TEMPLATE(class R, class S, class P, class NoValuePolicy)
- BOOST_OUTCOME_TREQUIRES(BOOST_OUTCOME_TPRED(std::is_copy_constructible<R>::value &&std::is_copy_constructible<P>::value &&
- (is_status_code<S>::value || is_errored_status_code<S>::value)))
- inline basic_outcome<R, S, P, NoValuePolicy> clone(const basic_outcome<R, S, P, NoValuePolicy> &v)
- {
- if(v)
- {
- return success_type<R>(v.assume_value());
- }
- if(v.has_error() && v.has_exception())
- {
- return failure_type<S, P>(v.assume_error().clone(), v.assume_exception(), hooks::spare_storage(&v));
- }
- if(v.has_exception())
- {
- return failure_type<S, P>(in_place_type<P>, v.assume_exception(), hooks::spare_storage(&v));
- }
- return failure_type<S, P>(in_place_type<S>, v.assume_error().clone(), hooks::spare_storage(&v));
- }
-
- BOOST_OUTCOME_TEMPLATE(class S, class P, class NoValuePolicy)
- BOOST_OUTCOME_TREQUIRES(BOOST_OUTCOME_TPRED(std::is_copy_constructible<P>::value && (is_status_code<S>::value || is_errored_status_code<S>::value)))
- inline basic_outcome<void, S, P, NoValuePolicy> clone(const basic_outcome<void, S, P, NoValuePolicy> &v)
- {
- if(v)
- {
- return success_type<void>();
- }
- if(v.has_error() && v.has_exception())
- {
- return failure_type<S, P>(v.assume_error().clone(), v.assume_exception(), hooks::spare_storage(&v));
- }
- if(v.has_exception())
- {
- return failure_type<S, P>(in_place_type<P>, v.assume_exception(), hooks::spare_storage(&v));
- }
- return failure_type<S, P>(in_place_type<S>, v.assume_error().clone(), hooks::spare_storage(&v));
- }
- namespace policy
- {
- template <class T, class DomainType, class E> struct status_code_throw<T, status_code<DomainType>, E> : base
- {
- using _base = base;
- template <class Impl> static constexpr void wide_value_check(Impl &&self)
- {
- if(!base::_has_value(static_cast<Impl &&>(self)))
- {
- if(base::_has_exception(static_cast<Impl &&>(self)))
- {
- BOOST_OUTCOME_V2_NAMESPACE::policy::detail::_rethrow_exception<trait::is_exception_ptr_available<E>::value>(
- base::_exception<T, status_code<DomainType>, E, status_code_throw>(static_cast<Impl &&>(self)));
- }
- if(base::_has_error(static_cast<Impl &&>(self)))
- {
- #ifndef BOOST_NO_EXCEPTIONS
- base::_error(static_cast<Impl &&>(self)).throw_exception();
- #else
- BOOST_OUTCOME_THROW_EXCEPTION("wide value check failed");
- #endif
- }
- }
- }
- template <class Impl> static constexpr void wide_error_check(Impl &&self) { _base::narrow_error_check(static_cast<Impl &&>(self)); }
- template <class Impl> static constexpr void wide_exception_check(Impl &&self) { _base::narrow_exception_check(static_cast<Impl &&>(self)); }
- };
- template <class T, class DomainType, class E>
- struct status_code_throw<T, errored_status_code<DomainType>, E> : status_code_throw<T, status_code<DomainType>, E>
- {
- status_code_throw() = default;
- using status_code_throw<T, status_code<DomainType>, E>::status_code_throw;
- };
- }
- }
- BOOST_OUTCOME_V2_NAMESPACE_END
- #endif
|