mirror of https://github.com/grpc/grpc.git
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.
999 lines
37 KiB
999 lines
37 KiB
// Copyright 2017 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_TEST_CPP_END2END_XDS_XDS_END2END_TEST_LIB_H |
|
#define GRPC_TEST_CPP_END2END_XDS_XDS_END2END_TEST_LIB_H |
|
|
|
#include <memory> |
|
#include <set> |
|
#include <string> |
|
#include <thread> |
|
#include <vector> |
|
|
|
#include <gmock/gmock.h> |
|
#include <gtest/gtest.h> |
|
|
|
#include "absl/log/check.h" |
|
#include "absl/log/log.h" |
|
#include "absl/status/statusor.h" |
|
#include "absl/strings/str_cat.h" |
|
#include "absl/strings/string_view.h" |
|
#include "absl/types/optional.h" |
|
|
|
#include <grpc/grpc.h> |
|
#include <grpc/grpc_security.h> |
|
#include <grpcpp/channel.h> |
|
#include <grpcpp/client_context.h> |
|
#include <grpcpp/ext/call_metric_recorder.h> |
|
#include <grpcpp/ext/server_metric_recorder.h> |
|
#include <grpcpp/xds_server_builder.h> |
|
|
|
#include "src/core/lib/security/credentials/fake/fake_credentials.h" |
|
#include "src/core/lib/security/security_connector/ssl_utils.h" |
|
#include "src/cpp/server/secure_server_credentials.h" |
|
#include "src/proto/grpc/testing/echo.grpc.pb.h" |
|
#include "src/proto/grpc/testing/xds/v3/http_connection_manager.grpc.pb.h" |
|
#include "src/proto/grpc/testing/xds/v3/http_filter_rbac.grpc.pb.h" |
|
#include "src/proto/grpc/testing/xds/v3/orca_load_report.pb.h" |
|
#include "src/proto/grpc/testing/xds/v3/rbac.pb.h" |
|
#include "test/core/test_util/port.h" |
|
#include "test/cpp/end2end/counted_service.h" |
|
#include "test/cpp/end2end/test_service_impl.h" |
|
#include "test/cpp/end2end/xds/xds_server.h" |
|
#include "test/cpp/end2end/xds/xds_utils.h" |
|
|
|
namespace grpc { |
|
namespace testing { |
|
|
|
// The parameter type for INSTANTIATE_TEST_SUITE_P(). |
|
class XdsTestType { |
|
public: |
|
enum HttpFilterConfigLocation { |
|
// Set the HTTP filter config directly in LDS. |
|
kHttpFilterConfigInListener, |
|
// Enable the HTTP filter in LDS, but override the filter config in route. |
|
kHttpFilterConfigInRoute, |
|
}; |
|
|
|
enum BootstrapSource { |
|
kBootstrapFromChannelArg, |
|
kBootstrapFromFile, |
|
kBootstrapFromEnvVar, |
|
}; |
|
|
|
XdsTestType& set_enable_load_reporting() { |
|
enable_load_reporting_ = true; |
|
return *this; |
|
} |
|
|
|
XdsTestType& set_enable_rds_testing() { |
|
enable_rds_testing_ = true; |
|
return *this; |
|
} |
|
|
|
XdsTestType& set_use_csds_streaming() { |
|
use_csds_streaming_ = true; |
|
return *this; |
|
} |
|
|
|
XdsTestType& set_filter_config_setup(HttpFilterConfigLocation setup) { |
|
filter_config_setup_ = setup; |
|
return *this; |
|
} |
|
|
|
XdsTestType& set_bootstrap_source(BootstrapSource bootstrap_source) { |
|
bootstrap_source_ = bootstrap_source; |
|
return *this; |
|
} |
|
|
|
XdsTestType& set_rbac_action(::envoy::config::rbac::v3::RBAC_Action action) { |
|
rbac_action_ = action; |
|
return *this; |
|
} |
|
|
|
XdsTestType& set_rbac_audit_condition( |
|
::envoy::config::rbac::v3::RBAC_AuditLoggingOptions_AuditCondition |
|
audit_condition) { |
|
rbac_audit_condition_ = audit_condition; |
|
return *this; |
|
} |
|
|
|
bool enable_load_reporting() const { return enable_load_reporting_; } |
|
bool enable_rds_testing() const { return enable_rds_testing_; } |
|
bool use_csds_streaming() const { return use_csds_streaming_; } |
|
HttpFilterConfigLocation filter_config_setup() const { |
|
return filter_config_setup_; |
|
} |
|
BootstrapSource bootstrap_source() const { return bootstrap_source_; } |
|
::envoy::config::rbac::v3::RBAC_Action rbac_action() const { |
|
return rbac_action_; |
|
} |
|
::envoy::config::rbac::v3::RBAC_AuditLoggingOptions_AuditCondition |
|
rbac_audit_condition() const { |
|
return rbac_audit_condition_; |
|
} |
|
|
|
std::string AsString() const { |
|
std::string retval = "V3"; |
|
if (enable_load_reporting_) retval += "WithLoadReporting"; |
|
if (enable_rds_testing_) retval += "Rds"; |
|
if (use_csds_streaming_) retval += "CsdsStreaming"; |
|
if (filter_config_setup_ == kHttpFilterConfigInRoute) { |
|
retval += "FilterPerRouteOverride"; |
|
} |
|
if (bootstrap_source_ == kBootstrapFromFile) { |
|
retval += "BootstrapFromFile"; |
|
} else if (bootstrap_source_ == kBootstrapFromEnvVar) { |
|
retval += "BootstrapFromEnvVar"; |
|
} |
|
if (rbac_action_ == ::envoy::config::rbac::v3::RBAC_Action_ALLOW) { |
|
retval += "RbacAllow"; |
|
} else if (rbac_action_ == ::envoy::config::rbac::v3::RBAC_Action_DENY) { |
|
retval += "RbacDeny"; |
|
} |
|
if (rbac_audit_condition_ != |
|
::envoy::config::rbac::v3:: |
|
RBAC_AuditLoggingOptions_AuditCondition_NONE) { |
|
retval += absl::StrCat("AuditCondition", |
|
::envoy::config::rbac::v3:: |
|
RBAC_AuditLoggingOptions_AuditCondition_Name( |
|
rbac_audit_condition_)); |
|
} |
|
return retval; |
|
} |
|
|
|
// For use as the final parameter in INSTANTIATE_TEST_SUITE_P(). |
|
static std::string Name(const ::testing::TestParamInfo<XdsTestType>& info) { |
|
return info.param.AsString(); |
|
} |
|
|
|
private: |
|
bool enable_load_reporting_ = false; |
|
bool enable_rds_testing_ = false; |
|
bool use_csds_streaming_ = false; |
|
HttpFilterConfigLocation filter_config_setup_ = kHttpFilterConfigInListener; |
|
BootstrapSource bootstrap_source_ = kBootstrapFromChannelArg; |
|
::envoy::config::rbac::v3::RBAC_Action rbac_action_ = |
|
::envoy::config::rbac::v3::RBAC_Action_LOG; |
|
::envoy::config::rbac::v3::RBAC_AuditLoggingOptions_AuditCondition |
|
rbac_audit_condition_ = ::envoy::config::rbac::v3:: |
|
RBAC_AuditLoggingOptions_AuditCondition_NONE; |
|
}; |
|
|
|
// A base class for xDS end-to-end tests. |
|
// |
|
// An xDS server is provided in balancer_. It is automatically started |
|
// for every test. Additional xDS servers can be started if needed by |
|
// calling CreateAndStartBalancer(). |
|
// |
|
// A default set of LDS, RDS, and CDS resources are created for gRPC |
|
// clients, available in default_listener_, default_route_config_, and |
|
// default_cluster_. These resources are automatically loaded into |
|
// balancer_ but can be modified by individual tests. No EDS resource |
|
// is provided by default. There are also default LDS and RDS resources |
|
// for the gRPC server side in default_server_listener_ and |
|
// default_server_route_config_. Methods are provided for constructing new |
|
// resources that can be added to the xDS server as needed. |
|
// |
|
// This class provides a mechanism for running backend servers, which will |
|
// be stored in backends_. No servers are created or started by default, |
|
// but tests can call CreateAndStartBackends() to start however many |
|
// backends they want. There are also a number of methods for accessing |
|
// backends by index, which is the index into the backends_ vector. |
|
// For methods that take a start_index and stop_index, this refers to |
|
// the indexes in the range [start_index, stop_index). If stop_index |
|
// is 0, backends_.size() is used. Backends may or may not be |
|
// xDS-enabled, at the discretion of the test. |
|
class XdsEnd2endTest : public ::testing::TestWithParam<XdsTestType>, |
|
public XdsResourceUtils { |
|
protected: |
|
// TLS certificate paths. |
|
static const char kCaCertPath[]; |
|
static const char kServerCertPath[]; |
|
static const char kServerKeyPath[]; |
|
|
|
// Message used in EchoRequest to the backend. |
|
static const char kRequestMessage[]; |
|
|
|
// A base class for server threads. |
|
class ServerThread { |
|
public: |
|
// A status notifier for xDS-enabled servers. |
|
class XdsServingStatusNotifier |
|
: public grpc::XdsServerServingStatusNotifierInterface { |
|
public: |
|
void OnServingStatusUpdate(std::string uri, |
|
ServingStatusUpdate update) override; |
|
|
|
void WaitOnServingStatusChange(std::string uri, |
|
grpc::StatusCode expected_status); |
|
|
|
private: |
|
grpc_core::Mutex mu_; |
|
grpc_core::CondVar cond_; |
|
std::map<std::string, grpc::Status> status_map ABSL_GUARDED_BY(mu_); |
|
}; |
|
|
|
// If use_xds_enabled_server is true, the server will use xDS. |
|
// If credentials is null, fake credentials will be used. |
|
explicit ServerThread( |
|
XdsEnd2endTest* test_obj, bool use_xds_enabled_server = false, |
|
std::shared_ptr<ServerCredentials> credentials = nullptr); |
|
|
|
virtual ~ServerThread() { |
|
// Shutdown should be called manually. Shutdown calls virtual methods and |
|
// can't be called from the base class destructor. |
|
CHECK(!running_); |
|
} |
|
|
|
void Start(); |
|
void Shutdown(); |
|
|
|
std::string target() const { return absl::StrCat("localhost:", port_); } |
|
|
|
int port() const { return port_; } |
|
|
|
XdsServingStatusNotifier* notifier() { return ¬ifier_; } |
|
|
|
void set_allow_put_requests(bool allow_put_requests) { |
|
allow_put_requests_ = allow_put_requests; |
|
} |
|
|
|
void StopListening(); |
|
|
|
void StopListeningAndSendGoaways(); |
|
|
|
private: |
|
class XdsChannelArgsServerBuilderOption; |
|
|
|
virtual const char* Type() = 0; |
|
virtual void RegisterAllServices(ServerBuilder* builder) = 0; |
|
virtual void StartAllServices() = 0; |
|
virtual void ShutdownAllServices() = 0; |
|
|
|
void Serve(grpc_core::Mutex* mu, grpc_core::CondVar* cond); |
|
|
|
XdsEnd2endTest* test_obj_; |
|
const bool use_xds_enabled_server_; |
|
const std::shared_ptr<ServerCredentials> credentials_; |
|
const int port_; |
|
std::unique_ptr<Server> server_; |
|
XdsServingStatusNotifier notifier_; |
|
std::unique_ptr<std::thread> thread_; |
|
bool running_ = false; |
|
bool allow_put_requests_ = false; |
|
}; |
|
|
|
// A server thread for a backend server. |
|
class BackendServerThread : public ServerThread { |
|
public: |
|
// A wrapper around the backend echo test service impl that counts |
|
// requests and responses. |
|
template <typename RpcService> |
|
class BackendServiceImpl |
|
: public CountedService<TestMultipleServiceImpl<RpcService>> { |
|
public: |
|
BackendServiceImpl() {} |
|
|
|
Status Echo(ServerContext* context, const EchoRequest* request, |
|
EchoResponse* response) override { |
|
auto peer_identity = context->auth_context()->GetPeerIdentity(); |
|
CountedService< |
|
TestMultipleServiceImpl<RpcService>>::IncreaseRequestCount(); |
|
{ |
|
grpc_core::MutexLock lock(&mu_); |
|
clients_.insert(context->peer()); |
|
last_peer_identity_.clear(); |
|
for (const auto& entry : peer_identity) { |
|
last_peer_identity_.emplace_back(entry.data(), entry.size()); |
|
} |
|
} |
|
if (request->has_param() && request->param().has_backend_metrics()) { |
|
const auto& request_metrics = request->param().backend_metrics(); |
|
auto* recorder = context->ExperimentalGetCallMetricRecorder(); |
|
for (const auto& p : request_metrics.named_metrics()) { |
|
char* key = static_cast<char*>( |
|
grpc_call_arena_alloc(context->c_call(), p.first.size() + 1)); |
|
strncpy(key, p.first.data(), p.first.size()); |
|
key[p.first.size()] = '\0'; |
|
recorder->RecordNamedMetric(key, p.second); |
|
} |
|
} |
|
const auto status = TestMultipleServiceImpl<RpcService>::Echo( |
|
context, request, response); |
|
CountedService< |
|
TestMultipleServiceImpl<RpcService>>::IncreaseResponseCount(); |
|
return status; |
|
} |
|
|
|
Status Echo1(ServerContext* context, const EchoRequest* request, |
|
EchoResponse* response) override { |
|
return Echo(context, request, response); |
|
} |
|
|
|
Status Echo2(ServerContext* context, const EchoRequest* request, |
|
EchoResponse* response) override { |
|
return Echo(context, request, response); |
|
} |
|
|
|
void Start() {} |
|
void Shutdown() {} |
|
|
|
std::set<std::string> clients() { |
|
grpc_core::MutexLock lock(&mu_); |
|
return clients_; |
|
} |
|
|
|
const std::vector<std::string>& last_peer_identity() { |
|
grpc_core::MutexLock lock(&mu_); |
|
return last_peer_identity_; |
|
} |
|
|
|
private: |
|
grpc_core::Mutex mu_; |
|
std::set<std::string> clients_ ABSL_GUARDED_BY(&mu_); |
|
std::vector<std::string> last_peer_identity_ ABSL_GUARDED_BY(&mu_); |
|
}; |
|
|
|
// If use_xds_enabled_server is true, the server will use xDS. |
|
BackendServerThread(XdsEnd2endTest* test_obj, bool use_xds_enabled_server, |
|
std::shared_ptr<ServerCredentials> credentials); |
|
|
|
BackendServiceImpl<grpc::testing::EchoTestService::Service>* |
|
backend_service() { |
|
return &backend_service_; |
|
} |
|
BackendServiceImpl<grpc::testing::EchoTest1Service::Service>* |
|
backend_service1() { |
|
return &backend_service1_; |
|
} |
|
BackendServiceImpl<grpc::testing::EchoTest2Service::Service>* |
|
backend_service2() { |
|
return &backend_service2_; |
|
} |
|
grpc::experimental::ServerMetricRecorder* server_metric_recorder() const { |
|
return server_metric_recorder_.get(); |
|
} |
|
|
|
private: |
|
const char* Type() override { return "Backend"; } |
|
void RegisterAllServices(ServerBuilder* builder) override; |
|
void StartAllServices() override; |
|
void ShutdownAllServices() override; |
|
|
|
BackendServiceImpl<grpc::testing::EchoTestService::Service> |
|
backend_service_; |
|
BackendServiceImpl<grpc::testing::EchoTest1Service::Service> |
|
backend_service1_; |
|
BackendServiceImpl<grpc::testing::EchoTest2Service::Service> |
|
backend_service2_; |
|
std::unique_ptr<experimental::ServerMetricRecorder> server_metric_recorder_; |
|
}; |
|
|
|
// A server thread for the xDS server. |
|
class BalancerServerThread : public ServerThread { |
|
public: |
|
explicit BalancerServerThread( |
|
XdsEnd2endTest* test_obj, absl::string_view debug_label, |
|
std::shared_ptr<ServerCredentials> credentials); |
|
|
|
AdsServiceImpl* ads_service() { return ads_service_.get(); } |
|
LrsServiceImpl* lrs_service() { return lrs_service_.get(); } |
|
|
|
private: |
|
const char* Type() override { return "Balancer"; } |
|
void RegisterAllServices(ServerBuilder* builder) override; |
|
void StartAllServices() override; |
|
void ShutdownAllServices() override; |
|
|
|
std::shared_ptr<AdsServiceImpl> ads_service_; |
|
std::shared_ptr<LrsServiceImpl> lrs_service_; |
|
}; |
|
|
|
// RPC services used to talk to the backends. |
|
enum RpcService { |
|
SERVICE_ECHO, |
|
SERVICE_ECHO1, |
|
SERVICE_ECHO2, |
|
}; |
|
|
|
// RPC methods used to talk to the backends. |
|
enum RpcMethod { |
|
METHOD_ECHO, |
|
METHOD_ECHO1, |
|
METHOD_ECHO2, |
|
}; |
|
|
|
// If balancer_credentials is null, it defaults to fake credentials. |
|
explicit XdsEnd2endTest( |
|
std::shared_ptr<ServerCredentials> balancer_credentials = nullptr); |
|
|
|
void SetUp() override { InitClient(); } |
|
void TearDown() override; |
|
|
|
// |
|
// xDS server management |
|
// |
|
|
|
// Creates and starts a new balancer, running in its own thread. |
|
// Most tests will not need to call this; instead, they can use |
|
// balancer_, which is already populated with default resources. |
|
// If credentials is null, it defaults to fake credentials. |
|
std::unique_ptr<BalancerServerThread> CreateAndStartBalancer( |
|
absl::string_view debug_label = "", |
|
std::shared_ptr<ServerCredentials> credentials = nullptr); |
|
|
|
// Sets the Listener and RouteConfiguration resource on the specified |
|
// balancer. If RDS is in use, they will be set as separate resources; |
|
// otherwise, the RouteConfig will be inlined into the Listener. |
|
void SetListenerAndRouteConfiguration( |
|
BalancerServerThread* balancer, Listener listener, |
|
const RouteConfiguration& route_config, |
|
const HcmAccessor& hcm_accessor = ClientHcmAccessor()) { |
|
XdsResourceUtils::SetListenerAndRouteConfiguration( |
|
balancer->ads_service(), std::move(listener), route_config, |
|
GetParam().enable_rds_testing(), hcm_accessor); |
|
} |
|
|
|
// A convenient wrapper for setting the Listener and |
|
// RouteConfiguration resources on the server side. |
|
void SetServerListenerNameAndRouteConfiguration( |
|
BalancerServerThread* balancer, Listener listener, int port, |
|
const RouteConfiguration& route_config) { |
|
SetListenerAndRouteConfiguration( |
|
balancer, PopulateServerListenerNameAndPort(listener, port), |
|
route_config, ServerHcmAccessor()); |
|
} |
|
|
|
// Sets the RouteConfiguration resource on the specified balancer. |
|
// If RDS is in use, it will be set directly as an independent |
|
// resource; otherwise, it will be inlined into a Listener resource |
|
// (either listener_to_copy, or if that is null, default_listener_). |
|
void SetRouteConfiguration(BalancerServerThread* balancer, |
|
const RouteConfiguration& route_config, |
|
const Listener* listener_to_copy = nullptr) { |
|
XdsResourceUtils::SetRouteConfiguration( |
|
balancer->ads_service(), route_config, GetParam().enable_rds_testing(), |
|
listener_to_copy); |
|
} |
|
|
|
// Helper method for generating an endpoint for a backend, for use in |
|
// constructing an EDS resource. |
|
EdsResourceArgs::Endpoint CreateEndpoint( |
|
size_t backend_idx, |
|
::envoy::config::core::v3::HealthStatus health_status = |
|
::envoy::config::core::v3::HealthStatus::UNKNOWN, |
|
int lb_weight = 1, std::vector<size_t> additional_backend_indxees = {}) { |
|
std::vector<int> additional_ports; |
|
additional_ports.reserve(additional_backend_indxees.size()); |
|
for (size_t idx : additional_backend_indxees) { |
|
additional_ports.push_back(backends_[idx]->port()); |
|
} |
|
return EdsResourceArgs::Endpoint(backends_[backend_idx]->port(), |
|
health_status, lb_weight, |
|
additional_ports); |
|
} |
|
|
|
// Creates a vector of endpoints for a specified range of backends, |
|
// for use in constructing an EDS resource. |
|
std::vector<EdsResourceArgs::Endpoint> CreateEndpointsForBackends( |
|
size_t start_index = 0, size_t stop_index = 0, |
|
::envoy::config::core::v3::HealthStatus health_status = |
|
::envoy::config::core::v3::HealthStatus::UNKNOWN, |
|
int lb_weight = 1); |
|
|
|
// Returns an endpoint for an unused port, for use in constructing an |
|
// EDS resource. |
|
EdsResourceArgs::Endpoint MakeNonExistantEndpoint() { |
|
return EdsResourceArgs::Endpoint(grpc_pick_unused_port_or_die()); |
|
} |
|
|
|
// |
|
// Backend management |
|
// |
|
|
|
// Creates num_backends backends and stores them in backends_, but does |
|
// not actually start them. If xds_enabled is true, the backends are |
|
// xDS-enabled. If credentials is null, it defaults to fake credentials. |
|
void CreateBackends( |
|
size_t num_backends, bool xds_enabled = false, |
|
std::shared_ptr<ServerCredentials> credentials = nullptr) { |
|
for (size_t i = 0; i < num_backends; ++i) { |
|
backends_.emplace_back( |
|
new BackendServerThread(this, xds_enabled, credentials)); |
|
} |
|
} |
|
|
|
// Starts all backends in backends_. |
|
void StartAllBackends() { |
|
for (auto& backend : backends_) backend->Start(); |
|
} |
|
|
|
// Same as CreateBackends(), but also starts the backends. |
|
void CreateAndStartBackends( |
|
size_t num_backends, bool xds_enabled = false, |
|
std::shared_ptr<ServerCredentials> credentials = nullptr) { |
|
CreateBackends(num_backends, xds_enabled, std::move(credentials)); |
|
StartAllBackends(); |
|
} |
|
|
|
// Starts the backend at backends_[index]. |
|
void StartBackend(size_t index) { backends_[index]->Start(); } |
|
|
|
// Shuts down all backends in backends_. |
|
void ShutdownAllBackends() { |
|
for (auto& backend : backends_) backend->Shutdown(); |
|
} |
|
|
|
// Shuts down the backend at backends_[index]. |
|
void ShutdownBackend(size_t index) { backends_[index]->Shutdown(); } |
|
|
|
// Resets the request counters for backends in the specified range. |
|
void ResetBackendCounters(size_t start_index = 0, size_t stop_index = 0); |
|
|
|
// Returns true if the specified backend has received requests for the |
|
// specified service. |
|
bool SeenBackend(size_t backend_idx, |
|
const RpcService rpc_service = SERVICE_ECHO); |
|
|
|
// Returns true if all backends in the specified range have received |
|
// requests for the specified service. |
|
bool SeenAllBackends(size_t start_index = 0, size_t stop_index = 0, |
|
const RpcService rpc_service = SERVICE_ECHO); |
|
|
|
// Returns a vector containing the port for every backend in the |
|
// specified range. |
|
std::vector<int> GetBackendPorts(size_t start_index = 0, |
|
size_t stop_index = 0) const; |
|
|
|
// |
|
// Client management |
|
// |
|
|
|
// Initializes global state for the client, such as the bootstrap file |
|
// and channel args for the XdsClient. Then calls ResetStub(). |
|
// All tests must call this exactly once at the start of the test. |
|
// If credentials is null, fake credentials will be used. |
|
void InitClient(absl::optional<XdsBootstrapBuilder> builder = absl::nullopt, |
|
std::string lb_expected_authority = "", |
|
int xds_resource_does_not_exist_timeout_ms = 0, |
|
std::string balancer_authority_override = "", |
|
std::shared_ptr<ChannelCredentials> credentials = nullptr); |
|
|
|
XdsBootstrapBuilder MakeBootstrapBuilder() { |
|
return XdsBootstrapBuilder().SetServers( |
|
{absl::StrCat("localhost:", balancer_->port())}); |
|
} |
|
|
|
// Sets channel_, stub_, stub1_, and stub2_. |
|
// If credentials is null, fake credentials will be used. |
|
void ResetStub(int failover_timeout_ms = 0, ChannelArguments* args = nullptr, |
|
std::shared_ptr<ChannelCredentials> credentials = nullptr); |
|
|
|
// Creates a new client channel. Requires that InitClient() has |
|
// already been called. |
|
// If credentials is null, fake credentials will be used. |
|
std::shared_ptr<Channel> CreateChannel( |
|
int failover_timeout_ms = 0, const char* server_name = kServerName, |
|
const char* xds_authority = "", ChannelArguments* args = nullptr, |
|
std::shared_ptr<ChannelCredentials> credentials = nullptr); |
|
|
|
// |
|
// Sending RPCs |
|
// |
|
|
|
// Options used for sending an RPC. |
|
struct RpcOptions { |
|
RpcService service = SERVICE_ECHO; |
|
RpcMethod method = METHOD_ECHO; |
|
// Will be multiplied by grpc_test_slowdown_factor(). |
|
int timeout_ms = 5000; |
|
bool wait_for_ready = false; |
|
std::vector<std::pair<std::string, std::string>> metadata; |
|
// These options are used by the backend service impl. |
|
bool server_fail = false; |
|
int server_sleep_us = 0; |
|
int client_cancel_after_us = 0; |
|
bool skip_cancelled_check = false; |
|
StatusCode server_expected_error = StatusCode::OK; |
|
absl::optional<xds::data::orca::v3::OrcaLoadReport> backend_metrics; |
|
bool server_notify_client_when_started = false; |
|
|
|
RpcOptions() {} |
|
|
|
RpcOptions& set_rpc_service(RpcService rpc_service) { |
|
service = rpc_service; |
|
return *this; |
|
} |
|
|
|
RpcOptions& set_rpc_method(RpcMethod rpc_method) { |
|
method = rpc_method; |
|
return *this; |
|
} |
|
|
|
RpcOptions& set_timeout_ms(int rpc_timeout_ms) { |
|
timeout_ms = rpc_timeout_ms; |
|
return *this; |
|
} |
|
|
|
RpcOptions& set_timeout(grpc_core::Duration timeout) { |
|
timeout_ms = timeout.millis(); |
|
return *this; |
|
} |
|
|
|
RpcOptions& set_wait_for_ready(bool rpc_wait_for_ready) { |
|
wait_for_ready = rpc_wait_for_ready; |
|
return *this; |
|
} |
|
|
|
RpcOptions& set_metadata( |
|
std::vector<std::pair<std::string, std::string>> rpc_metadata) { |
|
metadata = std::move(rpc_metadata); |
|
return *this; |
|
} |
|
|
|
RpcOptions& set_server_fail(bool rpc_server_fail) { |
|
server_fail = rpc_server_fail; |
|
return *this; |
|
} |
|
|
|
RpcOptions& set_server_sleep_us(int rpc_server_sleep_us) { |
|
server_sleep_us = rpc_server_sleep_us; |
|
return *this; |
|
} |
|
|
|
RpcOptions& set_client_cancel_after_us(int rpc_client_cancel_after_us) { |
|
client_cancel_after_us = rpc_client_cancel_after_us; |
|
return *this; |
|
} |
|
|
|
RpcOptions& set_skip_cancelled_check(bool rpc_skip_cancelled_check) { |
|
skip_cancelled_check = rpc_skip_cancelled_check; |
|
return *this; |
|
} |
|
|
|
RpcOptions& set_server_expected_error(StatusCode code) { |
|
server_expected_error = code; |
|
return *this; |
|
} |
|
|
|
RpcOptions& set_backend_metrics( |
|
absl::optional<xds::data::orca::v3::OrcaLoadReport> metrics) { |
|
backend_metrics = std::move(metrics); |
|
return *this; |
|
} |
|
|
|
RpcOptions& set_server_notify_client_when_started( |
|
bool rpc_server_notify_client_when_started) { |
|
server_notify_client_when_started = rpc_server_notify_client_when_started; |
|
return *this; |
|
} |
|
|
|
// Populates context and request. |
|
void SetupRpc(ClientContext* context, EchoRequest* request) const; |
|
}; |
|
|
|
// Sends an RPC with the specified options. |
|
// If response is non-null, it will be populated with the response. |
|
// Returns the status of the RPC. |
|
Status SendRpc(const RpcOptions& rpc_options = RpcOptions(), |
|
EchoResponse* response = nullptr, |
|
std::multimap<std::string, std::string>* |
|
server_initial_metadata = nullptr); |
|
|
|
// Internal helper function for SendRpc(). |
|
template <typename Stub> |
|
static Status SendRpcMethod(Stub* stub, const RpcOptions& rpc_options, |
|
ClientContext* context, EchoRequest& request, |
|
EchoResponse* response) { |
|
switch (rpc_options.method) { |
|
case METHOD_ECHO: |
|
return stub->Echo(context, request, response); |
|
case METHOD_ECHO1: |
|
return stub->Echo1(context, request, response); |
|
case METHOD_ECHO2: |
|
return stub->Echo2(context, request, response); |
|
} |
|
GPR_UNREACHABLE_CODE(return grpc::Status::OK); |
|
} |
|
|
|
// Send RPCs in a loop until either continue_predicate() returns false |
|
// or timeout_ms elapses. |
|
struct RpcResult { |
|
Status status; |
|
EchoResponse response; |
|
}; |
|
void SendRpcsUntil(const grpc_core::DebugLocation& debug_location, |
|
std::function<bool(const RpcResult&)> continue_predicate, |
|
int timeout_ms = 15000, |
|
const RpcOptions& rpc_options = RpcOptions()); |
|
|
|
// Sends the specified number of RPCs and fails if the RPC fails. |
|
void CheckRpcSendOk(const grpc_core::DebugLocation& debug_location, |
|
const size_t times = 1, |
|
const RpcOptions& rpc_options = RpcOptions()); |
|
|
|
// Sends one RPC, which must fail with the specified status code and |
|
// a message matching the specified regex. |
|
void CheckRpcSendFailure(const grpc_core::DebugLocation& debug_location, |
|
StatusCode expected_status, |
|
absl::string_view expected_message_regex, |
|
const RpcOptions& rpc_options = RpcOptions()); |
|
|
|
// Sends num_rpcs RPCs, counting how many of them fail with a message |
|
// matching the specfied expected_message_prefix. |
|
// Any failure with a non-matching status or message is a test failure. |
|
size_t SendRpcsAndCountFailuresWithMessage( |
|
const grpc_core::DebugLocation& debug_location, size_t num_rpcs, |
|
StatusCode expected_status, absl::string_view expected_message_prefix, |
|
const RpcOptions& rpc_options = RpcOptions()); |
|
|
|
// A class for running a long-running RPC in its own thread. |
|
// TODO(roth): Maybe consolidate this and SendConcurrentRpcs() |
|
// somehow? LongRunningRpc has a cleaner API, but SendConcurrentRpcs() |
|
// uses the callback API, which is probably better. |
|
class LongRunningRpc { |
|
public: |
|
// Starts the RPC. |
|
void StartRpc(grpc::testing::EchoTestService::Stub* stub, |
|
const RpcOptions& rpc_options = |
|
RpcOptions().set_timeout_ms(0).set_client_cancel_after_us( |
|
1 * 1000 * 1000)); |
|
|
|
// Cancels the RPC. |
|
void CancelRpc(); |
|
|
|
// Gets the RPC's status. Blocks if the RPC is not yet complete. |
|
Status GetStatus(); |
|
|
|
private: |
|
std::thread sender_thread_; |
|
ClientContext context_; |
|
Status status_; |
|
}; |
|
|
|
// Starts a set of concurrent RPCs. |
|
struct ConcurrentRpc { |
|
ClientContext context; |
|
Status status; |
|
grpc_core::Duration elapsed_time; |
|
EchoResponse response; |
|
}; |
|
std::vector<ConcurrentRpc> SendConcurrentRpcs( |
|
const grpc_core::DebugLocation& debug_location, |
|
grpc::testing::EchoTestService::Stub* stub, size_t num_rpcs, |
|
const RpcOptions& rpc_options); |
|
|
|
// |
|
// Waiting for individual backends to be seen by the client |
|
// |
|
|
|
struct WaitForBackendOptions { |
|
// If true, resets the backend counters before returning. |
|
bool reset_counters = true; |
|
// How long to wait for the backend(s) to see requests. |
|
// Will be multiplied by grpc_test_slowdown_factor(). |
|
int timeout_ms = 15000; |
|
|
|
WaitForBackendOptions() {} |
|
|
|
WaitForBackendOptions& set_reset_counters(bool enable) { |
|
reset_counters = enable; |
|
return *this; |
|
} |
|
|
|
WaitForBackendOptions& set_timeout_ms(int ms) { |
|
timeout_ms = ms; |
|
return *this; |
|
} |
|
}; |
|
|
|
// Sends RPCs until all of the backends in the specified range see requests. |
|
// The check_status callback will be invoked to check the status of |
|
// every RPC; if null, the default is to check that the RPC succeeded. |
|
// Returns the total number of RPCs sent. |
|
size_t WaitForAllBackends( |
|
const grpc_core::DebugLocation& debug_location, size_t start_index = 0, |
|
size_t stop_index = 0, |
|
std::function<void(const RpcResult&)> check_status = nullptr, |
|
const WaitForBackendOptions& wait_options = WaitForBackendOptions(), |
|
const RpcOptions& rpc_options = RpcOptions()); |
|
|
|
// Sends RPCs until the backend at index backend_idx sees requests. |
|
void WaitForBackend( |
|
const grpc_core::DebugLocation& debug_location, size_t backend_idx, |
|
std::function<void(const RpcResult&)> check_status = nullptr, |
|
const WaitForBackendOptions& wait_options = WaitForBackendOptions(), |
|
const RpcOptions& rpc_options = RpcOptions()) { |
|
WaitForAllBackends(debug_location, backend_idx, backend_idx + 1, |
|
check_status, wait_options, rpc_options); |
|
} |
|
|
|
// |
|
// Waiting for xDS NACKs |
|
// |
|
// These methods send RPCs in a loop until they see a NACK from the |
|
// xDS server, or until a timeout expires. |
|
|
|
// Sends RPCs until get_state() returns a response. |
|
absl::optional<AdsServiceImpl::ResponseState> WaitForNack( |
|
const grpc_core::DebugLocation& debug_location, |
|
std::function<absl::optional<AdsServiceImpl::ResponseState>()> get_state, |
|
const RpcOptions& rpc_options = RpcOptions(), |
|
StatusCode expected_status = StatusCode::UNAVAILABLE); |
|
|
|
// Sends RPCs until an LDS NACK is seen. |
|
absl::optional<AdsServiceImpl::ResponseState> WaitForLdsNack( |
|
const grpc_core::DebugLocation& debug_location, |
|
const RpcOptions& rpc_options = RpcOptions(), |
|
StatusCode expected_status = StatusCode::UNAVAILABLE) { |
|
return WaitForNack( |
|
debug_location, |
|
[&]() { return balancer_->ads_service()->lds_response_state(); }, |
|
rpc_options, expected_status); |
|
} |
|
|
|
// Sends RPCs until an RDS NACK is seen. |
|
absl::optional<AdsServiceImpl::ResponseState> WaitForRdsNack( |
|
const grpc_core::DebugLocation& debug_location, |
|
const RpcOptions& rpc_options = RpcOptions(), |
|
StatusCode expected_status = StatusCode::UNAVAILABLE) { |
|
return WaitForNack( |
|
debug_location, |
|
[&]() { return RouteConfigurationResponseState(balancer_.get()); }, |
|
rpc_options, expected_status); |
|
} |
|
|
|
// Sends RPCs until a CDS NACK is seen. |
|
absl::optional<AdsServiceImpl::ResponseState> WaitForCdsNack( |
|
const grpc_core::DebugLocation& debug_location, |
|
const RpcOptions& rpc_options = RpcOptions(), |
|
StatusCode expected_status = StatusCode::UNAVAILABLE) { |
|
return WaitForNack( |
|
debug_location, |
|
[&]() { return balancer_->ads_service()->cds_response_state(); }, |
|
rpc_options, expected_status); |
|
} |
|
|
|
// Sends RPCs until an EDS NACK is seen. |
|
absl::optional<AdsServiceImpl::ResponseState> WaitForEdsNack( |
|
const grpc_core::DebugLocation& debug_location, |
|
const RpcOptions& rpc_options = RpcOptions()) { |
|
return WaitForNack( |
|
debug_location, |
|
[&]() { return balancer_->ads_service()->eds_response_state(); }, |
|
rpc_options); |
|
} |
|
|
|
// Convenient front-end to wait for RouteConfiguration to be NACKed, |
|
// regardless of whether it's sent in LDS or RDS. |
|
absl::optional<AdsServiceImpl::ResponseState> WaitForRouteConfigNack( |
|
const grpc_core::DebugLocation& debug_location, |
|
const RpcOptions& rpc_options = RpcOptions(), |
|
StatusCode expected_status = StatusCode::UNAVAILABLE) { |
|
if (GetParam().enable_rds_testing()) { |
|
return WaitForRdsNack(debug_location, rpc_options, expected_status); |
|
} |
|
return WaitForLdsNack(debug_location, rpc_options, expected_status); |
|
} |
|
|
|
// Convenient front-end for accessing xDS response state for a |
|
// RouteConfiguration, regardless of whether it's sent in LDS or RDS. |
|
absl::optional<AdsServiceImpl::ResponseState> RouteConfigurationResponseState( |
|
BalancerServerThread* balancer) const { |
|
AdsServiceImpl* ads_service = balancer->ads_service(); |
|
if (GetParam().enable_rds_testing()) { |
|
return ads_service->rds_response_state(); |
|
} |
|
return ads_service->lds_response_state(); |
|
} |
|
|
|
// |
|
// Miscellaneous helper methods |
|
// |
|
|
|
// There is slight difference between time fetched by GPR and by C++ system |
|
// clock API. It's unclear if they are using the same syscall, but we do know |
|
// GPR round the number at millisecond-level. This creates a 1ms difference, |
|
// which could cause flake. |
|
static grpc_core::Timestamp NowFromCycleCounter() { |
|
return grpc_core::Timestamp::FromTimespecRoundDown( |
|
gpr_now(GPR_CLOCK_MONOTONIC)); |
|
} |
|
|
|
// Sets duration_proto to duration times grpc_test_slowdown_factor(). |
|
static void SetProtoDuration(grpc_core::Duration duration, |
|
google::protobuf::Duration* duration_proto); |
|
|
|
// Returns the number of RPCs needed to pass error_tolerance at 99.99994% |
|
// chance. Rolling dices in drop/fault-injection generates a binomial |
|
// distribution (if our code is not horribly wrong). Let's make "n" the number |
|
// of samples, "p" the probability. If we have np>5 & n(1-p)>5, we can |
|
// approximately treat the binomial distribution as a normal distribution. |
|
// |
|
// For normal distribution, we can easily look up how many standard deviation |
|
// we need to reach 99.995%. Based on Wiki's table |
|
// https://en.wikipedia.org/wiki/68%E2%80%9395%E2%80%9399.7_rule, we need 5.00 |
|
// sigma (standard deviation) to cover the probability area of 99.99994%. In |
|
// another word, for a sample with size "n" probability "p" error-tolerance |
|
// "k", we want the error always land within 5.00 sigma. The sigma of |
|
// binominal distribution and be computed as sqrt(np(1-p)). Hence, we have |
|
// the equation: |
|
// |
|
// kn <= 5.00 * sqrt(np(1-p)) |
|
// TODO(yashykt): The above explanation assumes a normal distribution, but we |
|
// use a uniform distribution instead. We need a better estimate of how many |
|
// RPCs are needed with what error tolerance. |
|
static size_t ComputeIdealNumRpcs(double p, double error_tolerance) { |
|
CHECK_GE(p, 0); |
|
CHECK_LE(p, 1); |
|
size_t num_rpcs = |
|
ceil(p * (1 - p) * 5.00 * 5.00 / error_tolerance / error_tolerance); |
|
num_rpcs += 1000; // Add 1K as a buffer to avoid flakiness. |
|
LOG(INFO) << "Sending " << num_rpcs << " RPCs for percentage=" << p |
|
<< " error_tolerance=" << error_tolerance; |
|
return num_rpcs; |
|
} |
|
|
|
// Returns a regex that can be matched against an RPC failure status |
|
// message for a connection failure. |
|
static std::string MakeConnectionFailureRegex(absl::string_view prefix); |
|
|
|
// Returns a private key pair, read from local files. |
|
static grpc_core::PemKeyCertPairList ReadTlsIdentityPair( |
|
const char* key_path, const char* cert_path); |
|
|
|
// Internal helper function for creating TLS and mTLS creds. |
|
// Not intended to be used by tests. |
|
static std::vector<experimental::IdentityKeyCertPair> |
|
MakeIdentityKeyCertPairForTlsCreds(); |
|
|
|
// Returns XdsCredentials with mTLS fallback creds. |
|
static std::shared_ptr<ChannelCredentials> CreateXdsChannelCredentials(); |
|
|
|
// Creates various types of server credentials. |
|
static std::shared_ptr<ServerCredentials> CreateFakeServerCredentials(); |
|
static std::shared_ptr<ServerCredentials> CreateMtlsServerCredentials(); |
|
static std::shared_ptr<ServerCredentials> CreateTlsServerCredentials(); |
|
|
|
std::unique_ptr<BalancerServerThread> balancer_; |
|
|
|
std::shared_ptr<Channel> channel_; |
|
std::unique_ptr<grpc::testing::EchoTestService::Stub> stub_; |
|
std::unique_ptr<grpc::testing::EchoTest1Service::Stub> stub1_; |
|
std::unique_ptr<grpc::testing::EchoTest2Service::Stub> stub2_; |
|
|
|
std::vector<std::unique_ptr<BackendServerThread>> backends_; |
|
|
|
// Default xDS resources. |
|
Listener default_listener_; |
|
RouteConfiguration default_route_config_; |
|
Listener default_server_listener_; |
|
RouteConfiguration default_server_route_config_; |
|
Cluster default_cluster_; |
|
|
|
int xds_drain_grace_time_ms_ = 10 * 60 * 1000; // 10 mins |
|
|
|
bool bootstrap_contents_from_env_var_; |
|
std::string bootstrap_; |
|
char* bootstrap_file_ = nullptr; |
|
absl::InlinedVector<grpc_arg, 3> xds_channel_args_to_add_; |
|
grpc_channel_args xds_channel_args_; |
|
}; |
|
|
|
} // namespace testing |
|
} // namespace grpc |
|
|
|
#endif // GRPC_TEST_CPP_END2END_XDS_XDS_END2END_TEST_LIB_H
|
|
|