Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Manage templates per session instead of globally #386

Merged
merged 2 commits into from
Dec 10, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
87 changes: 77 additions & 10 deletions pkg/collector/clock.go
Original file line number Diff line number Diff line change
Expand Up @@ -20,18 +20,28 @@ import (
)

// timer allows for injecting fake or real timers into code that needs to do arbitrary things based
// on time. We do not include the C() method, as we only support timers created with AfterFunc.
// on time.
type timer interface {
C() <-chan time.Time
Stop() bool
Reset(d time.Duration) bool
}

type realTimer struct {
*time.Timer
}

func (t *realTimer) C() <-chan time.Time {
return t.Timer.C
}

// clock allows for injecting fake or real clocks into code that needs to do arbitrary things based
// on time. We only support a very limited interface at the moment, with only the methods required
// by CollectingProcess.
type clock interface {
Now() time.Time
AfterFunc(d time.Duration, f func()) timer
NewTimer(d time.Duration) timer
}

// realClock implements the clock interface using functions from the time package.
Expand All @@ -42,21 +52,35 @@ func (realClock) Now() time.Time {
}

func (realClock) AfterFunc(d time.Duration, f func()) timer {
return time.AfterFunc(d, f)
return &realTimer{
Timer: time.AfterFunc(d, f),
}
}

func (realClock) NewTimer(d time.Duration) timer {
return &realTimer{
Timer: time.NewTimer(d),
}
}

type fakeTimer struct {
targetTime time.Time
f func()
ch chan time.Time
clock *fakeClock
}

func (t *fakeTimer) C() <-chan time.Time {
return t.ch
}

func (t *fakeTimer) Stop() bool {
clock := t.clock
clock.m.Lock()
defer clock.m.Unlock()
newTimers := make([]*fakeTimer, 0, len(clock.timers))
fired := true
t.targetTime = time.Time{}
for i := range clock.timers {
if clock.timers[i] != t {
newTimers = append(newTimers, t)
Expand All @@ -70,6 +94,9 @@ func (t *fakeTimer) Stop() bool {
}

func (t *fakeTimer) Reset(d time.Duration) bool {
if d <= 0 {
yuntanghsu marked this conversation as resolved.
Show resolved Hide resolved
panic("negative duration not supported")
}
clock := t.clock
clock.m.Lock()
defer clock.m.Unlock()
Expand Down Expand Up @@ -124,11 +151,28 @@ func (c *fakeClock) AfterFunc(d time.Duration, f func()) timer {
return t
}

func (c *fakeClock) NewTimer(d time.Duration) timer {
if d <= 0 {
yuntanghsu marked this conversation as resolved.
Show resolved Hide resolved
panic("negative duration not supported")
}
c.m.Lock()
defer c.m.Unlock()
t := &fakeTimer{
targetTime: c.now.Add(d),
// The channel is synchronous (unbuffered, capacity 0), as per Go documentation for
// time package.
ch: make(chan time.Time),
clock: c,
}
c.timers = append(c.timers, t)
return t
}

func (c *fakeClock) Step(d time.Duration) {
if d < 0 {
yuntanghsu marked this conversation as resolved.
Show resolved Hide resolved
panic("invalid duration")
}
timerFuncs := []func(){}
expiredTimers := []*fakeTimer{}
func() {
c.m.Lock()
defer c.m.Unlock()
Expand All @@ -140,18 +184,41 @@ func (c *fakeClock) Step(d time.Duration) {
// Collect timer functions to run and remove them from list.
newTimers := make([]*fakeTimer, 0, len(c.timers))
for _, t := range c.timers {
if !t.targetTime.After(c.now) {
timerFuncs = append(timerFuncs, t.f)
} else {
if t.targetTime.After(c.now) {
newTimers = append(newTimers, t)
} else {
expiredTimers = append(expiredTimers, t)
}
}
c.timers = newTimers
}()
// Run the timer functions, without holding a lock. This allows these functions to call
// clock.Now(), but also timer.Stop().
for _, f := range timerFuncs {
f()
for _, t := range expiredTimers {
if t.f != nil {
// Run the timer function, without holding a lock. This allows these
// functions to call clock.Now(), but also timer.Stop().
t.f()
} else {
retry := func() bool {
if !c.m.TryRLock() {
return true
}
defer c.m.RUnlock()
// If timer has been stopped or reset, do not fire.
if t.targetTime.IsZero() || t.targetTime.After(c.now) {
return false
}
select {
case t.ch <- c.now:
yuntanghsu marked this conversation as resolved.
Show resolved Hide resolved
return false
default:
}
return true
}
// Spin until we can acquire a read lock and write to the C channel: this
yuntanghsu marked this conversation as resolved.
Show resolved Hide resolved
// accounts for concurrent calls to Reset / Stop.
for retry() {
}
}
}
c.m.Lock()
defer c.m.Unlock()
Expand Down
64 changes: 64 additions & 0 deletions pkg/collector/clock_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -60,3 +60,67 @@ func TestFakeAfterFunc(t *testing.T) {
t.Fatalf("timer didn't fire")
}
}

func TestFakeNewTimer(t *testing.T) {
start := time.Now()
f := newFakeClock(start)
ch := make(chan time.Time, 1)
timer := f.NewTimer(1 * time.Second)
stopCh := make(chan struct{})
defer close(stopCh)
go func() {
for {
select {
case <-stopCh:
return
case t := <-timer.C():
ch <- t
}
}
}()
// After 1s, the timer should fire.
f.Step(1 * time.Second)
select {
case v := <-ch:
assert.Equal(t, start.Add(1*time.Second), v)
case <-time.After(100 * time.Millisecond):
t.Fatalf("timer didn't fire")
}
assert.False(t, timer.Stop(), "Stop should return false as timer has already been expired")
// After resetting the timer, it should fire again after another 1s.
assert.False(t, timer.Reset(1*time.Second), "Reset should return false as timer had expired")
f.Step(1 * time.Second)
select {
case v := <-ch:
assert.Equal(t, start.Add(2*time.Second), v)
case <-time.After(100 * time.Millisecond):
t.Fatalf("timer didn't fire")
}

assert.False(t, timer.Reset(1*time.Second), "Reset should return false as timer had expired")
assert.True(t, timer.Stop(), "Stop should return true as call stops the timer")
assert.False(t, timer.Stop(), "Stop should return false as timer has already been stopped")
assert.False(t, timer.Reset(1*time.Second), "Reset should return false as timer had been stopped")

// The timer should not fire until the target time is reached.
f.Step(999 * time.Millisecond)
select {
case <-ch:
t.Fatalf("timer should not have fired")
case <-time.After(100 * time.Millisecond):
}
assert.True(t, timer.Reset(1*time.Second), "Reset should return true as timer had been active")
f.Step(1 * time.Millisecond)
select {
case <-ch:
t.Fatalf("timer should not have fired")
case <-time.After(100 * time.Millisecond):
}
f.Step(999 * time.Millisecond)
select {
case v := <-ch:
assert.Equal(t, start.Add(3999*time.Millisecond), v)
case <-time.After(100 * time.Millisecond):
t.Fatalf("timer didn't fire")
}
}
Loading
Loading