From: Manuel Nickschas Date: Wed, 31 Oct 2018 23:39:21 +0000 (+0100) Subject: funchelpers: Support the invocation of non-void callables X-Git-Tag: test-travis-01~98 X-Git-Url: https://git.quassel-irc.org/?p=quassel.git;a=commitdiff_plain;h=a22e08480288685d73d9abd18c6a1087451c388b;ds=sidebyside funchelpers: Support the invocation of non-void callables Let invokeWithArgsList() handle and return the return value of non-void callables wrapped in a QVariant. To avoid special-casing, the invocation of a void callable also returns a (default-constructed, thus invalid) QVariant. To indicate failure, wrap the QVariant in a boost::optional that is empty if the callable could not be invoked. --- diff --git a/src/common/funchelpers.h b/src/common/funchelpers.h index fcd5aa47..ede4e43e 100644 --- a/src/common/funchelpers.h +++ b/src/common/funchelpers.h @@ -26,7 +26,10 @@ #include #include +#include + #include +#include #include // ---- Function traits -------------------------------------------------------------------------------------------------------------------- @@ -75,22 +78,38 @@ using FunctionTraits = detail::FuncHelper; namespace detail { +// Helper for invoking the callable, wrapping its return value in a QVariant (default-constructed if callable returns void). +// The correct overload is selected via SFINAE. +template +auto invokeWithArgs(const Callable& c, Args&&... args) + -> std::enable_if_t::ReturnType>::value, QVariant> +{ + c(std::forward(args)...); + return QVariant{}; +} + +template +auto invokeWithArgs(const Callable& c, Args&&... args) + -> std::enable_if_t::ReturnType>::value, QVariant> +{ + return QVariant::fromValue(c(std::forward(args)...)); +} + // Helper for unpacking the argument list via an index sequence template::ArgsTuple> -bool invokeWithArgsList(const Callable& c, const QVariantList& args, std::index_sequence) +boost::optional invokeWithArgsList(const Callable& c, const QVariantList& args, std::index_sequence) { // Sanity check that all types can be converted std::array::value> convertible{{args[Is].canConvert>>()...}}; for (size_t i = 0; i < convertible.size(); ++i) { if (!convertible[i]) { qWarning() << "Cannot convert parameter" << i << "from type" << args[static_cast(i)].typeName() << "to expected argument type"; - return false; + return boost::none; } } - // Invoke callable - c(args[Is].value>>()...); - return true; + // Invoke callable with unmarshalled arguments + return invokeWithArgs(c, args[Is].value>>()...); } } // detail @@ -99,22 +118,25 @@ bool invokeWithArgsList(const Callable& c, const QVariantList& args, std::index_ * Invokes the given callable with the arguments contained in the given variant list. * * The types contained in the given QVariantList are converted to the types expected by the callable. - * If the conversion fails, or if the argument count does not match, this function returns false and - * the callable is not invoked. + * If the invocation is successful, the returned optional contains a QVariant with the return value, + * or an invalid QVariant if the callable returns void. + * If the conversion fails, or if the argument count does not match, this function returns boost::none + * and the callable is not invoked. * * @param c Callable * @param args Arguments to be given to the callable - * @returns true if the callable could be invoked with the given list of arguments + * @returns An optional containing a QVariant with the return value if the callable could be invoked with + * the given list of arguments; otherwise boost::none */ template -bool invokeWithArgsList(const Callable& c, const QVariantList& args) +boost::optional invokeWithArgsList(const Callable& c, const QVariantList& args) { using ArgsTuple = typename FunctionTraits::ArgsTuple; constexpr auto tupleSize = std::tuple_size::value; if (tupleSize != args.size()) { qWarning().nospace() << "Argument count mismatch! Expected: " << tupleSize << ", actual: " << args.size(); - return false; + return boost::none; } return detail::invokeWithArgsList(c, args, std::make_index_sequence{}); } diff --git a/src/common/signalproxy.h b/src/common/signalproxy.h index 885ebe60..b58aae8f 100644 --- a/src/common/signalproxy.h +++ b/src/common/signalproxy.h @@ -422,7 +422,7 @@ public: qWarning() << "Cannot call slot in different thread!"; return false; } - return invokeWithArgsList(_callable, params); + return invokeWithArgsList(_callable, params) ? true : false; } private: diff --git a/tests/common/funchelperstest.cpp b/tests/common/funchelperstest.cpp index 14a78370..c40cadef 100644 --- a/tests/common/funchelperstest.cpp +++ b/tests/common/funchelperstest.cpp @@ -38,7 +38,9 @@ TEST(FuncHelpersTest, invokeWithArgsList) // Good case { QVariantList argsList{42, "Hello World"}; - ASSERT_TRUE(invokeWithArgsList(callable, argsList)); + auto ret = invokeWithArgsList(callable, argsList); + ASSERT_TRUE(ret); + EXPECT_FALSE(ret->isValid()); // Callable returns void, so the returned QVariant should be invalid EXPECT_EQ(42, intVal); EXPECT_EQ("Hello World", stringVal); } @@ -75,3 +77,35 @@ TEST(FuncHelpersTest, invokeWithArgsList) EXPECT_EQ("Hello World", stringVal); } } + +TEST(FuncHelpersTest, invokeWithArgsListAndReturnValue) +{ + { + int intVal{}; + QString stringVal{}; + + auto callable = [&intVal, &stringVal](int i, const QString& s) + { + intVal = i; + stringVal = s; + return -i; + }; + + // Good case + { + QVariantList argsList{42, "Hello World"}; + auto ret = invokeWithArgsList(callable, argsList); + ASSERT_TRUE(ret); + ASSERT_TRUE(ret->isValid()); + EXPECT_EQ(-42, *ret); + EXPECT_EQ(42, intVal); + EXPECT_EQ("Hello World", stringVal); + } + + // Failed invocation + { + QVariantList argsList{23}; + ASSERT_FALSE(invokeWithArgsList(callable, argsList)); + } + } +}