alist/pkg/task/manager.go

102 lines
1.7 KiB
Go
Raw Normal View History

2022-06-17 15:57:16 +08:00
package task
import (
2022-06-18 20:38:14 +08:00
log "github.com/sirupsen/logrus"
2022-06-17 15:57:16 +08:00
"sync/atomic"
"github.com/alist-org/alist/v3/pkg/generic_sync"
)
2022-06-17 21:52:31 +08:00
type Manager struct {
2022-06-18 20:38:14 +08:00
workerC chan struct{}
curID uint64
tasks generic_sync.MapOf[uint64, *Task]
2022-06-17 15:57:16 +08:00
}
2022-06-18 20:38:14 +08:00
func (tm *Manager) Submit(name string, f Func) uint64 {
2022-06-17 21:52:31 +08:00
task := newTask(name, f)
tm.addTask(task)
2022-06-18 20:38:14 +08:00
tm.do(task.ID)
2022-06-17 21:52:31 +08:00
return task.ID
2022-06-17 15:57:16 +08:00
}
2022-06-18 20:38:14 +08:00
func (tm *Manager) do(tid uint64) {
task := tm.MustGet(tid)
go func() {
log.Debugf("task [%s] waiting for worker", task.Name)
select {
case <-tm.workerC:
log.Debugf("task [%s] starting", task.Name)
task.run()
log.Debugf("task [%s] ended", task.Name)
}
tm.workerC <- struct{}{}
}()
}
2022-06-17 21:52:31 +08:00
func (tm *Manager) addTask(task *Task) {
2022-06-17 15:57:16 +08:00
task.ID = tm.curID
2022-06-17 21:52:31 +08:00
atomic.AddUint64(&tm.curID, 1)
2022-06-17 15:57:16 +08:00
tm.tasks.Store(task.ID, task)
}
2022-06-17 21:23:44 +08:00
func (tm *Manager) GetAll() []*Task {
2022-06-17 15:57:16 +08:00
return tm.tasks.Values()
}
2022-06-18 20:38:14 +08:00
func (tm *Manager) Get(tid uint64) (*Task, bool) {
return tm.tasks.Load(tid)
}
func (tm *Manager) MustGet(tid uint64) *Task {
task, _ := tm.Get(tid)
return task
2022-06-17 15:57:16 +08:00
}
2022-06-18 20:38:14 +08:00
func (tm *Manager) Retry(tid uint64) error {
t, ok := tm.Get(tid)
2022-06-17 22:09:20 +08:00
if !ok {
2022-06-18 20:38:14 +08:00
return ErrTaskNotFound
2022-06-17 22:09:20 +08:00
}
2022-06-18 20:38:14 +08:00
tm.do(t.ID)
2022-06-17 22:09:20 +08:00
return nil
}
2022-06-18 20:38:14 +08:00
func (tm *Manager) Cancel(tid uint64) error {
t, ok := tm.Get(tid)
2022-06-17 22:09:20 +08:00
if !ok {
2022-06-18 20:38:14 +08:00
return ErrTaskNotFound
2022-06-17 22:09:20 +08:00
}
t.Cancel()
return nil
}
2022-06-18 20:38:14 +08:00
func (tm *Manager) Remove(tid uint64) {
tm.tasks.Delete(tid)
2022-06-17 15:57:16 +08:00
}
2022-06-17 21:23:44 +08:00
func (tm *Manager) RemoveFinished() {
2022-06-17 15:57:16 +08:00
tasks := tm.GetAll()
for _, task := range tasks {
if task.Status == FINISHED {
tm.Remove(task.ID)
}
}
}
2022-06-17 21:23:44 +08:00
func (tm *Manager) RemoveError() {
2022-06-17 15:57:16 +08:00
tasks := tm.GetAll()
for _, task := range tasks {
if task.Error != nil {
tm.Remove(task.ID)
}
}
}
2022-06-17 21:52:31 +08:00
func NewTaskManager() *Manager {
return &Manager{
tasks: generic_sync.MapOf[uint64, *Task]{},
curID: 0,
}
2022-06-17 15:57:16 +08:00
}