107 lines
3.1 KiB
Go
107 lines
3.1 KiB
Go
|
/*
|
||
|
Copyright 2016 Google Inc. All Rights Reserved.
|
||
|
|
||
|
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.
|
||
|
*/
|
||
|
|
||
|
// This is ia snapshot from github.com/googleapis/gax-go with minor modifications.
|
||
|
package gax
|
||
|
|
||
|
import (
|
||
|
"time"
|
||
|
|
||
|
"google.golang.org/grpc/codes"
|
||
|
)
|
||
|
|
||
|
type CallOption interface {
|
||
|
Resolve(*CallSettings)
|
||
|
}
|
||
|
|
||
|
type callOptions []CallOption
|
||
|
|
||
|
func (opts callOptions) Resolve(s *CallSettings) *CallSettings {
|
||
|
for _, opt := range opts {
|
||
|
opt.Resolve(s)
|
||
|
}
|
||
|
return s
|
||
|
}
|
||
|
|
||
|
// Encapsulates the call settings for a particular API call.
|
||
|
type CallSettings struct {
|
||
|
Timeout time.Duration
|
||
|
RetrySettings RetrySettings
|
||
|
}
|
||
|
|
||
|
// Per-call configurable settings for retrying upon transient failure.
|
||
|
type RetrySettings struct {
|
||
|
RetryCodes map[codes.Code]bool
|
||
|
BackoffSettings BackoffSettings
|
||
|
}
|
||
|
|
||
|
// Parameters to the exponential backoff algorithm for retrying.
|
||
|
type BackoffSettings struct {
|
||
|
DelayTimeoutSettings MultipliableDuration
|
||
|
RPCTimeoutSettings MultipliableDuration
|
||
|
}
|
||
|
|
||
|
type MultipliableDuration struct {
|
||
|
Initial time.Duration
|
||
|
Max time.Duration
|
||
|
Multiplier float64
|
||
|
}
|
||
|
|
||
|
func (w CallSettings) Resolve(s *CallSettings) {
|
||
|
s.Timeout = w.Timeout
|
||
|
s.RetrySettings = w.RetrySettings
|
||
|
|
||
|
s.RetrySettings.RetryCodes = make(map[codes.Code]bool, len(w.RetrySettings.RetryCodes))
|
||
|
for key, value := range w.RetrySettings.RetryCodes {
|
||
|
s.RetrySettings.RetryCodes[key] = value
|
||
|
}
|
||
|
}
|
||
|
|
||
|
type withRetryCodes []codes.Code
|
||
|
|
||
|
func (w withRetryCodes) Resolve(s *CallSettings) {
|
||
|
s.RetrySettings.RetryCodes = make(map[codes.Code]bool)
|
||
|
for _, code := range w {
|
||
|
s.RetrySettings.RetryCodes[code] = true
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// WithRetryCodes sets a list of Google API canonical error codes upon which a
|
||
|
// retry should be attempted.
|
||
|
func WithRetryCodes(retryCodes []codes.Code) CallOption {
|
||
|
return withRetryCodes(retryCodes)
|
||
|
}
|
||
|
|
||
|
type withDelayTimeoutSettings MultipliableDuration
|
||
|
|
||
|
func (w withDelayTimeoutSettings) Resolve(s *CallSettings) {
|
||
|
s.RetrySettings.BackoffSettings.DelayTimeoutSettings = MultipliableDuration(w)
|
||
|
}
|
||
|
|
||
|
// WithDelayTimeoutSettings specifies:
|
||
|
// - The initial delay time, in milliseconds, between the completion of
|
||
|
// the first failed request and the initiation of the first retrying
|
||
|
// request.
|
||
|
// - The multiplier by which to increase the delay time between the
|
||
|
// completion of failed requests, and the initiation of the subsequent
|
||
|
// retrying request.
|
||
|
// - The maximum delay time, in milliseconds, between requests. When this
|
||
|
// value is reached, `RetryDelayMultiplier` will no longer be used to
|
||
|
// increase delay time.
|
||
|
func WithDelayTimeoutSettings(initial time.Duration, max time.Duration, multiplier float64) CallOption {
|
||
|
return withDelayTimeoutSettings(MultipliableDuration{initial, max, multiplier})
|
||
|
}
|