66 lines
1.1 KiB
Go
Raw Normal View History

2017-01-31 12:42:05 +01:00
package signal
import (
"context"
"time"
)
type ActivityUpdater interface {
2017-04-04 10:24:38 +02:00
Update()
}
type ActivityTimer struct {
2017-01-31 12:42:05 +01:00
updated chan bool
timeout chan time.Duration
2017-01-31 12:42:05 +01:00
ctx context.Context
cancel context.CancelFunc
}
func (t *ActivityTimer) Update() {
2017-01-31 12:42:05 +01:00
select {
case t.updated <- true:
default:
}
}
func (t *ActivityTimer) SetTimeout(timeout time.Duration) {
t.timeout <- timeout
}
func (t *ActivityTimer) run() {
ticker := time.NewTicker(<-t.timeout)
2017-05-08 17:09:21 +02:00
defer ticker.Stop()
2017-01-31 12:42:05 +01:00
for {
select {
2017-05-08 17:09:21 +02:00
case <-ticker.C:
2017-01-31 12:42:05 +01:00
case <-t.ctx.Done():
return
case timeout := <-t.timeout:
ticker.Stop()
ticker = time.NewTicker(timeout)
2017-01-31 14:15:34 +01:00
}
2017-05-08 17:09:21 +02:00
2017-01-31 14:15:34 +01:00
select {
case <-t.updated:
// Updated keep waiting.
2017-01-31 12:42:05 +01:00
default:
2017-01-31 14:15:34 +01:00
t.cancel()
return
2017-01-31 12:42:05 +01:00
}
}
}
func CancelAfterInactivity(ctx context.Context, timeout time.Duration) (context.Context, *ActivityTimer) {
2017-03-31 21:45:43 +02:00
ctx, cancel := context.WithCancel(ctx)
timer := &ActivityTimer{
2017-01-31 12:42:05 +01:00
ctx: ctx,
cancel: cancel,
timeout: make(chan time.Duration, 1),
2017-01-31 12:42:05 +01:00
updated: make(chan bool, 1),
}
timer.timeout <- timeout
2017-01-31 12:42:05 +01:00
go timer.run()
2017-03-31 21:45:43 +02:00
return ctx, timer
2017-01-31 12:42:05 +01:00
}