-
Notifications
You must be signed in to change notification settings - Fork 4.9k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Introduce Least Request LB active request bias config #11252
Conversation
…eighted hosts case Signed-off-by: Gastón Kleiman <gkleiman@lyft.com>
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
LGTM with small comments for a test. Thank you!
/wait
@@ -81,6 +81,7 @@ constexpr const char* runtime_features[] = { | |||
constexpr const char* disabled_runtime_features[] = { | |||
// Sentinel and test flag. | |||
"envoy.reloadable_features.test_feature_false", | |||
"envoy.reloadable_features.alternative_least_request_weights", |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
If we end up doing a real PR for this, it should be a real config option vs. one of these. I would just use a RuntimeFeatureFlag inside the LB config proto.
alternativeWeights_ = Runtime::runtimeFeatureEnabled( | ||
"envoy.reloadable_features.alternative_least_request_weights"); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
For perf reasons I would not re-snap this on every choice. I would just do it in refresh below. I think this is fine.
const uint32_t choice_count_; | ||
bool alternativeWeights_; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
alternative_weights_
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I recognize I have the unpopular opinion here, but this change seems like a hack to workaround for a lack of operational agility. I think a better approach would be to just add some coefficient (k) that makes the weights look something like:
new_weight = (configured_weight / outstanding_requests^k)
If k=1, things behave as they do today and normalize the weights by dividing it by outstanding requests.
If k=0, you get the behavior in this PR and the outstanding requests don't affect the configured weights.
You also have the freedom to configure anything in between and even make the outstanding requests have a more aggressive impact on the weights if you so choose. This would be trivial to implement, can be runtime configurable, and seems less hacky. WDYT?
const uint32_t choice_count_; | ||
bool alternativeWeights_; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Also explain what this means exactly in a comment. It's non-obvious from the name.
+1 I like this idea. Let's do that! |
Signed-off-by: Gastón Kleiman <gkleiman@lyft.com>
Good idea! I implemented this with a global runtime. If things work fine in the experiment, I'll make it a real config option inside |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Looks good to test. If the patch looks good I like this idea for a general config flag. Great idea @tonya11en!
/wait
ENVOY_LOG(debug, "cluster={} address={} active_requests_exponent={} weight={}", | ||
host.cluster().name(), host.address()->asString(), active_requests_exponent_, weight); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
definitely trace. I would also probably remove this in the final patch.
const double weight = static_cast<double>(host.weight()) / | ||
std::pow(host.stats().rq_active_.value() + 1, active_requests_exponent_); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
For perf reasons I would probably check if 1.0 and then branch and avoid the pow call.
Signed-off-by: Gastón Kleiman <gkleiman@lyft.com>
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
/wait
#include "common/upstream/edf_scheduler.h" | ||
|
||
namespace Envoy { | ||
namespace Upstream { | ||
|
||
static const std::string RuntimeLeastRequestsActiveRequestsExponent = |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
absl::string_view to avoid static fiasco.
Signed-off-by: Gastón Kleiman <gkleiman@lyft.com>
This pull request has been automatically marked as stale because it has not had activity in the last 7 days. It will be closed in 7 days if no further activity occurs. Please feel free to give a status update now, ping for review, or re-open when it's ready. Thank you for your contributions! |
EXPECT_CALL(random_, random()).WillRepeatedly(Return(0)); | ||
|
||
// We should see 2:1 ratio for hosts[1] to hosts[0], regardless of the active request count. | ||
hostSet().healthy_hosts_[1]->stats().rq_active_.set(1); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm surprised this works. Are the stats affecting the outstanding requests perceived by the LB when the exponent is non-zero?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yes, the active rq stat is used to calculate the effective weights when the exponent is > 0: https://github.com/envoyproxy/envoy/pull/11252/files/0a2b1fcdad5daeef1136ef7c2208f356d3fcf05f#diff-e746c5f817e7b535220c275b9dea666bR494
This pull request has been automatically marked as stale because it has not had activity in the last 7 days. It will be closed in 7 days if no further activity occurs. Please feel free to give a status update now, ping for review, or re-open when it's ready. Thank you for your contributions! |
This was just verified to have the desired effect in our infra, so I'm resetting stalebot. |
/wait |
Use a plain double instead of a runtime double to store the per-cluster active request bias. Note: The goal of this commit is to evaluate the memory overhead of this approach. A commit with te Least Requests LB changes might follow if we deem the memory overhead of this approach acceptable. Signed-off-by: Gastón Kleiman <gkleiman@lyft.com>
Per above I'm OK merging what you have temporarily. But I think what we want here is a RuntimeDouble for the process, not one for each cluster. I'm not 100% sure of the best data structure to hold that RuntimeDouble. |
Signed-off-by: Gastón Kleiman <gkleiman@lyft.com>
Signed-off-by: Gastón Kleiman <gkleiman@lyft.com>
@jmarantz 👍 I restored the functional changes and merged master — |
Signed-off-by: Gastón Kleiman <gkleiman@lyft.com>
/retest Looks like Azure Pipelines is degraded and the Linux coverage job failed with the following error:
|
🐴 hold your horses - no failures detected, yet. |
/azp run |
Azure Pipelines successfully started running 1 pipeline(s). |
Summarizing DM discussion on Slack, we are going to move the RuntimeDouble out of the cluster configuration and into bootstrap. The problem with my earlier suggestion to merge this is that it creates an API at the cluster level that will be hard to remove. @gkleiman indicated he didn't think it would be too bad to drive the config decision from the bootstrap proto. |
I've changed the place/way where the active request bias is stored a few times already since I began working on this, so I would love to reach consensus between Lyft & external maintainers before changing the implementation again. Here goes a summary of Lyft's requirements, findings about the extra per-cluster memory footprint and possible approaches. I hope this makes the conversation a bit easier — it turned out to be a bit long for a PR comment, so I would be also happy to move this to a Google Doc so people can comment on it. Lyft RequirementsIn order to improve Envoy’s load balancing behavior during squeeze tests, we would like to make it possible for the least request load balancer to ignore the active request count in the case that all host weights are not equal. We would do configure this change globally for all Envoy instances in our service mesh. Being able to override the active request bias for a particular cluster would be nice to have but is currently not required. Findings
Feedback@tonya11en and @mattklein123 initially proposed the current approach — per-cluster RuntimeDouble in `LeastRequestLbConfig (#1) @jmarantz suggested making the active request bias a global setting instead of a per-cluster setting (#4) @alyssawilk suggested using scalars for changing per-cluster values via CDS rather than using runtime overrides. (#2) Approaches1. Per-Cluster - RuntimeDouble field in CDS LeastRequestLbConfig (Current implementation)Pro: Easy to implement. Most flexible approach. 2. Per-Cluster - Scalar field in CDS LeastRequestLbConfigPro: Smaller memory footprint than using 3. Global Runtime - Hard-coded runtime key (Original implementation)We could define a global runtime key that can be used to globally configure the active request bias used by all least request LBs, e.g., Pro: No additional memory footprint. Very easy to implement. Easy to adopt, no dependency on a v3 xDS server. 4. Global Runtime - Move the RuntimeDouble out of the cluster configuration and into bootstrapWe could move the To avoid reducing the additional memory usage we could add a Pro: Small memory footprint. Easy to adopt, no dependency on a v3 xDS server. |
Thanks for the detailed note. It's clear we'll have to spend some memory to get the per-cluster configurability for this feature. I'm fine if that's the consensus. The amount of memory we are spending was surprising, which I believe is due to Envoy's usage of protobufs as retained data structures. I'm not sure how big a task it would be to move away from protobufs as a retained data structure and use them instead only for transport. But that could help get that memory back and perhaps then some. Obviously that would be outside the scope of the PR, obviously. Perhaps someone could own the task of evaluating that optimization as a follow-up. In the scheme of things, 2.5M memory is obviously not a lot for a proxy meant to handle 10k clusters, so I'm OK merging this in in its current form. I had pushed back earlier, because I thought maybe the per-cluster config was not strictly needed and would be hard to remove once it was added. Sounds like it would be useful at least. |
I think we should move this PR forward as-is with the per-cluster increase. If you want to leave some TODOs we can do that but I think the proto issues are out of scope for this PR? |
agreed; happy to see it merged and iterate from there. |
This reverts commit a6a285d. Signed-off-by: Gastón Kleiman <gkleiman@lyft.com>
Signed-off-by: Gastón Kleiman <gkleiman@lyft.com>
@mattklein123 |
FYI master merge should fix that (
#12067)
…On Tue, Jul 14, 2020 at 5:02 PM Gastón Kleiman ***@***.***> wrote:
@mattklein123 <https://github.com/mattklein123>
IpVersionsClientType/VhdsIntegrationTest.VhdsOnDemandUpdateHttpConnectionCloses/0
failed the tsan run due to a data race that doesn't seem to be related to
this PR.
—
You are receiving this because you were mentioned.
Reply to this email directly, view it on GitHub
<#11252 (comment)>,
or unsubscribe
<https://github.com/notifications/unsubscribe-auth/AELALPO6LGGHSY6522Q7KC3R3TBUZANCNFSM4NERGDPA>
.
|
Signed-off-by: Gastón Kleiman <gkleiman@lyft.com>
Signed-off-by: Gastón Kleiman <gkleiman@lyft.com>
* Add support for making Least Requests LB behave like Round Robin in weighted hosts case Signed-off-by: Gastón Kleiman <gkleiman@lyft.com> * Address feedback Signed-off-by: Gastón Kleiman <gkleiman@lyft.com> * Perf/logging improvements Signed-off-by: Gastón Kleiman <gkleiman@lyft.com> * Address feedback and cleanup BUILD file Signed-off-by: Gastón Kleiman <gkleiman@lyft.com> * Make active requests exponent configurable via CDS/runtime Signed-off-by: Gastón Kleiman <gkleiman@lyft.com> * Address feedback Signed-off-by: Gastón Kleiman <gkleiman@lyft.com> * Validate log message Signed-off-by: Gastón Kleiman <gkleiman@lyft.com> * Update cluster memory test golden values Signed-off-by: Gastón Kleiman <gkleiman@lyft.com> * Fix method name Signed-off-by: Gastón Kleiman <gkleiman@lyft.com> * Explicitly initialize active_request_bias_ Signed-off-by: Gastón Kleiman <gkleiman@lyft.com> * Try to make clang-tidy happy Signed-off-by: Gastón Kleiman <gkleiman@lyft.com> * Use unique_ptr instead of optional Signed-off-by: Gastón Kleiman <gkleiman@lyft.com> * Update stats integration test Signed-off-by: Gastón Kleiman <gkleiman@lyft.com> * Check whether memory footprint is reduced without LB changes Signed-off-by: Gastón Kleiman <gkleiman@lyft.com> * Use plain double for active request bias Use a plain double instead of a runtime double to store the per-cluster active request bias. Note: The goal of this commit is to evaluate the memory overhead of this approach. A commit with te Least Requests LB changes might follow if we deem the memory overhead of this approach acceptable. Signed-off-by: Gastón Kleiman <gkleiman@lyft.com> * Revert back to approved implementation using RuntimeDouble Signed-off-by: Gastón Kleiman <gkleiman@lyft.com> * Add extra fields to CDS cluster proto to check memory usage Signed-off-by: Gastón Kleiman <gkleiman@lyft.com> * Revert "Add extra fields to CDS cluster proto to check memory usage" This reverts commit a6a285d. Signed-off-by: Gastón Kleiman <gkleiman@lyft.com> * Add changelog entry Signed-off-by: Gastón Kleiman <gkleiman@lyft.com> Signed-off-by: Kevin Baichoo <kbaichoo@google.com>
* Add support for making Least Requests LB behave like Round Robin in weighted hosts case Signed-off-by: Gastón Kleiman <gkleiman@lyft.com> * Address feedback Signed-off-by: Gastón Kleiman <gkleiman@lyft.com> * Perf/logging improvements Signed-off-by: Gastón Kleiman <gkleiman@lyft.com> * Address feedback and cleanup BUILD file Signed-off-by: Gastón Kleiman <gkleiman@lyft.com> * Make active requests exponent configurable via CDS/runtime Signed-off-by: Gastón Kleiman <gkleiman@lyft.com> * Address feedback Signed-off-by: Gastón Kleiman <gkleiman@lyft.com> * Validate log message Signed-off-by: Gastón Kleiman <gkleiman@lyft.com> * Update cluster memory test golden values Signed-off-by: Gastón Kleiman <gkleiman@lyft.com> * Fix method name Signed-off-by: Gastón Kleiman <gkleiman@lyft.com> * Explicitly initialize active_request_bias_ Signed-off-by: Gastón Kleiman <gkleiman@lyft.com> * Try to make clang-tidy happy Signed-off-by: Gastón Kleiman <gkleiman@lyft.com> * Use unique_ptr instead of optional Signed-off-by: Gastón Kleiman <gkleiman@lyft.com> * Update stats integration test Signed-off-by: Gastón Kleiman <gkleiman@lyft.com> * Check whether memory footprint is reduced without LB changes Signed-off-by: Gastón Kleiman <gkleiman@lyft.com> * Use plain double for active request bias Use a plain double instead of a runtime double to store the per-cluster active request bias. Note: The goal of this commit is to evaluate the memory overhead of this approach. A commit with te Least Requests LB changes might follow if we deem the memory overhead of this approach acceptable. Signed-off-by: Gastón Kleiman <gkleiman@lyft.com> * Revert back to approved implementation using RuntimeDouble Signed-off-by: Gastón Kleiman <gkleiman@lyft.com> * Add extra fields to CDS cluster proto to check memory usage Signed-off-by: Gastón Kleiman <gkleiman@lyft.com> * Revert "Add extra fields to CDS cluster proto to check memory usage" This reverts commit a6a285d. Signed-off-by: Gastón Kleiman <gkleiman@lyft.com> * Add changelog entry Signed-off-by: Gastón Kleiman <gkleiman@lyft.com> Signed-off-by: scheler <santosh.cheler@appdynamics.com>
Commit Message:
Introduce Least Request LB active request bias config
Additional Description:
This patch adds a runtime configurable active request bias to the Least Request Load Balancer. This bias only takes effect when all host weights are not equal. The larger it is, the more aggressively active requests will lower the effective weight of a host.
The least request lb behaves like the round robin lb when the bias is set to
0.0
and all weights are not equal. This is very useful for squeeze/redline tests and fixes the issues described on #10602.Risk Level: S
Testing: Unit test. Validated on Lyft infra that this approach fixes the issues described in #10602.