v2ray-core/common/task/periodic.go

86 lines
1.3 KiB
Go
Raw Normal View History

2018-05-27 11:02:29 +00:00
package task
2018-02-08 14:39:46 +00:00
import (
"sync"
"time"
)
2018-05-27 11:02:29 +00:00
// Periodic is a task that runs periodically.
type Periodic struct {
2018-02-11 22:28:42 +00:00
// Interval of the task being run
2018-02-08 14:39:46 +00:00
Interval time.Duration
2018-02-11 22:28:42 +00:00
// Execute is the task function
Execute func() error
2018-02-08 14:39:46 +00:00
access sync.Mutex
timer *time.Timer
running bool
2018-05-27 12:42:53 +00:00
}
func (t *Periodic) hasClosed() bool {
t.access.Lock()
defer t.access.Unlock()
2018-05-27 12:42:53 +00:00
return !t.running
2018-05-27 12:42:53 +00:00
}
2018-02-08 14:39:46 +00:00
2018-05-27 12:42:53 +00:00
func (t *Periodic) checkedExecute() error {
if t.hasClosed() {
2018-02-08 14:39:46 +00:00
return nil
}
if err := t.Execute(); err != nil {
t.access.Lock()
t.running = false
t.access.Unlock()
2018-02-08 14:39:46 +00:00
return err
}
2018-05-27 12:42:53 +00:00
t.access.Lock()
defer t.access.Unlock()
if !t.running {
return nil
}
2018-02-08 14:39:46 +00:00
t.timer = time.AfterFunc(t.Interval, func() {
t.checkedExecute() // nolint: errcheck
2018-02-08 14:39:46 +00:00
})
return nil
}
// Start implements common.Runnable.
2018-05-27 11:02:29 +00:00
func (t *Periodic) Start() error {
t.access.Lock()
if t.running {
2018-08-29 21:12:47 +00:00
t.access.Unlock()
return nil
}
t.running = true
t.access.Unlock()
2018-02-08 14:39:46 +00:00
if err := t.checkedExecute(); err != nil {
t.access.Lock()
t.running = false
t.access.Unlock()
2018-02-08 14:39:46 +00:00
return err
}
return nil
}
2018-04-03 09:11:54 +00:00
// Close implements common.Closable.
2018-05-27 11:02:29 +00:00
func (t *Periodic) Close() error {
2018-02-08 14:39:46 +00:00
t.access.Lock()
defer t.access.Unlock()
t.running = false
2018-02-08 14:39:46 +00:00
if t.timer != nil {
t.timer.Stop()
t.timer = nil
}
return nil
}