-
Notifications
You must be signed in to change notification settings - Fork 4.4k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
balancer/weightedroundrobin: add load balancing policy (A58) (#6241)
- Loading branch information
Showing
10 changed files
with
1,545 additions
and
21 deletions.
There are no files selected for viewing
Large diffs are not rendered by default.
Oops, something went wrong.
Large diffs are not rendered by default.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,60 @@ | ||
/* | ||
* | ||
* Copyright 2023 gRPC authors. | ||
* | ||
* Licensed under the Apache License, Version 2.0 (the "License"); | ||
* you may not use this file except in compliance with the License. | ||
* You may obtain a copy of the License at | ||
* | ||
* http://www.apache.org/licenses/LICENSE-2.0 | ||
* | ||
* Unless required by applicable law or agreed to in writing, software | ||
* distributed under the License is distributed on an "AS IS" BASIS, | ||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||
* See the License for the specific language governing permissions and | ||
* limitations under the License. | ||
* | ||
*/ | ||
|
||
package weightedroundrobin | ||
|
||
import ( | ||
"time" | ||
|
||
"google.golang.org/grpc/serviceconfig" | ||
) | ||
|
||
type lbConfig struct { | ||
serviceconfig.LoadBalancingConfig `json:"-"` | ||
|
||
// Whether to enable out-of-band utilization reporting collection from the | ||
// endpoints. By default, per-request utilization reporting is used. | ||
EnableOOBLoadReport bool `json:"enableOobLoadReport,omitempty"` | ||
|
||
// Load reporting interval to request from the server. Note that the | ||
// server may not provide reports as frequently as the client requests. | ||
// Used only when enable_oob_load_report is true. Default is 10 seconds. | ||
OOBReportingPeriod time.Duration `json:"oobReportingPeriod,omitempty"` | ||
|
||
// A given endpoint must report load metrics continuously for at least this | ||
// long before the endpoint weight will be used. This avoids churn when | ||
// the set of endpoint addresses changes. Takes effect both immediately | ||
// after we establish a connection to an endpoint and after | ||
// weight_expiration_period has caused us to stop using the most recent | ||
// load metrics. Default is 10 seconds. | ||
BlackoutPeriod time.Duration `json:"blackoutPeriod,omitempty"` | ||
|
||
// If a given endpoint has not reported load metrics in this long, | ||
// then we stop using the reported weight. This ensures that we do | ||
// not continue to use very stale weights. Once we stop using a stale | ||
// value, if we later start seeing fresh reports again, the | ||
// blackout_period applies. Defaults to 3 minutes. | ||
WeightExpirationPeriod time.Duration `json:"weightExpirationPeriod,omitempty"` | ||
|
||
// How often endpoint weights are recalculated. Default is 1 second. | ||
WeightUpdatePeriod time.Duration `json:"weightUpdatePeriod,omitempty"` | ||
|
||
// The multiplier used to adjust endpoint weights with the error rate | ||
// calculated as eps/qps. Default is 1.0. | ||
ErrorUtilizationPenalty float64 `json:"errorUtilizationPenalty,omitempty"` | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,44 @@ | ||
/* | ||
* | ||
* Copyright 2023 gRPC authors. | ||
* | ||
* Licensed under the Apache License, Version 2.0 (the "License"); | ||
* you may not use this file except in compliance with the License. | ||
* You may obtain a copy of the License at | ||
* | ||
* http://www.apache.org/licenses/LICENSE-2.0 | ||
* | ||
* Unless required by applicable law or agreed to in writing, software | ||
* distributed under the License is distributed on an "AS IS" BASIS, | ||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||
* See the License for the specific language governing permissions and | ||
* limitations under the License. | ||
* | ||
*/ | ||
|
||
// Package internal allows for easier testing of the weightedroundrobin | ||
// package. | ||
package internal | ||
|
||
import ( | ||
"time" | ||
) | ||
|
||
// AllowAnyWeightUpdatePeriod permits any setting of WeightUpdatePeriod for | ||
// testing. Normally a minimum of 100ms is applied. | ||
var AllowAnyWeightUpdatePeriod bool | ||
|
||
// LBConfig allows tests to produce a JSON form of the config from the struct | ||
// instead of using a string. | ||
type LBConfig struct { | ||
EnableOOBLoadReport *bool `json:"enableOobLoadReport,omitempty"` | ||
OOBReportingPeriod *time.Duration `json:"oobReportingPeriod,omitempty"` | ||
BlackoutPeriod *time.Duration `json:"blackoutPeriod,omitempty"` | ||
WeightExpirationPeriod *time.Duration `json:"weightExpirationPeriod,omitempty"` | ||
WeightUpdatePeriod *time.Duration `json:"weightUpdatePeriod,omitempty"` | ||
ErrorUtilizationPenalty *float64 `json:"errorUtilizationPenalty,omitempty"` | ||
} | ||
|
||
// TimeNow can be overridden by tests to return a different value for the | ||
// current time. | ||
var TimeNow = time.Now |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,34 @@ | ||
/* | ||
* | ||
* Copyright 2023 gRPC authors. | ||
* | ||
* Licensed under the Apache License, Version 2.0 (the "License"); | ||
* you may not use this file except in compliance with the License. | ||
* You may obtain a copy of the License at | ||
* | ||
* http://www.apache.org/licenses/LICENSE-2.0 | ||
* | ||
* Unless required by applicable law or agreed to in writing, software | ||
* distributed under the License is distributed on an "AS IS" BASIS, | ||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||
* See the License for the specific language governing permissions and | ||
* limitations under the License. | ||
* | ||
*/ | ||
|
||
package weightedroundrobin | ||
|
||
import ( | ||
"fmt" | ||
|
||
"google.golang.org/grpc/grpclog" | ||
internalgrpclog "google.golang.org/grpc/internal/grpclog" | ||
) | ||
|
||
const prefix = "[%p] " | ||
|
||
var logger = grpclog.Component("weighted-round-robin") | ||
|
||
func prefixLogger(p *wrrBalancer) *internalgrpclog.PrefixLogger { | ||
return internalgrpclog.NewPrefixLogger(logger, fmt.Sprintf(prefix, p)) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,138 @@ | ||
/* | ||
* | ||
* Copyright 2023 gRPC authors. | ||
* | ||
* Licensed under the Apache License, Version 2.0 (the "License"); | ||
* you may not use this file except in compliance with the License. | ||
* You may obtain a copy of the License at | ||
* | ||
* http://www.apache.org/licenses/LICENSE-2.0 | ||
* | ||
* Unless required by applicable law or agreed to in writing, software | ||
* distributed under the License is distributed on an "AS IS" BASIS, | ||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||
* See the License for the specific language governing permissions and | ||
* limitations under the License. | ||
* | ||
*/ | ||
|
||
package weightedroundrobin | ||
|
||
import ( | ||
"math" | ||
) | ||
|
||
type scheduler interface { | ||
nextIndex() int | ||
} | ||
|
||
// newScheduler uses scWeights to create a new scheduler for selecting subconns | ||
// in a picker. It will return a round robin implementation if at least | ||
// len(scWeights)-1 are zero or there is only a single subconn, otherwise it | ||
// will return an Earliest Deadline First (EDF) scheduler implementation that | ||
// selects the subchannels according to their weights. | ||
func newScheduler(scWeights []float64, inc func() uint32) scheduler { | ||
n := len(scWeights) | ||
if n == 0 { | ||
return nil | ||
} | ||
if n == 1 { | ||
return &rrScheduler{numSCs: 1, inc: inc} | ||
} | ||
sum := float64(0) | ||
numZero := 0 | ||
max := float64(0) | ||
for _, w := range scWeights { | ||
sum += w | ||
if w > max { | ||
max = w | ||
} | ||
if w == 0 { | ||
numZero++ | ||
} | ||
} | ||
if numZero >= n-1 { | ||
return &rrScheduler{numSCs: uint32(n), inc: inc} | ||
} | ||
unscaledMean := sum / float64(n-numZero) | ||
scalingFactor := maxWeight / max | ||
mean := uint16(math.Round(scalingFactor * unscaledMean)) | ||
|
||
weights := make([]uint16, n) | ||
allEqual := true | ||
for i, w := range scWeights { | ||
if w == 0 { | ||
// Backends with weight = 0 use the mean. | ||
weights[i] = mean | ||
} else { | ||
scaledWeight := uint16(math.Round(scalingFactor * w)) | ||
weights[i] = scaledWeight | ||
if scaledWeight != mean { | ||
allEqual = false | ||
} | ||
} | ||
} | ||
|
||
if allEqual { | ||
return &rrScheduler{numSCs: uint32(n), inc: inc} | ||
} | ||
|
||
logger.Infof("using edf scheduler with weights: %v", weights) | ||
return &edfScheduler{weights: weights, inc: inc} | ||
} | ||
|
||
const maxWeight = math.MaxUint16 | ||
|
||
// edfScheduler implements EDF using the same algorithm as grpc-c++ here: | ||
// | ||
// https://github.com/grpc/grpc/blob/master/src/core/ext/filters/client_channel/lb_policy/weighted_round_robin/static_stride_scheduler.cc | ||
type edfScheduler struct { | ||
inc func() uint32 | ||
weights []uint16 | ||
} | ||
|
||
// Returns the index in s.weights for the picker to choose. | ||
func (s *edfScheduler) nextIndex() int { | ||
const offset = maxWeight / 2 | ||
|
||
for { | ||
idx := uint64(s.inc()) | ||
|
||
// The sequence number (idx) is split in two: the lower %n gives the | ||
// index of the backend, and the rest gives the number of times we've | ||
// iterated through all backends. `generation` is used to | ||
// deterministically decide whether we pick or skip the backend on this | ||
// iteration, in proportion to the backend's weight. | ||
|
||
backendIndex := idx % uint64(len(s.weights)) | ||
generation := idx / uint64(len(s.weights)) | ||
weight := uint64(s.weights[backendIndex]) | ||
|
||
// We pick a backend `weight` times per `maxWeight` generations. The | ||
// multiply and modulus ~evenly spread out the picks for a given | ||
// backend between different generations. The offset by `backendIndex` | ||
// helps to reduce the chance of multiple consecutive non-picks: if we | ||
// have two consecutive backends with an equal, say, 80% weight of the | ||
// max, with no offset we would see 1/5 generations that skipped both. | ||
// TODO(b/190488683): add test for offset efficacy. | ||
mod := uint64(weight*generation+backendIndex*offset) % maxWeight | ||
|
||
if mod < maxWeight-weight { | ||
continue | ||
} | ||
return int(backendIndex) | ||
} | ||
} | ||
|
||
// A simple RR scheduler to use for fallback when fewer than two backends have | ||
// non-zero weights, or all backends have the the same weight, or when only one | ||
// subconn exists. | ||
type rrScheduler struct { | ||
inc func() uint32 | ||
numSCs uint32 | ||
} | ||
|
||
func (s *rrScheduler) nextIndex() int { | ||
idx := s.inc() | ||
return int(idx % s.numSCs) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters