-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathtask_manager.go
119 lines (98 loc) · 2.75 KB
/
task_manager.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
package main
import "sync/atomic"
type TaskState int
const (
TaskStateIdle TaskState = iota
TaskStateRecording
TaskStateTranscribing
)
const maxHistoryLength = 100
// TaskManager is a thread safe manager for global task state
type TaskManager struct {
currentTask atomic.Pointer[TranscribeTask]
transcriptionRes chan *TranscriptionResult
stateCh chan TaskState
context atomic.Pointer[string]
history atomic.Pointer[[]*TranscriptionResult]
}
// task managers ensures only only one task is running at a time and cancels
// the current task if a new one is started
var taskManager = TaskManager{
currentTask: atomic.Pointer[TranscribeTask]{}, // Initialize as nil
transcriptionRes: make(chan *TranscriptionResult),
stateCh: make(chan TaskState, 128),
context: atomic.Pointer[string]{},
history: atomic.Pointer[[]*TranscriptionResult]{},
}
func (tm *TaskManager) StartNewTask() *TranscribeTask {
newTask := NewTranscribeTask()
oldTask := tm.currentTask.Swap(newTask)
if oldTask != nil {
oldTask.Abort()
<-oldTask.waitForCompletion
}
stateCh := newTask.Start()
go func() {
// this waits for task to fish, state is closed when task is done
for state := range stateCh {
tm.stateCh <- state
}
tm.stateCh <- TaskStateIdle
if tm.currentTask.CompareAndSwap(newTask, nil) {
if result := newTask.GetResult(); result != nil {
tm.transcriptionRes <- result
}
}
}()
return newTask
}
func (tm *TaskManager) StartOrStopTask() {
if currentTask := tm.currentTask.Load(); currentTask != nil {
tm.StopRecording()
} else {
tm.StartNewTask()
}
}
func (tm *TaskManager) StopRecording() {
if currentTask := tm.currentTask.Load(); currentTask != nil {
currentTask.StopRecording()
}
}
func (tm *TaskManager) Abort() {
if currentTask := tm.currentTask.Load(); currentTask != nil {
currentTask.Abort()
}
}
func (tm *TaskManager) GetContext() string {
return *tm.context.Load()
}
func (tm *TaskManager) SetContext(ctx string) {
tm.context.Store(&ctx)
}
func (tm *TaskManager) AppendToHistory(entry *TranscriptionResult) {
for {
oldHistory := tm.history.Load()
if oldHistory == nil {
newHistory := []*TranscriptionResult{entry}
if tm.history.CompareAndSwap(nil, &newHistory) {
break
}
} else {
newHistory := append(*oldHistory, entry)
if len(newHistory) > maxHistoryLength {
newHistory = newHistory[len(newHistory)-maxHistoryLength:]
}
if tm.history.CompareAndSwap(oldHistory, &newHistory) {
break
}
}
}
}
// get a copy of the current history
func (tm *TaskManager) GetHistory() []*TranscriptionResult {
history := tm.history.Load()
if history == nil {
return []*TranscriptionResult{}
}
return append([]*TranscriptionResult(nil), *history...)
}