mirror of
https://github.com/status-im/status-go.git
synced 2025-01-12 15:45:07 +00:00
38308d48f2
* feat_: log error and stacktrace when panic in goroutine * test_: add test TestSafeGo * chore_: rename logAndCall to call * chore_: rename SafeGo to Go * chore_: make lint-fix * chore_: use t.Cleanup * chore_: Revert "chore_: use t.Cleanup" This reverts commit 4eb420d179cc0e208e84c13cb941e6b3d1ed9819. * chore_: Revert "chore_: make lint-fix" This reverts commit fcc995f157e671a4229b47419c3a0e4004b5fdab. * chore_: Revert "chore_: rename SafeGo to Go" This reverts commit a6d73d6df583f313032d79aac62f66328039cb55. * chore_: Revert "chore_: rename logAndCall to call" This reverts commit 8fbe993bedb9fbba67349a44f151e2dd5e3bc4cc. * chore_: Revert "test_: add test TestSafeGo" This reverts commit a1fa91839f3960398980c6bf456e6462ec944819. * chore_: Revert "feat_: log error and stacktrace when panic in goroutine" This reverts commit f612dd828fa2ce410d0e806fe773ecbe3e86a68a. * feat_: log error and stacktrace when panic in goroutine * chore_: make lint-fix * chore_: rename logAndCall to call * chore_: renaming LogOnPanic * chore_: update rest goroutine function calls * chore_: make lint-fix
113 lines
2.3 KiB
Go
113 lines
2.3 KiB
Go
package transactions
|
|
|
|
import (
|
|
"context"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/status-im/status-go/common"
|
|
)
|
|
|
|
// TaskFunc defines the task to be run. The context is canceled when Stop is
|
|
// called to early stop scheduled task.
|
|
type TaskFunc func(ctx context.Context) (done bool)
|
|
|
|
const (
|
|
WorkNotDone = false
|
|
WorkDone = true
|
|
)
|
|
|
|
// ConditionalRepeater runs a task at regular intervals until the task returns
|
|
// true. It doesn't allow running task in parallel and can be triggered early
|
|
// by call to RunUntilDone.
|
|
type ConditionalRepeater struct {
|
|
interval time.Duration
|
|
task TaskFunc
|
|
// nil if not running
|
|
ctx context.Context
|
|
ctxMu sync.Mutex
|
|
cancel context.CancelFunc
|
|
runNowCh chan bool
|
|
runNowMu sync.Mutex
|
|
}
|
|
|
|
func NewConditionalRepeater(interval time.Duration, task TaskFunc) *ConditionalRepeater {
|
|
return &ConditionalRepeater{
|
|
interval: interval,
|
|
task: task,
|
|
runNowCh: make(chan bool, 1),
|
|
}
|
|
}
|
|
|
|
// RunUntilDone starts the task immediately and continues to run it at the defined
|
|
// interval until the task returns true. Can be called multiple times but it
|
|
// does not allow multiple concurrent executions of the task.
|
|
func (t *ConditionalRepeater) RunUntilDone() {
|
|
t.ctxMu.Lock()
|
|
defer func() {
|
|
t.runNowMu.Lock()
|
|
if len(t.runNowCh) == 0 {
|
|
t.runNowCh <- true
|
|
}
|
|
t.runNowMu.Unlock()
|
|
t.ctxMu.Unlock()
|
|
}()
|
|
|
|
if t.ctx != nil {
|
|
return
|
|
}
|
|
t.ctx, t.cancel = context.WithCancel(context.Background())
|
|
|
|
go func() {
|
|
defer common.LogOnPanic()
|
|
defer func() {
|
|
t.ctxMu.Lock()
|
|
defer t.ctxMu.Unlock()
|
|
t.cancel()
|
|
t.ctx = nil
|
|
}()
|
|
|
|
ticker := time.NewTicker(t.interval)
|
|
defer ticker.Stop()
|
|
|
|
for {
|
|
select {
|
|
// Stop was called or task returned true
|
|
case <-t.ctx.Done():
|
|
return
|
|
// Scheduled execution
|
|
case <-ticker.C:
|
|
if t.task(t.ctx) {
|
|
return
|
|
}
|
|
// Start right away if requested
|
|
case <-t.runNowCh:
|
|
ticker.Reset(t.interval)
|
|
if t.task(t.ctx) {
|
|
t.runNowMu.Lock()
|
|
if len(t.runNowCh) == 0 {
|
|
t.runNowMu.Unlock()
|
|
return
|
|
}
|
|
t.runNowMu.Unlock()
|
|
}
|
|
}
|
|
}
|
|
}()
|
|
}
|
|
|
|
// Stop forcefully stops the running task by canceling its context.
|
|
func (t *ConditionalRepeater) Stop() {
|
|
t.ctxMu.Lock()
|
|
defer t.ctxMu.Unlock()
|
|
if t.ctx != nil {
|
|
t.cancel()
|
|
}
|
|
}
|
|
|
|
func (t *ConditionalRepeater) IsRunning() bool {
|
|
t.ctxMu.Lock()
|
|
defer t.ctxMu.Unlock()
|
|
return t.ctx != nil
|
|
}
|