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.

205 lines
6.3 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 {
auto reactor = context->DefaultReactor();
MatchRequestsAndStatuses(reactor, absl::nullopt);
return reactor;
}
void SetReturnStatus(const Status& status) {
MatchRequestsAndStatuses(absl::nullopt, status);
}
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_);
done_ = true;
}
MatchRequestsAndStatuses(absl::nullopt, absl::nullopt);
}
private:
void MatchRequestsAndStatuses(absl::optional<ServerUnaryReactor*> new_request,
absl::optional<Status> new_status) {
grpc_core::MutexLock lock(&mu_);
if (new_request.has_value()) {
pending_requests_.push_back(*new_request);
}
if (new_status.has_value()) {
pending_statuses_.push_back(std::move(*new_status));
}
while (!pending_requests_.empty() && !pending_statuses_.empty()) {
pending_requests_.front()->Finish(std::move(pending_statuses_.front()));
pending_requests_.erase(pending_requests_.begin());
pending_statuses_.erase(pending_statuses_.begin());
}
while (!pending_requests_.empty() && done_) {
pending_requests_.front()->Finish(
Status(StatusCode::ABORTED, "Shutting down"));
pending_requests_.erase(pending_requests_.begin());
}
request_var_.SignalAll();
}
grpc_core::Mutex mu_;
grpc_core::CondVar request_var_ ABSL_GUARDED_BY(&mu_);
std::vector<ServerUnaryReactor*> pending_requests_ ABSL_GUARDED_BY(&mu_);
std::vector<Status> pending_statuses_ 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();
WaitForState(State::kDone, absl::Seconds(5));
server_thread_.detach();
}
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