mirror of https://github.com/grpc/grpc.git
Promise library context management helpers (#26912)
* Promise library context management helpers * Automated change: Fix sanity tests * Automated change: Fix sanity tests * fixes * Automated change: Fix sanity tests * missing file * Automated change: Fix sanity tests * compatibility * build * Automated change: Fix sanity tests * propagate * Automated change: Fix sanity tests * merge with head * Automated change: Fix sanity tests Co-authored-by: ctiller <ctiller@users.noreply.github.com>pull/27073/head
parent
4744aceb68
commit
aacd32c721
9 changed files with 229 additions and 9 deletions
@ -0,0 +1,84 @@ |
||||
// Copyright 2021 gRPC authors.
|
||||
//
|
||||
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||
// you may not use this file except in compliance with the License.
|
||||
// You may obtain a copy of the License at
|
||||
//
|
||||
// http://www.apache.org/licenses/LICENSE-2.0
|
||||
//
|
||||
// Unless required by applicable law or agreed to in writing, software
|
||||
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
// See the License for the specific language governing permissions and
|
||||
// limitations under the License.
|
||||
|
||||
#ifndef GRPC_CORE_LIB_PROMISE_CONTEXT_H |
||||
#define GRPC_CORE_LIB_PROMISE_CONTEXT_H |
||||
|
||||
#include <grpc/impl/codegen/port_platform.h> |
||||
|
||||
#include <utility> |
||||
#include "src/core/lib/gpr/tls.h" |
||||
|
||||
namespace grpc_core { |
||||
|
||||
// To avoid accidentally creating context types, we require an explicit
|
||||
// specialization of this template per context type. The specialization need
|
||||
// not contain any members, only exist.
|
||||
// The reason for avoiding this is that context types each use a thread local.
|
||||
template <typename T> |
||||
struct ContextType; |
||||
|
||||
namespace promise_detail { |
||||
|
||||
template <typename T> |
||||
class Context : public ContextType<T> { |
||||
public: |
||||
explicit Context(T* p) : old_(current_) { current_ = p; } |
||||
~Context() { current_ = old_; } |
||||
Context(const Context&) = delete; |
||||
Context& operator=(const Context&) = delete; |
||||
|
||||
static T* get() { return current_; } |
||||
|
||||
private: |
||||
T* const old_; |
||||
static GPR_THREAD_LOCAL(T*) current_; |
||||
}; |
||||
|
||||
template <typename T> |
||||
GPR_THREAD_LOCAL(T*) |
||||
Context<T>::current_; |
||||
|
||||
template <typename T, typename F> |
||||
class WithContext { |
||||
public: |
||||
WithContext(F f, T* context) : context_(context), f_(std::move(f)) {} |
||||
|
||||
decltype(std::declval<F>()()) operator()() { |
||||
Context<T> ctx(context_); |
||||
return f_(); |
||||
} |
||||
|
||||
private: |
||||
T* context_; |
||||
F f_; |
||||
}; |
||||
|
||||
} // namespace promise_detail
|
||||
|
||||
// Retrieve the current value of a context.
|
||||
template <typename T> |
||||
T* GetContext() { |
||||
return promise_detail::Context<T>::get(); |
||||
} |
||||
|
||||
// Given a promise and a context, return a promise that has that context set.
|
||||
template <typename T, typename F> |
||||
promise_detail::WithContext<T, F> WithContext(F f, T* context) { |
||||
return promise_detail::WithContext<T, F>(f, context); |
||||
} |
||||
|
||||
} // namespace grpc_core
|
||||
|
||||
#endif // GRPC_CORE_LIB_PROMISE_CONTEXT_H
|
@ -0,0 +1,41 @@ |
||||
// Copyright 2021 gRPC authors.
|
||||
//
|
||||
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||
// you may not use this file except in compliance with the License.
|
||||
// You may obtain a copy of the License at
|
||||
//
|
||||
// http://www.apache.org/licenses/LICENSE-2.0
|
||||
//
|
||||
// Unless required by applicable law or agreed to in writing, software
|
||||
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
// See the License for the specific language governing permissions and
|
||||
// limitations under the License.
|
||||
|
||||
#include "src/core/lib/promise/context.h" |
||||
#include <gtest/gtest.h> |
||||
|
||||
namespace grpc_core { |
||||
|
||||
struct TestContext { |
||||
bool done = false; |
||||
}; |
||||
|
||||
template <> |
||||
struct ContextType<TestContext> {}; |
||||
|
||||
TEST(Context, WithContext) { |
||||
EXPECT_EQ(GetContext<TestContext>(), nullptr); |
||||
TestContext test; |
||||
EXPECT_EQ(GetContext<TestContext>(), nullptr); |
||||
EXPECT_EQ(test.done, false); |
||||
WithContext([]() { GetContext<TestContext>()->done = true; }, &test)(); |
||||
EXPECT_EQ(test.done, true); |
||||
} |
||||
|
||||
} // namespace grpc_core
|
||||
|
||||
int main(int argc, char** argv) { |
||||
::testing::InitGoogleTest(&argc, argv); |
||||
return RUN_ALL_TESTS(); |
||||
} |
Loading…
Reference in new issue