From 33907348cb9d1dbefaef5c817395cb6e1414149d Mon Sep 17 00:00:00 2001 From: Stephan Zuercher Date: Fri, 23 Aug 2019 11:54:43 -0700 Subject: [PATCH 1/8] subset lb: allow ring hash/maglev LB to work with subsets Skip initializing the thread aware LB for a cluster when the subset load balancer is enabled. Also adds some extra checks for LB policies that are incompatible with the subset load balancer. Risk Level: low Testing: test additional checks Docs Changes: updated docs w.r.t subset lb compatibility Release Notes: n/a Fixes: #7651 Signed-off-by: Stephan Zuercher --- .../upstream/load_balancing/subsets.rst | 20 +++++++++++----- .../common/upstream/cluster_manager_impl.cc | 22 ++++++++++-------- source/common/upstream/upstream_impl.cc | 12 ++++++++++ .../upstream/cluster_manager_impl_test.cc | 23 ++++++++++++++++++- 4 files changed, 61 insertions(+), 16 deletions(-) diff --git a/docs/root/intro/arch_overview/upstream/load_balancing/subsets.rst b/docs/root/intro/arch_overview/upstream/load_balancing/subsets.rst index 710a72f43b40..942903ceb691 100644 --- a/docs/root/intro/arch_overview/upstream/load_balancing/subsets.rst +++ b/docs/root/intro/arch_overview/upstream/load_balancing/subsets.rst @@ -35,12 +35,20 @@ therefore, contain a definition that has the same keys as a given route in order balancing to occur. This feature can only be enabled using the V2 configuration API. Furthermore, host metadata is only -supported when using the EDS discovery type for clusters. Host metadata for subset load balancing -must be placed under the filter name ``"envoy.lb"``. Similarly, route metadata match criteria use -the ``"envoy.lb"`` filter name. Host metadata may be hierarchical (e.g., the value for a top-level -key may be a structured value or list), but the subset load balancer only compares top-level keys -and values. Therefore when using structured values, a route's match criteria will only match if an -identical structured value appears in the host's metadata. +supported when hosts are defined using +:ref:`ClusterLoadAssignments `. ClusterLoadAssignments are +available via EDS or the Cluster :ref:`load_assignment ` +field. Host metadata for subset load balancing must be placed under the filter name ``"envoy.lb"``. +Similarly, route metadata match criteria use ``"envoy.lb"`` filter name. Host metadata may be +hierarchical (e.g., the value for a top-level key may be a structured value or list), but the +subset load balancer only compares top-level keys and values. Therefore when using structured +values, a route's match criteria will only match if an identical structured value appears in the +host's metadata. + +Finally, note that subset load balancing is not available for the +:ref:`ORIGINAL_DST_LB ` or +:ref:`CLUSTER_PROVIDED ` load balancer +policies. Examples ^^^^^^^^ diff --git a/source/common/upstream/cluster_manager_impl.cc b/source/common/upstream/cluster_manager_impl.cc index 1dd78f2dabfa..90baf33e22c3 100644 --- a/source/common/upstream/cluster_manager_impl.cc +++ b/source/common/upstream/cluster_manager_impl.cc @@ -640,17 +640,21 @@ void ClusterManagerImpl::loadCluster(const envoy::api::v2::Cluster& cluster, const auto cluster_entry_it = cluster_map.find(cluster_reference.info()->name()); // If an LB is thread aware, create it here. The LB is not initialized until cluster pre-init - // finishes. + // finishes. For RingHash/Maglev don't create the LB here if subset balancing is enabled. if (cluster_reference.info()->lbType() == LoadBalancerType::RingHash) { - cluster_entry_it->second->thread_aware_lb_ = std::make_unique( - cluster_reference.prioritySet(), cluster_reference.info()->stats(), - cluster_reference.info()->statsScope(), runtime_, random_, - cluster_reference.info()->lbRingHashConfig(), cluster_reference.info()->lbConfig()); + if (!cluster_reference.info()->lbSubsetInfo().isEnabled()) { + cluster_entry_it->second->thread_aware_lb_ = std::make_unique( + cluster_reference.prioritySet(), cluster_reference.info()->stats(), + cluster_reference.info()->statsScope(), runtime_, random_, + cluster_reference.info()->lbRingHashConfig(), cluster_reference.info()->lbConfig()); + } } else if (cluster_reference.info()->lbType() == LoadBalancerType::Maglev) { - cluster_entry_it->second->thread_aware_lb_ = std::make_unique( - cluster_reference.prioritySet(), cluster_reference.info()->stats(), - cluster_reference.info()->statsScope(), runtime_, random_, - cluster_reference.info()->lbConfig()); + if (!cluster_reference.info()->lbSubsetInfo().isEnabled()) { + cluster_entry_it->second->thread_aware_lb_ = std::make_unique( + cluster_reference.prioritySet(), cluster_reference.info()->stats(), + cluster_reference.info()->statsScope(), runtime_, random_, + cluster_reference.info()->lbConfig()); + } } else if (cluster_reference.info()->lbType() == LoadBalancerType::ClusterProvided) { cluster_entry_it->second->thread_aware_lb_ = std::move(new_cluster_pair.second); } diff --git a/source/common/upstream/upstream_impl.cc b/source/common/upstream/upstream_impl.cc index 8443868f68f0..93421a92a21c 100644 --- a/source/common/upstream/upstream_impl.cc +++ b/source/common/upstream/upstream_impl.cc @@ -643,12 +643,24 @@ ClusterInfoImpl::ClusterInfoImpl( envoy::api::v2::Cluster_LbPolicy_Name(config.lb_policy()), envoy::api::v2::Cluster_DiscoveryType_Name(config.type()))); } + if (config.has_lb_subset_config()) { + throw EnvoyException( + fmt::format("cluster: LB policy {} cannot be combined with lb_subset_config", + envoy::api::v2::Cluster_LbPolicy_Name(config.lb_policy()))); + } + lb_type_ = LoadBalancerType::ClusterProvided; break; case envoy::api::v2::Cluster::MAGLEV: lb_type_ = LoadBalancerType::Maglev; break; case envoy::api::v2::Cluster::CLUSTER_PROVIDED: + if (config.has_lb_subset_config()) { + throw EnvoyException( + fmt::format("cluster: LB policy {} cannot be combined with lb_subset_config", + envoy::api::v2::Cluster_LbPolicy_Name(config.lb_policy()))); + } + lb_type_ = LoadBalancerType::ClusterProvided; break; default: diff --git a/test/common/upstream/cluster_manager_impl_test.cc b/test/common/upstream/cluster_manager_impl_test.cc index d8a85b0e13a2..b29f7f2bf84d 100644 --- a/test/common/upstream/cluster_manager_impl_test.cc +++ b/test/common/upstream/cluster_manager_impl_test.cc @@ -610,7 +610,7 @@ TEST_F(ClusterManagerImplTest, SubsetLoadBalancerInitialization) { factory_.tls_.shutdownThread(); } -TEST_F(ClusterManagerImplTest, SubsetLoadBalancerRestriction) { +TEST_F(ClusterManagerImplTest, SubsetLoadBalancerOriginalDstRestriction) { const std::string yaml = R"EOF( static_resources: clusters: @@ -631,6 +631,27 @@ TEST_F(ClusterManagerImplTest, SubsetLoadBalancerRestriction) { "cluster: cluster type 'original_dst' may not be used with lb_subset_config"); } +TEST_F(ClusterManagerImplTest, SubsetLoadBalancerClusterProvidedLbRestriction) { + const std::string yaml = R"EOF( +static_resources: + clusters: + - name: cluster_1 + connect_timeout: 0.250s + type: static + lb_policy: cluster_provided + )EOF"; + + envoy::config::bootstrap::v2::Bootstrap bootstrap = parseBootstrapFromV2Yaml(yaml); + envoy::api::v2::Cluster::LbSubsetConfig* subset_config = + bootstrap.mutable_static_resources()->mutable_clusters(0)->mutable_lb_subset_config(); + subset_config->set_fallback_policy(envoy::api::v2::Cluster::LbSubsetConfig::ANY_ENDPOINT); + subset_config->add_subset_selectors()->add_keys("x"); + + EXPECT_THROW_WITH_MESSAGE( + create(bootstrap), EnvoyException, + "cluster: LB policy CLUSTER_PROVIDED cannot be combined with lb_subset_config"); +} + TEST_F(ClusterManagerImplTest, SubsetLoadBalancerLocalityAware) { const std::string yaml = R"EOF( static_resources: From 0908fbe0ea9daea5a474b2a7ff51a7043cb5bc8e Mon Sep 17 00:00:00 2001 From: Stephan Zuercher Date: Mon, 26 Aug 2019 12:52:18 -0700 Subject: [PATCH 2/8] Prove that subset load balancer is instantiated for all valid lb policies (and clean up some other tests). Signed-off-by: Stephan Zuercher --- test/common/upstream/BUILD | 5 + .../upstream/cluster_manager_impl_test.cc | 107 +++++++++++++----- 2 files changed, 81 insertions(+), 31 deletions(-) diff --git a/test/common/upstream/BUILD b/test/common/upstream/BUILD index 4e5d8b4679ba..fcfbf17a6e94 100644 --- a/test/common/upstream/BUILD +++ b/test/common/upstream/BUILD @@ -29,6 +29,9 @@ envoy_cc_test( envoy_cc_test( name = "cluster_manager_impl_test", srcs = ["cluster_manager_impl_test.cc"], + external_deps = [ + "abseil_optional", + ], deps = [ ":utility_lib", "//include/envoy/stats:stats_interface", @@ -43,6 +46,7 @@ envoy_cc_test( "//source/common/stats:stats_lib", "//source/common/upstream:cluster_factory_lib", "//source/common/upstream:cluster_manager_lib", + "//source/common/upstream:subset_lb_lib", "//source/extensions/transport_sockets/raw_buffer:config", "//source/extensions/transport_sockets/tls:context_lib", "//test/mocks/access_log:access_log_mocks", @@ -61,6 +65,7 @@ envoy_cc_test( "//test/test_common:simulated_time_system_lib", "//test/test_common:threadsafe_singleton_injector_lib", "//test/test_common:utility_lib", + "@envoy_api//envoy/api/v2:cds_cc", ], ) diff --git a/test/common/upstream/cluster_manager_impl_test.cc b/test/common/upstream/cluster_manager_impl_test.cc index b29f7f2bf84d..b34dab904d3f 100644 --- a/test/common/upstream/cluster_manager_impl_test.cc +++ b/test/common/upstream/cluster_manager_impl_test.cc @@ -2,6 +2,7 @@ #include #include "envoy/admin/v2alpha/config_dump.pb.h" +#include "envoy/api/v2/cds.pb.h" #include "envoy/api/v2/core/base.pb.h" #include "envoy/network/listen_socket.h" #include "envoy/upstream/upstream.h" @@ -17,6 +18,7 @@ #include "common/singleton/manager_impl.h" #include "common/upstream/cluster_factory_impl.h" #include "common/upstream/cluster_manager_impl.h" +#include "common/upstream/subset_lb.h" #include "extensions/transport_sockets/tls/context_manager_impl.h" @@ -38,6 +40,7 @@ #include "test/test_common/threadsafe_singleton_injector.h" #include "test/test_common/utility.h" +#include "absl/strings/str_replace.h" #include "gmock/gmock.h" #include "gtest/gtest.h" @@ -575,14 +578,45 @@ TEST_F(ClusterManagerImplTest, OriginalDstLbRestriction2) { "'ORIGINAL_DST_LB' is allowed only with cluster type 'ORIGINAL_DST'"); } -TEST_F(ClusterManagerImplTest, SubsetLoadBalancerInitialization) { - const std::string yaml = R"EOF( +class ClusterManagerSubsetInitializationTest + : public ClusterManagerImplTest, + public testing::WithParamInterface { +public: + ClusterManagerSubsetInitializationTest() {} + + static std::vector lbPolicies() { + int first = static_cast(envoy::api::v2::Cluster_LbPolicy_LbPolicy_MIN); + int last = static_cast(envoy::api::v2::Cluster_LbPolicy_LbPolicy_MAX); + ASSERT(first < last); + + std::vector policies; + for (int i = first; i <= last; i++) { + if (envoy::api::v2::Cluster_LbPolicy_IsValid(i)) { + auto policy = static_cast(i); + policies.push_back(policy); + } + } + return policies; + } + + static std::string paramName(const testing::TestParamInfo& info) { + std::string name = envoy::api::v2::Cluster_LbPolicy_Name(info.param); + return absl::StrReplaceAll(name, {{"_", ""}}); + } +}; + +TEST_P(ClusterManagerSubsetInitializationTest, SubsetLoadBalancerInitialization) { + const std::string yamlPattern = R"EOF( static_resources: clusters: - name: cluster_1 connect_timeout: 0.250s type: static - lb_policy: round_robin + lb_policy: "{}" + lb_subset_config: + fallback_policy: ANY_ENDPOINT + subset_selectors: + - keys: [ "x" ] load_assignment: endpoints: - lb_endpoints: @@ -598,18 +632,33 @@ TEST_F(ClusterManagerImplTest, SubsetLoadBalancerInitialization) { port_value: 8001 )EOF"; - envoy::config::bootstrap::v2::Bootstrap bootstrap = parseBootstrapFromV2Yaml(yaml); - envoy::api::v2::Cluster::LbSubsetConfig* subset_config = - bootstrap.mutable_static_resources()->mutable_clusters(0)->mutable_lb_subset_config(); - subset_config->set_fallback_policy(envoy::api::v2::Cluster::LbSubsetConfig::ANY_ENDPOINT); - subset_config->add_subset_selectors()->add_keys("x"); + const std::string policy_name = envoy::api::v2::Cluster_LbPolicy_Name(GetParam()); + const std::string yaml = fmt::format(yamlPattern, policy_name); - create(bootstrap); - checkStats(1 /*added*/, 0 /*modified*/, 0 /*removed*/, 1 /*active*/, 0 /*warming*/); + if (GetParam() == envoy::api::v2::Cluster_LbPolicy_ORIGINAL_DST_LB || + GetParam() == envoy::api::v2::Cluster_LbPolicy_CLUSTER_PROVIDED) { + EXPECT_THROW(create(parseBootstrapFromV2Yaml(yaml)), EnvoyException); + } else { + create(parseBootstrapFromV2Yaml(yaml)); + checkStats(1 /*added*/, 0 /*modified*/, 0 /*removed*/, 1 /*active*/, 0 /*warming*/); - factory_.tls_.shutdownThread(); + Upstream::ThreadLocalCluster* tlc = cluster_manager_->get("cluster_1"); + EXPECT_NE(nullptr, tlc); + + if (tlc) { + Upstream::LoadBalancer& lb = tlc->loadBalancer(); + EXPECT_NE(nullptr, dynamic_cast(&lb)); + } + + factory_.tls_.shutdownThread(); + } } +INSTANTIATE_TEST_SUITE_P(ClusterManagerSubsetInitializationTest, + ClusterManagerSubsetInitializationTest, + testing::ValuesIn(ClusterManagerSubsetInitializationTest::lbPolicies()), + ClusterManagerSubsetInitializationTest::paramName); + TEST_F(ClusterManagerImplTest, SubsetLoadBalancerOriginalDstRestriction) { const std::string yaml = R"EOF( static_resources: @@ -618,16 +667,14 @@ TEST_F(ClusterManagerImplTest, SubsetLoadBalancerOriginalDstRestriction) { connect_timeout: 0.250s type: original_dst lb_policy: original_dst_lb + lb_subset_config: + fallback_policy: ANY_ENDPOINT + subset_selectors: + - keys: [ "x" ] )EOF"; - envoy::config::bootstrap::v2::Bootstrap bootstrap = parseBootstrapFromV2Yaml(yaml); - envoy::api::v2::Cluster::LbSubsetConfig* subset_config = - bootstrap.mutable_static_resources()->mutable_clusters(0)->mutable_lb_subset_config(); - subset_config->set_fallback_policy(envoy::api::v2::Cluster::LbSubsetConfig::ANY_ENDPOINT); - subset_config->add_subset_selectors()->add_keys("x"); - EXPECT_THROW_WITH_MESSAGE( - create(bootstrap), EnvoyException, + create(parseBootstrapFromV2Yaml(yaml)), EnvoyException, "cluster: cluster type 'original_dst' may not be used with lb_subset_config"); } @@ -639,16 +686,14 @@ TEST_F(ClusterManagerImplTest, SubsetLoadBalancerClusterProvidedLbRestriction) { connect_timeout: 0.250s type: static lb_policy: cluster_provided + lb_subset_config: + fallback_policy: ANY_ENDPOINT + subset_selectors: + - keys: [ "x" ] )EOF"; - envoy::config::bootstrap::v2::Bootstrap bootstrap = parseBootstrapFromV2Yaml(yaml); - envoy::api::v2::Cluster::LbSubsetConfig* subset_config = - bootstrap.mutable_static_resources()->mutable_clusters(0)->mutable_lb_subset_config(); - subset_config->set_fallback_policy(envoy::api::v2::Cluster::LbSubsetConfig::ANY_ENDPOINT); - subset_config->add_subset_selectors()->add_keys("x"); - EXPECT_THROW_WITH_MESSAGE( - create(bootstrap), EnvoyException, + create(parseBootstrapFromV2Yaml(yaml)), EnvoyException, "cluster: LB policy CLUSTER_PROVIDED cannot be combined with lb_subset_config"); } @@ -660,6 +705,11 @@ TEST_F(ClusterManagerImplTest, SubsetLoadBalancerLocalityAware) { connect_timeout: 0.250s type: STATIC lb_policy: ROUND_ROBIN + lb_subset_config: + fallback_policy: ANY_ENDPOINT + subset_selectors: + - keys: [ "x" ] + locality_weight_aware: true load_assignment: endpoints: - lb_endpoints: @@ -675,12 +725,7 @@ TEST_F(ClusterManagerImplTest, SubsetLoadBalancerLocalityAware) { port_value: 8001 )EOF"; - envoy::config::bootstrap::v2::Bootstrap bootstrap = parseBootstrapFromV2Yaml(yaml); - envoy::api::v2::Cluster::LbSubsetConfig* subset_config = - bootstrap.mutable_static_resources()->mutable_clusters(0)->mutable_lb_subset_config(); - subset_config->set_locality_weight_aware(true); - - EXPECT_THROW_WITH_MESSAGE(create(bootstrap), EnvoyException, + EXPECT_THROW_WITH_MESSAGE(create(parseBootstrapFromV2Yaml(yaml)), EnvoyException, "Locality weight aware subset LB requires that a " "locality_weighted_lb_config be set in cluster_1"); } From e259f4751340f9960ac0c3d9b3079f77aa34488c Mon Sep 17 00:00:00 2001 From: Stephan Zuercher Date: Mon, 26 Aug 2019 19:51:21 -0700 Subject: [PATCH 3/8] fix tidy errors Signed-off-by: Stephan Zuercher --- test/common/upstream/cluster_manager_impl_test.cc | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/test/common/upstream/cluster_manager_impl_test.cc b/test/common/upstream/cluster_manager_impl_test.cc index 81f32c26e81b..df1b1cec3397 100644 --- a/test/common/upstream/cluster_manager_impl_test.cc +++ b/test/common/upstream/cluster_manager_impl_test.cc @@ -584,7 +584,7 @@ class ClusterManagerSubsetInitializationTest : public ClusterManagerImplTest, public testing::WithParamInterface { public: - ClusterManagerSubsetInitializationTest() {} + ClusterManagerSubsetInitializationTest() = default; static std::vector lbPolicies() { int first = static_cast(envoy::api::v2::Cluster_LbPolicy_LbPolicy_MIN); @@ -602,7 +602,7 @@ class ClusterManagerSubsetInitializationTest } static std::string paramName(const testing::TestParamInfo& info) { - std::string name = envoy::api::v2::Cluster_LbPolicy_Name(info.param); + const std::string& name = envoy::api::v2::Cluster_LbPolicy_Name(info.param); return absl::StrReplaceAll(name, {{"_", ""}}); } }; @@ -634,7 +634,7 @@ TEST_P(ClusterManagerSubsetInitializationTest, SubsetLoadBalancerInitialization) port_value: 8001 )EOF"; - const std::string policy_name = envoy::api::v2::Cluster_LbPolicy_Name(GetParam()); + const std::string& policy_name = envoy::api::v2::Cluster_LbPolicy_Name(GetParam()); const std::string yaml = fmt::format(yamlPattern, policy_name); if (GetParam() == envoy::api::v2::Cluster_LbPolicy_ORIGINAL_DST_LB || From cd2bb3a8dea5bb0107b9a8301aceab20170b6c19 Mon Sep 17 00:00:00 2001 From: Stephan Zuercher Date: Tue, 3 Sep 2019 14:26:42 -0700 Subject: [PATCH 4/8] additional comments Signed-off-by: Stephan Zuercher --- source/common/upstream/cluster_manager_impl.cc | 3 ++- test/common/upstream/cluster_manager_impl_test.cc | 1 + 2 files changed, 3 insertions(+), 1 deletion(-) diff --git a/source/common/upstream/cluster_manager_impl.cc b/source/common/upstream/cluster_manager_impl.cc index f98cd7ea1f9c..e5c06231c4ca 100644 --- a/source/common/upstream/cluster_manager_impl.cc +++ b/source/common/upstream/cluster_manager_impl.cc @@ -667,7 +667,8 @@ void ClusterManagerImpl::loadCluster(const envoy::api::v2::Cluster& cluster, const auto cluster_entry_it = cluster_map.find(cluster_reference.info()->name()); // If an LB is thread aware, create it here. The LB is not initialized until cluster pre-init - // finishes. For RingHash/Maglev don't create the LB here if subset balancing is enabled. + // finishes. For RingHash/Maglev don't create the LB here if subset balancing is enabled, + // because the thread_aware_lb_ field takes precedence over the subset lb). if (cluster_reference.info()->lbType() == LoadBalancerType::RingHash) { if (!cluster_reference.info()->lbSubsetInfo().isEnabled()) { cluster_entry_it->second->thread_aware_lb_ = std::make_unique( diff --git a/test/common/upstream/cluster_manager_impl_test.cc b/test/common/upstream/cluster_manager_impl_test.cc index 78948e78d767..ddba7e732f45 100644 --- a/test/common/upstream/cluster_manager_impl_test.cc +++ b/test/common/upstream/cluster_manager_impl_test.cc @@ -607,6 +607,7 @@ class ClusterManagerSubsetInitializationTest } }; +// Test initialization of subset load balancer with every possible load balancer policy. TEST_P(ClusterManagerSubsetInitializationTest, SubsetLoadBalancerInitialization) { const std::string yamlPattern = R"EOF( static_resources: From 9cfabc6a2d1136504e85eda7344d3eb1d46cc7ef Mon Sep 17 00:00:00 2001 From: Stephan Zuercher Date: Mon, 9 Sep 2019 11:20:38 -0700 Subject: [PATCH 5/8] remove duplicate validation of original_dst_lb+subset_lb; test cluster_provided+subset_lb Signed-off-by: Stephan Zuercher --- .../common/upstream/original_dst_cluster.cc | 4 ---- test/common/upstream/BUILD | 1 + .../upstream/cluster_manager_impl_test.cc | 22 +++++++++++++++---- 3 files changed, 19 insertions(+), 8 deletions(-) diff --git a/source/common/upstream/original_dst_cluster.cc b/source/common/upstream/original_dst_cluster.cc index ec24c36e0e47..04b91f58f236 100644 --- a/source/common/upstream/original_dst_cluster.cc +++ b/source/common/upstream/original_dst_cluster.cc @@ -184,10 +184,6 @@ OriginalDstClusterFactory::createClusterImpl( envoy::api::v2::Cluster_LbPolicy_Name(cluster.lb_policy()), envoy::api::v2::Cluster_DiscoveryType_Name(cluster.type()))); } - if (cluster.has_lb_subset_config() && cluster.lb_subset_config().subset_selectors_size() != 0) { - throw EnvoyException( - fmt::format("cluster: cluster type 'original_dst' may not be used with lb_subset_config")); - } // TODO(mattklein123): The original DST load balancer type should be deprecated and instead // the cluster should directly supply the load balancer. This will remove diff --git a/test/common/upstream/BUILD b/test/common/upstream/BUILD index fcfbf17a6e94..1ac3067c7ef2 100644 --- a/test/common/upstream/BUILD +++ b/test/common/upstream/BUILD @@ -49,6 +49,7 @@ envoy_cc_test( "//source/common/upstream:subset_lb_lib", "//source/extensions/transport_sockets/raw_buffer:config", "//source/extensions/transport_sockets/tls:context_lib", + "//test/integration/clusters:custom_static_cluster", "//test/mocks/access_log:access_log_mocks", "//test/mocks/api:api_mocks", "//test/mocks/http:http_mocks", diff --git a/test/common/upstream/cluster_manager_impl_test.cc b/test/common/upstream/cluster_manager_impl_test.cc index ddba7e732f45..e546295ff5b7 100644 --- a/test/common/upstream/cluster_manager_impl_test.cc +++ b/test/common/upstream/cluster_manager_impl_test.cc @@ -23,6 +23,7 @@ #include "extensions/transport_sockets/tls/context_manager_impl.h" #include "test/common/upstream/utility.h" +#include "test/integration/clusters/custom_static_cluster.h" #include "test/mocks/access_log/mocks.h" #include "test/mocks/api/mocks.h" #include "test/mocks/http/mocks.h" @@ -614,7 +615,7 @@ TEST_P(ClusterManagerSubsetInitializationTest, SubsetLoadBalancerInitialization) clusters: - name: cluster_1 connect_timeout: 0.250s - type: static + {} lb_policy: "{}" lb_subset_config: fallback_policy: ANY_ENDPOINT @@ -636,11 +637,24 @@ TEST_P(ClusterManagerSubsetInitializationTest, SubsetLoadBalancerInitialization) )EOF"; const std::string& policy_name = envoy::api::v2::Cluster_LbPolicy_Name(GetParam()); - const std::string yaml = fmt::format(yamlPattern, policy_name); + + std::string cluster_type = "type: STATIC"; + if (GetParam() == envoy::api::v2::Cluster_LbPolicy_ORIGINAL_DST_LB) { + cluster_type = "type: ORIGINAL_DST"; + } else if (GetParam() == envoy::api::v2::Cluster_LbPolicy_CLUSTER_PROVIDED) { + // This custom cluster type is registered by linking test/integration/custom/static_cluster.cc. + cluster_type = "cluster_type: { name: envoy.clusters.custom_static_with_lb }"; + } + + const std::string yaml = fmt::format(yamlPattern, cluster_type, policy_name); if (GetParam() == envoy::api::v2::Cluster_LbPolicy_ORIGINAL_DST_LB || GetParam() == envoy::api::v2::Cluster_LbPolicy_CLUSTER_PROVIDED) { - EXPECT_THROW(create(parseBootstrapFromV2Yaml(yaml)), EnvoyException); + EXPECT_THROW_WITH_MESSAGE( + create(parseBootstrapFromV2Yaml(yaml)), EnvoyException, + fmt::format("cluster: LB policy {} cannot be combined with lb_subset_config", + envoy::api::v2::Cluster_LbPolicy_Name(GetParam()))); + } else { create(parseBootstrapFromV2Yaml(yaml)); checkStats(1 /*added*/, 0 /*modified*/, 0 /*removed*/, 1 /*active*/, 0 /*warming*/); @@ -678,7 +692,7 @@ TEST_F(ClusterManagerImplTest, SubsetLoadBalancerOriginalDstRestriction) { EXPECT_THROW_WITH_MESSAGE( create(parseBootstrapFromV2Yaml(yaml)), EnvoyException, - "cluster: cluster type 'original_dst' may not be used with lb_subset_config"); + "cluster: LB policy ORIGINAL_DST_LB cannot be combined with lb_subset_config"); } TEST_F(ClusterManagerImplTest, SubsetLoadBalancerClusterProvidedLbRestriction) { From d77bfc2616183e35da2a89366114627f59d3ad8f Mon Sep 17 00:00:00 2001 From: Stephan Zuercher Date: Mon, 9 Sep 2019 16:10:36 -0700 Subject: [PATCH 6/8] integration test for subset lb vs. compatible policies Signed-off-by: Stephan Zuercher --- test/integration/BUILD | 11 + .../http_load_balancer_integration_test.cc | 233 ++++++++++++++++++ 2 files changed, 244 insertions(+) create mode 100644 test/integration/http_load_balancer_integration_test.cc diff --git a/test/integration/BUILD b/test/integration/BUILD index 70082725f8d9..a10b678bc838 100644 --- a/test/integration/BUILD +++ b/test/integration/BUILD @@ -252,6 +252,17 @@ envoy_cc_test( ], ) +envoy_cc_test( + name = "http_load_balancer_integration_test", + srcs = [ + "http_load_balancer_integration_test.cc", + ], + deps = [ + ":http_integration_lib", + "//test/common/upstream:utility_lib", + ], +) + envoy_cc_test( name = "http_timeout_integration_test", srcs = [ diff --git a/test/integration/http_load_balancer_integration_test.cc b/test/integration/http_load_balancer_integration_test.cc new file mode 100644 index 000000000000..5214dc0ea949 --- /dev/null +++ b/test/integration/http_load_balancer_integration_test.cc @@ -0,0 +1,233 @@ +#include "test/integration/http_integration.h" + +#include "absl/strings/str_replace.h" +#include "gtest/gtest.h" + +namespace Envoy { +namespace { + +const std::string SUBSET_CONFIG = R"EOF( +admin: + access_log_path: /dev/null + address: + socket_address: + address: 127.0.0.1 + port_value: 0 +static_resources: + clusters: + name: cluster_0 + lb_policy: round_robin + lb_subset_config: + subset_selectors: + - keys: [ "type" ] + load_assignment: + cluster_name: cluster_0 + endpoints: + - lb_endpoints: + - endpoint: + address: + socket_address: + address: 127.0.0.1 + port_value: 0 + metadata: + filter_metadata: + "envoy.lb": { "type": "a" } + - endpoint: + address: + socket_address: + address: 127.0.0.1 + port_value: 0 + metadata: + filter_metadata: + "envoy.lb": { "type": "a" } + - endpoint: + address: + socket_address: + address: 127.0.0.1 + port_value: 0 + metadata: + filter_metadata: + "envoy.lb": { "type": "b" } + - endpoint: + address: + socket_address: + address: 127.0.0.1 + port_value: 0 + metadata: + filter_metadata: + "envoy.lb": { "type": "b" } + listeners: + name: listener_0 + address: + socket_address: + address: 127.0.0.1 + port_value: 0 + filter_chains: + filters: + name: envoy.http_connection_manager + config: + stat_prefix: config_test + http_filters: + name: envoy.router + codec_type: HTTP1 + access_log: + name: envoy.file_access_log + filter: + not_health_check_filter: {} + config: + path: /dev/null + route_config: + name: route_config_0 + virtual_hosts: + name: integration + domains: "*" + routes: + - match: + prefix: "/" + headers: + - name: "x-type" + exact_match: "a" + route: + cluster: cluster_0 + metadata_match: + filter_metadata: + "envoy.lb": { "type": "a" } + hash_policy: + - header: + header_name: "x-hash" + - match: + prefix: "/" + headers: + - name: "x-type" + exact_match: "b" + route: + cluster: cluster_0 + metadata_match: + filter_metadata: + "envoy.lb": { "type": "b" } + hash_policy: + - header: + header_name: "x-hash" + response_headers_to_add: + - header: + key: "x-host-type" + value: '%UPSTREAM_METADATA(["envoy.lb", "type"])%' + - header: + key: "x-host" + value: '%UPSTREAM_REMOTE_ADDRESS%' +)EOF"; + +} // namespace +class HttpSubsetLbIntegrationTest : public testing::TestWithParam, + public HttpIntegrationTest { +public: + // Returns all load balancer types except ORIGINAL_DST_LB and CLUSTER_PROVIDED. + static std::vector getSubsetLbTestParams() { + int first = static_cast(envoy::api::v2::Cluster_LbPolicy_LbPolicy_MIN); + int last = static_cast(envoy::api::v2::Cluster_LbPolicy_LbPolicy_MAX); + ASSERT(first < last); + + std::vector ret; + for (int i = first; i <= last; i++) { + if (!envoy::api::v2::Cluster_LbPolicy_IsValid(i)) { + continue; + } + + auto policy = static_cast(i); + + if (policy == envoy::api::v2::Cluster_LbPolicy_ORIGINAL_DST_LB || + policy == envoy::api::v2::Cluster_LbPolicy_CLUSTER_PROVIDED) { + continue; + } + + ret.push_back(policy); + } + + return ret; + } + + // Converts an LbPolicy to strings suitable for test names. + static std::string + subsetLbTestParamsToString(const testing::TestParamInfo& p) { + const std::string& policy_name = envoy::api::v2::Cluster_LbPolicy_Name(p.param); + return absl::StrReplaceAll(policy_name, {{"_", ""}}); + } + + HttpSubsetLbIntegrationTest() + : HttpIntegrationTest(Http::CodecClient::Type::HTTP1, Network::Address::IpVersion::v4, + SUBSET_CONFIG) { + autonomous_upstream_ = true; + setUpstreamCount(4); + config_helper_.addConfigModifier([](envoy::config::bootstrap::v2::Bootstrap& bootstrap) { + auto* static_resources = bootstrap.mutable_static_resources(); + for (int i = 0; i < static_resources->clusters_size(); ++i) { + auto* cluster = static_resources->mutable_clusters(i); + cluster->set_lb_policy(GetParam()); + } + }); + } + + void SetUp() override { + setDownstreamProtocol(Http::CodecClient::Type::HTTP1); + setUpstreamProtocol(FakeHttpConnection::Type::HTTP1); + } + + // Runs a subset lb test with the given request headers, expecting the x-host-type header to + // the given type ("a" or "b"). If expect_unique_host, verifies that a single host is selected + // over n iterations (e.g. for maglev/hash-ring policies). Otherwise, expected more than one + // host to be selected over n iterations. + void runTest(Http::TestHeaderMapImpl& request_headers, const std::string expected_host_type, + const bool expect_unique_host, const int n = 10) { + std::set hosts; + for (int i = 0; i < n; i++) { + Http::TestHeaderMapImpl response_headers{{":status", "200"}}; + + // Send header only request. + IntegrationStreamDecoderPtr response = codec_client_->makeHeaderOnlyRequest(request_headers); + response->waitForEndStream(); + + // Expect a response from a host in the correct subset. + EXPECT_EQ(response->headers() + .get(Envoy::Http::LowerCaseString{"x-host-type"}) + ->value() + .getStringView(), + expected_host_type); + + hosts.emplace( + response->headers().get(Envoy::Http::LowerCaseString{"x-host"})->value().getStringView()); + } + + if (expect_unique_host) { + EXPECT_EQ(hosts.size(), 1) << "Expected a single unique host to be selected for " + << envoy::api::v2::Cluster_LbPolicy_Name(GetParam()); + } else { + EXPECT_GT(hosts.size(), 1) << "Expected multiple hosts to be selected" + << envoy::api::v2::Cluster_LbPolicy_Name(GetParam()); + } + } + + Http::TestHeaderMapImpl type_a_request_headers_{{":method", "GET"}, {":path", "/test"}, + {":scheme", "http"}, {":authority", "host"}, + {"x-type", "a"}, {"x-hash", "hash-a"}}; + Http::TestHeaderMapImpl type_b_request_headers_{{":method", "GET"}, {":path", "/test"}, + {":scheme", "http"}, {":authority", "host"}, + {"x-type", "b"}, {"x-hash", "hash-b"}}; +}; + +INSTANTIATE_TEST_SUITE_P(SubsetCompatibleLoadBalancers, HttpSubsetLbIntegrationTest, + testing::ValuesIn(HttpSubsetLbIntegrationTest::getSubsetLbTestParams()), + HttpSubsetLbIntegrationTest::subsetLbTestParamsToString); + +// Tests each subset-compatible load balancer policy with 4 hosts divided into 2 subsets. +TEST_P(HttpSubsetLbIntegrationTest, SubsetLoadBalancer) { + initialize(); + codec_client_ = makeHttpConnection(lookupPort("http")); + + const bool expect_unique_host = (GetParam() == envoy::api::v2::Cluster_LbPolicy_RING_HASH || + GetParam() == envoy::api::v2::Cluster_LbPolicy_MAGLEV); + + runTest(type_a_request_headers_, "a", expect_unique_host); + runTest(type_b_request_headers_, "b", expect_unique_host); +} + +} // namespace Envoy From 4161d823b0bfcd9a15eecafc5b786035f4caceb9 Mon Sep 17 00:00:00 2001 From: Stephan Zuercher Date: Mon, 9 Sep 2019 16:13:35 -0700 Subject: [PATCH 7/8] rename test file Signed-off-by: Stephan Zuercher --- test/integration/BUILD | 4 ++-- ...integration_test.cc => http_subset_lb_integration_test.cc} | 0 2 files changed, 2 insertions(+), 2 deletions(-) rename test/integration/{http_load_balancer_integration_test.cc => http_subset_lb_integration_test.cc} (100%) diff --git a/test/integration/BUILD b/test/integration/BUILD index a10b678bc838..fd69958d4b4f 100644 --- a/test/integration/BUILD +++ b/test/integration/BUILD @@ -253,9 +253,9 @@ envoy_cc_test( ) envoy_cc_test( - name = "http_load_balancer_integration_test", + name = "http_subset_lb_integration_test", srcs = [ - "http_load_balancer_integration_test.cc", + "http_subset_lb_integration_test.cc", ], deps = [ ":http_integration_lib", diff --git a/test/integration/http_load_balancer_integration_test.cc b/test/integration/http_subset_lb_integration_test.cc similarity index 100% rename from test/integration/http_load_balancer_integration_test.cc rename to test/integration/http_subset_lb_integration_test.cc From b6f44cb53183df6533e23b7592b5510deba32fbf Mon Sep 17 00:00:00 2001 From: Stephan Zuercher Date: Mon, 9 Sep 2019 16:42:48 -0700 Subject: [PATCH 8/8] handle LOAD_BALANCING_POLICY_CONFIG Signed-off-by: Stephan Zuercher --- test/common/upstream/cluster_manager_impl_test.cc | 4 +++- test/integration/http_subset_lb_integration_test.cc | 3 ++- 2 files changed, 5 insertions(+), 2 deletions(-) diff --git a/test/common/upstream/cluster_manager_impl_test.cc b/test/common/upstream/cluster_manager_impl_test.cc index e3c536bb11a0..1534ba3f3b2d 100644 --- a/test/common/upstream/cluster_manager_impl_test.cc +++ b/test/common/upstream/cluster_manager_impl_test.cc @@ -594,7 +594,9 @@ class ClusterManagerSubsetInitializationTest for (int i = first; i <= last; i++) { if (envoy::api::v2::Cluster_LbPolicy_IsValid(i)) { auto policy = static_cast(i); - policies.push_back(policy); + if (policy != envoy::api::v2::Cluster_LbPolicy_LOAD_BALANCING_POLICY_CONFIG) { + policies.push_back(policy); + } } } return policies; diff --git a/test/integration/http_subset_lb_integration_test.cc b/test/integration/http_subset_lb_integration_test.cc index 5214dc0ea949..427f3f5a3c64 100644 --- a/test/integration/http_subset_lb_integration_test.cc +++ b/test/integration/http_subset_lb_integration_test.cc @@ -136,7 +136,8 @@ class HttpSubsetLbIntegrationTest : public testing::TestWithParam(i); if (policy == envoy::api::v2::Cluster_LbPolicy_ORIGINAL_DST_LB || - policy == envoy::api::v2::Cluster_LbPolicy_CLUSTER_PROVIDED) { + policy == envoy::api::v2::Cluster_LbPolicy_CLUSTER_PROVIDED || + policy == envoy::api::v2::Cluster_LbPolicy_LOAD_BALANCING_POLICY_CONFIG) { continue; }