[core] Expand core attributes to hold values of any type (#32835)

pull/32981/head
Eugene Ostroukhov 2 years ago committed by GitHub
parent f02ce240d7
commit ac228814a0
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
  1. 4
      src/core/BUILD
  2. 12
      src/core/ext/filters/client_channel/client_channel.cc
  3. 4
      src/core/ext/filters/client_channel/client_channel_internal.h
  4. 9
      src/core/ext/filters/client_channel/lb_policy/ring_hash/ring_hash.cc
  5. 19
      src/core/ext/filters/client_channel/lb_policy/ring_hash/ring_hash.h
  6. 15
      src/core/ext/filters/client_channel/lb_policy/xds/xds_cluster_manager.cc
  7. 7
      src/core/ext/filters/client_channel/lb_policy/xds/xds_override_host.cc
  8. 11
      src/core/ext/filters/client_channel/resolver/xds/xds_resolver.cc
  9. 17
      src/core/ext/filters/client_channel/resolver/xds/xds_resolver.h
  10. 6
      src/core/ext/filters/stateful_session/stateful_session_filter.cc
  11. 17
      src/core/ext/filters/stateful_session/stateful_session_filter.h
  12. 22
      src/core/lib/service_config/service_config_call_data.h
  13. 54
      test/core/client_channel/lb_policy/lb_policy_test_lib.h
  14. 1
      test/core/client_channel/lb_policy/weighted_round_robin_test.cc
  15. 137
      test/core/client_channel/lb_policy/xds_override_host_test.cc

@ -4553,6 +4553,7 @@ grpc_cc_library(
"closure", "closure",
"error", "error",
"grpc_lb_subchannel_list", "grpc_lb_subchannel_list",
"grpc_service_config",
"json", "json",
"json_args", "json_args",
"json_object_loader", "json_object_loader",
@ -5142,8 +5143,10 @@ grpc_cc_library(
hdrs = [ hdrs = [
"ext/filters/client_channel/resolver/xds/xds_resolver.h", "ext/filters/client_channel/resolver/xds/xds_resolver.h",
], ],
external_deps = ["absl/strings"],
language = "c++", language = "c++",
deps = [ deps = [
"grpc_service_config",
"unique_type_name", "unique_type_name",
"//:gpr_platform", "//:gpr_platform",
], ],
@ -5174,6 +5177,7 @@ grpc_cc_library(
"channel_fwd", "channel_fwd",
"dual_ref_counted", "dual_ref_counted",
"grpc_lb_policy_ring_hash", "grpc_lb_policy_ring_hash",
"grpc_resolver_xds_header",
"grpc_service_config", "grpc_service_config",
"grpc_xds_client", "grpc_xds_client",
"iomgr_fwd", "iomgr_fwd",

@ -2335,7 +2335,8 @@ class ClientChannel::LoadBalancedCall::LbCallState
// Internal API to allow first-party LB policies to access per-call // Internal API to allow first-party LB policies to access per-call
// attributes set by the ConfigSelector. // attributes set by the ConfigSelector.
absl::string_view GetCallAttribute(UniqueTypeName type) override; ServiceConfigCallData::CallAttributeInterface* GetCallAttribute(
UniqueTypeName type) const override;
private: private:
LoadBalancedCall* lb_call_; LoadBalancedCall* lb_call_;
@ -2420,15 +2421,12 @@ class ClientChannel::LoadBalancedCall::Metadata
// ClientChannel::LoadBalancedCall::LbCallState // ClientChannel::LoadBalancedCall::LbCallState
// //
absl::string_view ServiceConfigCallData::CallAttributeInterface*
ClientChannel::LoadBalancedCall::LbCallState::GetCallAttribute( ClientChannel::LoadBalancedCall::LbCallState::GetCallAttribute(
UniqueTypeName type) { UniqueTypeName type) const {
auto* service_config_call_data = static_cast<ServiceConfigCallData*>( auto* service_config_call_data = static_cast<ServiceConfigCallData*>(
lb_call_->call_context()[GRPC_CONTEXT_SERVICE_CONFIG_CALL_DATA].value); lb_call_->call_context()[GRPC_CONTEXT_SERVICE_CONFIG_CALL_DATA].value);
auto& call_attributes = service_config_call_data->call_attributes(); return service_config_call_data->GetCallAttribute(type);
auto it = call_attributes.find(type);
if (it == call_attributes.end()) return absl::string_view();
return it->second;
} }
// //

@ -22,7 +22,6 @@
#include <utility> #include <utility>
#include "absl/functional/any_invocable.h" #include "absl/functional/any_invocable.h"
#include "absl/strings/string_view.h"
#include "src/core/lib/channel/context.h" #include "src/core/lib/channel/context.h"
#include "src/core/lib/gprpp/ref_counted_ptr.h" #include "src/core/lib/gprpp/ref_counted_ptr.h"
@ -48,7 +47,8 @@ namespace grpc_core {
// LB policies to access internal call attributes. // LB policies to access internal call attributes.
class ClientChannelLbCallState : public LoadBalancingPolicy::CallState { class ClientChannelLbCallState : public LoadBalancingPolicy::CallState {
public: public:
virtual absl::string_view GetCallAttribute(UniqueTypeName type) = 0; virtual ServiceConfigCallData::CallAttributeInterface* GetCallAttribute(
UniqueTypeName type) const = 0;
}; };
// Internal type for ServiceConfigCallData. Handles call commits. // Internal type for ServiceConfigCallData. Handles call commits.

@ -70,7 +70,7 @@ namespace grpc_core {
TraceFlag grpc_lb_ring_hash_trace(false, "ring_hash_lb"); TraceFlag grpc_lb_ring_hash_trace(false, "ring_hash_lb");
UniqueTypeName RequestHashAttributeName() { UniqueTypeName RequestHashAttribute::TypeName() {
static UniqueTypeName::Factory kFactory("request_hash"); static UniqueTypeName::Factory kFactory("request_hash");
return kFactory.Create(); return kFactory.Create();
} }
@ -345,7 +345,12 @@ class RingHash : public LoadBalancingPolicy {
RingHash::PickResult RingHash::Picker::Pick(PickArgs args) { RingHash::PickResult RingHash::Picker::Pick(PickArgs args) {
auto* call_state = static_cast<ClientChannelLbCallState*>(args.call_state); auto* call_state = static_cast<ClientChannelLbCallState*>(args.call_state);
auto hash = call_state->GetCallAttribute(RequestHashAttributeName()); auto* hash_attribute = static_cast<RequestHashAttribute*>(
call_state->GetCallAttribute(RequestHashAttribute::TypeName()));
absl::string_view hash;
if (hash_attribute != nullptr) {
hash = hash_attribute->request_hash();
}
uint64_t h; uint64_t h;
if (!absl::SimpleAtoi(hash, &h)) { if (!absl::SimpleAtoi(hash, &h)) {
return PickResult::Fail( return PickResult::Fail(

@ -21,15 +21,32 @@
#include <stdint.h> #include <stdint.h>
#include "absl/strings/string_view.h"
#include "src/core/lib/gprpp/unique_type_name.h" #include "src/core/lib/gprpp/unique_type_name.h"
#include "src/core/lib/gprpp/validation_errors.h" #include "src/core/lib/gprpp/validation_errors.h"
#include "src/core/lib/json/json.h" #include "src/core/lib/json/json.h"
#include "src/core/lib/json/json_args.h" #include "src/core/lib/json/json_args.h"
#include "src/core/lib/json/json_object_loader.h" #include "src/core/lib/json/json_object_loader.h"
#include "src/core/lib/service_config/service_config_call_data.h"
namespace grpc_core { namespace grpc_core {
UniqueTypeName RequestHashAttributeName(); class RequestHashAttribute
: public ServiceConfigCallData::CallAttributeInterface {
public:
static UniqueTypeName TypeName();
explicit RequestHashAttribute(absl::string_view request_hash)
: request_hash_(request_hash) {}
absl::string_view request_hash() const { return request_hash_; }
private:
UniqueTypeName type() const override { return TypeName(); }
absl::string_view request_hash_;
};
// Helper Parsing method to parse ring hash policy configs; for example, ring // Helper Parsing method to parse ring hash policy configs; for example, ring
// hash size validity. // hash size validity.

@ -19,6 +19,7 @@
#include <stddef.h> #include <stddef.h>
#include <algorithm> #include <algorithm>
#include <functional>
#include <map> #include <map>
#include <memory> #include <memory>
#include <string> #include <string>
@ -123,8 +124,8 @@ class XdsClusterManagerLb : public LoadBalancingPolicy {
class ClusterPicker : public SubchannelPicker { class ClusterPicker : public SubchannelPicker {
public: public:
// Maintains a map of cluster names to pickers. // Maintains a map of cluster names to pickers.
using ClusterMap = using ClusterMap = std::map<std::string /*cluster_name*/,
std::map<std::string /*cluster_name*/, RefCountedPtr<SubchannelPicker>>; RefCountedPtr<SubchannelPicker>, std::less<>>;
// It is required that the keys of cluster_map have to live at least as long // It is required that the keys of cluster_map have to live at least as long
// as the ClusterPicker instance. // as the ClusterPicker instance.
@ -230,9 +231,13 @@ class XdsClusterManagerLb : public LoadBalancingPolicy {
XdsClusterManagerLb::PickResult XdsClusterManagerLb::ClusterPicker::Pick( XdsClusterManagerLb::PickResult XdsClusterManagerLb::ClusterPicker::Pick(
PickArgs args) { PickArgs args) {
auto* call_state = static_cast<ClientChannelLbCallState*>(args.call_state); auto* call_state = static_cast<ClientChannelLbCallState*>(args.call_state);
auto cluster_name = auto* cluster_name_attribute = static_cast<XdsClusterAttribute*>(
call_state->GetCallAttribute(XdsClusterAttributeTypeName()); call_state->GetCallAttribute(XdsClusterAttribute::TypeName()));
auto it = cluster_map_.find(std::string(cluster_name)); absl::string_view cluster_name;
if (cluster_name_attribute != nullptr) {
cluster_name = cluster_name_attribute->cluster();
}
auto it = cluster_map_.find(cluster_name);
if (it != cluster_map_.end()) { if (it != cluster_map_.end()) {
return it->second->Pick(args); return it->second->Pick(args);
} }

@ -370,8 +370,11 @@ XdsOverrideHostLb::Picker::PickOverridenHost(absl::string_view override_host) {
LoadBalancingPolicy::PickResult XdsOverrideHostLb::Picker::Pick( LoadBalancingPolicy::PickResult XdsOverrideHostLb::Picker::Pick(
LoadBalancingPolicy::PickArgs args) { LoadBalancingPolicy::PickArgs args) {
auto* call_state = static_cast<ClientChannelLbCallState*>(args.call_state); auto* call_state = static_cast<ClientChannelLbCallState*>(args.call_state);
auto override_host = call_state->GetCallAttribute(XdsOverrideHostTypeName()); auto* override_host = static_cast<XdsOverrideHostAttribute*>(
auto overridden_host_pick = PickOverridenHost(override_host); call_state->GetCallAttribute(XdsOverrideHostAttribute::TypeName()));
auto overridden_host_pick =
PickOverridenHost(override_host != nullptr ? override_host->host_name()
: absl::string_view());
if (overridden_host_pick.has_value()) { if (overridden_host_pick.has_value()) {
return std::move(*overridden_host_pick); return std::move(*overridden_host_pick);
} }

@ -16,11 +16,11 @@
#include <grpc/support/port_platform.h> #include <grpc/support/port_platform.h>
#include <stdint.h>
#include <stdlib.h> #include <stdlib.h>
#include <string.h> #include <string.h>
#include <algorithm> #include <algorithm>
#include <cstdint>
#include <initializer_list> #include <initializer_list>
#include <map> #include <map>
#include <memory> #include <memory>
@ -59,6 +59,7 @@
#include "src/core/ext/filters/client_channel/config_selector.h" #include "src/core/ext/filters/client_channel/config_selector.h"
#include "src/core/ext/filters/client_channel/lb_policy/ring_hash/ring_hash.h" #include "src/core/ext/filters/client_channel/lb_policy/ring_hash/ring_hash.h"
#include "src/core/ext/filters/client_channel/resolver/xds/xds_resolver.h"
#include "src/core/ext/xds/xds_bootstrap.h" #include "src/core/ext/xds/xds_bootstrap.h"
#include "src/core/ext/xds/xds_bootstrap_grpc.h" #include "src/core/ext/xds/xds_bootstrap_grpc.h"
#include "src/core/ext/xds/xds_client_grpc.h" #include "src/core/ext/xds/xds_client_grpc.h"
@ -93,7 +94,7 @@ namespace grpc_core {
TraceFlag grpc_xds_resolver_trace(false, "xds_resolver"); TraceFlag grpc_xds_resolver_trace(false, "xds_resolver");
UniqueTypeName XdsClusterAttributeTypeName() { UniqueTypeName XdsClusterAttribute::TypeName() {
static UniqueTypeName::Factory kFactory("xds_cluster_name"); static UniqueTypeName::Factory kFactory("xds_cluster_name");
return kFactory.Create(); return kFactory.Create();
} }
@ -731,13 +732,15 @@ XdsResolver::XdsConfigSelector::GetCallConfig(GetCallConfigArgs args) {
method_config->GetMethodParsedConfigVector(grpc_empty_slice()); method_config->GetMethodParsedConfigVector(grpc_empty_slice());
call_config.service_config = std::move(method_config); call_config.service_config = std::move(method_config);
} }
call_config.call_attributes[XdsClusterAttributeTypeName()] = it->first; call_config.call_attributes[XdsClusterAttribute::TypeName()] =
args.arena->New<XdsClusterAttribute>(it->first);
std::string hash_string = absl::StrCat(hash.value()); std::string hash_string = absl::StrCat(hash.value());
char* hash_value = char* hash_value =
static_cast<char*>(args.arena->Alloc(hash_string.size() + 1)); static_cast<char*>(args.arena->Alloc(hash_string.size() + 1));
memcpy(hash_value, hash_string.c_str(), hash_string.size()); memcpy(hash_value, hash_string.c_str(), hash_string.size());
hash_value[hash_string.size()] = '\0'; hash_value[hash_string.size()] = '\0';
call_config.call_attributes[RequestHashAttributeName()] = hash_value; call_config.call_attributes[RequestHashAttribute::TypeName()] =
args.arena->New<RequestHashAttribute>(hash_value);
call_config.on_commit = [cluster_state = it->second->Ref()]() mutable { call_config.on_commit = [cluster_state = it->second->Ref()]() mutable {
cluster_state.reset(); cluster_state.reset();
}; };

@ -19,12 +19,27 @@
#include <grpc/support/port_platform.h> #include <grpc/support/port_platform.h>
#include "absl/strings/string_view.h"
#include "src/core/lib/gprpp/unique_type_name.h" #include "src/core/lib/gprpp/unique_type_name.h"
#include "src/core/lib/service_config/service_config_call_data.h"
namespace grpc_core { namespace grpc_core {
UniqueTypeName XdsClusterAttributeTypeName(); class XdsClusterAttribute
: public ServiceConfigCallData::CallAttributeInterface {
public:
static UniqueTypeName TypeName();
explicit XdsClusterAttribute(absl::string_view cluster) : cluster_(cluster) {}
absl::string_view cluster() const { return cluster_; }
private:
UniqueTypeName type() const override { return TypeName(); }
absl::string_view cluster_;
};
} // namespace grpc_core } // namespace grpc_core
#endif // GRPC_SRC_CORE_EXT_FILTERS_CLIENT_CHANNEL_RESOLVER_XDS_XDS_RESOLVER_H #endif // GRPC_SRC_CORE_EXT_FILTERS_CLIENT_CHANNEL_RESOLVER_XDS_XDS_RESOLVER_H

@ -59,7 +59,7 @@ namespace grpc_core {
TraceFlag grpc_stateful_session_filter_trace(false, "stateful_session_filter"); TraceFlag grpc_stateful_session_filter_trace(false, "stateful_session_filter");
UniqueTypeName XdsOverrideHostTypeName() { UniqueTypeName XdsOverrideHostAttribute::TypeName() {
static UniqueTypeName::Factory kFactory("xds_override_host"); static UniqueTypeName::Factory kFactory("xds_override_host");
return kFactory.Create(); return kFactory.Create();
} }
@ -160,8 +160,8 @@ ArenaPromise<ServerMetadataHandle> StatefulSessionFilter::MakeCallPromise(
} }
// We have a valid cookie, so add the call attribute to be used by the // We have a valid cookie, so add the call attribute to be used by the
// xds_override_host LB policy. // xds_override_host LB policy.
service_config_call_data->SetCallAttribute(XdsOverrideHostTypeName(), service_config_call_data->SetCallAttribute(
*cookie_value); GetContext<Arena>()->New<XdsOverrideHostAttribute>(*cookie_value));
} }
// Intercept server initial metadata. // Intercept server initial metadata.
call_args.server_initial_metadata->InterceptAndMap( call_args.server_initial_metadata->InterceptAndMap(

@ -30,11 +30,26 @@
#include "src/core/lib/channel/promise_based_filter.h" #include "src/core/lib/channel/promise_based_filter.h"
#include "src/core/lib/gprpp/unique_type_name.h" #include "src/core/lib/gprpp/unique_type_name.h"
#include "src/core/lib/promise/arena_promise.h" #include "src/core/lib/promise/arena_promise.h"
#include "src/core/lib/service_config/service_config_call_data.h"
#include "src/core/lib/transport/transport.h" #include "src/core/lib/transport/transport.h"
namespace grpc_core { namespace grpc_core {
UniqueTypeName XdsOverrideHostTypeName(); class XdsOverrideHostAttribute
: public ServiceConfigCallData::CallAttributeInterface {
public:
static UniqueTypeName TypeName();
explicit XdsOverrideHostAttribute(absl::string_view host_name)
: host_name_(host_name) {}
absl::string_view host_name() const { return host_name_; }
private:
UniqueTypeName type() const override { return TypeName(); }
absl::string_view host_name_;
};
// A filter to provide cookie-based stateful session affinity. // A filter to provide cookie-based stateful session affinity.
class StatefulSessionFilter : public ChannelFilter { class StatefulSessionFilter : public ChannelFilter {

@ -25,8 +25,6 @@
#include <memory> #include <memory>
#include <utility> #include <utility>
#include "absl/strings/string_view.h"
#include "src/core/lib/gprpp/ref_counted_ptr.h" #include "src/core/lib/gprpp/ref_counted_ptr.h"
#include "src/core/lib/gprpp/unique_type_name.h" #include "src/core/lib/gprpp/unique_type_name.h"
#include "src/core/lib/service_config/service_config.h" #include "src/core/lib/service_config/service_config.h"
@ -40,7 +38,13 @@ namespace grpc_core {
/// easily access method and global parameters for the call. /// easily access method and global parameters for the call.
class ServiceConfigCallData { class ServiceConfigCallData {
public: public:
using CallAttributes = std::map<UniqueTypeName, absl::string_view>; class CallAttributeInterface {
public:
virtual ~CallAttributeInterface() = default;
virtual UniqueTypeName type() const = 0;
};
using CallAttributes = std::map<UniqueTypeName, CallAttributeInterface*>;
ServiceConfigCallData() : method_configs_(nullptr) {} ServiceConfigCallData() : method_configs_(nullptr) {}
@ -63,12 +67,16 @@ class ServiceConfigCallData {
return service_config_->GetGlobalParsedConfig(index); return service_config_->GetGlobalParsedConfig(index);
} }
const CallAttributes& call_attributes() const { return call_attributes_; }
// Must be called when holding the call combiner (legacy filter) or from // Must be called when holding the call combiner (legacy filter) or from
// inside the activity (promise-based filter). // inside the activity (promise-based filter).
void SetCallAttribute(UniqueTypeName name, absl::string_view value) { void SetCallAttribute(CallAttributeInterface* value) {
call_attributes_[name] = value; call_attributes_.emplace(value->type(), value);
}
CallAttributeInterface* GetCallAttribute(UniqueTypeName name) const {
auto it = call_attributes_.find(name);
if (it == call_attributes_.end()) return nullptr;
return it->second;
} }
private: private:

@ -76,6 +76,7 @@
#include "src/core/lib/load_balancing/lb_policy_registry.h" #include "src/core/lib/load_balancing/lb_policy_registry.h"
#include "src/core/lib/load_balancing/subchannel_interface.h" #include "src/core/lib/load_balancing/subchannel_interface.h"
#include "src/core/lib/resolver/server_address.h" #include "src/core/lib/resolver/server_address.h"
#include "src/core/lib/service_config/service_config_call_data.h"
#include "src/core/lib/transport/connectivity_state.h" #include "src/core/lib/transport/connectivity_state.h"
#include "src/core/lib/uri/uri_parser.h" #include "src/core/lib/uri/uri_parser.h"
@ -84,6 +85,9 @@ namespace testing {
class LoadBalancingPolicyTest : public ::testing::Test { class LoadBalancingPolicyTest : public ::testing::Test {
protected: protected:
using CallAttributes = std::vector<
std::unique_ptr<ServiceConfigCallData::CallAttributeInterface>>;
// Channel-level subchannel state for a specific address and channel args. // Channel-level subchannel state for a specific address and channel args.
// This is analogous to the real subchannel in the ClientChannel code. // This is analogous to the real subchannel in the ClientChannel code.
class SubchannelState { class SubchannelState {
@ -464,10 +468,9 @@ class LoadBalancingPolicyTest : public ::testing::Test {
// A fake CallState implementation, for use in PickArgs. // A fake CallState implementation, for use in PickArgs.
class FakeCallState : public ClientChannelLbCallState { class FakeCallState : public ClientChannelLbCallState {
public: public:
explicit FakeCallState( explicit FakeCallState(const CallAttributes& attributes) {
const std::map<UniqueTypeName, absl::string_view>& attributes) {
for (const auto& p : attributes) { for (const auto& p : attributes) {
attributes_.emplace(p.first, std::string(p.second)); attributes_.emplace(p->type(), p.get());
} }
} }
@ -484,12 +487,18 @@ class LoadBalancingPolicyTest : public ::testing::Test {
return allocation; return allocation;
} }
absl::string_view GetCallAttribute(UniqueTypeName type) override { ServiceConfigCallData::CallAttributeInterface* GetCallAttribute(
return attributes_[type]; UniqueTypeName type) const override {
auto it = attributes_.find(type);
if (it != attributes_.end()) {
return it->second;
}
return nullptr;
} }
std::vector<void*> allocations_; std::vector<void*> allocations_;
std::map<UniqueTypeName, std::string> attributes_; std::map<UniqueTypeName, ServiceConfigCallData::CallAttributeInterface*>
attributes_;
}; };
// A fake BackendMetricAccessor implementation, for passing to // A fake BackendMetricAccessor implementation, for passing to
@ -699,11 +708,11 @@ class LoadBalancingPolicyTest : public ::testing::Test {
// the old list followed by one READY update where the picker is using the // the old list followed by one READY update where the picker is using the
// new list. Returns a picker if the reported states match expectations. // new list. Returns a picker if the reported states match expectations.
RefCountedPtr<LoadBalancingPolicy::SubchannelPicker> RefCountedPtr<LoadBalancingPolicy::SubchannelPicker>
WaitForRoundRobinListChange( WaitForRoundRobinListChange(absl::Span<const absl::string_view> old_addresses,
absl::Span<const absl::string_view> old_addresses, absl::Span<const absl::string_view> new_addresses,
absl::Span<const absl::string_view> new_addresses, const CallAttributes& call_attributes = {},
const std::map<UniqueTypeName, absl::string_view>& call_attributes = {}, size_t num_iterations = 3,
size_t num_iterations = 3, SourceLocation location = SourceLocation()) { SourceLocation location = SourceLocation()) {
gpr_log(GPR_INFO, "Waiting for expected RR addresses..."); gpr_log(GPR_INFO, "Waiting for expected RR addresses...");
RefCountedPtr<LoadBalancingPolicy::SubchannelPicker> retval; RefCountedPtr<LoadBalancingPolicy::SubchannelPicker> retval;
size_t num_picks = size_t num_picks =
@ -762,7 +771,7 @@ class LoadBalancingPolicyTest : public ::testing::Test {
// Does a pick and returns the result. // Does a pick and returns the result.
LoadBalancingPolicy::PickResult DoPick( LoadBalancingPolicy::PickResult DoPick(
LoadBalancingPolicy::SubchannelPicker* picker, LoadBalancingPolicy::SubchannelPicker* picker,
const std::map<UniqueTypeName, absl::string_view>& call_attributes = {}) { const CallAttributes& call_attributes = {}) {
ExecCtx exec_ctx; ExecCtx exec_ctx;
FakeMetadata metadata({}); FakeMetadata metadata({});
FakeCallState call_state(call_attributes); FakeCallState call_state(call_attributes);
@ -770,10 +779,9 @@ class LoadBalancingPolicyTest : public ::testing::Test {
} }
// Requests a pick on picker and expects a Queue result. // Requests a pick on picker and expects a Queue result.
void ExpectPickQueued( void ExpectPickQueued(LoadBalancingPolicy::SubchannelPicker* picker,
LoadBalancingPolicy::SubchannelPicker* picker, const CallAttributes call_attributes = {},
const std::map<UniqueTypeName, absl::string_view>& call_attributes = {}, SourceLocation location = SourceLocation()) {
SourceLocation location = SourceLocation()) {
ASSERT_NE(picker, nullptr); ASSERT_NE(picker, nullptr);
auto pick_result = DoPick(picker, call_attributes); auto pick_result = DoPick(picker, call_attributes);
ASSERT_TRUE(absl::holds_alternative<LoadBalancingPolicy::PickResult::Queue>( ASSERT_TRUE(absl::holds_alternative<LoadBalancingPolicy::PickResult::Queue>(
@ -791,7 +799,7 @@ class LoadBalancingPolicyTest : public ::testing::Test {
// automatically to represent a complete call with no backend metric data. // automatically to represent a complete call with no backend metric data.
absl::optional<std::string> ExpectPickComplete( absl::optional<std::string> ExpectPickComplete(
LoadBalancingPolicy::SubchannelPicker* picker, LoadBalancingPolicy::SubchannelPicker* picker,
const std::map<UniqueTypeName, absl::string_view>& call_attributes = {}, const CallAttributes& call_attributes = {},
std::unique_ptr<LoadBalancingPolicy::SubchannelCallTrackerInterface>* std::unique_ptr<LoadBalancingPolicy::SubchannelCallTrackerInterface>*
subchannel_call_tracker = nullptr, subchannel_call_tracker = nullptr,
SourceLocation location = SourceLocation()) { SourceLocation location = SourceLocation()) {
@ -827,7 +835,7 @@ class LoadBalancingPolicyTest : public ::testing::Test {
// list of addresses, or nullopt if a non-complete pick was returned. // list of addresses, or nullopt if a non-complete pick was returned.
absl::optional<std::vector<std::string>> GetCompletePicks( absl::optional<std::vector<std::string>> GetCompletePicks(
LoadBalancingPolicy::SubchannelPicker* picker, size_t num_picks, LoadBalancingPolicy::SubchannelPicker* picker, size_t num_picks,
const std::map<UniqueTypeName, absl::string_view>& call_attributes = {}, const CallAttributes& call_attributes = {},
std::vector< std::vector<
std::unique_ptr<LoadBalancingPolicy::SubchannelCallTrackerInterface>>* std::unique_ptr<LoadBalancingPolicy::SubchannelCallTrackerInterface>>*
subchannel_call_trackers = nullptr, subchannel_call_trackers = nullptr,
@ -874,11 +882,11 @@ class LoadBalancingPolicyTest : public ::testing::Test {
// Checks that the picker has round-robin behavior over the specified // Checks that the picker has round-robin behavior over the specified
// set of addresses. // set of addresses.
void ExpectRoundRobinPicks( void ExpectRoundRobinPicks(LoadBalancingPolicy::SubchannelPicker* picker,
LoadBalancingPolicy::SubchannelPicker* picker, absl::Span<const absl::string_view> addresses,
absl::Span<const absl::string_view> addresses, const CallAttributes& call_attributes = {},
const std::map<UniqueTypeName, absl::string_view>& call_attributes = {}, size_t num_iterations = 3,
size_t num_iterations = 3, SourceLocation location = SourceLocation()) { SourceLocation location = SourceLocation()) {
auto picks = GetCompletePicks(picker, num_iterations * addresses.size(), auto picks = GetCompletePicks(picker, num_iterations * addresses.size(),
call_attributes, nullptr, location); call_attributes, nullptr, location);
ASSERT_TRUE(picks.has_value()) << location.file() << ":" << location.line(); ASSERT_TRUE(picks.has_value()) << location.file() << ":" << location.line();

@ -47,7 +47,6 @@
#include "src/core/lib/gprpp/orphanable.h" #include "src/core/lib/gprpp/orphanable.h"
#include "src/core/lib/gprpp/ref_counted_ptr.h" #include "src/core/lib/gprpp/ref_counted_ptr.h"
#include "src/core/lib/gprpp/time.h" #include "src/core/lib/gprpp/time.h"
#include "src/core/lib/gprpp/unique_type_name.h"
#include "src/core/lib/json/json.h" #include "src/core/lib/json/json.h"
#include "src/core/lib/json/json_writer.h" #include "src/core/lib/json/json_writer.h"
#include "src/core/lib/load_balancing/lb_policy.h" #include "src/core/lib/load_balancing/lb_policy.h"

@ -16,6 +16,7 @@
#include <stddef.h> #include <stddef.h>
#include <algorithm>
#include <array> #include <array>
#include <map> #include <map>
#include <memory> #include <memory>
@ -35,7 +36,6 @@
#include "src/core/ext/xds/xds_health_status.h" #include "src/core/ext/xds/xds_health_status.h"
#include "src/core/lib/gprpp/orphanable.h" #include "src/core/lib/gprpp/orphanable.h"
#include "src/core/lib/gprpp/ref_counted_ptr.h" #include "src/core/lib/gprpp/ref_counted_ptr.h"
#include "src/core/lib/gprpp/unique_type_name.h"
#include "src/core/lib/json/json.h" #include "src/core/lib/json/json.h"
#include "src/core/lib/load_balancing/lb_policy.h" #include "src/core/lib/load_balancing/lb_policy.h"
#include "src/core/lib/resolver/server_address.h" #include "src/core/lib/resolver/server_address.h"
@ -112,6 +112,13 @@ class XdsOverrideHostTest : public LoadBalancingPolicyTest {
EXPECT_EQ(ApplyUpdate(update, policy_.get()), absl::OkStatus()); EXPECT_EQ(ApplyUpdate(update, policy_.get()), absl::OkStatus());
} }
CallAttributes MakeOverrideHostAttribute(absl::string_view host) {
CallAttributes override_host_attributes;
override_host_attributes.emplace_back(
std::make_unique<XdsOverrideHostAttribute>(host));
return override_host_attributes;
}
OrphanablePtr<LoadBalancingPolicy> policy_; OrphanablePtr<LoadBalancingPolicy> policy_;
}; };
@ -134,13 +141,18 @@ TEST_F(XdsOverrideHostTest, OverrideHost) {
auto picker = ExpectStartupWithRoundRobin(kAddresses); auto picker = ExpectStartupWithRoundRobin(kAddresses);
ASSERT_NE(picker, nullptr); ASSERT_NE(picker, nullptr);
// Check that the host is overridden // Check that the host is overridden
std::map<UniqueTypeName, absl::string_view> call_attributes{ EXPECT_EQ(ExpectPickComplete(picker.get(),
{XdsOverrideHostTypeName(), kAddresses[1]}}; MakeOverrideHostAttribute(kAddresses[1])),
EXPECT_EQ(ExpectPickComplete(picker.get(), call_attributes), kAddresses[1]); kAddresses[1]);
EXPECT_EQ(ExpectPickComplete(picker.get(), call_attributes), kAddresses[1]); EXPECT_EQ(ExpectPickComplete(picker.get(),
call_attributes[XdsOverrideHostTypeName()] = kAddresses[0]; MakeOverrideHostAttribute(kAddresses[1])),
EXPECT_EQ(ExpectPickComplete(picker.get(), call_attributes), kAddresses[0]); kAddresses[1]);
EXPECT_EQ(ExpectPickComplete(picker.get(), call_attributes), kAddresses[0]); EXPECT_EQ(ExpectPickComplete(picker.get(),
MakeOverrideHostAttribute(kAddresses[0])),
kAddresses[0]);
EXPECT_EQ(ExpectPickComplete(picker.get(),
MakeOverrideHostAttribute(kAddresses[0])),
kAddresses[0]);
} }
TEST_F(XdsOverrideHostTest, SubchannelNotFound) { TEST_F(XdsOverrideHostTest, SubchannelNotFound) {
@ -149,9 +161,8 @@ TEST_F(XdsOverrideHostTest, SubchannelNotFound) {
"ipv4:127.0.0.1:441", "ipv4:127.0.0.1:442", "ipv4:127.0.0.1:443"}; "ipv4:127.0.0.1:441", "ipv4:127.0.0.1:442", "ipv4:127.0.0.1:443"};
auto picker = ExpectStartupWithRoundRobin(kAddresses); auto picker = ExpectStartupWithRoundRobin(kAddresses);
ASSERT_NE(picker, nullptr); ASSERT_NE(picker, nullptr);
std::map<UniqueTypeName, absl::string_view> call_attributes{ ExpectRoundRobinPicks(picker.get(), kAddresses,
{XdsOverrideHostTypeName(), "no such host"}}; MakeOverrideHostAttribute("no such host"));
ExpectRoundRobinPicks(picker.get(), kAddresses, call_attributes);
} }
TEST_F(XdsOverrideHostTest, SubchannelsComeAndGo) { TEST_F(XdsOverrideHostTest, SubchannelsComeAndGo) {
@ -160,9 +171,8 @@ TEST_F(XdsOverrideHostTest, SubchannelsComeAndGo) {
auto picker = ExpectStartupWithRoundRobin(kAddresses); auto picker = ExpectStartupWithRoundRobin(kAddresses);
ASSERT_NE(picker, nullptr); ASSERT_NE(picker, nullptr);
// Check that the host is overridden // Check that the host is overridden
std::map<UniqueTypeName, absl::string_view> call_attributes{ ExpectRoundRobinPicks(picker.get(), {kAddresses[1]},
{XdsOverrideHostTypeName(), kAddresses[1]}}; MakeOverrideHostAttribute(kAddresses[1]));
ExpectRoundRobinPicks(picker.get(), {kAddresses[1]}, call_attributes);
// Some other address is gone // Some other address is gone
EXPECT_EQ(ApplyUpdate(BuildUpdate({kAddresses[0], kAddresses[1]}, EXPECT_EQ(ApplyUpdate(BuildUpdate({kAddresses[0], kAddresses[1]},
MakeXdsOverrideHostConfig()), MakeXdsOverrideHostConfig()),
@ -175,7 +185,8 @@ TEST_F(XdsOverrideHostTest, SubchannelsComeAndGo) {
picker = picker =
WaitForRoundRobinListChange(kAddresses, {kAddresses[0], kAddresses[1]}); WaitForRoundRobinListChange(kAddresses, {kAddresses[0], kAddresses[1]});
// Make sure host override still works. // Make sure host override still works.
ExpectRoundRobinPicks(picker.get(), {kAddresses[1]}, call_attributes); ExpectRoundRobinPicks(picker.get(), {kAddresses[1]},
MakeOverrideHostAttribute(kAddresses[1]));
// "Our" address is gone so others get returned in round-robin order // "Our" address is gone so others get returned in round-robin order
EXPECT_EQ(ApplyUpdate(BuildUpdate({kAddresses[0], kAddresses[2]}, EXPECT_EQ(ApplyUpdate(BuildUpdate({kAddresses[0], kAddresses[2]},
MakeXdsOverrideHostConfig()), MakeXdsOverrideHostConfig()),
@ -186,7 +197,8 @@ TEST_F(XdsOverrideHostTest, SubchannelsComeAndGo) {
// checking again afterward, because the host override won't actually // checking again afterward, because the host override won't actually
// be used. // be used.
WaitForRoundRobinListChange({kAddresses[0], kAddresses[1]}, WaitForRoundRobinListChange({kAddresses[0], kAddresses[1]},
{kAddresses[0], kAddresses[2]}, call_attributes); {kAddresses[0], kAddresses[2]},
MakeOverrideHostAttribute(kAddresses[1]));
// And now it is back // And now it is back
EXPECT_EQ(ApplyUpdate(BuildUpdate({kAddresses[1], kAddresses[2]}, EXPECT_EQ(ApplyUpdate(BuildUpdate({kAddresses[1], kAddresses[2]},
MakeXdsOverrideHostConfig()), MakeXdsOverrideHostConfig()),
@ -196,7 +208,8 @@ TEST_F(XdsOverrideHostTest, SubchannelsComeAndGo) {
picker = WaitForRoundRobinListChange({kAddresses[0], kAddresses[2]}, picker = WaitForRoundRobinListChange({kAddresses[0], kAddresses[2]},
{kAddresses[1], kAddresses[2]}); {kAddresses[1], kAddresses[2]});
// Make sure host override works. // Make sure host override works.
ExpectRoundRobinPicks(picker.get(), {kAddresses[1]}, call_attributes); ExpectRoundRobinPicks(picker.get(), {kAddresses[1]},
MakeOverrideHostAttribute(kAddresses[1]));
} }
TEST_F(XdsOverrideHostTest, FailedSubchannelIsNotPicked) { TEST_F(XdsOverrideHostTest, FailedSubchannelIsNotPicked) {
@ -206,9 +219,9 @@ TEST_F(XdsOverrideHostTest, FailedSubchannelIsNotPicked) {
auto picker = ExpectStartupWithRoundRobin(kAddresses); auto picker = ExpectStartupWithRoundRobin(kAddresses);
ASSERT_NE(picker, nullptr); ASSERT_NE(picker, nullptr);
// Check that the host is overridden // Check that the host is overridden
std::map<UniqueTypeName, absl::string_view> pick_arg{ EXPECT_EQ(ExpectPickComplete(picker.get(),
{XdsOverrideHostTypeName(), kAddresses[1]}}; MakeOverrideHostAttribute(kAddresses[1])),
EXPECT_EQ(ExpectPickComplete(picker.get(), pick_arg), kAddresses[1]); kAddresses[1]);
auto subchannel = FindSubchannel(kAddresses[1]); auto subchannel = FindSubchannel(kAddresses[1]);
ASSERT_NE(subchannel, nullptr); ASSERT_NE(subchannel, nullptr);
subchannel->SetConnectivityState(GRPC_CHANNEL_IDLE); subchannel->SetConnectivityState(GRPC_CHANNEL_IDLE);
@ -222,7 +235,8 @@ TEST_F(XdsOverrideHostTest, FailedSubchannelIsNotPicked) {
absl::ResourceExhaustedError("Hmmmm")); absl::ResourceExhaustedError("Hmmmm"));
ExpectReresolutionRequest(); ExpectReresolutionRequest();
picker = ExpectState(GRPC_CHANNEL_READY); picker = ExpectState(GRPC_CHANNEL_READY);
ExpectRoundRobinPicks(picker.get(), {kAddresses[0], kAddresses[2]}, pick_arg); ExpectRoundRobinPicks(picker.get(), {kAddresses[0], kAddresses[2]},
MakeOverrideHostAttribute(kAddresses[1]));
} }
TEST_F(XdsOverrideHostTest, ConnectingSubchannelIsQueued) { TEST_F(XdsOverrideHostTest, ConnectingSubchannelIsQueued) {
@ -232,19 +246,25 @@ TEST_F(XdsOverrideHostTest, ConnectingSubchannelIsQueued) {
auto picker = ExpectStartupWithRoundRobin(kAddresses); auto picker = ExpectStartupWithRoundRobin(kAddresses);
ASSERT_NE(picker, nullptr); ASSERT_NE(picker, nullptr);
// Check that the host is overridden // Check that the host is overridden
std::map<UniqueTypeName, absl::string_view> pick_arg{ EXPECT_EQ(ExpectPickComplete(picker.get(),
{XdsOverrideHostTypeName(), kAddresses[1]}}; {
EXPECT_EQ(ExpectPickComplete(picker.get(), pick_arg), kAddresses[1]); MakeOverrideHostAttribute(kAddresses[1]),
}),
kAddresses[1]);
auto subchannel = FindSubchannel(kAddresses[1]); auto subchannel = FindSubchannel(kAddresses[1]);
ASSERT_NE(subchannel, nullptr); ASSERT_NE(subchannel, nullptr);
subchannel->SetConnectivityState(GRPC_CHANNEL_IDLE); subchannel->SetConnectivityState(GRPC_CHANNEL_IDLE);
ExpectReresolutionRequest(); ExpectReresolutionRequest();
EXPECT_TRUE(subchannel->ConnectionRequested()); EXPECT_TRUE(subchannel->ConnectionRequested());
picker = ExpectState(GRPC_CHANNEL_READY); picker = ExpectState(GRPC_CHANNEL_READY);
ExpectPickQueued(picker.get(), pick_arg); ExpectPickQueued(picker.get(), {
MakeOverrideHostAttribute(kAddresses[1]),
});
subchannel->SetConnectivityState(GRPC_CHANNEL_CONNECTING); subchannel->SetConnectivityState(GRPC_CHANNEL_CONNECTING);
picker = ExpectState(GRPC_CHANNEL_READY); picker = ExpectState(GRPC_CHANNEL_READY);
ExpectPickQueued(picker.get(), pick_arg); ExpectPickQueued(picker.get(), {
MakeOverrideHostAttribute(kAddresses[1]),
});
} }
TEST_F(XdsOverrideHostTest, DrainingState) { TEST_F(XdsOverrideHostTest, DrainingState) {
@ -262,16 +282,17 @@ TEST_F(XdsOverrideHostTest, DrainingState) {
ExpectRoundRobinPicks(picker.get(), {kAddresses[0], kAddresses[2]}); ExpectRoundRobinPicks(picker.get(), {kAddresses[0], kAddresses[2]});
ExpectQueueEmpty(); ExpectQueueEmpty();
// Draining subchannel is returned // Draining subchannel is returned
std::map<UniqueTypeName, absl::string_view> pick_arg{ EXPECT_EQ(ExpectPickComplete(picker.get(),
{XdsOverrideHostTypeName(), kAddresses[1]}}; MakeOverrideHostAttribute(kAddresses[1])),
EXPECT_EQ(ExpectPickComplete(picker.get(), pick_arg), kAddresses[1]); kAddresses[1]);
ApplyUpdateWithHealthStatuses( ApplyUpdateWithHealthStatuses(
{{kAddresses[0], XdsHealthStatus::HealthStatus::kUnknown}, {{kAddresses[0], XdsHealthStatus::HealthStatus::kUnknown},
{kAddresses[2], XdsHealthStatus::HealthStatus::kHealthy}}); {kAddresses[2], XdsHealthStatus::HealthStatus::kHealthy}});
picker = ExpectState(GRPC_CHANNEL_READY); picker = ExpectState(GRPC_CHANNEL_READY);
ASSERT_NE(picker, nullptr); ASSERT_NE(picker, nullptr);
// Gone! // Gone!
ExpectRoundRobinPicks(picker.get(), {kAddresses[0], kAddresses[2]}, pick_arg); ExpectRoundRobinPicks(picker.get(), {kAddresses[0], kAddresses[2]},
MakeOverrideHostAttribute(kAddresses[1]));
} }
TEST_F(XdsOverrideHostTest, DrainingSubchannelIsConnecting) { TEST_F(XdsOverrideHostTest, DrainingSubchannelIsConnecting) {
@ -281,9 +302,9 @@ TEST_F(XdsOverrideHostTest, DrainingSubchannelIsConnecting) {
auto picker = ExpectStartupWithRoundRobin(kAddresses); auto picker = ExpectStartupWithRoundRobin(kAddresses);
ASSERT_NE(picker, nullptr); ASSERT_NE(picker, nullptr);
// Check that the host is overridden // Check that the host is overridden
std::map<UniqueTypeName, absl::string_view> pick_arg{ EXPECT_EQ(ExpectPickComplete(picker.get(),
{XdsOverrideHostTypeName(), kAddresses[1]}}; MakeOverrideHostAttribute(kAddresses[1])),
EXPECT_EQ(ExpectPickComplete(picker.get(), pick_arg), kAddresses[1]); kAddresses[1]);
ApplyUpdateWithHealthStatuses( ApplyUpdateWithHealthStatuses(
{{kAddresses[0], XdsHealthStatus::HealthStatus::kUnknown}, {{kAddresses[0], XdsHealthStatus::HealthStatus::kUnknown},
{kAddresses[1], XdsHealthStatus::HealthStatus::kDraining}, {kAddresses[1], XdsHealthStatus::HealthStatus::kDraining},
@ -294,21 +315,25 @@ TEST_F(XdsOverrideHostTest, DrainingSubchannelIsConnecting) {
// There are two notifications - one from child policy and one from the parent // There are two notifications - one from child policy and one from the parent
// policy due to draining channel update // policy due to draining channel update
picker = ExpectState(GRPC_CHANNEL_READY); picker = ExpectState(GRPC_CHANNEL_READY);
EXPECT_EQ(ExpectPickComplete(picker.get(), pick_arg), kAddresses[1]); EXPECT_EQ(ExpectPickComplete(picker.get(),
MakeOverrideHostAttribute(kAddresses[1])),
kAddresses[1]);
ExpectRoundRobinPicks(picker.get(), {kAddresses[0], kAddresses[2]}); ExpectRoundRobinPicks(picker.get(), {kAddresses[0], kAddresses[2]});
subchannel->SetConnectivityState(GRPC_CHANNEL_IDLE); subchannel->SetConnectivityState(GRPC_CHANNEL_IDLE);
picker = ExpectState(GRPC_CHANNEL_READY); picker = ExpectState(GRPC_CHANNEL_READY);
ExpectPickQueued(picker.get(), pick_arg); ExpectPickQueued(picker.get(), MakeOverrideHostAttribute(kAddresses[1]));
ExpectRoundRobinPicks(picker.get(), {kAddresses[0], kAddresses[2]}); ExpectRoundRobinPicks(picker.get(), {kAddresses[0], kAddresses[2]});
EXPECT_TRUE(subchannel->ConnectionRequested()); EXPECT_TRUE(subchannel->ConnectionRequested());
ExpectQueueEmpty(); ExpectQueueEmpty();
subchannel->SetConnectivityState(GRPC_CHANNEL_CONNECTING); subchannel->SetConnectivityState(GRPC_CHANNEL_CONNECTING);
picker = ExpectState(GRPC_CHANNEL_READY); picker = ExpectState(GRPC_CHANNEL_READY);
ExpectPickQueued(picker.get(), pick_arg); ExpectPickQueued(picker.get(), MakeOverrideHostAttribute(kAddresses[1]));
ExpectRoundRobinPicks(picker.get(), {kAddresses[0], kAddresses[2]}); ExpectRoundRobinPicks(picker.get(), {kAddresses[0], kAddresses[2]});
subchannel->SetConnectivityState(GRPC_CHANNEL_READY); subchannel->SetConnectivityState(GRPC_CHANNEL_READY);
picker = ExpectState(GRPC_CHANNEL_READY); picker = ExpectState(GRPC_CHANNEL_READY);
EXPECT_EQ(ExpectPickComplete(picker.get(), pick_arg), kAddresses[1]); EXPECT_EQ(ExpectPickComplete(picker.get(),
MakeOverrideHostAttribute(kAddresses[1])),
kAddresses[1]);
ExpectRoundRobinPicks(picker.get(), {kAddresses[0], kAddresses[2]}); ExpectRoundRobinPicks(picker.get(), {kAddresses[0], kAddresses[2]});
} }
@ -326,9 +351,9 @@ TEST_F(XdsOverrideHostTest, DrainingToHealthy) {
ASSERT_NE(picker, nullptr); ASSERT_NE(picker, nullptr);
ExpectRoundRobinPicks(picker.get(), {kAddresses[0], kAddresses[2]}); ExpectRoundRobinPicks(picker.get(), {kAddresses[0], kAddresses[2]});
ExpectQueueEmpty(); ExpectQueueEmpty();
std::map<UniqueTypeName, absl::string_view> pick_arg{ EXPECT_EQ(ExpectPickComplete(picker.get(),
{XdsOverrideHostTypeName(), kAddresses[1]}}; MakeOverrideHostAttribute(kAddresses[1])),
EXPECT_EQ(ExpectPickComplete(picker.get(), pick_arg), kAddresses[1]); kAddresses[1]);
ApplyUpdateWithHealthStatuses( ApplyUpdateWithHealthStatuses(
{{kAddresses[0], XdsHealthStatus::HealthStatus::kHealthy}, {{kAddresses[0], XdsHealthStatus::HealthStatus::kHealthy},
{kAddresses[1], XdsHealthStatus::HealthStatus::kHealthy}, {kAddresses[1], XdsHealthStatus::HealthStatus::kHealthy},
@ -336,8 +361,12 @@ TEST_F(XdsOverrideHostTest, DrainingToHealthy) {
{"UNKNOWN", "HEALTHY", "DRAINING"}); {"UNKNOWN", "HEALTHY", "DRAINING"});
picker = ExpectState(GRPC_CHANNEL_READY); picker = ExpectState(GRPC_CHANNEL_READY);
ASSERT_NE(picker, nullptr); ASSERT_NE(picker, nullptr);
EXPECT_EQ(ExpectPickComplete(picker.get(), pick_arg), kAddresses[1]); EXPECT_EQ(ExpectPickComplete(picker.get(),
EXPECT_EQ(ExpectPickComplete(picker.get(), pick_arg), kAddresses[1]); MakeOverrideHostAttribute(kAddresses[1])),
kAddresses[1]);
EXPECT_EQ(ExpectPickComplete(picker.get(),
MakeOverrideHostAttribute(kAddresses[1])),
kAddresses[1]);
} }
TEST_F(XdsOverrideHostTest, OverrideHostStatus) { TEST_F(XdsOverrideHostTest, OverrideHostStatus) {
@ -353,13 +382,13 @@ TEST_F(XdsOverrideHostTest, OverrideHostStatus) {
ASSERT_NE(picker, nullptr); ASSERT_NE(picker, nullptr);
ExpectRoundRobinPicks(picker.get(), {kAddresses[0], kAddresses[1]}); ExpectRoundRobinPicks(picker.get(), {kAddresses[0], kAddresses[1]});
EXPECT_EQ(ExpectPickComplete(picker.get(), EXPECT_EQ(ExpectPickComplete(picker.get(),
{{XdsOverrideHostTypeName(), kAddresses[0]}}), MakeOverrideHostAttribute(kAddresses[0])),
kAddresses[0]); kAddresses[0]);
EXPECT_EQ(ExpectPickComplete(picker.get(), EXPECT_EQ(ExpectPickComplete(picker.get(),
{{XdsOverrideHostTypeName(), kAddresses[1]}}), MakeOverrideHostAttribute(kAddresses[1])),
kAddresses[1]); kAddresses[1]);
EXPECT_EQ(ExpectPickComplete(picker.get(), EXPECT_EQ(ExpectPickComplete(picker.get(),
{{XdsOverrideHostTypeName(), kAddresses[2]}}), MakeOverrideHostAttribute(kAddresses[2])),
kAddresses[2]); kAddresses[2]);
// UNKNOWN excluded - first chanel does not get overridden // UNKNOWN excluded - first chanel does not get overridden
ApplyUpdateWithHealthStatuses( ApplyUpdateWithHealthStatuses(
@ -371,12 +400,12 @@ TEST_F(XdsOverrideHostTest, OverrideHostStatus) {
ASSERT_NE(picker, nullptr); ASSERT_NE(picker, nullptr);
ExpectRoundRobinPicks(picker.get(), {kAddresses[0], kAddresses[1]}); ExpectRoundRobinPicks(picker.get(), {kAddresses[0], kAddresses[1]});
ExpectRoundRobinPicks(picker.get(), {kAddresses[0], kAddresses[1]}, ExpectRoundRobinPicks(picker.get(), {kAddresses[0], kAddresses[1]},
{{XdsOverrideHostTypeName(), kAddresses[0]}}); MakeOverrideHostAttribute(kAddresses[0]));
EXPECT_EQ(ExpectPickComplete(picker.get(), EXPECT_EQ(ExpectPickComplete(picker.get(),
{{XdsOverrideHostTypeName(), kAddresses[1]}}), MakeOverrideHostAttribute(kAddresses[1])),
kAddresses[1]); kAddresses[1]);
EXPECT_EQ(ExpectPickComplete(picker.get(), EXPECT_EQ(ExpectPickComplete(picker.get(),
{{XdsOverrideHostTypeName(), kAddresses[2]}}), MakeOverrideHostAttribute(kAddresses[2])),
kAddresses[2]); kAddresses[2]);
// HEALTHY excluded - second chanel does not get overridden // HEALTHY excluded - second chanel does not get overridden
ApplyUpdateWithHealthStatuses( ApplyUpdateWithHealthStatuses(
@ -388,13 +417,13 @@ TEST_F(XdsOverrideHostTest, OverrideHostStatus) {
ASSERT_NE(picker, nullptr); ASSERT_NE(picker, nullptr);
ExpectRoundRobinPicks(picker.get(), {kAddresses[0], kAddresses[1]}); ExpectRoundRobinPicks(picker.get(), {kAddresses[0], kAddresses[1]});
EXPECT_EQ(ExpectPickComplete(picker.get(), EXPECT_EQ(ExpectPickComplete(picker.get(),
{{XdsOverrideHostTypeName(), kAddresses[0]}}), MakeOverrideHostAttribute(kAddresses[0])),
kAddresses[0]); kAddresses[0]);
EXPECT_EQ(ExpectPickComplete(picker.get(), EXPECT_EQ(ExpectPickComplete(picker.get(),
{{XdsOverrideHostTypeName(), kAddresses[1]}}), MakeOverrideHostAttribute(kAddresses[1])),
kAddresses[1]); kAddresses[1]);
ExpectRoundRobinPicks(picker.get(), {kAddresses[0], kAddresses[1]}, ExpectRoundRobinPicks(picker.get(), {kAddresses[0], kAddresses[1]},
{{XdsOverrideHostTypeName(), kAddresses[2]}}); MakeOverrideHostAttribute(kAddresses[2]));
// DRAINING excluded - third chanel does not get overridden // DRAINING excluded - third chanel does not get overridden
ApplyUpdateWithHealthStatuses( ApplyUpdateWithHealthStatuses(
{{kAddresses[0], XdsHealthStatus::HealthStatus::kUnknown}, {{kAddresses[0], XdsHealthStatus::HealthStatus::kUnknown},
@ -405,13 +434,13 @@ TEST_F(XdsOverrideHostTest, OverrideHostStatus) {
ASSERT_NE(picker, nullptr); ASSERT_NE(picker, nullptr);
ExpectRoundRobinPicks(picker.get(), {kAddresses[0], kAddresses[1]}); ExpectRoundRobinPicks(picker.get(), {kAddresses[0], kAddresses[1]});
EXPECT_EQ(ExpectPickComplete(picker.get(), EXPECT_EQ(ExpectPickComplete(picker.get(),
{{XdsOverrideHostTypeName(), kAddresses[0]}}), MakeOverrideHostAttribute(kAddresses[0])),
kAddresses[0]); kAddresses[0]);
EXPECT_EQ(ExpectPickComplete(picker.get(), EXPECT_EQ(ExpectPickComplete(picker.get(),
{{XdsOverrideHostTypeName(), kAddresses[1]}}), MakeOverrideHostAttribute(kAddresses[1])),
kAddresses[1]); kAddresses[1]);
ExpectRoundRobinPicks(picker.get(), {kAddresses[0], kAddresses[1]}, ExpectRoundRobinPicks(picker.get(), {kAddresses[0], kAddresses[1]},
{{XdsOverrideHostTypeName(), kAddresses[2]}}); MakeOverrideHostAttribute(kAddresses[2]));
} }
} // namespace } // namespace

Loading…
Cancel
Save