fix: IAM store fallback to list users and policies from disk (#10787)
Bonus fixes, remove package retry it is harder to get it right, also manage context remove it such that we don't have to rely on it anymore instead use a simple Jitter retry.master
parent
4ea31da889
commit
68de5a6f6a
@ -1,133 +0,0 @@ |
|||||||
/* |
|
||||||
* Minio Cloud Storage, (C) 2020 MinIO, Inc. |
|
||||||
* |
|
||||||
* 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 retry |
|
||||||
|
|
||||||
import ( |
|
||||||
"context" |
|
||||||
"math" |
|
||||||
"math/rand" |
|
||||||
"time" |
|
||||||
) |
|
||||||
|
|
||||||
// MaxJitter will randomize over the full exponential backoff time
|
|
||||||
const MaxJitter = 1.0 |
|
||||||
|
|
||||||
// NoJitter disables the use of jitter for randomizing the
|
|
||||||
// exponential backoff time
|
|
||||||
const NoJitter = 0.0 |
|
||||||
|
|
||||||
// defaultTimer implements Timer interface using time.Timer
|
|
||||||
type defaultTimer struct { |
|
||||||
timer *time.Timer |
|
||||||
} |
|
||||||
|
|
||||||
// C returns the timers channel which receives the current time when the timer fires.
|
|
||||||
func (t *defaultTimer) C() <-chan time.Time { |
|
||||||
return t.timer.C |
|
||||||
} |
|
||||||
|
|
||||||
// Start starts the timer to fire after the given duration
|
|
||||||
// don't use this code concurrently.
|
|
||||||
func (t *defaultTimer) Start(duration time.Duration) { |
|
||||||
if t.timer == nil { |
|
||||||
t.timer = time.NewTimer(duration) |
|
||||||
} else { |
|
||||||
t.timer.Reset(duration) |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
// Stop is called when the timer is not used anymore and resources may be freed.
|
|
||||||
func (t *defaultTimer) Stop() { |
|
||||||
if t.timer != nil { |
|
||||||
t.timer.Stop() |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
// NewTimerWithJitter creates a timer with exponentially increasing delays
|
|
||||||
// until the maximum retry attempts are reached. - this function is a fully
|
|
||||||
// configurable version, meant for only advanced use cases. For the most part
|
|
||||||
// one should use newRetryTimerSimple and newRetryTimer.
|
|
||||||
func NewTimerWithJitter(ctx context.Context, unit time.Duration, cap time.Duration, jitter float64) <-chan int { |
|
||||||
attemptCh := make(chan int) |
|
||||||
|
|
||||||
// normalize jitter to the range [0, 1.0]
|
|
||||||
jitter = math.Max(NoJitter, math.Min(MaxJitter, jitter)) |
|
||||||
|
|
||||||
// computes the exponential backoff duration according to
|
|
||||||
// https://www.awsarchitectureblog.com/2015/03/backoff.html
|
|
||||||
exponentialBackoffWait := func(attempt int) time.Duration { |
|
||||||
// 1<<uint(attempt) below could overflow, so limit the value of attempt
|
|
||||||
const maxAttempt = 30 |
|
||||||
if attempt > maxAttempt { |
|
||||||
attempt = maxAttempt |
|
||||||
} |
|
||||||
//sleep = random_between(0, min(cap, base * 2 ** attempt))
|
|
||||||
sleep := unit * 1 << uint(attempt) |
|
||||||
if sleep > cap { |
|
||||||
sleep = cap |
|
||||||
} |
|
||||||
if jitter > NoJitter { |
|
||||||
sleep -= time.Duration(rand.Float64() * float64(sleep) * jitter) |
|
||||||
} |
|
||||||
return sleep |
|
||||||
} |
|
||||||
|
|
||||||
go func() { |
|
||||||
nextBackoff := 0 |
|
||||||
t := &defaultTimer{} |
|
||||||
|
|
||||||
defer func() { |
|
||||||
t.Stop() |
|
||||||
}() |
|
||||||
|
|
||||||
defer close(attemptCh) |
|
||||||
|
|
||||||
// Channel used to signal after the expiry of backoff wait seconds.
|
|
||||||
for { |
|
||||||
select { |
|
||||||
case <-ctx.Done(): |
|
||||||
return |
|
||||||
case attemptCh <- nextBackoff: |
|
||||||
nextBackoff++ |
|
||||||
} |
|
||||||
|
|
||||||
t.Start(exponentialBackoffWait(nextBackoff)) |
|
||||||
|
|
||||||
select { |
|
||||||
case <-ctx.Done(): |
|
||||||
return |
|
||||||
case <-t.C(): |
|
||||||
} |
|
||||||
} |
|
||||||
}() |
|
||||||
|
|
||||||
// Start reading..
|
|
||||||
return attemptCh |
|
||||||
} |
|
||||||
|
|
||||||
// Default retry constants.
|
|
||||||
const ( |
|
||||||
defaultRetryUnit = 50 * time.Millisecond // 50 millisecond.
|
|
||||||
defaultRetryCap = 500 * time.Millisecond // 500 millisecond.
|
|
||||||
) |
|
||||||
|
|
||||||
// NewTimer creates a timer with exponentially increasing delays
|
|
||||||
// until the maximum retry attempts are reached. - this function is a
|
|
||||||
// simpler version with all default values.
|
|
||||||
func NewTimer(ctx context.Context) <-chan int { |
|
||||||
return NewTimerWithJitter(ctx, defaultRetryUnit, defaultRetryCap, MaxJitter) |
|
||||||
} |
|
@ -1,85 +0,0 @@ |
|||||||
/* |
|
||||||
* Minio Cloud Storage, (C) 2020 Minio, Inc. |
|
||||||
* |
|
||||||
* 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 retry |
|
||||||
|
|
||||||
import ( |
|
||||||
"context" |
|
||||||
"testing" |
|
||||||
"time" |
|
||||||
) |
|
||||||
|
|
||||||
// Tests for retry timer.
|
|
||||||
func TestRetryTimerSimple(t *testing.T) { |
|
||||||
retryCtx, cancel := context.WithCancel(context.Background()) |
|
||||||
attemptCh := NewTimer(retryCtx) |
|
||||||
i := <-attemptCh |
|
||||||
if i != 0 { |
|
||||||
cancel() |
|
||||||
t.Fatalf("Invalid attempt counter returned should be 0, found %d instead", i) |
|
||||||
} |
|
||||||
i = <-attemptCh |
|
||||||
if i <= 0 { |
|
||||||
cancel() |
|
||||||
t.Fatalf("Invalid attempt counter returned should be greater than 0, found %d instead", i) |
|
||||||
} |
|
||||||
cancel() |
|
||||||
_, ok := <-attemptCh |
|
||||||
if ok { |
|
||||||
t.Fatal("Attempt counter should be closed") |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
// Test retry time with no jitter.
|
|
||||||
func TestRetryTimerWithNoJitter(t *testing.T) { |
|
||||||
retryCtx, cancel := context.WithCancel(context.Background()) |
|
||||||
defer cancel() |
|
||||||
|
|
||||||
// No jitter
|
|
||||||
attemptCh := NewTimerWithJitter(retryCtx, time.Millisecond, 5*time.Millisecond, NoJitter) |
|
||||||
i := <-attemptCh |
|
||||||
if i != 0 { |
|
||||||
cancel() |
|
||||||
t.Fatalf("Invalid attempt counter returned should be 0, found %d instead", i) |
|
||||||
} |
|
||||||
// Loop through the maximum possible attempt.
|
|
||||||
for i = range attemptCh { |
|
||||||
if i == 30 { |
|
||||||
cancel() |
|
||||||
} |
|
||||||
} |
|
||||||
_, ok := <-attemptCh |
|
||||||
if ok { |
|
||||||
t.Fatal("Attempt counter should be closed") |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
// Test retry time with Jitter greater than MaxJitter.
|
|
||||||
func TestRetryTimerWithJitter(t *testing.T) { |
|
||||||
retryCtx, cancel := context.WithCancel(context.Background()) |
|
||||||
// Jitter will be set back to 1.0
|
|
||||||
attemptCh := NewTimerWithJitter(retryCtx, time.Second, 30*time.Second, 2.0) |
|
||||||
i := <-attemptCh |
|
||||||
if i != 0 { |
|
||||||
cancel() |
|
||||||
t.Fatalf("Invalid attempt counter returned should be 0, found %d instead", i) |
|
||||||
} |
|
||||||
cancel() |
|
||||||
_, ok := <-attemptCh |
|
||||||
if ok { |
|
||||||
t.Fatal("Attempt counter should be closed") |
|
||||||
} |
|
||||||
} |
|
Loading…
Reference in new issue