The C based gRPC (C++, Python, Ruby, Objective-C, PHP, C#)
https://grpc.io/
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
198 lines
5.9 KiB
198 lines
5.9 KiB
// |
|
// |
|
// Copyright 2023 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 "test/cpp/interop/pre_stop_hook_server.h" |
|
|
|
#include <thread> |
|
|
|
#include "absl/strings/str_format.h" |
|
|
|
#include <grpcpp/grpcpp.h> |
|
|
|
#include "src/proto/grpc/testing/test.grpc.pb.h" |
|
|
|
namespace grpc { |
|
namespace testing { |
|
namespace { |
|
|
|
class HookServiceImpl final : public HookService::CallbackService { |
|
public: |
|
ServerUnaryReactor* Hook(CallbackServerContext* context, |
|
const Empty* /* request */, |
|
Empty* /* reply */) override { |
|
grpc_core::MutexLock lock(&mu_); |
|
auto reactor = context->DefaultReactor(); |
|
if (pending_status_) { |
|
reactor->Finish(std::move(*pending_status_)); |
|
pending_status_ = absl::nullopt; |
|
} else if (done_) { |
|
reactor->Finish(Status(StatusCode::ABORTED, "Shutting down")); |
|
} else { |
|
pending_requests_.push_back(reactor); |
|
} |
|
request_var_.SignalAll(); |
|
return reactor; |
|
} |
|
|
|
void SetReturnStatus(const Status& status) { |
|
grpc_core::MutexLock lock(&mu_); |
|
if (pending_requests_.empty()) { |
|
pending_status_ = status; |
|
} else { |
|
auto reactor = pending_requests_.begin(); |
|
(*reactor)->Finish(status); |
|
pending_requests_.erase(reactor); |
|
} |
|
request_var_.SignalAll(); |
|
} |
|
|
|
bool TestOnlyExpectRequests(size_t expected_requests_count, |
|
const absl::Duration& timeout) { |
|
grpc_core::MutexLock lock(&mu_); |
|
auto deadline = absl::Now() + timeout; |
|
while (pending_requests_.size() < expected_requests_count && |
|
!request_var_.WaitWithDeadline(&mu_, deadline)) { |
|
} |
|
return pending_requests_.size() >= expected_requests_count; |
|
} |
|
|
|
void Stop() { |
|
grpc_core::MutexLock lock(&mu_); |
|
for (auto request : pending_requests_) { |
|
request->Finish(Status(StatusCode::ABORTED, "Shutting down")); |
|
} |
|
pending_requests_.clear(); |
|
} |
|
|
|
private: |
|
grpc_core::Mutex mu_; |
|
grpc_core::CondVar request_var_ ABSL_GUARDED_BY(&mu_); |
|
absl::optional<Status> pending_status_ ABSL_GUARDED_BY(&mu_); |
|
std::vector<ServerUnaryReactor*> pending_requests_ ABSL_GUARDED_BY(&mu_); |
|
bool done_ ABSL_GUARDED_BY(&mu_) = false; |
|
}; |
|
|
|
enum class State { kNew, kWaiting, kDone, kShuttingDown }; |
|
|
|
std::unique_ptr<Server> BuildHookServer(HookServiceImpl* service, int port) { |
|
ServerBuilder builder; |
|
builder.AddListeningPort(absl::StrFormat("0.0.0.0:%d", port), |
|
grpc::InsecureServerCredentials()); |
|
builder.RegisterService(service); |
|
return builder.BuildAndStart(); |
|
} |
|
|
|
} // namespace |
|
|
|
class PreStopHookServer { |
|
public: |
|
explicit PreStopHookServer(int port, const absl::Duration& startup_timeout) |
|
: server_(BuildHookServer(&hook_service_, port)), |
|
server_thread_(PreStopHookServer::ServerThread, this) { |
|
WaitForState(State::kWaiting, startup_timeout); |
|
} |
|
|
|
~PreStopHookServer() { |
|
hook_service_.Stop(); |
|
SetState(State::kShuttingDown); |
|
server_->Shutdown(); |
|
server_thread_.join(); |
|
} |
|
|
|
State GetState() { |
|
grpc_core::MutexLock lock(&mu_); |
|
return state_; |
|
} |
|
|
|
void SetState(State state) { |
|
grpc_core::MutexLock lock(&mu_); |
|
state_ = state; |
|
condition_.SignalAll(); |
|
} |
|
|
|
void SetReturnStatus(const Status& status) { |
|
hook_service_.SetReturnStatus(status); |
|
} |
|
|
|
bool TestOnlyExpectRequests(size_t expected_requests_count, |
|
absl::Duration timeout) { |
|
return hook_service_.TestOnlyExpectRequests(expected_requests_count, |
|
timeout); |
|
} |
|
|
|
private: |
|
bool WaitForState(State state, const absl::Duration& timeout) { |
|
grpc_core::MutexLock lock(&mu_); |
|
auto deadline = absl::Now() + timeout; |
|
while (state_ != state && !condition_.WaitWithDeadline(&mu_, deadline)) { |
|
} |
|
return state_ == state; |
|
} |
|
|
|
static void ServerThread(PreStopHookServer* server) { |
|
server->SetState(State::kWaiting); |
|
server->server_->Wait(); |
|
server->SetState(State::kDone); |
|
} |
|
|
|
HookServiceImpl hook_service_; |
|
grpc_core::Mutex mu_; |
|
grpc_core::CondVar condition_ ABSL_GUARDED_BY(mu_); |
|
State state_ ABSL_GUARDED_BY(mu_) = State::kNew; |
|
std::unique_ptr<Server> server_; |
|
std::thread server_thread_; |
|
}; |
|
|
|
Status PreStopHookServerManager::Start(int port, size_t timeout_s) { |
|
if (server_) { |
|
return Status(StatusCode::ALREADY_EXISTS, |
|
"Pre hook server is already running"); |
|
} |
|
server_ = std::unique_ptr<PreStopHookServer, PreStopHookServerDeleter>( |
|
new PreStopHookServer(port, absl::Seconds(timeout_s)), |
|
PreStopHookServerDeleter()); |
|
return server_->GetState() == State::kWaiting |
|
? Status::OK |
|
: Status(StatusCode::DEADLINE_EXCEEDED, "Server have not started"); |
|
} |
|
|
|
Status PreStopHookServerManager::Stop() { |
|
if (!server_) { |
|
return Status(StatusCode::UNAVAILABLE, "Pre hook server is not running"); |
|
} |
|
server_.reset(); |
|
return Status::OK; |
|
} |
|
|
|
void PreStopHookServerManager::Return(StatusCode code, |
|
absl::string_view description) { |
|
server_->SetReturnStatus(Status(code, std::string(description))); |
|
} |
|
|
|
bool PreStopHookServerManager::TestOnlyExpectRequests( |
|
size_t expected_requests_count, const absl::Duration& timeout) { |
|
return server_->TestOnlyExpectRequests(expected_requests_count, timeout); |
|
} |
|
|
|
void PreStopHookServerManager::PreStopHookServerDeleter::operator()( |
|
PreStopHookServer* server) { |
|
delete server; |
|
} |
|
|
|
} // namespace testing |
|
} // namespace grpc
|
|
|