Drop the AGPL license in favor of a source-available license. See the blog post [1] for details. [1] https://www.scylladb.com/2024/12/18/why-were-moving-to-a-source-available-license/
159 lines
5.4 KiB
C++
159 lines
5.4 KiB
C++
/*
|
|
* Copyright 2022-present ScyllaDB
|
|
*/
|
|
|
|
/*
|
|
* SPDX-License-Identifier: LicenseRef-ScyllaDB-Source-Available-1.0
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
#include <utility>
|
|
#include <seastar/core/future.hh>
|
|
#include "utils/result.hh"
|
|
|
|
namespace utils {
|
|
|
|
// Converts a result into either a ready or an exceptional future.
|
|
// Supports only results which has an exception_container as their error type.
|
|
template<typename R>
|
|
requires ExceptionContainerResult<R>
|
|
seastar::future<typename R::value_type> result_into_future(R&& res) {
|
|
if (res) {
|
|
if constexpr (std::is_void_v<typename R::value_type>) {
|
|
return seastar::make_ready_future<>();
|
|
} else {
|
|
return seastar::make_ready_future<typename R::value_type>(std::move(res).assume_value());
|
|
}
|
|
} else {
|
|
return std::move(res).assume_error().template into_exception_future<typename R::value_type>();
|
|
}
|
|
}
|
|
|
|
// Converts a non-result future to return a successful result.
|
|
// It _does not_ convert exceptional futures to failed results.
|
|
template<typename R>
|
|
requires ExceptionContainerResult<R>
|
|
seastar::future<R> then_ok_result(seastar::future<typename R::value_type>&& f) {
|
|
using T = typename R::value_type;
|
|
if constexpr (std::is_void_v<T>) {
|
|
return f.then([] {
|
|
return seastar::make_ready_future<R>(bo::success());
|
|
});
|
|
} else {
|
|
return f.then([] (T&& t) {
|
|
return seastar::make_ready_future<R>(bo::success(std::move(t)));
|
|
});
|
|
}
|
|
}
|
|
|
|
// Takes a result<T>, discards the inner value and returns result<>.
|
|
template<ExceptionContainerResult R>
|
|
rebind_result<void, R> result_discard_value(R&& res) {
|
|
if (res) {
|
|
return bo::success();
|
|
} else {
|
|
return std::move(res).as_failure();
|
|
}
|
|
}
|
|
|
|
namespace internal {
|
|
|
|
template<typename C, typename Arg, bool Unpack>
|
|
struct result_wrapped_call_traits {
|
|
};
|
|
|
|
template<typename C, typename Arg>
|
|
struct result_wrapped_call_traits<C, Arg, false> {
|
|
static_assert(ExceptionContainerResult<Arg>);
|
|
using return_type = decltype(seastar::futurize_invoke(std::declval<C>(), std::declval<typename Arg::value_type>()));
|
|
|
|
static auto invoke_with_value(C& c, Arg&& arg) {
|
|
// Arg must have a value
|
|
return seastar::futurize_invoke(c, std::move(arg).value());
|
|
}
|
|
};
|
|
|
|
template<typename C, typename... Exs>
|
|
struct result_wrapped_call_traits<C, result_with_exception<void, Exs...>, false> {
|
|
using return_type = decltype(seastar::futurize_invoke(std::declval<C>()));
|
|
|
|
static auto invoke_with_value(C& c, result_with_exception<void, Exs...>&& arg) {
|
|
return seastar::futurize_invoke(c);
|
|
}
|
|
};
|
|
|
|
template<typename C, ExceptionContainer ExCont, typename... Args>
|
|
struct result_wrapped_call_traits<C, bo::result<std::tuple<Args...>, ExCont, exception_container_throw_policy>, true> {
|
|
private:
|
|
using result_type = bo::result<std::tuple<Args...>, ExCont, exception_container_throw_policy>;
|
|
|
|
public:
|
|
using return_type = decltype(seastar::futurize_apply(std::declval<C>(), std::declval<std::tuple<Args...>>()));
|
|
|
|
static auto invoke_with_value(C& c, result_type&& args) {
|
|
// Arg must have a value
|
|
return seastar::futurize_apply(c, std::move(args).value());
|
|
}
|
|
};
|
|
|
|
template<typename C, typename Arg>
|
|
struct result_wrapped_call_traits<C, Arg, true> {
|
|
static_assert(false && sizeof(Arg), "result_wrap_apply must be called with a result<std::tuple<...>> as a second argument");
|
|
};
|
|
|
|
template<typename C, bool Unpack>
|
|
struct result_wrapper {
|
|
C c;
|
|
|
|
result_wrapper(C&& c) : c(std::move(c)) {}
|
|
|
|
template<typename InputResult>
|
|
requires ExceptionContainerResult<InputResult>
|
|
auto operator()(InputResult arg) {
|
|
using traits = internal::result_wrapped_call_traits<C, InputResult, Unpack>;
|
|
using return_type = typename traits::return_type;
|
|
static_assert(ExceptionContainerResultFuture<return_type>,
|
|
"the return type of the call must be a future<result<T>> for some T");
|
|
|
|
using return_result_type = typename return_type::value_type;
|
|
static_assert(std::is_same_v<typename InputResult::error_type, typename return_result_type::error_type>,
|
|
"the error type of accepted result and returned result are not the same");
|
|
|
|
if (arg) {
|
|
return traits::invoke_with_value(c, std::move(arg));
|
|
} else {
|
|
return seastar::make_ready_future<return_result_type>(bo::failure(std::move(arg).assume_error()));
|
|
}
|
|
}
|
|
};
|
|
|
|
}
|
|
|
|
// Converts a callable to accept a result<T> instead of T.
|
|
// Given a callable which can be called with following arguments and return type:
|
|
//
|
|
// (T) -> future<result<U>>
|
|
//
|
|
// ...returns a new callable which can be called with the following signature:
|
|
//
|
|
// (result<T>) -> future<result<U>>
|
|
//
|
|
// On call, the adapted callable is run only if the result<T> contains a value.
|
|
// Otherwise, the adapted callable is not called and the error is returned immediately.
|
|
// The resulting callable must receive result<> as an argument when being called,
|
|
// it won't be automatically converted to result<>.
|
|
template<typename C>
|
|
auto result_wrap(C&& c) {
|
|
return internal::result_wrapper<C, false>(std::move(c));
|
|
}
|
|
|
|
// Similar to result_wrap, but the resulting callable takes a result<tuple<...>>,
|
|
// unpacks the tuple and provides each argument separately to the wrapped callable.
|
|
template<typename C>
|
|
auto result_wrap_unpack(C&& c) {
|
|
return internal::result_wrapper<C, true>(std::move(c));
|
|
}
|
|
|
|
}
|