From f3b35fae2450c4b299d5e18ad2aba2edfab45cd0 Mon Sep 17 00:00:00 2001 From: Tobias Krischer Date: Wed, 14 Sep 2022 01:08:17 +0200 Subject: [PATCH] add EventuallyWithT assertion --- assert/assertion_format.go | 34 +++++++++++++++ assert/assertion_forward.go | 68 +++++++++++++++++++++++++++++ assert/assertions.go | 86 +++++++++++++++++++++++++++++++++++++ assert/assertions_test.go | 32 ++++++++++++++ require/require.go | 74 +++++++++++++++++++++++++++++++ require/require_forward.go | 68 +++++++++++++++++++++++++++++ 6 files changed, 362 insertions(+) diff --git a/assert/assertion_format.go b/assert/assertion_format.go index 7880b8f94..de70fef88 100644 --- a/assert/assertion_format.go +++ b/assert/assertion_format.go @@ -155,6 +155,40 @@ func Eventuallyf(t TestingT, condition func() bool, waitFor time.Duration, tick return Eventually(t, condition, waitFor, tick, append([]interface{}{msg}, args...)...) } +// EventuallyWithTf asserts that given condition will be met in waitFor time, +// periodically checking target function each tick. In contrast to Eventually, +// it supplies a CollectT to the condition function, so that the condition +// function can use the CollectT to call other assertions. +// The supplied CollectT collects all errors from one tick (if there are any). +// If the condition is not met before waitFor, the collected errors of +// the last tick are copied to t. +// +// falseThenTrue := func(falses int) func() bool { +// count := 0 +// return func() bool { +// if count < falses { +// count++ +// return false +// } +// return true +// } +// } +// f := falseThenTrue(5) +// assert.EventuallyWithTf(t, func(mockT *assert.CollectT) (success bool, "error message %s", "formatted") { +// defer func() { +// r := recover() +// success = (r == nil) +// }() +// assert.True(mockT, f()) +// return +// }, 50*time.Millisecond, 10*time.Millisecond) +func EventuallyWithTf(t TestingT, condition func(collect *CollectT) bool, waitFor time.Duration, tick time.Duration, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return EventuallyWithT(t, condition, waitFor, tick, append([]interface{}{msg}, args...)...) +} + // Exactlyf asserts that two objects are equal in value and type. // // assert.Exactlyf(t, int32(123), int64(123), "error message %s", "formatted") diff --git a/assert/assertion_forward.go b/assert/assertion_forward.go index 339515b8b..8a9ed97fd 100644 --- a/assert/assertion_forward.go +++ b/assert/assertion_forward.go @@ -288,6 +288,74 @@ func (a *Assertions) Eventually(condition func() bool, waitFor time.Duration, ti return Eventually(a.t, condition, waitFor, tick, msgAndArgs...) } +// EventuallyWithT asserts that given condition will be met in waitFor time, +// periodically checking target function each tick. In contrast to Eventually, +// it supplies a CollectT to the condition function, so that the condition +// function can use the CollectT to call other assertions. +// The supplied CollectT collects all errors from one tick (if there are any). +// If the condition is not met before waitFor, the collected errors of +// the last tick are copied to t. +// +// falseThenTrue := func(falses int) func() bool { +// count := 0 +// return func() bool { +// if count < falses { +// count++ +// return false +// } +// return true +// } +// } +// f := falseThenTrue(5) +// a.EventuallyWithT(func(mockT *assert.CollectT) (success bool) { +// defer func() { +// r := recover() +// success = (r == nil) +// }() +// assert.True(mockT, f()) +// return +// }, 50*time.Millisecond, 10*time.Millisecond) +func (a *Assertions) EventuallyWithT(condition func(collect *CollectT) bool, waitFor time.Duration, tick time.Duration, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return EventuallyWithT(a.t, condition, waitFor, tick, msgAndArgs...) +} + +// EventuallyWithTf asserts that given condition will be met in waitFor time, +// periodically checking target function each tick. In contrast to Eventually, +// it supplies a CollectT to the condition function, so that the condition +// function can use the CollectT to call other assertions. +// The supplied CollectT collects all errors from one tick (if there are any). +// If the condition is not met before waitFor, the collected errors of +// the last tick are copied to t. +// +// falseThenTrue := func(falses int) func() bool { +// count := 0 +// return func() bool { +// if count < falses { +// count++ +// return false +// } +// return true +// } +// } +// f := falseThenTrue(5) +// a.EventuallyWithTf(func(mockT *assert.CollectT) (success bool, "error message %s", "formatted") { +// defer func() { +// r := recover() +// success = (r == nil) +// }() +// assert.True(mockT, f()) +// return +// }, 50*time.Millisecond, 10*time.Millisecond) +func (a *Assertions) EventuallyWithTf(condition func(collect *CollectT) bool, waitFor time.Duration, tick time.Duration, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return EventuallyWithTf(a.t, condition, waitFor, tick, msg, args...) +} + // Eventuallyf asserts that given condition will be met in waitFor time, // periodically checking target function each tick. // diff --git a/assert/assertions.go b/assert/assertions.go index fa1245b18..71c5c5a4d 100644 --- a/assert/assertions.go +++ b/assert/assertions.go @@ -1756,6 +1756,92 @@ func Eventually(t TestingT, condition func() bool, waitFor time.Duration, tick t } } +// CollectT implements the TestingT interface and collects all errors. +type CollectT struct { + errors []error +} + +// Errorf collects the error. +func (c *CollectT) Errorf(format string, args ...interface{}) { + c.errors = append(c.errors, fmt.Errorf(format, args...)) +} + +// FailNow panics. +func (c *CollectT) FailNow() { + panic("Assertion failed") +} + +// Reset clears the collected errors. +func (c *CollectT) Reset() { + c.errors = nil +} + +// Copy copies the collected errors to the supplied t. +func (c *CollectT) Copy(t TestingT) { + for _, err := range c.errors { + t.Errorf("%v", err) + } +} + +// EventuallyWithT asserts that given condition will be met in waitFor time, +// periodically checking target function each tick. In contrast to Eventually, +// it supplies a CollectT to the condition function, so that the condition +// function can use the CollectT to call other assertions. +// The supplied CollectT collects all errors from one tick (if there are any). +// If the condition is not met before waitFor, the collected errors of +// the last tick are copied to t. +// +// falseThenTrue := func(falses int) func() bool { +// count := 0 +// return func() bool { +// if count < falses { +// count++ +// return false +// } +// return true +// } +// } +// f := falseThenTrue(5) +// assert.EventuallyWithT(t, func(mockT *assert.CollectT) (success bool) { +// defer func() { +// r := recover() +// success = (r == nil) +// }() +// assert.True(mockT, f()) +// return +// }, 50*time.Millisecond, 10*time.Millisecond) +func EventuallyWithT(t TestingT, condition func(collect *CollectT) bool, waitFor time.Duration, tick time.Duration, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + + collect := new(CollectT) + ch := make(chan bool, 1) + + timer := time.NewTimer(waitFor) + defer timer.Stop() + + ticker := time.NewTicker(tick) + defer ticker.Stop() + + for tick := ticker.C; ; { + select { + case <-timer.C: + collect.Copy(t) + return Fail(t, "Condition never satisfied", msgAndArgs...) + case <-tick: + tick = nil + collect.Reset() + go func() { ch <- condition(collect) }() + case v := <-ch: + if v { + return true + } + tick = ticker.C + } + } +} + // Never asserts that the given condition doesn't satisfy in waitFor time, // periodically checking the target function each tick. // diff --git a/assert/assertions_test.go b/assert/assertions_test.go index bdab184c1..9c324f1c0 100644 --- a/assert/assertions_test.go +++ b/assert/assertions_test.go @@ -2428,6 +2428,38 @@ func TestEventuallyTrue(t *testing.T) { True(t, Eventually(t, condition, 100*time.Millisecond, 20*time.Millisecond)) } +func TestEventuallyWithTFalse(t *testing.T) { + mockT := new(CollectT) + + condition := func(collect *CollectT) bool { + return True(collect, false) + } + + False(t, EventuallyWithT(mockT, condition, 100*time.Millisecond, 20*time.Millisecond)) + Len(t, mockT.errors, 2) +} + +func TestEventuallyWithTTrue(t *testing.T) { + mockT := new(CollectT) + + state := 0 + condition := func(collect *CollectT) bool { + defer func() { + state += 1 + }() + + if state == 2 { + True(collect, true) + return true + } + + return True(collect, false) + } + + True(t, EventuallyWithT(mockT, condition, 100*time.Millisecond, 20*time.Millisecond)) + Len(t, mockT.errors, 0) +} + func TestNeverFalse(t *testing.T) { condition := func() bool { return false diff --git a/require/require.go b/require/require.go index 880853f5a..1b622802a 100644 --- a/require/require.go +++ b/require/require.go @@ -364,6 +364,80 @@ func Eventually(t TestingT, condition func() bool, waitFor time.Duration, tick t t.FailNow() } +// EventuallyWithT asserts that given condition will be met in waitFor time, +// periodically checking target function each tick. In contrast to Eventually, +// it supplies a CollectT to the condition function, so that the condition +// function can use the CollectT to call other assertions. +// The supplied CollectT collects all errors from one tick (if there are any). +// If the condition is not met before waitFor, the collected errors of +// the last tick are copied to t. +// +// falseThenTrue := func(falses int) func() bool { +// count := 0 +// return func() bool { +// if count < falses { +// count++ +// return false +// } +// return true +// } +// } +// f := falseThenTrue(5) +// assert.EventuallyWithT(t, func(mockT *assert.CollectT) (success bool) { +// defer func() { +// r := recover() +// success = (r == nil) +// }() +// assert.True(mockT, f()) +// return +// }, 50*time.Millisecond, 10*time.Millisecond) +func EventuallyWithT(t TestingT, condition func(collect *assert.CollectT) bool, waitFor time.Duration, tick time.Duration, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.EventuallyWithT(t, condition, waitFor, tick, msgAndArgs...) { + return + } + t.FailNow() +} + +// EventuallyWithTf asserts that given condition will be met in waitFor time, +// periodically checking target function each tick. In contrast to Eventually, +// it supplies a CollectT to the condition function, so that the condition +// function can use the CollectT to call other assertions. +// The supplied CollectT collects all errors from one tick (if there are any). +// If the condition is not met before waitFor, the collected errors of +// the last tick are copied to t. +// +// falseThenTrue := func(falses int) func() bool { +// count := 0 +// return func() bool { +// if count < falses { +// count++ +// return false +// } +// return true +// } +// } +// f := falseThenTrue(5) +// assert.EventuallyWithTf(t, func(mockT *assert.CollectT) (success bool, "error message %s", "formatted") { +// defer func() { +// r := recover() +// success = (r == nil) +// }() +// assert.True(mockT, f()) +// return +// }, 50*time.Millisecond, 10*time.Millisecond) +func EventuallyWithTf(t TestingT, condition func(collect *assert.CollectT) bool, waitFor time.Duration, tick time.Duration, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.EventuallyWithTf(t, condition, waitFor, tick, msg, args...) { + return + } + t.FailNow() +} + // Eventuallyf asserts that given condition will be met in waitFor time, // periodically checking target function each tick. // diff --git a/require/require_forward.go b/require/require_forward.go index 960bf6f2c..e9b2b01d1 100644 --- a/require/require_forward.go +++ b/require/require_forward.go @@ -289,6 +289,74 @@ func (a *Assertions) Eventually(condition func() bool, waitFor time.Duration, ti Eventually(a.t, condition, waitFor, tick, msgAndArgs...) } +// EventuallyWithT asserts that given condition will be met in waitFor time, +// periodically checking target function each tick. In contrast to Eventually, +// it supplies a CollectT to the condition function, so that the condition +// function can use the CollectT to call other assertions. +// The supplied CollectT collects all errors from one tick (if there are any). +// If the condition is not met before waitFor, the collected errors of +// the last tick are copied to t. +// +// falseThenTrue := func(falses int) func() bool { +// count := 0 +// return func() bool { +// if count < falses { +// count++ +// return false +// } +// return true +// } +// } +// f := falseThenTrue(5) +// a.EventuallyWithT(func(mockT *assert.CollectT) (success bool) { +// defer func() { +// r := recover() +// success = (r == nil) +// }() +// assert.True(mockT, f()) +// return +// }, 50*time.Millisecond, 10*time.Millisecond) +func (a *Assertions) EventuallyWithT(condition func(collect *assert.CollectT) bool, waitFor time.Duration, tick time.Duration, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + EventuallyWithT(a.t, condition, waitFor, tick, msgAndArgs...) +} + +// EventuallyWithTf asserts that given condition will be met in waitFor time, +// periodically checking target function each tick. In contrast to Eventually, +// it supplies a CollectT to the condition function, so that the condition +// function can use the CollectT to call other assertions. +// The supplied CollectT collects all errors from one tick (if there are any). +// If the condition is not met before waitFor, the collected errors of +// the last tick are copied to t. +// +// falseThenTrue := func(falses int) func() bool { +// count := 0 +// return func() bool { +// if count < falses { +// count++ +// return false +// } +// return true +// } +// } +// f := falseThenTrue(5) +// a.EventuallyWithTf(func(mockT *assert.CollectT) (success bool, "error message %s", "formatted") { +// defer func() { +// r := recover() +// success = (r == nil) +// }() +// assert.True(mockT, f()) +// return +// }, 50*time.Millisecond, 10*time.Millisecond) +func (a *Assertions) EventuallyWithTf(condition func(collect *assert.CollectT) bool, waitFor time.Duration, tick time.Duration, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + EventuallyWithTf(a.t, condition, waitFor, tick, msg, args...) +} + // Eventuallyf asserts that given condition will be met in waitFor time, // periodically checking target function each tick. //