Unverified Commit e7c65f70 authored by Kenton Varda's avatar Kenton Varda Committed by GitHub

Merge pull request #698 from capnproto/reduce-remote-promise

Automatically reduce `Promise<RemotePromise<T>>` -> `RemotePromise<T>`.
parents 30c7d54b 6f4d69ad
...@@ -7,8 +7,8 @@ ...@@ -7,8 +7,8 @@
# support for. # support for.
# - Use CMake to ... # - Use CMake to ...
# build Cap'n Proto with MinGW. # build Cap'n Proto with MinGW.
# build Cap'n Proto with VS2015 and VS2017. # build Cap'n Proto with VS2017.
# build Cap'n Proto samples with VS2015 and VS2017. # build Cap'n Proto samples with VS2017.
version: "{build}" version: "{build}"
...@@ -42,10 +42,6 @@ environment: ...@@ -42,10 +42,6 @@ environment:
# TODO(someday): Right now /maxcpucount occasionally expresses a filesystem-related race: # TODO(someday): Right now /maxcpucount occasionally expresses a filesystem-related race:
# capnp-capnpc++ complains that it can't create test.capnp.h. # capnp-capnpc++ complains that it can't create test.capnp.h.
- CMAKE_GENERATOR: Visual Studio 14 2015
BUILD_NAME: vs2015
EXTRA_BUILD_FLAGS: # /maxcpucount
- CMAKE_GENERATOR: MinGW Makefiles - CMAKE_GENERATOR: MinGW Makefiles
BUILD_NAME: mingw BUILD_NAME: mingw
EXTRA_BUILD_FLAGS: -j2 EXTRA_BUILD_FLAGS: -j2
......
...@@ -1031,6 +1031,55 @@ TEST(Capability, TransferCap) { ...@@ -1031,6 +1031,55 @@ TEST(Capability, TransferCap) {
}).wait(waitScope); }).wait(waitScope);
} }
KJ_TEST("Promise<RemotePromise<T>> automatically reduces to RemotePromise<T>") {
kj::EventLoop loop;
kj::WaitScope waitScope(loop);
int callCount = 0;
test::TestInterface::Client client(kj::heap<TestInterfaceImpl>(callCount));
RemotePromise<test::TestInterface::FooResults> promise = kj::evalLater([&]() {
auto request = client.fooRequest();
request.setI(123);
request.setJ(true);
return request.send();
});
EXPECT_EQ(0, callCount);
auto response = promise.wait(waitScope);
EXPECT_EQ("foo", response.getX());
EXPECT_EQ(1, callCount);
}
KJ_TEST("Promise<RemotePromise<T>> automatically reduces to RemotePromise<T> with pipelining") {
kj::EventLoop loop;
kj::WaitScope waitScope(loop);
int callCount = 0;
int chainedCallCount = 0;
test::TestPipeline::Client client(kj::heap<TestPipelineImpl>(callCount));
auto promise = kj::evalLater([&]() {
auto request = client.getCapRequest();
request.setN(234);
request.setInCap(test::TestInterface::Client(kj::heap<TestInterfaceImpl>(chainedCallCount)));
return request.send();
});
auto pipelineRequest = promise.getOutBox().getCap().fooRequest();
pipelineRequest.setI(321);
auto pipelinePromise = pipelineRequest.send();
EXPECT_EQ(0, callCount);
EXPECT_EQ(0, chainedCallCount);
auto response = pipelinePromise.wait(waitScope);
EXPECT_EQ("bar", response.getX());
EXPECT_EQ(2, callCount);
EXPECT_EQ(1, chainedCallCount);
}
} // namespace } // namespace
} // namespace _ } // namespace _
} // namespace capnp } // namespace capnp
...@@ -60,6 +60,9 @@ public: ...@@ -60,6 +60,9 @@ public:
KJ_DISALLOW_COPY(RemotePromise); KJ_DISALLOW_COPY(RemotePromise);
RemotePromise(RemotePromise&& other) = default; RemotePromise(RemotePromise&& other) = default;
RemotePromise& operator=(RemotePromise&& other) = default; RemotePromise& operator=(RemotePromise&& other) = default;
static RemotePromise<T> reducePromise(kj::Promise<RemotePromise>&& promise);
// Hook for KJ so that Promise<RemotePromise<T>> automatically reduces to RemotePromise<T>.
}; };
class LocalClient; class LocalClient;
...@@ -737,6 +740,23 @@ private: ...@@ -737,6 +740,23 @@ private:
// ======================================================================================= // =======================================================================================
// Inline implementation details // Inline implementation details
template <typename T>
RemotePromise<T> RemotePromise<T>::reducePromise(kj::Promise<RemotePromise>&& promise) {
kj::Tuple<kj::Promise<Response<T>>, kj::Promise<kj::Own<PipelineHook>>> splitPromise =
promise.then([](RemotePromise&& inner) {
// `inner` is multiply-inherited, and we want to move away each superclass separately.
// Let's create two references to make clear what we're doing (though this is not strictly
// necessary).
kj::Promise<Response<T>>& innerPromise = inner;
typename T::Pipeline& innerPipeline = inner;
return kj::tuple(kj::mv(innerPromise), PipelineHook::from(kj::mv(innerPipeline)));
}).split();
return RemotePromise(kj::mv(kj::get<0>(splitPromise)),
typename T::Pipeline(AnyPointer::Pipeline(
newLocalPromisePipeline(kj::mv(kj::get<1>(splitPromise))))));
}
template <typename Params, typename Results> template <typename Params, typename Results>
RemotePromise<Results> Request<Params, Results>::send() { RemotePromise<Results> Request<Params, Results>::send() {
auto typelessPromise = hook->send(); auto typelessPromise = hook->send();
......
...@@ -524,8 +524,8 @@ private: ...@@ -524,8 +524,8 @@ private:
} }
template <size_t index> template <size_t index>
Promise<JoinPromises<typename SplitBranch<T, index>::Element>> addSplit() { ReducePromises<typename SplitBranch<T, index>::Element> addSplit() {
return Promise<JoinPromises<typename SplitBranch<T, index>::Element>>( return ReducePromises<typename SplitBranch<T, index>::Element>(
false, maybeChain(kj::heap<SplitBranch<T, index>>(addRef(*this)), false, maybeChain(kj::heap<SplitBranch<T, index>>(addRef(*this)),
implicitCast<typename SplitBranch<T, index>::Element*>(nullptr))); implicitCast<typename SplitBranch<T, index>::Element*>(nullptr)));
} }
...@@ -580,6 +580,16 @@ Own<PromiseNode>&& maybeChain(Own<PromiseNode>&& node, T*) { ...@@ -580,6 +580,16 @@ Own<PromiseNode>&& maybeChain(Own<PromiseNode>&& node, T*) {
return kj::mv(node); return kj::mv(node);
} }
template <typename T, typename Result = decltype(T::reducePromise(instance<Promise<T>>()))>
inline Result maybeReduce(Promise<T>&& promise, bool) {
return T::reducePromise(kj::mv(promise));
}
template <typename T>
inline Promise<T> maybeReduce(Promise<T>&& promise, ...) {
return kj::mv(promise);
}
// ------------------------------------------------------------------- // -------------------------------------------------------------------
class ExclusiveJoinPromiseNode final: public PromiseNode { class ExclusiveJoinPromiseNode final: public PromiseNode {
...@@ -809,8 +819,9 @@ PromiseForResult<Func, T> Promise<T>::then(Func&& func, ErrorFunc&& errorHandler ...@@ -809,8 +819,9 @@ PromiseForResult<Func, T> Promise<T>::then(Func&& func, ErrorFunc&& errorHandler
Own<_::PromiseNode> intermediate = Own<_::PromiseNode> intermediate =
heap<_::TransformPromiseNode<ResultT, _::FixVoid<T>, Func, ErrorFunc>>( heap<_::TransformPromiseNode<ResultT, _::FixVoid<T>, Func, ErrorFunc>>(
kj::mv(node), kj::fwd<Func>(func), kj::fwd<ErrorFunc>(errorHandler)); kj::mv(node), kj::fwd<Func>(func), kj::fwd<ErrorFunc>(errorHandler));
return PromiseForResult<Func, T>(false, auto result = _::ChainPromises<_::ReturnType<Func, T>>(false,
_::maybeChain(kj::mv(intermediate), implicitCast<ResultT*>(nullptr))); _::maybeChain(kj::mv(intermediate), implicitCast<ResultT*>(nullptr)));
return _::maybeReduce(kj::mv(result), false);
} }
namespace _ { // private namespace _ { // private
...@@ -1108,7 +1119,7 @@ PromiseFulfillerPair<T> newPromiseAndFulfiller() { ...@@ -1108,7 +1119,7 @@ PromiseFulfillerPair<T> newPromiseAndFulfiller() {
Own<_::PromiseNode> intermediate( Own<_::PromiseNode> intermediate(
heap<_::AdapterPromiseNode<_::FixVoid<T>, _::PromiseAndFulfillerAdapter<T>>>(*wrapper)); heap<_::AdapterPromiseNode<_::FixVoid<T>, _::PromiseAndFulfillerAdapter<T>>>(*wrapper));
Promise<_::JoinPromises<T>> promise(false, _::ReducePromises<T> promise(false,
_::maybeChain(kj::mv(intermediate), implicitCast<T*>(nullptr))); _::maybeChain(kj::mv(intermediate), implicitCast<T*>(nullptr)));
return PromiseFulfillerPair<T> { kj::mv(promise), kj::mv(wrapper) }; return PromiseFulfillerPair<T> { kj::mv(promise), kj::mv(wrapper) };
......
...@@ -45,15 +45,28 @@ Promise<void> joinPromises(Array<Promise<void>>&& promises); ...@@ -45,15 +45,28 @@ Promise<void> joinPromises(Array<Promise<void>>&& promises);
namespace _ { // private namespace _ { // private
template <typename T> struct JoinPromises_ { typedef T Type; }; template <typename T>
template <typename T> struct JoinPromises_<Promise<T>> { typedef T Type; }; Promise<T> chainPromiseType(T*);
template <typename T>
Promise<T> chainPromiseType(Promise<T>*);
template <typename T> template <typename T>
using JoinPromises = typename JoinPromises_<T>::Type; using ChainPromises = decltype(chainPromiseType((T*)nullptr));
// If T is Promise<U>, resolves to U, otherwise resolves to T. // Constructs a promise for T, reducing double-promises. That is, if T is Promise<U>, resolves to
// // Promise<U>, otherwise resolves to Promise<T>.
// TODO(cleanup): Rename to avoid confusion with joinPromises() call which is completely
// unrelated. template <typename T>
Promise<T> reducePromiseType(T*, ...);
template <typename T>
Promise<T> reducePromiseType(Promise<T>*, ...);
template <typename T, typename Reduced = decltype(T::reducePromise(kj::instance<Promise<T>>()))>
Reduced reducePromiseType(T*, bool);
template <typename T>
using ReducePromises = decltype(reducePromiseType((T*)nullptr, false));
// Like ChainPromises, but also takes into account whether T has a method `reducePromise` that
// reduces Promise<T> to something else. In particular this allows Promise<capnp::RemotePromise<U>>
// to reduce to capnp::RemotePromise<U>.
class PropagateException { class PropagateException {
// A functor which accepts a kj::Exception as a parameter and returns a broken promise of // A functor which accepts a kj::Exception as a parameter and returns a broken promise of
...@@ -90,7 +103,7 @@ using ReturnType = typename ReturnType_<Func, T>::Type; ...@@ -90,7 +103,7 @@ using ReturnType = typename ReturnType_<Func, T>::Type;
template <typename T> struct SplitTuplePromise_ { typedef Promise<T> Type; }; template <typename T> struct SplitTuplePromise_ { typedef Promise<T> Type; };
template <typename... T> template <typename... T>
struct SplitTuplePromise_<kj::_::Tuple<T...>> { struct SplitTuplePromise_<kj::_::Tuple<T...>> {
typedef kj::Tuple<Promise<JoinPromises<T>>...> Type; typedef kj::Tuple<ReducePromises<T>...> Type;
}; };
template <typename T> template <typename T>
......
...@@ -44,7 +44,7 @@ template <typename T> ...@@ -44,7 +44,7 @@ template <typename T>
struct PromiseFulfillerPair; struct PromiseFulfillerPair;
template <typename Func, typename T> template <typename Func, typename T>
using PromiseForResult = Promise<_::JoinPromises<_::ReturnType<Func, T>>>; using PromiseForResult = _::ReducePromises<_::ReturnType<Func, T>>;
// Evaluates to the type of Promise for the result of calling functor type Func with parameter type // Evaluates to the type of Promise for the result of calling functor type Func with parameter type
// T. If T is void, then the promise is for the result of calling Func with no arguments. If // T. If T is void, then the promise is for the result of calling Func with no arguments. If
// Func itself returns a promise, the promises are joined, so you never get Promise<Promise<T>>. // Func itself returns a promise, the promises are joined, so you never get Promise<Promise<T>>.
...@@ -482,7 +482,7 @@ Promise<T> newAdaptedPromise(Params&&... adapterConstructorParams); ...@@ -482,7 +482,7 @@ Promise<T> newAdaptedPromise(Params&&... adapterConstructorParams);
template <typename T> template <typename T>
struct PromiseFulfillerPair { struct PromiseFulfillerPair {
Promise<_::JoinPromises<T>> promise; _::ReducePromises<T> promise;
Own<PromiseFulfiller<T>> fulfiller; Own<PromiseFulfiller<T>> fulfiller;
}; };
......
...@@ -60,8 +60,8 @@ ...@@ -60,8 +60,8 @@
#endif #endif
#endif #endif
#elif defined(_MSC_VER) #elif defined(_MSC_VER)
#if _MSC_VER < 1900 #if _MSC_VER < 1910
#error "You need Visual Studio 2015 or better to compile this code." #error "You need Visual Studio 2017 or better to compile this code."
#endif #endif
#else #else
#warning "I don't recognize your compiler. As of this writing, Clang and GCC are the only "\ #warning "I don't recognize your compiler. As of this writing, Clang and GCC are the only "\
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment