mirror of https://github.com/hashicorp/consul
sdk/retry: support ending the iteration early
I've found this feature to be very useful in https://pkg.go.dev/gotest.tools/v3/poll#WaitOn I have encountered a few cases where I wanted that same support, so this commit adds it.pull/10112/head
parent
034c5c5f8c
commit
a2986ebb9c
|
@ -34,6 +34,7 @@ type Failer interface {
|
||||||
// R provides context for the retryer.
|
// R provides context for the retryer.
|
||||||
type R struct {
|
type R struct {
|
||||||
fail bool
|
fail bool
|
||||||
|
done bool
|
||||||
output []string
|
output []string
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -77,6 +78,12 @@ func (r *R) log(s string) {
|
||||||
r.output = append(r.output, decorate(s))
|
r.output = append(r.output, decorate(s))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Stop retrying, and fail the test with the specified error.
|
||||||
|
func (r *R) Stop(err error) {
|
||||||
|
r.log(err.Error())
|
||||||
|
r.done = true
|
||||||
|
}
|
||||||
|
|
||||||
func decorate(s string) string {
|
func decorate(s string) string {
|
||||||
_, file, line, ok := runtime.Caller(3)
|
_, file, line, ok := runtime.Caller(3)
|
||||||
if ok {
|
if ok {
|
||||||
|
@ -120,6 +127,16 @@ func dedup(a []string) string {
|
||||||
func run(r Retryer, t Failer, f func(r *R)) {
|
func run(r Retryer, t Failer, f func(r *R)) {
|
||||||
t.Helper()
|
t.Helper()
|
||||||
rr := &R{}
|
rr := &R{}
|
||||||
|
|
||||||
|
fail := func() {
|
||||||
|
t.Helper()
|
||||||
|
out := dedup(rr.output)
|
||||||
|
if out != "" {
|
||||||
|
t.Log(out)
|
||||||
|
}
|
||||||
|
t.FailNow()
|
||||||
|
}
|
||||||
|
|
||||||
for r.Continue() {
|
for r.Continue() {
|
||||||
func() {
|
func() {
|
||||||
defer func() {
|
defer func() {
|
||||||
|
@ -129,17 +146,17 @@ func run(r Retryer, t Failer, f func(r *R)) {
|
||||||
}()
|
}()
|
||||||
f(rr)
|
f(rr)
|
||||||
}()
|
}()
|
||||||
if !rr.fail {
|
|
||||||
|
switch {
|
||||||
|
case rr.done:
|
||||||
|
fail()
|
||||||
|
return
|
||||||
|
case !rr.fail:
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
rr.fail = false
|
rr.fail = false
|
||||||
}
|
}
|
||||||
|
fail()
|
||||||
out := dedup(rr.output)
|
|
||||||
if out != "" {
|
|
||||||
t.Log(out)
|
|
||||||
}
|
|
||||||
t.FailNow()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// DefaultFailer provides default retry.Run() behavior for unit tests.
|
// DefaultFailer provides default retry.Run() behavior for unit tests.
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
package retry
|
package retry
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"fmt"
|
||||||
"testing"
|
"testing"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
@ -52,15 +53,35 @@ func TestRunWith(t *testing.T) {
|
||||||
require.Equal(t, 3, iter)
|
require.Equal(t, 3, iter)
|
||||||
require.Equal(t, 1, ft.fails)
|
require.Equal(t, 1, ft.fails)
|
||||||
})
|
})
|
||||||
|
|
||||||
|
t.Run("Stop ends the retrying", func(t *testing.T) {
|
||||||
|
ft := &fakeT{}
|
||||||
|
iter := 0
|
||||||
|
RunWith(&Counter{Count: 5, Wait: time.Millisecond}, ft, func(r *R) {
|
||||||
|
iter++
|
||||||
|
if iter == 2 {
|
||||||
|
r.Stop(fmt.Errorf("do not proceed"))
|
||||||
|
}
|
||||||
|
r.Fatalf("not yet")
|
||||||
|
})
|
||||||
|
|
||||||
|
require.Equal(t, 2, iter)
|
||||||
|
require.Equal(t, 1, ft.fails)
|
||||||
|
require.Len(t, ft.out, 1)
|
||||||
|
require.Contains(t, ft.out[0], "not yet\n")
|
||||||
|
require.Contains(t, ft.out[0], "do not proceed\n")
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
type fakeT struct {
|
type fakeT struct {
|
||||||
fails int
|
fails int
|
||||||
|
out []string
|
||||||
}
|
}
|
||||||
|
|
||||||
func (f *fakeT) Helper() {}
|
func (f *fakeT) Helper() {}
|
||||||
|
|
||||||
func (f *fakeT) Log(args ...interface{}) {
|
func (f *fakeT) Log(args ...interface{}) {
|
||||||
|
f.out = append(f.out, fmt.Sprint(args...))
|
||||||
}
|
}
|
||||||
|
|
||||||
func (f *fakeT) FailNow() {
|
func (f *fakeT) FailNow() {
|
||||||
|
|
Loading…
Reference in New Issue