2023-08-29 00:51:04 +02:00
|
|
|
package runners
|
|
|
|
|
|
|
|
import (
|
2024-04-12 12:21:05 +02:00
|
|
|
"net/http"
|
|
|
|
|
2024-12-14 09:18:17 +01:00
|
|
|
"github.com/gorilla/context"
|
2024-10-26 14:56:17 +02:00
|
|
|
"github.com/semaphoreui/semaphore/api/helpers"
|
|
|
|
"github.com/semaphoreui/semaphore/db"
|
|
|
|
"github.com/semaphoreui/semaphore/pkg/task_logger"
|
|
|
|
"github.com/semaphoreui/semaphore/services/runners"
|
|
|
|
"github.com/semaphoreui/semaphore/util"
|
2023-08-29 00:51:04 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
func RunnerMiddleware(next http.Handler) http.Handler {
|
|
|
|
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
|
|
|
2024-09-26 14:52:31 +02:00
|
|
|
token := r.Header.Get("X-Runner-Token")
|
2024-01-07 17:32:30 +01:00
|
|
|
|
2024-01-07 20:50:37 +01:00
|
|
|
if token == "" {
|
|
|
|
helpers.WriteJSON(w, http.StatusUnauthorized, map[string]string{
|
|
|
|
"error": "Invalid token",
|
|
|
|
})
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2023-08-29 00:51:04 +02:00
|
|
|
store := helpers.Store(r)
|
|
|
|
|
2024-09-29 12:40:07 +02:00
|
|
|
runner, err := store.GetGlobalRunnerByToken(token)
|
2023-08-29 00:51:04 +02:00
|
|
|
|
2024-01-07 20:50:37 +01:00
|
|
|
if err != nil {
|
|
|
|
helpers.WriteJSON(w, http.StatusNotFound, map[string]string{
|
|
|
|
"error": "Runner not found",
|
2024-01-07 17:32:30 +01:00
|
|
|
})
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-01-07 20:50:37 +01:00
|
|
|
if runner.Token != token {
|
|
|
|
helpers.WriteJSON(w, http.StatusUnauthorized, map[string]string{
|
|
|
|
"error": "Invalid token",
|
2023-08-29 00:51:04 +02:00
|
|
|
})
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
context.Set(r, "runner", runner)
|
|
|
|
next.ServeHTTP(w, r)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func GetRunner(w http.ResponseWriter, r *http.Request) {
|
|
|
|
runner := context.Get(r, "runner").(db.Runner)
|
|
|
|
|
|
|
|
data := runners.RunnerState{
|
|
|
|
AccessKeys: make(map[int]db.AccessKey),
|
|
|
|
}
|
|
|
|
|
|
|
|
tasks := helpers.TaskPool(r).GetRunningTasks()
|
|
|
|
|
|
|
|
for _, tsk := range tasks {
|
|
|
|
if tsk.RunnerID != runner.ID {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2024-04-12 12:32:54 +02:00
|
|
|
if tsk.Task.Status == task_logger.TaskStartingStatus {
|
2023-08-29 00:51:04 +02:00
|
|
|
|
|
|
|
data.NewJobs = append(data.NewJobs, runners.JobData{
|
2024-05-27 22:00:40 +02:00
|
|
|
Username: tsk.Username,
|
|
|
|
IncomingVersion: tsk.IncomingVersion,
|
2024-12-16 16:27:42 +01:00
|
|
|
Alias: tsk.Alias,
|
2024-05-27 22:00:40 +02:00
|
|
|
Task: tsk.Task,
|
|
|
|
Template: tsk.Template,
|
|
|
|
Inventory: tsk.Inventory,
|
|
|
|
InventoryRepository: tsk.Inventory.Repository,
|
|
|
|
Repository: tsk.Repository,
|
|
|
|
Environment: tsk.Environment,
|
2023-08-29 00:51:04 +02:00
|
|
|
})
|
|
|
|
|
|
|
|
if tsk.Inventory.SSHKeyID != nil {
|
2023-09-11 03:15:33 +02:00
|
|
|
err := tsk.Inventory.SSHKey.DeserializeSecret()
|
|
|
|
if err != nil {
|
|
|
|
// TODO: return error
|
|
|
|
}
|
2023-08-29 00:51:04 +02:00
|
|
|
data.AccessKeys[*tsk.Inventory.SSHKeyID] = tsk.Inventory.SSHKey
|
|
|
|
}
|
|
|
|
|
|
|
|
if tsk.Inventory.BecomeKeyID != nil {
|
2023-09-11 03:15:33 +02:00
|
|
|
err := tsk.Inventory.BecomeKey.DeserializeSecret()
|
|
|
|
if err != nil {
|
|
|
|
// TODO: return error
|
|
|
|
}
|
2023-08-29 00:51:04 +02:00
|
|
|
data.AccessKeys[*tsk.Inventory.BecomeKeyID] = tsk.Inventory.BecomeKey
|
|
|
|
}
|
|
|
|
|
2024-10-03 21:41:36 +02:00
|
|
|
if tsk.Template.Vaults != nil {
|
|
|
|
for _, vault := range tsk.Template.Vaults {
|
2024-10-22 21:50:31 +02:00
|
|
|
if vault.VaultKeyID != nil {
|
|
|
|
err := vault.Vault.DeserializeSecret()
|
|
|
|
if err != nil {
|
|
|
|
// TODO: return error
|
|
|
|
}
|
|
|
|
data.AccessKeys[*vault.VaultKeyID] = *vault.Vault
|
2024-10-03 21:41:36 +02:00
|
|
|
}
|
2023-09-11 02:30:40 +02:00
|
|
|
}
|
2023-09-11 02:00:10 +02:00
|
|
|
}
|
|
|
|
|
2024-05-27 22:00:40 +02:00
|
|
|
if tsk.Inventory.RepositoryID != nil {
|
|
|
|
err := tsk.Inventory.Repository.SSHKey.DeserializeSecret()
|
|
|
|
if err != nil {
|
|
|
|
// TODO: return error
|
|
|
|
}
|
|
|
|
data.AccessKeys[tsk.Inventory.Repository.SSHKeyID] = tsk.Inventory.Repository.SSHKey
|
|
|
|
}
|
|
|
|
|
2023-08-29 00:51:04 +02:00
|
|
|
data.AccessKeys[tsk.Repository.SSHKeyID] = tsk.Repository.SSHKey
|
|
|
|
|
|
|
|
} else {
|
|
|
|
data.CurrentJobs = append(data.CurrentJobs, runners.JobState{
|
|
|
|
ID: tsk.Task.ID,
|
|
|
|
Status: tsk.Task.Status,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
helpers.WriteJSON(w, http.StatusOK, data)
|
|
|
|
}
|
|
|
|
|
|
|
|
func UpdateRunner(w http.ResponseWriter, r *http.Request) {
|
2023-12-25 10:49:47 +01:00
|
|
|
|
|
|
|
runner := context.Get(r, "runner").(db.Runner)
|
|
|
|
|
2023-08-29 00:51:04 +02:00
|
|
|
var body runners.RunnerProgress
|
|
|
|
|
|
|
|
if !helpers.Bind(w, r, &body) {
|
|
|
|
helpers.WriteJSON(w, http.StatusBadRequest, map[string]string{
|
|
|
|
"error": "Invalid format",
|
|
|
|
})
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
taskPool := helpers.TaskPool(r)
|
|
|
|
|
|
|
|
if body.Jobs == nil {
|
|
|
|
w.WriteHeader(http.StatusNoContent)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, job := range body.Jobs {
|
|
|
|
tsk := taskPool.GetTask(job.ID)
|
|
|
|
|
|
|
|
if tsk == nil {
|
|
|
|
// TODO: log
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2023-12-25 10:49:47 +01:00
|
|
|
if tsk.RunnerID != runner.ID {
|
|
|
|
// TODO: add error message
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2023-08-29 00:51:04 +02:00
|
|
|
for _, logRecord := range job.LogRecords {
|
2024-04-12 12:21:05 +02:00
|
|
|
tsk.LogWithTime(logRecord.Time, logRecord.Message)
|
2023-08-29 00:51:04 +02:00
|
|
|
}
|
2023-09-10 23:18:25 +02:00
|
|
|
|
|
|
|
tsk.SetStatus(job.Status)
|
2024-12-14 09:18:17 +01:00
|
|
|
|
|
|
|
if job.Commit != nil {
|
|
|
|
tsk.SetCommit(job.Commit.Hash, job.Commit.Message)
|
|
|
|
}
|
2023-08-29 00:51:04 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
w.WriteHeader(http.StatusNoContent)
|
|
|
|
}
|
|
|
|
|
|
|
|
func RegisterRunner(w http.ResponseWriter, r *http.Request) {
|
|
|
|
var register runners.RunnerRegistration
|
|
|
|
|
|
|
|
if !helpers.Bind(w, r, ®ister) {
|
|
|
|
helpers.WriteJSON(w, http.StatusBadRequest, map[string]string{
|
|
|
|
"error": "Invalid format",
|
|
|
|
})
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if util.Config.RunnerRegistrationToken == "" || register.RegistrationToken != util.Config.RunnerRegistrationToken {
|
|
|
|
helpers.WriteJSON(w, http.StatusBadRequest, map[string]string{
|
|
|
|
"error": "Invalid registration token",
|
|
|
|
})
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
runner, err := helpers.Store(r).CreateRunner(db.Runner{
|
2023-09-16 23:47:06 +02:00
|
|
|
Webhook: register.Webhook,
|
|
|
|
MaxParallelTasks: register.MaxParallelTasks,
|
2023-08-29 00:51:04 +02:00
|
|
|
})
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
helpers.WriteJSON(w, http.StatusInternalServerError, map[string]string{
|
|
|
|
"error": "Unexpected error",
|
|
|
|
})
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-09-29 11:57:02 +02:00
|
|
|
var res struct {
|
|
|
|
Token string `json:"token"`
|
2023-08-29 00:51:04 +02:00
|
|
|
}
|
|
|
|
|
2024-09-29 11:57:02 +02:00
|
|
|
res.Token = runner.Token
|
|
|
|
|
2023-08-29 00:51:04 +02:00
|
|
|
helpers.WriteJSON(w, http.StatusOK, res)
|
|
|
|
}
|
2024-03-27 12:11:33 +01:00
|
|
|
|
|
|
|
func UnregisterRunner(w http.ResponseWriter, r *http.Request) {
|
|
|
|
|
|
|
|
runner := context.Get(r, "runner").(db.Runner)
|
|
|
|
|
|
|
|
err := helpers.Store(r).DeleteGlobalRunner(runner.ID)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
helpers.WriteJSON(w, http.StatusInternalServerError, map[string]string{
|
|
|
|
"error": "Unknown error",
|
|
|
|
})
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
w.WriteHeader(http.StatusNoContent)
|
|
|
|
}
|