Skip to content

Commit 5e6e95d

Browse files
committed
feat: add assert.Consistentlyf
This changeset adds the `assert.Consistently` and it's associated functions to assert that a condition is true over the entire period of `waitFor`. This is useful when testing the behavior of asynchronous functions. Closes #1087
1 parent 1b4fca7 commit 5e6e95d

6 files changed

+400
-0
lines changed

assert/assertion_format.go

+36
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

assert/assertion_forward.go

+72
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

assert/assertions.go

+87
Original file line numberDiff line numberDiff line change
@@ -2038,6 +2038,93 @@ func EventuallyWithT(t TestingT, condition func(collect *CollectT), waitFor time
20382038
}
20392039
}
20402040

2041+
// Consistently asserts that given condition will be met for the entire
2042+
// duration of waitFor time, periodically checking target function each tick.
2043+
//
2044+
// assert.Consistently(t, func() bool { return true; }, time.Second, 10*time.Millisecond)
2045+
func Consistently(t TestingT, condition func() bool, waitFor time.Duration, tick time.Duration, msgAndArgs ...interface{}) bool {
2046+
if h, ok := t.(tHelper); ok {
2047+
h.Helper()
2048+
}
2049+
2050+
ch := make(chan bool, 1)
2051+
2052+
timer := time.NewTimer(waitFor)
2053+
defer timer.Stop()
2054+
2055+
ticker := time.NewTicker(tick)
2056+
defer ticker.Stop()
2057+
2058+
for tick := ticker.C; ; {
2059+
select {
2060+
case <-timer.C:
2061+
return true
2062+
case <-tick:
2063+
tick = nil
2064+
go func() { ch <- condition() }()
2065+
case v := <-ch:
2066+
if !v {
2067+
return Fail(t, "Condition never satisfied", msgAndArgs...)
2068+
}
2069+
tick = ticker.C
2070+
}
2071+
}
2072+
}
2073+
2074+
// ConsistentlyWithT asserts that given condition will be met for the entire
2075+
// waitFor time, periodically checking target function each tick. In contrast
2076+
// to Consistently, it supplies a CollectT to the condition function, so that
2077+
// the condition function can use the CollectT to call other assertions. The
2078+
// condition is considered "met" if no errors are raised across all ticks. The
2079+
// supplied CollectT collects all errors from one tick (if there are any). If
2080+
// the condition is not met once before waitFor, the collected error of the
2081+
// failing tick are copied to t.
2082+
//
2083+
// externalValue := false
2084+
// go func() {
2085+
// time.Sleep(8*time.Second)
2086+
// externalValue = true
2087+
// }()
2088+
// assert.ConsistentlyWithT(t, func(c *assert.CollectT) {
2089+
// // add assertions as needed; any assertion failure will fail the current tick
2090+
// assert.True(c, externalValue, "expected 'externalValue' to be true")
2091+
// }, 10*time.Second, 1*time.Second, "external state has not changed to 'true'; still false")
2092+
func ConsistentlyWithT(t TestingT, condition func(collect *CollectT), waitFor time.Duration, tick time.Duration, msgAndArgs ...interface{}) bool {
2093+
if h, ok := t.(tHelper); ok {
2094+
h.Helper()
2095+
}
2096+
2097+
ch := make(chan []error, 1)
2098+
2099+
timer := time.NewTimer(waitFor)
2100+
defer timer.Stop()
2101+
2102+
ticker := time.NewTicker(tick)
2103+
defer ticker.Stop()
2104+
2105+
for tick := ticker.C; ; {
2106+
select {
2107+
case <-timer.C:
2108+
return true
2109+
case <-tick:
2110+
tick = nil
2111+
go func() {
2112+
collect := new(CollectT)
2113+
defer func() {
2114+
ch <- collect.errors
2115+
}()
2116+
condition(collect)
2117+
}()
2118+
case errs := <-ch:
2119+
if len(errs) > 0 {
2120+
return Fail(t, "Condition never satisfied", msgAndArgs...)
2121+
}
2122+
2123+
tick = ticker.C
2124+
}
2125+
}
2126+
}
2127+
20412128
// Never asserts that the given condition doesn't satisfy in waitFor time,
20422129
// periodically checking the target function each tick.
20432130
//

assert/assertions_test.go

+49
Original file line numberDiff line numberDiff line change
@@ -2898,6 +2898,29 @@ func TestEventuallyTrue(t *testing.T) {
28982898
True(t, Eventually(t, condition, 100*time.Millisecond, 20*time.Millisecond))
28992899
}
29002900

2901+
func TestConsistentlyTrue(t *testing.T) {
2902+
condition := func() bool {
2903+
return true
2904+
}
2905+
2906+
True(t, Consistently(t, condition, 100*time.Millisecond, 20*time.Millisecond))
2907+
}
2908+
2909+
func TestConsistentlyFalse(t *testing.T) {
2910+
mockT := new(testing.T)
2911+
2912+
state := 0
2913+
condition := func() bool {
2914+
defer func() {
2915+
state += 1
2916+
}()
2917+
2918+
return state != 2
2919+
}
2920+
2921+
False(t, Consistently(mockT, condition, 100*time.Millisecond, 20*time.Millisecond))
2922+
}
2923+
29012924
// errorsCapturingT is a mock implementation of TestingT that captures errors reported with Errorf.
29022925
type errorsCapturingT struct {
29032926
errors []error
@@ -2970,6 +2993,32 @@ func TestEventuallyWithT_ReturnsTheLatestFinishedConditionErrors(t *testing.T) {
29702993
Len(t, mockT.errors, 2)
29712994
}
29722995

2996+
func TestConsistentlyWithTTrue(t *testing.T) {
2997+
mockT := new(errorsCapturingT)
2998+
2999+
condition := func(collect *CollectT) {
3000+
True(collect, true)
3001+
}
3002+
3003+
True(t, ConsistentlyWithT(mockT, condition, 100*time.Millisecond, 20*time.Millisecond))
3004+
Len(t, mockT.errors, 0)
3005+
}
3006+
3007+
func TestConsistentlyWithTFalse(t *testing.T) {
3008+
mockT := new(errorsCapturingT)
3009+
3010+
state := 0
3011+
condition := func(collect *CollectT) {
3012+
defer func() {
3013+
state += 1
3014+
}()
3015+
False(collect, state == 2)
3016+
}
3017+
3018+
False(t, ConsistentlyWithT(mockT, condition, 100*time.Millisecond, 20*time.Millisecond))
3019+
Len(t, mockT.errors, 1)
3020+
}
3021+
29733022
func TestNeverFalse(t *testing.T) {
29743023
condition := func() bool {
29753024
return false

require/require.go

+84
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

0 commit comments

Comments
 (0)