mirror of
https://github.com/coder/coder.git
synced 2025-07-12 00:14:10 +00:00
chore: add usage information to clock library README (#13594)
Adds a Usage section to the README of the clock testing library.
This commit is contained in:
522
clock/README.md
522
clock/README.md
@ -5,6 +5,362 @@ A Go time testing library for writing deterministic unit tests
|
|||||||
_Note: Quartz is the name I'm targeting for the standalone open source project when we spin this
|
_Note: Quartz is the name I'm targeting for the standalone open source project when we spin this
|
||||||
out._
|
out._
|
||||||
|
|
||||||
|
Our high level goal is to write unit tests that
|
||||||
|
|
||||||
|
1. execute quickly
|
||||||
|
2. don't flake
|
||||||
|
3. are straightforward to write and understand
|
||||||
|
|
||||||
|
For tests to execute quickly without flakes, we want to focus on _determinism_: the test should run
|
||||||
|
the same each time, and it should be easy to force the system into a known state (no races) before
|
||||||
|
executing test assertions. `time.Sleep`, `runtime.Gosched()`, and
|
||||||
|
polling/[Eventually](https://pkg.go.dev/github.com/stretchr/testify/assert#Eventually) are all
|
||||||
|
symptoms of an inability to do this easily.
|
||||||
|
|
||||||
|
## Usage
|
||||||
|
|
||||||
|
### `Clock` interface
|
||||||
|
|
||||||
|
In your application code, maintain a reference to a `quartz.Clock` instance to start timers and
|
||||||
|
tickers, instead of the bare `time` standard library.
|
||||||
|
|
||||||
|
```go
|
||||||
|
import "github.com/coder/quartz"
|
||||||
|
|
||||||
|
type Component struct {
|
||||||
|
...
|
||||||
|
|
||||||
|
// for testing
|
||||||
|
clock quartz.Clock
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Whenever you would call into `time` to start a timer or ticker, call `Component`'s `clock` instead.
|
||||||
|
|
||||||
|
In production, set this clock to `quartz.NewReal()` to create a clock that just transparently passes
|
||||||
|
through to the standard `time` library.
|
||||||
|
|
||||||
|
### Mocking
|
||||||
|
|
||||||
|
In your tests, you can use a `*Mock` to control the tickers and timers your code under test gets.
|
||||||
|
|
||||||
|
```go
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
"github.com/coder/quartz"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestComponent(t *testing.T) {
|
||||||
|
mClock := quartz.NewMock(t)
|
||||||
|
comp := &Component{
|
||||||
|
...
|
||||||
|
clock: mClock,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
The `*Mock` clock starts at Jan 1, 2024, 00:00 UTC by default, but you can set any start time you'd like prior to your test.
|
||||||
|
|
||||||
|
```go
|
||||||
|
mClock := quartz.NewMock(t)
|
||||||
|
mClock.Set(time.Date(2021, 6, 18, 12, 0, 0, 0, time.UTC)) // June 18, 2021 @ 12pm UTC
|
||||||
|
```
|
||||||
|
|
||||||
|
#### Advancing the clock
|
||||||
|
|
||||||
|
Once you begin setting timers or tickers, you cannot change the time backward, only advance it
|
||||||
|
forward. You may continue to use `Set()`, but it is often easier and clearer to use `Advance()`.
|
||||||
|
|
||||||
|
For example, with a timer:
|
||||||
|
|
||||||
|
```go
|
||||||
|
fired := false
|
||||||
|
|
||||||
|
tmr := mClock.Afterfunc(time.Second, func() {
|
||||||
|
fired = true
|
||||||
|
})
|
||||||
|
mClock.Advance(time.Second)
|
||||||
|
```
|
||||||
|
|
||||||
|
When you call `Advance()` it immediately moves the clock forward the given amount, and triggers any
|
||||||
|
tickers or timers that are scheduled to happen at that time. Any triggered events happen on separate
|
||||||
|
goroutines, so _do not_ immediately assert the results:
|
||||||
|
|
||||||
|
```go
|
||||||
|
fired := false
|
||||||
|
|
||||||
|
tmr := mClock.Afterfunc(time.Second, func() {
|
||||||
|
fired = true
|
||||||
|
})
|
||||||
|
mClock.Advance(time.Second)
|
||||||
|
|
||||||
|
// RACE CONDITION, DO NOT DO THIS!
|
||||||
|
if !fired {
|
||||||
|
t.Fatal("didn't fire")
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
`Advance()` (and `Set()` for that matter) return an `AdvanceWaiter` object you can use to wait for
|
||||||
|
all triggered events to complete.
|
||||||
|
|
||||||
|
```go
|
||||||
|
fired := false
|
||||||
|
// set a test timeout so we don't wait the default `go test` timeout for a failure
|
||||||
|
ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second)
|
||||||
|
|
||||||
|
tmr := mClock.Afterfunc(time.Second, func() {
|
||||||
|
fired = true
|
||||||
|
})
|
||||||
|
|
||||||
|
w := mClock.Advance(time.Second)
|
||||||
|
err := w.Wait(ctx)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal("AfterFunc f never completed")
|
||||||
|
}
|
||||||
|
if !fired {
|
||||||
|
t.Fatal("didn't fire")
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
The construction of waiting for the triggered events and failing the test if they don't complete is
|
||||||
|
very common, so there is a shorthand:
|
||||||
|
|
||||||
|
```go
|
||||||
|
w := mClock.Advance(time.Second)
|
||||||
|
err := w.Wait(ctx)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal("AfterFunc f never completed")
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
is equivalent to:
|
||||||
|
|
||||||
|
```go
|
||||||
|
w := mClock.Advance(time.Second)
|
||||||
|
w.MustWait(ctx)
|
||||||
|
```
|
||||||
|
|
||||||
|
or even more briefly:
|
||||||
|
|
||||||
|
```go
|
||||||
|
mClock.Advance(time.Second).MustWait(ctx)
|
||||||
|
```
|
||||||
|
|
||||||
|
### Advance only to the next event
|
||||||
|
|
||||||
|
One important restriction on advancing the clock is that you may only advance forward to the next
|
||||||
|
timer or ticker event and no further. The following will result in a test failure:
|
||||||
|
|
||||||
|
```go
|
||||||
|
func TestAdvanceTooFar(t *testing.T) {
|
||||||
|
ctx, cancel := context.WithTimeout(10*time.Second)
|
||||||
|
defer cancel()
|
||||||
|
mClock := quartz.NewMock(t)
|
||||||
|
var firedAt time.Time
|
||||||
|
mClock.AfterFunc(time.Second, func() {
|
||||||
|
firedAt := mClock.Now()
|
||||||
|
})
|
||||||
|
mClock.Advance(2*time.Second).MustWait(ctx)
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
This is a deliberate design decision to allow `Advance()` to immediately and synchronously move the
|
||||||
|
clock forward (even without calling `Wait()` on returned waiter). This helps meet Quartz's design
|
||||||
|
goals of writing deterministic and easy to understand unit tests. It also allows the clock to be
|
||||||
|
advanced, deterministically _during_ the execution of a tick or timer function, as explained in the
|
||||||
|
next sections on Traps.
|
||||||
|
|
||||||
|
Advancing multiple events can be accomplished via looping. E.g. if you have a 1-second ticker
|
||||||
|
|
||||||
|
```go
|
||||||
|
for i := 0; i < 10; i++ {
|
||||||
|
mClock.Advance(time.Second).MustWait(ctx)
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
will advance 10 ticks.
|
||||||
|
|
||||||
|
If you don't know or don't want to compute the time to the next event, you can use `AdvanceNext()`.
|
||||||
|
|
||||||
|
```go
|
||||||
|
d, w := mClock.AdvanceNext()
|
||||||
|
w.MustWait(ctx)
|
||||||
|
// d contains the duration we advanced
|
||||||
|
```
|
||||||
|
|
||||||
|
### Traps
|
||||||
|
|
||||||
|
A trap allows you to match specific calls into the library while mocking, block their return,
|
||||||
|
inspect their arguments, then release them to allow them to return. They help you write
|
||||||
|
deterministic unit tests even when the code under test executes asynchronously from the test.
|
||||||
|
|
||||||
|
You set your traps prior to executing code under test, and then wait for them to be triggered.
|
||||||
|
|
||||||
|
```go
|
||||||
|
func TestTrap(t *testing.T) {
|
||||||
|
ctx, cancel := context.WithTimeout(10*time.Second)
|
||||||
|
defer cancel()
|
||||||
|
mClock := quartz.NewMock(t)
|
||||||
|
trap := mClock.Trap().AfterFunc()
|
||||||
|
defer trap.Close() // stop trapping AfterFunc calls
|
||||||
|
|
||||||
|
count := 0
|
||||||
|
go mClock.AfterFunc(time.Hour, func(){
|
||||||
|
count++
|
||||||
|
})
|
||||||
|
call := trap.MustWait(ctx)
|
||||||
|
call.Release()
|
||||||
|
if call.Duration != time.Hour {
|
||||||
|
t.Fatal("wrong duration")
|
||||||
|
}
|
||||||
|
|
||||||
|
// Now that the async call to AfterFunc has occurred, we can advance the clock to trigger it
|
||||||
|
mClock.Advance(call.Duration).MustWait(ctx)
|
||||||
|
if count != 1 {
|
||||||
|
t.Fatal("wrong count")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
In this test, the trap serves 2 purposes. Firstly, it allows us to capture and assert the duration
|
||||||
|
passed to the `AfterFunc` call. Secondly, it prevents a race between setting the timer and advancing
|
||||||
|
it. Since these things happen on different goroutines, if `Advance()` completes before
|
||||||
|
`AfterFunc()` is called, then the timer never pops in this test.
|
||||||
|
|
||||||
|
Any untrapped calls immediately complete using the current time, and calling `Close()` on a trap
|
||||||
|
causes the mock clock to stop trapping those calls.
|
||||||
|
|
||||||
|
You may also `Advance()` the clock between trapping a call and releasing it. The call uses the
|
||||||
|
current (mocked) time at the moment it is released.
|
||||||
|
|
||||||
|
```go
|
||||||
|
func TestTrap2(t *testing.T) {
|
||||||
|
ctx, cancel := context.WithTimeout(10*time.Second)
|
||||||
|
defer cancel()
|
||||||
|
mClock := quartz.NewMock(t)
|
||||||
|
trap := mClock.Trap().Now()
|
||||||
|
defer trap.Close() // stop trapping AfterFunc calls
|
||||||
|
|
||||||
|
var logs []string
|
||||||
|
done := make(chan struct{})
|
||||||
|
go func(clk quartz.Clock){
|
||||||
|
defer close(done)
|
||||||
|
start := clk.Now()
|
||||||
|
phase1()
|
||||||
|
p1end := clk.Now()
|
||||||
|
logs = append(fmt.Sprintf("Phase 1 took %s", p1end.Sub(start).String()))
|
||||||
|
phase2()
|
||||||
|
p2end := clk.Now()
|
||||||
|
logs = append(fmt.Sprintf("Phase 2 took %s", p2end.Sub(p1end).String()))
|
||||||
|
}(mClock)
|
||||||
|
|
||||||
|
// start
|
||||||
|
trap.MustWait(ctx).Release()
|
||||||
|
// phase 1
|
||||||
|
call := trap.MustWait(ctx)
|
||||||
|
mClock.Advance(3*time.Second).MustWait(ctx)
|
||||||
|
call.Release()
|
||||||
|
// phase 2
|
||||||
|
call = trap.MustWait(ctx)
|
||||||
|
mClock.Advance(5*time.Second).MustWait(ctx)
|
||||||
|
call.Release()
|
||||||
|
|
||||||
|
<-done
|
||||||
|
// Now logs contains []string{"Phase 1 took 3s", "Phase 2 took 5s"}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### Tags
|
||||||
|
|
||||||
|
When multiple goroutines in the code under test call into the Clock, you can use `tags` to
|
||||||
|
distinguish them in your traps.
|
||||||
|
|
||||||
|
```go
|
||||||
|
trap := mClock.Trap.Now("foo") // traps any calls that contain "foo"
|
||||||
|
defer trap.Close()
|
||||||
|
|
||||||
|
foo := make(chan time.Time)
|
||||||
|
go func(){
|
||||||
|
foo <- mClock.Now("foo", "bar")
|
||||||
|
}()
|
||||||
|
baz := make(chan time.Time)
|
||||||
|
go func(){
|
||||||
|
baz <- mClock.Now("baz")
|
||||||
|
}()
|
||||||
|
call := trap.MustWait(ctx)
|
||||||
|
mClock.Advance(time.Second).MustWait(ctx)
|
||||||
|
call.Release()
|
||||||
|
// call.Tags contains []string{"foo", "bar"}
|
||||||
|
|
||||||
|
gotFoo := <-foo // 1s after start
|
||||||
|
gotBaz := <-baz // ?? never trapped, so races with Advance()
|
||||||
|
```
|
||||||
|
|
||||||
|
Tags appear as an optional suffix on all `Clock` methods (type `...string`) and are ignored entirely
|
||||||
|
by the real clock. They also appear on all methods on returned timers and tickers.
|
||||||
|
|
||||||
|
## Recommended Patterns
|
||||||
|
|
||||||
|
### Options
|
||||||
|
|
||||||
|
We use the Option pattern to inject the mock clock for testing, keeping the call signature in
|
||||||
|
production clean. The option pattern is compatible with other optional fields as well.
|
||||||
|
|
||||||
|
```go
|
||||||
|
type Option func(*Thing)
|
||||||
|
|
||||||
|
// WithTestClock is used in tests to inject a mock Clock
|
||||||
|
func WithTestClock(clk quartz.Clock) Option {
|
||||||
|
return func(t *Thing) {
|
||||||
|
t.clock = clk
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewThing(<required args>, opts ...Option) *Thing {
|
||||||
|
t := &Thing{
|
||||||
|
...
|
||||||
|
clock: quartz.NewReal()
|
||||||
|
}
|
||||||
|
for _, o := range opts {
|
||||||
|
o(t)
|
||||||
|
}
|
||||||
|
return t
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
In tests, this becomes
|
||||||
|
|
||||||
|
```go
|
||||||
|
func TestThing(t *testing.T) {
|
||||||
|
mClock := quartz.NewMock(t)
|
||||||
|
thing := NewThing(<required args>, WithTestClock(mClock))
|
||||||
|
...
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### Tagging convention
|
||||||
|
|
||||||
|
Tag your `Clock` method calls as:
|
||||||
|
|
||||||
|
```go
|
||||||
|
func (c *Component) Method() {
|
||||||
|
now := c.clock.Now("Component", "Method")
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
or
|
||||||
|
|
||||||
|
```go
|
||||||
|
func (c *Component) Method() {
|
||||||
|
start := c.clock.Now("Component", "Method", "start")
|
||||||
|
...
|
||||||
|
end := c.clock.Now("Component", "Method", "end")
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
This makes it much less likely that code changes that introduce new components or methods will spoil
|
||||||
|
existing unit tests.
|
||||||
|
|
||||||
## Why another time testing library?
|
## Why another time testing library?
|
||||||
|
|
||||||
Writing good unit tests for components and functions that use the `time` package is difficult, even
|
Writing good unit tests for components and functions that use the `time` package is difficult, even
|
||||||
@ -18,7 +374,7 @@ Quartz shares the high level design of a `Clock` interface that closely resemble
|
|||||||
the `time` standard library, and a "real" clock passes thru to the standard library in production,
|
the `time` standard library, and a "real" clock passes thru to the standard library in production,
|
||||||
while a mock clock gives precise control in testing.
|
while a mock clock gives precise control in testing.
|
||||||
|
|
||||||
Our high level goal is to write unit tests that
|
As mentioned in our introduction, our high level goal is to write unit tests that
|
||||||
|
|
||||||
1. execute quickly
|
1. execute quickly
|
||||||
2. don't flake
|
2. don't flake
|
||||||
@ -27,12 +383,6 @@ Our high level goal is to write unit tests that
|
|||||||
For several reasons, this is a tall order when it comes to code that depends on time, and we found
|
For several reasons, this is a tall order when it comes to code that depends on time, and we found
|
||||||
the existing libraries insufficient for our goals.
|
the existing libraries insufficient for our goals.
|
||||||
|
|
||||||
For tests to execute quickly without flakes, we want to focus on _determinism_: the test should run
|
|
||||||
the same each time, and it should be easy to force the system into a known state (no races) before
|
|
||||||
executing test assertions. `time.Sleep`, `runtime.Gosched()`, and
|
|
||||||
polling/[Eventually](https://pkg.go.dev/github.com/stretchr/testify/assert#Eventually) are all
|
|
||||||
symptoms of an inability to do this easily.
|
|
||||||
|
|
||||||
### Preventing test flakes
|
### Preventing test flakes
|
||||||
|
|
||||||
The following example comes from the README from benbjohnson/clock:
|
The following example comes from the README from benbjohnson/clock:
|
||||||
@ -43,11 +393,11 @@ count := 0
|
|||||||
|
|
||||||
// Kick off a timer to increment every 1 mock second.
|
// Kick off a timer to increment every 1 mock second.
|
||||||
go func() {
|
go func() {
|
||||||
ticker := mock.Ticker(1 * time.Second)
|
ticker := mock.Ticker(1 * time.Second)
|
||||||
for {
|
for {
|
||||||
<-ticker.C
|
<-ticker.C
|
||||||
count++
|
count++
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
runtime.Gosched()
|
runtime.Gosched()
|
||||||
|
|
||||||
@ -76,15 +426,15 @@ with ticks in one and context expiring in another, i.e.
|
|||||||
```go
|
```go
|
||||||
t := time.NewTicker(duration)
|
t := time.NewTicker(duration)
|
||||||
for {
|
for {
|
||||||
select {
|
select {
|
||||||
case <-ctx.Done():
|
case <-ctx.Done():
|
||||||
return ctx.Err()
|
return ctx.Err()
|
||||||
case <-t.C:
|
case <-t.C:
|
||||||
err := do()
|
err := do()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
@ -111,16 +461,16 @@ for calls that access the clock.
|
|||||||
|
|
||||||
```go
|
```go
|
||||||
func TestTicker(t *testing.T) {
|
func TestTicker(t *testing.T) {
|
||||||
mClock := quartz.NewMock(t)
|
mClock := quartz.NewMock(t)
|
||||||
trap := mClock.Trap().TickerFunc()
|
trap := mClock.Trap().TickerFunc()
|
||||||
defer trap.Close() // stop trapping at end
|
defer trap.Close() // stop trapping at end
|
||||||
go runMyTicker(mClock) // async calls TickerFunc()
|
go runMyTicker(mClock) // async calls TickerFunc()
|
||||||
call := trap.Wait(context.Background()) // waits for a call and blocks its return
|
call := trap.Wait(context.Background()) // waits for a call and blocks its return
|
||||||
call.Release() // allow the TickerFunc() call to return
|
call.Release() // allow the TickerFunc() call to return
|
||||||
// optionally check the duration using call.Duration
|
// optionally check the duration using call.Duration
|
||||||
// Move the clock forward 1 tick
|
// Move the clock forward 1 tick
|
||||||
mClock.Advance(time.Second).MustWait(context.Background())
|
mClock.Advance(time.Second).MustWait(context.Background())
|
||||||
// assert results of the tick
|
// assert results of the tick
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
@ -139,9 +489,9 @@ A very basic example is measuring how long something took:
|
|||||||
```go
|
```go
|
||||||
var measurement time.Duration
|
var measurement time.Duration
|
||||||
go func(clock quartz.Clock) {
|
go func(clock quartz.Clock) {
|
||||||
start := clock.Now()
|
start := clock.Now()
|
||||||
doSomething()
|
doSomething()
|
||||||
measurement = clock.Since(start)
|
measurement = clock.Since(start)
|
||||||
}(mClock)
|
}(mClock)
|
||||||
|
|
||||||
// how to get measurement to be, say, 5 seconds?
|
// how to get measurement to be, say, 5 seconds?
|
||||||
@ -159,9 +509,9 @@ control the time each call sees.
|
|||||||
trap := mClock.Trap().Since()
|
trap := mClock.Trap().Since()
|
||||||
var measurement time.Duration
|
var measurement time.Duration
|
||||||
go func(clock quartz.Clock) {
|
go func(clock quartz.Clock) {
|
||||||
start := clock.Now()
|
start := clock.Now()
|
||||||
doSomething()
|
doSomething()
|
||||||
measurement = clock.Since(start)
|
measurement = clock.Since(start)
|
||||||
}(mClock)
|
}(mClock)
|
||||||
|
|
||||||
c := trap.Wait(ctx)
|
c := trap.Wait(ctx)
|
||||||
@ -179,25 +529,25 @@ no activity recorded for some period, say 10 minutes in the following example:
|
|||||||
|
|
||||||
```go
|
```go
|
||||||
type InactivityTimer struct {
|
type InactivityTimer struct {
|
||||||
mu sync.Mutex
|
mu sync.Mutex
|
||||||
activity time.Time
|
activity time.Time
|
||||||
clock quartz.Clock
|
clock quartz.Clock
|
||||||
}
|
}
|
||||||
|
|
||||||
func (i *InactivityTimer) Start() {
|
func (i *InactivityTimer) Start() {
|
||||||
i.mu.Lock()
|
i.mu.Lock()
|
||||||
defer i.mu.Unlock()
|
defer i.mu.Unlock()
|
||||||
next := i.clock.Until(i.activity.Add(10*time.Minute))
|
next := i.clock.Until(i.activity.Add(10*time.Minute))
|
||||||
t := i.clock.AfterFunc(next, func() {
|
t := i.clock.AfterFunc(next, func() {
|
||||||
i.mu.Lock()
|
i.mu.Lock()
|
||||||
defer i.mu.Unlock()
|
defer i.mu.Unlock()
|
||||||
next := i.clock.Until(i.activity.Add(10*time.Minute))
|
next := i.clock.Until(i.activity.Add(10*time.Minute))
|
||||||
if next == 0 {
|
if next == 0 {
|
||||||
i.timeoutLocked()
|
i.timeoutLocked()
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
t.Reset(next)
|
t.Reset(next)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
@ -217,19 +567,19 @@ initial one, so to make testing easier we can "tag" the call we want. Like this:
|
|||||||
|
|
||||||
```go
|
```go
|
||||||
func (i *InactivityTimer) Start() {
|
func (i *InactivityTimer) Start() {
|
||||||
i.mu.Lock()
|
i.mu.Lock()
|
||||||
defer i.mu.Unlock()
|
defer i.mu.Unlock()
|
||||||
next := i.clock.Until(i.activity.Add(10*time.Minute))
|
next := i.clock.Until(i.activity.Add(10*time.Minute))
|
||||||
t := i.clock.AfterFunc(next, func() {
|
t := i.clock.AfterFunc(next, func() {
|
||||||
i.mu.Lock()
|
i.mu.Lock()
|
||||||
defer i.mu.Unlock()
|
defer i.mu.Unlock()
|
||||||
next := i.clock.Until(i.activity.Add(10*time.Minute), "inner")
|
next := i.clock.Until(i.activity.Add(10*time.Minute), "inner")
|
||||||
if next == 0 {
|
if next == 0 {
|
||||||
i.timeoutLocked()
|
i.timeoutLocked()
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
t.Reset(next)
|
t.Reset(next)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
@ -238,29 +588,29 @@ string tags that allow traps to match on them.
|
|||||||
|
|
||||||
```go
|
```go
|
||||||
func TestInactivityTimer_Late(t *testing.T) {
|
func TestInactivityTimer_Late(t *testing.T) {
|
||||||
// set a timeout on the test itself, so that if Wait functions get blocked, we don't have to
|
// set a timeout on the test itself, so that if Wait functions get blocked, we don't have to
|
||||||
// wait for the default test timeout of 10 minutes.
|
// wait for the default test timeout of 10 minutes.
|
||||||
ctx, cancel := context.WithTimeout(10*time.Second)
|
ctx, cancel := context.WithTimeout(10*time.Second)
|
||||||
defer cancel()
|
defer cancel()
|
||||||
mClock := quartz.NewMock(t)
|
mClock := quartz.NewMock(t)
|
||||||
trap := mClock.Trap.Until("inner")
|
trap := mClock.Trap.Until("inner")
|
||||||
defer trap.Close()
|
defer trap.Close()
|
||||||
|
|
||||||
it := &InactivityTimer{
|
it := &InactivityTimer{
|
||||||
activity: mClock.Now(),
|
activity: mClock.Now(),
|
||||||
clock: mClock,
|
clock: mClock,
|
||||||
}
|
}
|
||||||
it.Start()
|
it.Start()
|
||||||
|
|
||||||
// Trigger the AfterFunc
|
// Trigger the AfterFunc
|
||||||
w := mClock.Advance(10*time.Minute)
|
w := mClock.Advance(10*time.Minute)
|
||||||
c := trap.Wait(ctx)
|
c := trap.Wait(ctx)
|
||||||
// Advance the clock a few ms to simulate a busy system
|
// Advance the clock a few ms to simulate a busy system
|
||||||
mClock.Advance(3*time.Millisecond)
|
mClock.Advance(3*time.Millisecond)
|
||||||
c.Release() // Until() returns
|
c.Release() // Until() returns
|
||||||
w.MustWait(ctx) // Wait for the AfterFunc to wrap up
|
w.MustWait(ctx) // Wait for the AfterFunc to wrap up
|
||||||
|
|
||||||
// Assert that the timeoutLocked() function was called
|
// Assert that the timeoutLocked() function was called
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user