123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163 |
- // SiYuan - Build Your Eternal Digital Garden
- // Copyright (c) 2020-present, b3log.org
- //
- // This program is free software: you can redistribute it and/or modify
- // it under the terms of the GNU Affero General Public License as published by
- // the Free Software Foundation, either version 3 of the License, or
- // (at your option) any later version.
- //
- // This program is distributed in the hope that it will be useful,
- // but WITHOUT ANY WARRANTY; without even the implied warranty of
- // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- // GNU Affero General Public License for more details.
- //
- // You should have received a copy of the GNU Affero General Public License
- // along with this program. If not, see <https://www.gnu.org/licenses/>.
- package task
- import (
- "reflect"
- "sync"
- "time"
- "github.com/siyuan-note/logging"
- )
- var (
- taskQueue []*Task
- taskQueueStatus int
- queueLock = sync.Mutex{}
- taskLock = sync.Mutex{}
- )
- const (
- QueueStatusRunning = iota
- QueueStatusClosing
- )
- type Task struct {
- Action string
- Handler reflect.Value
- Args []interface{}
- Created time.Time
- }
- func PrependTask(action string, handler interface{}, args ...interface{}) {
- queueLock.Lock()
- defer queueLock.Unlock()
- if QueueStatusRunning != taskQueueStatus {
- logging.LogWarnf("task queue is paused, action [%s] will be ignored", action)
- return
- }
- cancelTask(action)
- taskQueue = append([]*Task{newTask(action, handler, args...)}, taskQueue...)
- }
- func AppendTask(action string, handler interface{}, args ...interface{}) {
- queueLock.Lock()
- defer queueLock.Unlock()
- if QueueStatusRunning != taskQueueStatus {
- logging.LogWarnf("task queue is paused, action [%s] will be ignored", action)
- return
- }
- cancelTask(action)
- taskQueue = append(taskQueue, newTask(action, handler, args...))
- }
- func CancelTask(actions ...string) {
- queueLock.Lock()
- defer queueLock.Unlock()
- cancelTask(actions...)
- }
- func cancelTask(actions ...string) {
- for i := len(taskQueue) - 1; i >= 0; i-- {
- task := taskQueue[i]
- for _, action := range actions {
- if action == task.Action {
- taskQueue = append(taskQueue[:i], taskQueue[i+1:]...)
- break
- }
- }
- }
- }
- func newTask(action string, handler interface{}, args ...interface{}) *Task {
- return &Task{
- Action: action,
- Handler: reflect.ValueOf(handler),
- Args: args,
- Created: time.Now(),
- }
- }
- const (
- CloudSync = "task.cloud.sync" // 数据同步
- RepoCheckout = "task.repo.checkout" // 从快照中检出
- DatabaseIndexFull = "task.database.index.full" // 重建索引
- DatabaseIndex = "task.database.index" // 数据库所以队列
- DatabaseIndexFix = "task.database.index.fix" // 数据库索引订正
- OCRImage = "task.ocr.image" // 图片 OCR 提取文本
- HistoryGenerateDoc = "task.history.generateDoc" // 生成文件历史
- DatabaseIndexEmbedBlock = "task.database.index.embedBlock" // 数据库索引嵌入块
- )
- func Loop() {
- for {
- time.Sleep(10 * time.Millisecond)
- task := popTask()
- if nil == task {
- continue
- }
- execTask(task)
- }
- }
- func CloseWait() {
- queueLock.Lock()
- defer queueLock.Unlock()
- taskQueueStatus = QueueStatusClosing
- for {
- time.Sleep(10 * time.Millisecond)
- if 1 > len(taskQueue) {
- break
- }
- }
- }
- func popTask() (ret *Task) {
- queueLock.Lock()
- defer queueLock.Unlock()
- if 0 == len(taskQueue) {
- return
- }
- ret = taskQueue[0]
- taskQueue = taskQueue[1:]
- return
- }
- func execTask(task *Task) {
- taskLock.Lock()
- defer taskLock.Unlock()
- defer logging.Recover()
- args := make([]reflect.Value, len(task.Args))
- for i, v := range task.Args {
- if nil == v {
- args[i] = reflect.New(task.Handler.Type().In(i)).Elem()
- } else {
- args[i] = reflect.ValueOf(v)
- }
- }
- task.Handler.Call(args)
- }
|