|
|
|
package signal
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"time"
|
|
|
|
)
|
|
|
|
|
|
|
|
type ActivityUpdater interface {
|
|
|
|
Update()
|
|
|
|
}
|
|
|
|
|
|
|
|
type ActivityTimer struct {
|
|
|
|
updated chan struct{}
|
|
|
|
timeout chan time.Duration
|
|
|
|
closing chan struct{}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (t *ActivityTimer) Update() {
|
|
|
|
select {
|
|
|
|
case t.updated <- struct{}{}:
|
|
|
|
default:
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (t *ActivityTimer) SetTimeout(timeout time.Duration) {
|
|
|
|
select {
|
|
|
|
case <-t.closing:
|
|
|
|
case t.timeout <- timeout:
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (t *ActivityTimer) run(ctx context.Context, cancel context.CancelFunc) {
|
|
|
|
defer func() {
|
|
|
|
cancel()
|
|
|
|
close(t.closing)
|
|
|
|
}()
|
|
|
|
|
|
|
|
timeout := <-t.timeout
|
|
|
|
if timeout == 0 {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
ticker := time.NewTicker(timeout)
|
|
|
|
defer func() {
|
|
|
|
ticker.Stop()
|
|
|
|
}()
|
|
|
|
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case <-ticker.C:
|
|
|
|
case <-ctx.Done():
|
|
|
|
return
|
|
|
|
case timeout := <-t.timeout:
|
|
|
|
if timeout == 0 {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
ticker.Stop()
|
|
|
|
ticker = time.NewTicker(timeout)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
select {
|
|
|
|
case <-t.updated:
|
|
|
|
// Updated keep waiting.
|
|
|
|
default:
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func CancelAfterInactivity(ctx context.Context, cancel context.CancelFunc, timeout time.Duration) *ActivityTimer {
|
|
|
|
timer := &ActivityTimer{
|
|
|
|
timeout: make(chan time.Duration, 1),
|
|
|
|
updated: make(chan struct{}, 1),
|
|
|
|
closing: make(chan struct{}),
|
|
|
|
}
|
|
|
|
timer.timeout <- timeout
|
|
|
|
go timer.run(ctx, cancel)
|
|
|
|
return timer
|
|
|
|
}
|