v2ray-core/common/signal/timer.go

74 lines
1.1 KiB
Go
Raw Normal View History

2017-01-31 11:42:05 +00:00
package signal
import (
"context"
"time"
)
type ActivityUpdater interface {
2017-04-04 08:24:38 +00:00
Update()
}
type ActivityTimer struct {
2017-01-31 11:42:05 +00:00
updated chan bool
timeout chan time.Duration
2017-01-31 11:42:05 +00:00
}
func (t *ActivityTimer) Update() {
2017-01-31 11:42:05 +00:00
select {
case t.updated <- true:
default:
}
}
func (t *ActivityTimer) SetTimeout(timeout time.Duration) {
t.timeout <- timeout
}
2017-11-30 23:47:17 +00:00
func (t *ActivityTimer) run(ctx context.Context, cancel context.CancelFunc) {
2017-12-14 16:39:58 +00:00
defer cancel()
timeout := <-t.timeout
if timeout == 0 {
return
}
2017-12-14 22:57:04 +00:00
ticker := time.NewTicker(timeout)
2017-10-06 07:42:46 +00:00
defer func() {
ticker.Stop()
}()
2017-05-08 15:09:21 +00:00
2017-01-31 11:42:05 +00:00
for {
select {
2017-05-08 15:09:21 +00:00
case <-ticker.C:
2017-11-30 23:47:17 +00:00
case <-ctx.Done():
2017-01-31 11:42:05 +00:00
return
case timeout := <-t.timeout:
2017-11-23 13:58:35 +00:00
if timeout == 0 {
return
}
ticker.Stop()
ticker = time.NewTicker(timeout)
2017-11-30 23:47:17 +00:00
continue
2017-01-31 13:15:34 +00:00
}
2017-05-08 15:09:21 +00:00
2017-01-31 13:15:34 +00:00
select {
case <-t.updated:
// Updated keep waiting.
2017-01-31 11:42:05 +00:00
default:
2017-01-31 13:15:34 +00:00
return
2017-01-31 11:42:05 +00:00
}
}
}
2017-11-14 23:36:14 +00:00
func CancelAfterInactivity(ctx context.Context, cancel context.CancelFunc, timeout time.Duration) *ActivityTimer {
timer := &ActivityTimer{
timeout: make(chan time.Duration, 1),
2017-01-31 11:42:05 +00:00
updated: make(chan bool, 1),
}
timer.timeout <- timeout
2017-11-30 23:47:17 +00:00
go timer.run(ctx, cancel)
2017-11-14 23:36:14 +00:00
return timer
2017-01-31 11:42:05 +00:00
}