2024-01-20 23:46:43 +01:00
|
|
|
package project
|
|
|
|
|
|
|
|
import (
|
2024-10-06 00:16:12 +02:00
|
|
|
"encoding/json"
|
2024-01-20 23:46:43 +01:00
|
|
|
"fmt"
|
2024-10-06 00:16:12 +02:00
|
|
|
"reflect"
|
2024-04-12 10:00:44 +02:00
|
|
|
|
2024-10-26 14:56:17 +02:00
|
|
|
"github.com/semaphoreui/semaphore/db"
|
|
|
|
"github.com/semaphoreui/semaphore/pkg/random"
|
2024-01-20 23:46:43 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
func findNameByID[T db.BackupEntity](ID int, items []T) (*string, error) {
|
|
|
|
for _, o := range items {
|
|
|
|
if o.GetID() == ID {
|
|
|
|
name := o.GetName()
|
|
|
|
return &name, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil, fmt.Errorf("item %d does not exist", ID)
|
|
|
|
}
|
|
|
|
func findEntityByName[T db.BackupEntity](name *string, items []T) *T {
|
|
|
|
if name == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
for _, o := range items {
|
|
|
|
if o.GetName() == *name {
|
|
|
|
return &o
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func getSchedulesByProject(projectID int, schedules []db.Schedule) []db.Schedule {
|
|
|
|
result := make([]db.Schedule, 0)
|
|
|
|
for _, o := range schedules {
|
|
|
|
if o.ProjectID == projectID {
|
|
|
|
result = append(result, o)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
|
|
|
func getScheduleByTemplate(templateID int, schedules []db.Schedule) *string {
|
|
|
|
for _, o := range schedules {
|
|
|
|
if o.TemplateID == templateID {
|
|
|
|
return &o.CronFormat
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2024-02-10 14:21:39 +01:00
|
|
|
func getRandomName(name string) string {
|
2024-04-12 10:00:44 +02:00
|
|
|
return name + " - " + random.String(10)
|
2024-02-10 14:21:39 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func makeUniqueNames[T any](items []T, getter func(item *T) string, setter func(item *T, name string)) {
|
|
|
|
for i := len(items) - 1; i >= 0; i-- {
|
|
|
|
for k, other := range items {
|
|
|
|
if k == i {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
|
|
|
name := getter(&items[i])
|
|
|
|
|
|
|
|
if name == getter(&other) {
|
|
|
|
randomName := getRandomName(name)
|
|
|
|
setter(&items[i], randomName)
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *BackupDB) makeUniqueNames() {
|
|
|
|
|
|
|
|
makeUniqueNames(b.templates, func(item *db.Template) string {
|
|
|
|
return item.Name
|
|
|
|
}, func(item *db.Template, name string) {
|
|
|
|
item.Name = name
|
|
|
|
})
|
|
|
|
|
|
|
|
makeUniqueNames(b.repositories, func(item *db.Repository) string {
|
|
|
|
return item.Name
|
|
|
|
}, func(item *db.Repository, name string) {
|
|
|
|
item.Name = name
|
|
|
|
})
|
|
|
|
|
2024-02-10 14:23:11 +01:00
|
|
|
makeUniqueNames(b.inventories, func(item *db.Inventory) string {
|
|
|
|
return item.Name
|
|
|
|
}, func(item *db.Inventory, name string) {
|
|
|
|
item.Name = name
|
|
|
|
})
|
|
|
|
|
|
|
|
makeUniqueNames(b.environments, func(item *db.Environment) string {
|
|
|
|
return item.Name
|
|
|
|
}, func(item *db.Environment, name string) {
|
|
|
|
item.Name = name
|
|
|
|
})
|
|
|
|
|
2024-02-10 14:21:39 +01:00
|
|
|
makeUniqueNames(b.keys, func(item *db.AccessKey) string {
|
|
|
|
return item.Name
|
|
|
|
}, func(item *db.AccessKey, name string) {
|
|
|
|
item.Name = name
|
|
|
|
})
|
2024-02-10 14:23:11 +01:00
|
|
|
|
|
|
|
makeUniqueNames(b.views, func(item *db.View) string {
|
|
|
|
return item.Title
|
|
|
|
}, func(item *db.View, name string) {
|
|
|
|
item.Title = name
|
|
|
|
})
|
2024-10-07 19:53:51 +02:00
|
|
|
|
|
|
|
makeUniqueNames(b.integrations, func(item *db.Integration) string {
|
|
|
|
return item.Name
|
|
|
|
}, func(item *db.Integration, name string) {
|
|
|
|
item.Name = name
|
|
|
|
})
|
2024-02-10 14:21:39 +01:00
|
|
|
}
|
|
|
|
|
2024-10-07 19:53:51 +02:00
|
|
|
func (b *BackupDB) load(projectID int, store db.Store) (err error) {
|
2024-01-20 23:46:43 +01:00
|
|
|
|
|
|
|
b.templates, err = store.GetTemplates(projectID, db.TemplateFilter{}, db.RetrieveQueryParams{})
|
|
|
|
if err != nil {
|
2024-10-07 19:53:51 +02:00
|
|
|
return
|
2024-01-20 23:46:43 +01:00
|
|
|
}
|
|
|
|
|
2024-10-29 13:45:05 +01:00
|
|
|
for i := range b.templates {
|
|
|
|
var vaults []db.TemplateVault
|
|
|
|
vaults, err = store.GetTemplateVaults(b.templates[i].ProjectID, b.templates[i].ID)
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
b.templates[i].Vaults = vaults
|
|
|
|
}
|
|
|
|
|
2024-01-20 23:46:43 +01:00
|
|
|
b.repositories, err = store.GetRepositories(projectID, db.RetrieveQueryParams{})
|
|
|
|
if err != nil {
|
2024-10-07 19:53:51 +02:00
|
|
|
return
|
2024-01-20 23:46:43 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
b.keys, err = store.GetAccessKeys(projectID, db.RetrieveQueryParams{})
|
|
|
|
if err != nil {
|
2024-10-07 19:53:51 +02:00
|
|
|
return
|
2024-01-20 23:46:43 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
b.views, err = store.GetViews(projectID)
|
|
|
|
if err != nil {
|
2024-10-07 19:53:51 +02:00
|
|
|
return
|
2024-01-20 23:46:43 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
b.inventories, err = store.GetInventories(projectID, db.RetrieveQueryParams{})
|
|
|
|
if err != nil {
|
2024-10-07 19:53:51 +02:00
|
|
|
return
|
2024-01-20 23:46:43 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
b.environments, err = store.GetEnvironments(projectID, db.RetrieveQueryParams{})
|
|
|
|
if err != nil {
|
2024-10-07 19:53:51 +02:00
|
|
|
return
|
2024-01-20 23:46:43 +01:00
|
|
|
}
|
2024-10-07 19:53:51 +02:00
|
|
|
|
2024-01-20 23:46:43 +01:00
|
|
|
schedules, err := store.GetSchedules()
|
|
|
|
if err != nil {
|
2024-10-07 19:53:51 +02:00
|
|
|
return
|
2024-01-20 23:46:43 +01:00
|
|
|
}
|
2024-10-07 19:53:51 +02:00
|
|
|
|
2024-01-20 23:46:43 +01:00
|
|
|
b.schedules = getSchedulesByProject(projectID, schedules)
|
2024-10-07 19:53:51 +02:00
|
|
|
|
2024-01-20 23:46:43 +01:00
|
|
|
b.meta, err = store.GetProject(projectID)
|
|
|
|
if err != nil {
|
2024-10-07 19:53:51 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
b.integrationProjAliases, err = store.GetIntegrationAliases(projectID, nil)
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
b.integrations, err = store.GetIntegrations(projectID, db.RetrieveQueryParams{})
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
b.integrationAliases = make(map[int][]db.IntegrationAlias)
|
|
|
|
b.integrationMatchers = make(map[int][]db.IntegrationMatcher)
|
|
|
|
b.integrationExtractValues = make(map[int][]db.IntegrationExtractValue)
|
|
|
|
for _, o := range b.integrations {
|
|
|
|
b.integrationAliases[o.ID], err = store.GetIntegrationAliases(projectID, &o.ID)
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
b.integrationMatchers[o.ID], err = store.GetIntegrationMatchers(projectID, db.RetrieveQueryParams{}, o.ID)
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
b.integrationExtractValues[o.ID], err = store.GetIntegrationExtractValues(projectID, db.RetrieveQueryParams{}, o.ID)
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
2024-01-20 23:46:43 +01:00
|
|
|
}
|
2024-02-10 14:21:39 +01:00
|
|
|
|
|
|
|
b.makeUniqueNames()
|
|
|
|
|
2024-10-07 19:53:51 +02:00
|
|
|
return
|
2024-01-20 23:46:43 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func (b *BackupDB) format() (*BackupFormat, error) {
|
2024-10-05 22:16:25 +02:00
|
|
|
keys := make([]BackupAccessKey, len(b.keys))
|
2024-01-20 23:46:43 +01:00
|
|
|
for i, o := range b.keys {
|
2024-10-05 22:16:25 +02:00
|
|
|
keys[i] = BackupAccessKey{
|
|
|
|
o,
|
2024-01-20 23:46:43 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
environments := make([]BackupEnvironment, len(b.environments))
|
|
|
|
for i, o := range b.environments {
|
|
|
|
environments[i] = BackupEnvironment{
|
2024-10-05 22:16:25 +02:00
|
|
|
o,
|
2024-01-20 23:46:43 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
inventories := make([]BackupInventory, len(b.inventories))
|
|
|
|
for i, o := range b.inventories {
|
|
|
|
var SSHKey *string = nil
|
|
|
|
if o.SSHKeyID != nil {
|
|
|
|
SSHKey, _ = findNameByID[db.AccessKey](*o.SSHKeyID, b.keys)
|
|
|
|
}
|
|
|
|
var BecomeKey *string = nil
|
|
|
|
if o.BecomeKeyID != nil {
|
|
|
|
BecomeKey, _ = findNameByID[db.AccessKey](*o.BecomeKeyID, b.keys)
|
|
|
|
}
|
|
|
|
inventories[i] = BackupInventory{
|
2024-10-05 22:16:25 +02:00
|
|
|
Inventory: o,
|
2024-01-20 23:46:43 +01:00
|
|
|
SSHKey: SSHKey,
|
|
|
|
BecomeKey: BecomeKey,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
views := make([]BackupView, len(b.views))
|
|
|
|
for i, o := range b.views {
|
|
|
|
views[i] = BackupView{
|
2024-10-05 22:16:25 +02:00
|
|
|
o,
|
2024-01-20 23:46:43 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
repositories := make([]BackupRepository, len(b.repositories))
|
|
|
|
for i, o := range b.repositories {
|
|
|
|
SSHKey, _ := findNameByID[db.AccessKey](o.SSHKeyID, b.keys)
|
|
|
|
repositories[i] = BackupRepository{
|
2024-10-05 22:16:25 +02:00
|
|
|
Repository: o,
|
|
|
|
SSHKey: SSHKey,
|
2024-01-20 23:46:43 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
templates := make([]BackupTemplate, len(b.templates))
|
|
|
|
for i, o := range b.templates {
|
|
|
|
var View *string = nil
|
|
|
|
if o.ViewID != nil {
|
|
|
|
View, _ = findNameByID[db.View](*o.ViewID, b.views)
|
|
|
|
}
|
2024-10-03 21:41:36 +02:00
|
|
|
var vaults []BackupTemplateVault = nil
|
|
|
|
for _, vault := range o.Vaults {
|
|
|
|
var vaultKey *string = nil
|
2024-10-22 21:50:31 +02:00
|
|
|
if vault.VaultKeyID != nil {
|
|
|
|
vaultKey, _ = findNameByID[db.AccessKey](*vault.VaultKeyID, b.keys)
|
|
|
|
}
|
2024-10-03 21:41:36 +02:00
|
|
|
vaults = append(vaults, BackupTemplateVault{
|
2024-10-05 22:16:25 +02:00
|
|
|
TemplateVault: vault,
|
2024-10-22 21:50:31 +02:00
|
|
|
VaultKey: vaultKey,
|
2024-10-03 21:41:36 +02:00
|
|
|
})
|
|
|
|
|
2024-01-20 23:46:43 +01:00
|
|
|
}
|
|
|
|
var Environment *string = nil
|
|
|
|
if o.EnvironmentID != nil {
|
|
|
|
Environment, _ = findNameByID[db.Environment](*o.EnvironmentID, b.environments)
|
|
|
|
}
|
|
|
|
var BuildTemplate *string = nil
|
|
|
|
if o.BuildTemplateID != nil {
|
|
|
|
BuildTemplate, _ = findNameByID[db.Template](*o.BuildTemplateID, b.templates)
|
|
|
|
}
|
|
|
|
Repository, _ := findNameByID[db.Repository](o.RepositoryID, b.repositories)
|
2024-04-19 18:47:08 +02:00
|
|
|
|
|
|
|
var Inventory *string = nil
|
|
|
|
if o.InventoryID != nil {
|
|
|
|
Inventory, _ = findNameByID[db.Inventory](*o.InventoryID, b.inventories)
|
|
|
|
}
|
2024-01-20 23:46:43 +01:00
|
|
|
|
|
|
|
templates[i] = BackupTemplate{
|
2024-10-05 22:16:25 +02:00
|
|
|
Template: o,
|
|
|
|
View: View,
|
|
|
|
Repository: *Repository,
|
|
|
|
Inventory: Inventory,
|
|
|
|
Environment: Environment,
|
|
|
|
BuildTemplate: BuildTemplate,
|
|
|
|
Cron: getScheduleByTemplate(o.ID, b.schedules),
|
|
|
|
Vaults: vaults,
|
2024-01-20 23:46:43 +01:00
|
|
|
}
|
|
|
|
}
|
2024-10-07 19:53:51 +02:00
|
|
|
|
|
|
|
integrations := make([]BackupIntegration, len(b.integrations))
|
|
|
|
for i, o := range b.integrations {
|
|
|
|
|
|
|
|
var aliases []string
|
|
|
|
|
|
|
|
for _, a := range b.integrationAliases[o.ID] {
|
|
|
|
aliases = append(aliases, a.Alias)
|
|
|
|
}
|
|
|
|
|
|
|
|
tplName, _ := findNameByID[db.Template](o.TemplateID, b.templates)
|
|
|
|
|
|
|
|
if tplName == nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
var keyName *string
|
|
|
|
|
|
|
|
if o.AuthSecretID != nil {
|
|
|
|
keyName, _ = findNameByID[db.AccessKey](*o.AuthSecretID, b.keys)
|
|
|
|
}
|
|
|
|
|
|
|
|
integrations[i] = BackupIntegration{
|
|
|
|
Integration: o,
|
|
|
|
Aliases: aliases,
|
|
|
|
Matchers: b.integrationMatchers[o.ID],
|
|
|
|
ExtractValues: b.integrationExtractValues[o.ID],
|
|
|
|
Template: *tplName,
|
|
|
|
AuthSecret: keyName,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
var integrationAliases []string
|
|
|
|
|
|
|
|
for _, alias := range b.integrationProjAliases {
|
|
|
|
integrationAliases = append(integrationAliases, alias.Alias)
|
|
|
|
}
|
|
|
|
|
2024-01-20 23:46:43 +01:00
|
|
|
return &BackupFormat{
|
|
|
|
Meta: BackupMeta{
|
2024-10-05 22:16:25 +02:00
|
|
|
b.meta,
|
2024-01-20 23:46:43 +01:00
|
|
|
},
|
2024-10-07 19:53:51 +02:00
|
|
|
Inventories: inventories,
|
|
|
|
Environments: environments,
|
|
|
|
Views: views,
|
|
|
|
Repositories: repositories,
|
|
|
|
Keys: keys,
|
|
|
|
Templates: templates,
|
|
|
|
Integration: integrations,
|
|
|
|
IntegrationAliases: integrationAliases,
|
2024-01-20 23:46:43 +01:00
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func GetBackup(projectID int, store db.Store) (*BackupFormat, error) {
|
|
|
|
backup := BackupDB{}
|
2024-10-07 19:53:51 +02:00
|
|
|
if err := backup.load(projectID, store); err != nil {
|
2024-01-20 23:46:43 +01:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return backup.format()
|
|
|
|
}
|
2024-10-06 00:16:12 +02:00
|
|
|
|
|
|
|
func (b *BackupFormat) Marshal() (res string, err error) {
|
|
|
|
data, err := marshalValue(reflect.ValueOf(b))
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
bytes, err := json.Marshal(data)
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
res = string(bytes)
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *BackupFormat) Unmarshal(res string) (err error) {
|
2024-10-06 00:58:18 +02:00
|
|
|
// Parse the JSON data into a map
|
|
|
|
var jsonData interface{}
|
|
|
|
if err = json.Unmarshal([]byte(res), &jsonData); err != nil {
|
|
|
|
return
|
|
|
|
}
|
2024-10-06 00:16:12 +02:00
|
|
|
|
2024-10-06 00:58:18 +02:00
|
|
|
// Start the recursive unmarshaling process
|
|
|
|
err = unmarshalValueWithBackupTags(jsonData, reflect.ValueOf(b))
|
2024-10-06 00:16:12 +02:00
|
|
|
return
|
|
|
|
}
|