Semaphore/services/tasks/TaskRunner_test.go

647 lines
13 KiB
Go
Raw Normal View History

package tasks
import (
"math/rand"
"os"
"path"
2021-09-13 15:22:08 +02:00
"strings"
"testing"
2024-04-19 15:24:24 +02:00
"github.com/semaphoreui/semaphore/db_lib"
"github.com/semaphoreui/semaphore/db"
"github.com/semaphoreui/semaphore/db/bolt"
"github.com/semaphoreui/semaphore/util"
)
2022-11-20 10:10:33 +01:00
func TestTaskRunnerRun(t *testing.T) {
util.Config = &util.ConfigType{
TmpPath: "/tmp",
}
2024-07-14 19:14:57 +02:00
store := bolt.CreateTestStore()
2022-11-20 10:10:33 +01:00
pool := CreateTaskPool(store)
go pool.Run()
var task db.Task
var err error
db.StoreSession(store, "", func() {
2024-06-30 09:48:36 +02:00
task, err = store.CreateTask(db.Task{}, 0)
2022-11-20 10:10:33 +01:00
})
if err != nil {
t.Fatal(err)
}
taskRunner := TaskRunner{
Task: task,
2022-11-20 10:10:33 +01:00
pool: &pool,
}
taskRunner.job = &LocalJob{
Task: taskRunner.Task,
Template: taskRunner.Template,
Inventory: taskRunner.Inventory,
Repository: taskRunner.Repository,
Environment: taskRunner.Environment,
Logger: &taskRunner,
App: &db_lib.AnsibleApp{
Template: taskRunner.Template,
Repository: taskRunner.Repository,
Logger: &taskRunner,
Playbook: &db_lib.AnsiblePlaybook{
Logger: &taskRunner,
TemplateID: taskRunner.Template.ID,
Repository: taskRunner.Repository,
},
},
}
2022-11-20 10:10:33 +01:00
taskRunner.run()
}
2022-10-16 20:30:26 +02:00
func TestGetRepoPath(t *testing.T) {
util.Config = &util.ConfigType{
TmpPath: "/tmp",
}
inventoryID := 1
tsk := TaskRunner{
Task: db.Task{},
Inventory: db.Inventory{
SSHKeyID: &inventoryID,
SSHKey: db.AccessKey{
ID: 12345,
Type: db.AccessKeySSH,
},
Type: db.InventoryStatic,
},
Template: db.Template{
2022-10-16 20:30:26 +02:00
Playbook: "deploy/test.yml",
},
}
tsk.job = &LocalJob{
Task: tsk.Task,
Template: tsk.Template,
Inventory: tsk.Inventory,
Repository: tsk.Repository,
Environment: tsk.Environment,
Logger: &tsk,
App: &db_lib.AnsibleApp{
Template: tsk.Template,
Repository: tsk.Repository,
Logger: &tsk,
Playbook: &db_lib.AnsiblePlaybook{
Logger: &tsk,
TemplateID: tsk.Template.ID,
Repository: tsk.Repository,
},
},
}
dir := tsk.job.(*LocalJob).App.(*db_lib.AnsibleApp).GetPlaybookDir()
if dir != "/tmp/repository_0_0/deploy" {
t.Fatal("Invalid playbook dir: " + dir)
}
}
func TestGetRepoPath_whenStartsWithSlash(t *testing.T) {
util.Config = &util.ConfigType{
TmpPath: "/tmp",
}
inventoryID := 1
tsk := TaskRunner{
Task: db.Task{},
Inventory: db.Inventory{
2022-10-16 20:30:26 +02:00
SSHKeyID: &inventoryID,
SSHKey: db.AccessKey{
ID: 12345,
Type: db.AccessKeySSH,
},
Type: db.InventoryStatic,
},
Template: db.Template{
Playbook: "/deploy/test.yml",
2022-10-16 20:30:26 +02:00
},
}
tsk.job = &LocalJob{
Task: tsk.Task,
Template: tsk.Template,
Inventory: tsk.Inventory,
Repository: tsk.Repository,
Environment: tsk.Environment,
Logger: &tsk,
App: &db_lib.AnsibleApp{
Template: tsk.Template,
Repository: tsk.Repository,
Logger: &tsk,
Playbook: &db_lib.AnsiblePlaybook{
Logger: &tsk,
TemplateID: tsk.Template.ID,
Repository: tsk.Repository,
},
},
}
2022-10-16 20:30:26 +02:00
dir := tsk.job.(*LocalJob).App.(*db_lib.AnsibleApp).GetPlaybookDir()
2022-10-16 20:30:26 +02:00
if dir != "/tmp/repository_0_0/deploy" {
t.Fatal("Invalid playbook dir: " + dir)
}
}
func TestPopulateDetails(t *testing.T) {
2024-07-14 19:14:57 +02:00
store := bolt.CreateTestStore()
proj, err := store.CreateProject(db.Project{})
if err != nil {
t.Fatal(err)
}
key, err := store.CreateAccessKey(db.AccessKey{
ProjectID: &proj.ID,
2022-01-27 15:16:58 +01:00
Type: db.AccessKeyNone,
})
if err != nil {
t.Fatal(err)
}
repo, err := store.CreateRepository(db.Repository{
ProjectID: proj.ID,
SSHKeyID: key.ID,
2022-01-22 09:21:11 +01:00
Name: "Test",
GitURL: "git@example.com:test/test",
GitBranch: "master",
})
if err != nil {
t.Fatal(err)
}
inv, err := store.CreateInventory(db.Inventory{
ProjectID: proj.ID,
})
if err != nil {
t.Fatal(err)
}
env, err := store.CreateEnvironment(db.Environment{
ProjectID: proj.ID,
Name: "test",
JSON: `{"author": "Denis", "comment": "Hello, World!"}`,
})
if err != nil {
t.Fatal(err)
}
tpl, err := store.CreateTemplate(db.Template{
2022-02-03 08:05:13 +01:00
Name: "Test",
Playbook: "test.yml",
ProjectID: proj.ID,
RepositoryID: repo.ID,
InventoryID: &inv.ID,
EnvironmentID: &env.ID,
})
if err != nil {
t.Fatal(err)
}
2022-11-20 10:10:33 +01:00
pool := TaskPool{store: store}
tsk := TaskRunner{
pool: &pool,
Task: db.Task{
TemplateID: tpl.ID,
ProjectID: proj.ID,
Environment: `{"comment": "Just do it!", "time": "2021-11-02"}`,
},
}
tsk.job = &LocalJob{
Task: tsk.Task,
Template: tsk.Template,
Inventory: tsk.Inventory,
Repository: tsk.Repository,
Environment: tsk.Environment,
Logger: &tsk,
App: &db_lib.AnsibleApp{
Template: tsk.Template,
Repository: tsk.Repository,
Logger: &tsk,
Playbook: &db_lib.AnsiblePlaybook{
Logger: &tsk,
TemplateID: tsk.Template.ID,
Repository: tsk.Repository,
},
},
}
err = tsk.populateDetails()
if err != nil {
t.Fatal(err)
}
if tsk.Environment.JSON != `{"author":"Denis","comment":"Hello, World!","time":"2021-11-02"}` {
t.Fatal(err)
}
}
func TestPopulateDetailsInventory(t *testing.T) {
store := bolt.CreateTestStore()
proj, err := store.CreateProject(db.Project{})
if err != nil {
t.Fatal(err)
}
key, err := store.CreateAccessKey(db.AccessKey{
ProjectID: &proj.ID,
Type: db.AccessKeyNone,
})
if err != nil {
t.Fatal(err)
}
repo, err := store.CreateRepository(db.Repository{
ProjectID: proj.ID,
SSHKeyID: key.ID,
Name: "Test",
GitURL: "git@example.com:test/test",
GitBranch: "master",
})
if err != nil {
t.Fatal(err)
}
inv, err := store.CreateInventory(db.Inventory{
ProjectID: proj.ID,
ID: 1,
})
if err != nil {
t.Fatal(err)
}
inv2, err := store.CreateInventory(db.Inventory{
ProjectID: proj.ID,
ID: 2,
})
if err != nil {
t.Fatal(err)
}
env, err := store.CreateEnvironment(db.Environment{
ProjectID: proj.ID,
Name: "test",
JSON: `{"author": "Denis", "comment": "Hello, World!"}`,
})
if err != nil {
t.Fatal(err)
}
tpl, err := store.CreateTemplate(db.Template{
Name: "Test",
Playbook: "test.yml",
ProjectID: proj.ID,
RepositoryID: repo.ID,
InventoryID: &inv.ID,
EnvironmentID: &env.ID,
})
if err != nil {
t.Fatal(err)
}
pool := TaskPool{store: store}
tsk := TaskRunner{
pool: &pool,
Task: db.Task{
TemplateID: tpl.ID,
ProjectID: proj.ID,
Environment: `{"comment": "Just do it!", "time": "2021-11-02"}`,
InventoryID: &inv2.ID,
},
}
tsk.job = &LocalJob{
Task: tsk.Task,
Template: tsk.Template,
Repository: tsk.Repository,
Environment: tsk.Environment,
Logger: &tsk,
App: &db_lib.AnsibleApp{
Template: tsk.Template,
Repository: tsk.Repository,
Logger: &tsk,
Playbook: &db_lib.AnsiblePlaybook{
Logger: &tsk,
TemplateID: tsk.Template.ID,
Repository: tsk.Repository,
},
},
}
err = tsk.populateDetails()
if err != nil {
t.Fatal(err)
}
if tsk.Inventory.ID != 2 {
t.Fatal(err)
}
}
func TestPopulateDetailsInventory1(t *testing.T) {
store := bolt.CreateTestStore()
proj, err := store.CreateProject(db.Project{})
if err != nil {
t.Fatal(err)
}
key, err := store.CreateAccessKey(db.AccessKey{
ProjectID: &proj.ID,
Type: db.AccessKeyNone,
})
if err != nil {
t.Fatal(err)
}
repo, err := store.CreateRepository(db.Repository{
ProjectID: proj.ID,
SSHKeyID: key.ID,
Name: "Test",
GitURL: "git@example.com:test/test",
GitBranch: "master",
})
if err != nil {
t.Fatal(err)
}
inv, err := store.CreateInventory(db.Inventory{
ProjectID: proj.ID,
ID: 1,
})
if err != nil {
t.Fatal(err)
}
env, err := store.CreateEnvironment(db.Environment{
ProjectID: proj.ID,
Name: "test",
JSON: `{"author": "Denis", "comment": "Hello, World!"}`,
})
if err != nil {
t.Fatal(err)
}
tpl, err := store.CreateTemplate(db.Template{
Name: "Test",
Playbook: "test.yml",
ProjectID: proj.ID,
RepositoryID: repo.ID,
InventoryID: &inv.ID,
EnvironmentID: &env.ID,
})
if err != nil {
t.Fatal(err)
}
pool := TaskPool{store: store}
tsk := TaskRunner{
pool: &pool,
Task: db.Task{
TemplateID: tpl.ID,
ProjectID: proj.ID,
Environment: `{"comment": "Just do it!", "time": "2021-11-02"}`,
},
}
tsk.job = &LocalJob{
Task: tsk.Task,
Template: tsk.Template,
Repository: tsk.Repository,
Environment: tsk.Environment,
Logger: &tsk,
App: &db_lib.AnsibleApp{
Template: tsk.Template,
Repository: tsk.Repository,
Logger: &tsk,
Playbook: &db_lib.AnsiblePlaybook{
Logger: &tsk,
TemplateID: tsk.Template.ID,
Repository: tsk.Repository,
},
},
}
err = tsk.populateDetails()
if err != nil {
t.Fatal(err)
}
if tsk.Inventory.ID != 1 {
t.Fatal(err)
}
}
2021-09-13 15:22:08 +02:00
func TestTaskGetPlaybookArgs(t *testing.T) {
util.Config = &util.ConfigType{
TmpPath: "/tmp",
}
inventoryID := 1
tsk := TaskRunner{
Task: db.Task{},
Inventory: db.Inventory{
2021-09-13 15:22:08 +02:00
SSHKeyID: &inventoryID,
SSHKey: db.AccessKey{
ID: 12345,
2021-09-13 15:22:08 +02:00
Type: db.AccessKeySSH,
},
Type: db.InventoryStatic,
2021-09-13 15:22:08 +02:00
},
Template: db.Template{
2021-09-13 15:22:08 +02:00
Playbook: "test.yml",
},
}
tsk.job = &LocalJob{
Task: tsk.Task,
Template: tsk.Template,
Inventory: tsk.Inventory,
Repository: tsk.Repository,
Environment: tsk.Environment,
Logger: &tsk,
App: &db_lib.AnsibleApp{
Template: tsk.Template,
Repository: tsk.Repository,
Logger: &tsk,
Playbook: &db_lib.AnsiblePlaybook{
Logger: &tsk,
TemplateID: tsk.Template.ID,
Repository: tsk.Repository,
},
},
}
2021-09-13 15:22:08 +02:00
args, _, err := tsk.job.(*LocalJob).getPlaybookArgs("", nil)
2021-09-13 15:22:08 +02:00
if err != nil {
t.Fatal(err)
}
res := strings.Join(args, " ")
2024-04-19 15:24:24 +02:00
if res != "-i /tmp/inventory_0 --extra-vars {\"semaphore_vars\":{\"task_details\":{\"id\":0,\"url\":null,\"username\":\"\"}}} test.yml" {
2021-09-13 15:22:08 +02:00
t.Fatal("incorrect result")
}
}
func TestTaskGetPlaybookArgs2(t *testing.T) {
util.Config = &util.ConfigType{
TmpPath: "/tmp",
}
inventoryID := 1
tsk := TaskRunner{
Task: db.Task{},
Inventory: db.Inventory{
Type: db.InventoryStatic,
2021-09-13 15:22:08 +02:00
SSHKeyID: &inventoryID,
SSHKey: db.AccessKey{
ID: 12345,
2021-09-13 15:22:08 +02:00
Type: db.AccessKeyLoginPassword,
LoginPassword: db.LoginPassword{
Password: "123456",
Login: "root",
2021-09-13 15:22:08 +02:00
},
},
},
Template: db.Template{
2021-09-13 15:22:08 +02:00
Playbook: "test.yml",
},
}
tsk.job = &LocalJob{
Task: tsk.Task,
Template: tsk.Template,
Inventory: tsk.Inventory,
Repository: tsk.Repository,
Environment: tsk.Environment,
Logger: &tsk,
App: &db_lib.AnsibleApp{
Template: tsk.Template,
Repository: tsk.Repository,
Logger: &tsk,
Playbook: &db_lib.AnsiblePlaybook{
Logger: &tsk,
TemplateID: tsk.Template.ID,
Repository: tsk.Repository,
},
},
}
2021-09-13 15:22:08 +02:00
args, _, err := tsk.job.(*LocalJob).getPlaybookArgs("", nil)
2021-09-13 15:22:08 +02:00
if err != nil {
t.Fatal(err)
}
res := strings.Join(args, " ")
2024-04-19 15:24:24 +02:00
if res != "-i /tmp/inventory_0 --extra-vars {\"semaphore_vars\":{\"task_details\":{\"id\":0,\"url\":null,\"username\":\"\"}}} test.yml" {
2021-09-13 15:22:08 +02:00
t.Fatal("incorrect result")
}
}
func TestTaskGetPlaybookArgs3(t *testing.T) {
util.Config = &util.ConfigType{
TmpPath: "/tmp",
}
inventoryID := 1
tsk := TaskRunner{
Task: db.Task{},
Inventory: db.Inventory{
Type: db.InventoryStatic,
2021-09-13 15:22:08 +02:00
BecomeKeyID: &inventoryID,
BecomeKey: db.AccessKey{
ID: 12345,
2021-09-13 15:22:08 +02:00
Type: db.AccessKeyLoginPassword,
LoginPassword: db.LoginPassword{
Password: "123456",
Login: "root",
2021-09-13 15:22:08 +02:00
},
},
},
Template: db.Template{
2021-09-13 15:22:08 +02:00
Playbook: "test.yml",
},
}
tsk.job = &LocalJob{
Task: tsk.Task,
Template: tsk.Template,
Inventory: tsk.Inventory,
Repository: tsk.Repository,
Environment: tsk.Environment,
Logger: &tsk,
App: &db_lib.AnsibleApp{
Template: tsk.Template,
Repository: tsk.Repository,
Logger: &tsk,
Playbook: &db_lib.AnsiblePlaybook{
Logger: &tsk,
TemplateID: tsk.Template.ID,
Repository: tsk.Repository,
},
},
}
2021-09-13 15:22:08 +02:00
args, _, err := tsk.job.(*LocalJob).getPlaybookArgs("", nil)
2021-09-13 15:22:08 +02:00
if err != nil {
t.Fatal(err)
}
res := strings.Join(args, " ")
2024-04-19 15:24:24 +02:00
if res != "-i /tmp/inventory_0 --extra-vars {\"semaphore_vars\":{\"task_details\":{\"id\":0,\"url\":null,\"username\":\"\"}}} test.yml" {
2021-09-13 15:22:08 +02:00
t.Fatal("incorrect result")
}
}
func TestCheckTmpDir(t *testing.T) {
//It should be able to create a random dir in /tmp
dirName := path.Join(os.TempDir(), util.RandString(rand.Intn(10-4)+4))
err := checkTmpDir(dirName)
if err != nil {
t.Fatal(err)
}
//checking again for this directory should return no error, as it exists
err = checkTmpDir(dirName)
if err != nil {
t.Fatal(err)
}
err = os.Chmod(dirName, os.FileMode(int(0550)))
if err != nil {
t.Fatal(err)
}
//nolint: vetshadow
if stat, err := os.Stat(dirName); err != nil {
t.Fatal(err)
} else if stat.Mode() != os.FileMode(int(0550)) {
// File System is not support 0550 mode, skip this test
return
}
err = checkTmpDir(dirName + "/noway")
if err == nil {
t.Fatal("You should not be able to write in this folder, causing an error")
}
err = os.Remove(dirName)
if err != nil {
t.Log(err)
}
}