I think this is the right direction generally. I lean a bit more toward the ignore_on_error (but maybe called bypass on error?) construction due to how it fits into the usual .then() usage.
The bit of MonClient I was thinking of was the users of Connection::AuthResult in crimson/mon/MonClient.cc -- it ended up being threaded through several callback chains and might be a good opportunity to prove out one of these approaches.
-Sam
On Tue, Jul 9, 2019 at 10:11 AM kefu chai <tchaikov@xxxxxxxxx> wrote:
On Sun, Jul 7, 2019 at 10:17 PM Ronen Friedman <rfriedma@xxxxxxxxxx> wrote:... and now it compiles in godbolt:(I've also made some changes to the code, to better demonstrate the possible error-handling paths)Comments/corrections anyone?Ronen, IIUC,WpassR<> is actually a variant of ignore_on_error<>. the difference istemplate<typename Func, typename Output, typename... Args>concept _OnResult_ = requires(Func&& func, Args&&... args) {{ std::move(func)(std::move(args)...) -> std::Same<Output>;};template<typename Func, typename Output, typename... Args>requires OnResult<Func, Output, Args...>Func ignore_on_error(Func&& func)// ignore_on_error is a decorator in python, which decorates the given `func`. the returned function will ignore the args if it is an errortemplate<typename Func, typename Output, typename... Args>requires OnResult<Func, Output, Args...>outcome<Output> WpassR(Func&& func, Args&&... args)// WPassR evaluates func(args) only if args is a value instead of an error or an exception.// and Wsplit evaluates happy_days(args) if args is a value, otherwise it evaluates on_error(args).i don't have preference either way. probably we will their pros and cons when we put either of them into practice.ThanksRonenOn Thu, Jul 4, 2019 at 5:34 PM Ronen Friedman <rfriedma@xxxxxxxxxx> wrote:Hi Kefu,Like I said in the meeting: I really think this is the right direction.I am not sure I fully understand the specifics you've proposed. Here is my take (which may be exactly what you meant):Suppose this is the "happy path":In the face of an error we may want to have this behavior (1'st option):
- let's assume that the values of type T are wrapped in outcome::outcome<T, error-code, exception-code> (I'd rather using that over outcome::result);
- Thus, f1() should be written to return the wrapper type (outcome<T1>),
- f2() is now a function from outcome<T1> to outcome<T2>.
- as we may want to have error codes "pass thru" f2(), without requiring f2() to be modified with the boiler-plat code for the pass-thru, let us wrap f2() into pass_thru<T1,T2>(f2). I have attached code that presents one way of doing just that.
note that while f2() is just a "pass thru" for an incoming error code, we still need to "re-code" the error value in the incoming outcome<T1> into the same value in outcome<T2>.Another desired behaviour, as you mention in the email, is to split the logic into a separate "happy days" and "error path":We can use a separate wrapper around the two paths (note that both paths are expected to return optional<T2>).Here is some code that demonstrates what I mean (compiled with g++9, but the link in Godbolt is only as a display mechanism):https://godbolt.org/z/i-d4TU <- use the new link above, not this oneNote that the code is not polished yet. For example: I think I would be able to replace the callreturn WpassR<Oint, Ofloat>(of2_ps, x)with
return WpassR(of2_ps, x)using some magic (deduction guides? that will require building some structs around) - if we think this direction is worth our time.RonenOn Wed, Jul 3, 2019 at 7:13 AM kefu chai <tchaikov@xxxxxxxxx> wrote:hi guys,
i just came across boost::outcome[0]. it reminded me the discussion we
had back in Barcelona regarding to the error handling in crimson.
well, strictly speaking, it's not limited to errors. it covers the
non-error handling as well.
the question is: shall we start prototyping the crimson variant of
outcome<> now? if yes, probably we can leverage boost::outcome<>?
a little bit background:
seastar uses exception for propagating the error. but it incurs
runtime overhead. because, to throw an exception, the libstdc++
runtime needs to acquire a global lock.
well, some of us might want to argue, why not just return a
future<Result, Error>? let me use an example here, imagine we are
handling a write request in OSD. we might need to go through following
steps:
1. perform some sanity tests, for instance, to see if the OSD is ready
for handling the write request
2. try to read the object info of the object from local storage to see
if it already exists
3. write to the object to the local storage, and send write requests
to replica OSDs (assuming it's in a replicated pool), wait for the
completions of these write ops.
4. update the statistics
5. reply to the client
and it's intuitive to structure these steps using chained continuation like
do_with(std::move(request), [this](auto request) {
return perform_tests(request->object_id).then([request, this] {
return read_object_info(request->object_id);
}).then([request, this](optional<object_info> object_info) {
return when_all(
write_local(request->object_id, request->offset, request->data),
parallel_for_each(replica_osds, [request](auto replica_osd) {
return replica_osd->write_remote(request->object_id,
request->offset, request->data)
}));
}).then([write_size=request.data.size(), this] {
update_statistics(write_size);
return reply_to(reply_t::success, request);
});
}).handle_exception([](auto exception) {
return reply_to(reply_t::failure, exception.error_code, request);
});
in which, if any test fails in step#1, we either need to wait until
the OSD is ready, or just need to bail out, and skip the following
steps. the "handle_exception()" clause is used to handle the "bail
out" case, where we cannot do anything to serve the request. for
instance, the request is invalid.
we want to differentiate two types of errors. one of them are actually
exceptions which does not happen often in real world, and we don't
need/want to optimize for this case. but the other case could be
normal. for instance, it's fairly normal that an object does not exist
yet, when we are trying to write to it. and we do want to be
performant when handling these "errors" in this category, and also, we
want to do this in a convenient way just like handling exceptions.
because, we need an efficient way to convey the message to caller that
"please skip the following continuations, and i would go to this
handling route instead". if my memory serves me correctly, we think
that we need to create a wrapper around seastar::future<> to allow the
caller to do something like
// a helper to run func or skip it
template<typename Func>
auto ignore_on_error(Func&& f) {
return [f=std::move(f)](auto&& t) {
return t.is_value() ? f(t.value()) ? t;
}
}
return read_object_info(oid).then(
return ignore_on_error([](object_info& oi) {
return handle_write_with_object_info(std::move(oi));
}).then([](auto t) {
return handle_write_without_object_info();
});
);
in the example above, i assume we will do something very different
depending on if the object's existence.
cheers,
---
[0] https://www.boost.org/doc/libs/1_70_0/libs/outcome/doc/html/index.html
--
Regards
Kefu Chai
--Regards
Kefu Chai
_______________________________________________ Dev mailing list -- dev@xxxxxxx To unsubscribe send an email to dev-leave@xxxxxxx