mirror of
https://github.com/semaphoreui/semaphore.git
synced 2025-01-21 07:49:34 +01:00
143 lines
2.7 KiB
Go
143 lines
2.7 KiB
Go
package tasks
|
|
|
|
import (
|
|
"strconv"
|
|
"time"
|
|
|
|
log "github.com/Sirupsen/logrus"
|
|
"github.com/ansible-semaphore/semaphore/util"
|
|
)
|
|
|
|
type taskPool struct {
|
|
queue []*task
|
|
register chan *task
|
|
activeProj map[int]*task
|
|
activeNodes map[string]*task
|
|
running int
|
|
runningTasks map[int]*task
|
|
}
|
|
|
|
type resourceLock struct {
|
|
lock bool
|
|
holder *task
|
|
}
|
|
|
|
var pool = taskPool{
|
|
queue: make([]*task, 0),
|
|
register: make(chan *task),
|
|
activeProj: make(map[int]*task),
|
|
activeNodes: make(map[string]*task),
|
|
running: 0,
|
|
runningTasks: make(map[int]*task),
|
|
}
|
|
|
|
var resourceLocker = make(chan *resourceLock)
|
|
|
|
//nolint: gocyclo
|
|
func (p *taskPool) run() {
|
|
ticker := time.NewTicker(5 * time.Second)
|
|
|
|
defer func() {
|
|
close(resourceLocker)
|
|
ticker.Stop()
|
|
}()
|
|
|
|
// Lock or unlock resources when running a task
|
|
go func(locker <-chan *resourceLock) {
|
|
for l := range locker {
|
|
t := l.holder
|
|
|
|
if l.lock {
|
|
if p.blocks(t) {
|
|
panic("Trying to lock an already locked resource!")
|
|
}
|
|
|
|
p.activeProj[t.projectID] = t
|
|
|
|
for _, node := range t.hosts {
|
|
p.activeNodes[node] = t
|
|
}
|
|
|
|
p.running++
|
|
p.runningTasks[t.task.ID] = t
|
|
continue
|
|
}
|
|
|
|
if p.activeProj[t.projectID] == t {
|
|
delete(p.activeProj, t.projectID)
|
|
}
|
|
|
|
for _, node := range t.hosts {
|
|
delete(p.activeNodes, node)
|
|
}
|
|
|
|
p.running--
|
|
delete(p.runningTasks, t.task.ID)
|
|
}
|
|
}(resourceLocker)
|
|
|
|
for {
|
|
select {
|
|
case task := <-p.register:
|
|
p.queue = append(p.queue, task)
|
|
log.Debug(task)
|
|
msg := "Task " + strconv.Itoa(task.task.ID) + " added to queue"
|
|
task.log(msg)
|
|
log.Info(msg)
|
|
case <-ticker.C:
|
|
if len(p.queue) == 0 {
|
|
continue
|
|
}
|
|
|
|
//get task from top of queue
|
|
t := p.queue[0]
|
|
if t.task.Status == taskFailStatus {
|
|
//delete failed task from queue
|
|
p.queue = p.queue[1:]
|
|
log.Info("Task " + strconv.Itoa(t.task.ID) + " removed from queue")
|
|
continue
|
|
}
|
|
if p.blocks(t) {
|
|
//move blocked task to end of queue
|
|
p.queue = append(p.queue[1:], t)
|
|
continue
|
|
}
|
|
log.Info("Set resource locker with task " + strconv.Itoa(t.task.ID))
|
|
resourceLocker <- &resourceLock{lock: true, holder: t}
|
|
if !t.prepared {
|
|
go t.prepareRun()
|
|
continue
|
|
}
|
|
go t.run()
|
|
p.queue = p.queue[1:]
|
|
log.Info("Task " + strconv.Itoa(t.task.ID) + " removed from queue")
|
|
}
|
|
}
|
|
}
|
|
|
|
func (p *taskPool) blocks(t *task) bool {
|
|
if p.running >= util.Config.MaxParallelTasks {
|
|
return true
|
|
}
|
|
|
|
switch util.Config.ConcurrencyMode {
|
|
case "project":
|
|
return p.activeProj[t.projectID] != nil
|
|
case "node":
|
|
for _, node := range t.hosts {
|
|
if p.activeNodes[node] != nil {
|
|
return true
|
|
}
|
|
}
|
|
|
|
return false
|
|
default:
|
|
return p.running > 0
|
|
}
|
|
}
|
|
|
|
// StartRunner begins the task pool, used as a goroutine
|
|
func StartRunner() {
|
|
pool.run()
|
|
}
|