mirror of
https://github.com/semaphoreui/semaphore.git
synced 2024-12-03 14:51:05 +01:00
211 lines
5.5 KiB
Go
211 lines
5.5 KiB
Go
package sql
|
|
|
|
import (
|
|
"fmt"
|
|
"github.com/go-gorp/gorp/v3"
|
|
"path"
|
|
"regexp"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/ansible-semaphore/semaphore/db"
|
|
log "github.com/sirupsen/logrus"
|
|
)
|
|
|
|
var (
|
|
autoIncrementRE = regexp.MustCompile(`(?i)\bautoincrement\b`)
|
|
serialRE = regexp.MustCompile(`(?i)\binteger primary key autoincrement\b`)
|
|
dateTimeTypeRE = regexp.MustCompile(`(?i)\bdatetime\b`)
|
|
tinyintRE = regexp.MustCompile(`(?i)\btinyint\b`)
|
|
longtextRE = regexp.MustCompile(`(?i)\blongtext\b`)
|
|
ifExistsRE = regexp.MustCompile(`(?i)\bif exists\b`)
|
|
changeRE = regexp.MustCompile(`^alter table \x60(\w+)\x60 change \x60(\w+)\x60 \x60(\w+)\x60 ([\w\(\)]+)( not null)?$`)
|
|
//dropForeignKeyRE = regexp.MustCompile(`^alter table \x60(\w+)\x60 drop foreign key \x60(\w+)\x60 /\* postgres:\x60(\w*)\x60 mysql:\x60(\w*)\x60 \*/$`)
|
|
dropForeignKey2RE = regexp.MustCompile(`(?i)\bdrop foreign key\b`)
|
|
)
|
|
|
|
// getVersionPath is the humanoid version with the file format appended
|
|
func getVersionPath(version db.Migration) string {
|
|
return version.HumanoidVersion() + ".sql"
|
|
}
|
|
|
|
// getVersionErrPath is the humanoid version with '.err' and file format appended
|
|
func getVersionErrPath(version db.Migration) string {
|
|
return version.HumanoidVersion() + ".err.sql"
|
|
}
|
|
|
|
// getVersionSQL takes a path to an SQL file and returns it from embed.FS
|
|
// a slice of strings separated by newlines
|
|
func getVersionSQL(name string) (queries []string) {
|
|
sql, err := dbAssets.ReadFile(path.Join("migrations", name))
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
queries = strings.Split(strings.ReplaceAll(string(sql), ";\r\n", ";\n"), ";\n")
|
|
for i := range queries {
|
|
queries[i] = strings.Trim(queries[i], "\r\n\t ")
|
|
}
|
|
return
|
|
}
|
|
|
|
// prepareMigration converts migration SQLite-query to current dialect.
|
|
// Supported MySQL and Postgres dialects.
|
|
func (d *SqlDb) prepareMigration(query string) string {
|
|
switch d.sql.Dialect.(type) {
|
|
case gorp.MySQLDialect:
|
|
query = autoIncrementRE.ReplaceAllString(query, "auto_increment")
|
|
query = ifExistsRE.ReplaceAllString(query, "")
|
|
case gorp.PostgresDialect:
|
|
m := changeRE.FindStringSubmatch(query)
|
|
if m != nil {
|
|
tableName := m[1]
|
|
oldColumnName := m[2]
|
|
newColumnName := m[3]
|
|
columnType := m[4]
|
|
columnNotNull := m[5] != ""
|
|
|
|
var queries []string
|
|
queries = append(queries,
|
|
"alter table `"+tableName+"` alter column `"+oldColumnName+"` type "+columnType)
|
|
|
|
if columnNotNull {
|
|
queries = append(queries,
|
|
"alter table `"+tableName+"` alter column `"+oldColumnName+"` set not null")
|
|
} else {
|
|
queries = append(queries,
|
|
"alter table `"+tableName+"` alter column `"+oldColumnName+"` drop not null")
|
|
}
|
|
|
|
if oldColumnName != newColumnName {
|
|
queries = append(queries,
|
|
"alter table `"+tableName+"` rename column `"+oldColumnName+"` to `"+newColumnName+"`")
|
|
}
|
|
|
|
query = strings.Join(queries, "; ")
|
|
}
|
|
|
|
query = dateTimeTypeRE.ReplaceAllString(query, "timestamp")
|
|
query = tinyintRE.ReplaceAllString(query, "smallint")
|
|
query = longtextRE.ReplaceAllString(query, "text")
|
|
query = serialRE.ReplaceAllString(query, "serial primary key")
|
|
query = dropForeignKey2RE.ReplaceAllString(query, "drop constraint")
|
|
query = identifierQuoteRE.ReplaceAllString(query, "\"")
|
|
}
|
|
return query
|
|
}
|
|
|
|
// IsMigrationApplied queries the database to see if a migration table with this version id exists already
|
|
func (d *SqlDb) IsMigrationApplied(migration db.Migration) (bool, error) {
|
|
initialized, err := d.IsInitialized()
|
|
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
|
|
if !initialized {
|
|
return false, nil
|
|
}
|
|
|
|
exists, err := d.sql.SelectInt(
|
|
d.PrepareQuery("select count(1) as ex from migrations where version = ?"),
|
|
migration.Version)
|
|
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
|
|
return exists > 0, nil
|
|
}
|
|
|
|
// ApplyMigration runs executes a database migration
|
|
func (d *SqlDb) ApplyMigration(migration db.Migration) error {
|
|
initialized, err := d.IsInitialized()
|
|
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if !initialized {
|
|
fmt.Println("Creating migrations table")
|
|
query := d.prepareMigration(initialSQL)
|
|
if query == "" {
|
|
return nil
|
|
}
|
|
_, err = d.exec(query)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
tx, err := d.sql.Begin()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
queries := getVersionSQL(getVersionPath(migration))
|
|
for i, query := range queries {
|
|
fmt.Printf("\r [%d/%d]", i+1, len(query))
|
|
|
|
if len(query) == 0 {
|
|
continue
|
|
}
|
|
|
|
q := d.prepareMigration(query)
|
|
if q == "" {
|
|
continue
|
|
}
|
|
|
|
_, err = tx.Exec(q)
|
|
if err != nil {
|
|
handleRollbackError(tx.Rollback())
|
|
log.Warnf("\n ERR! Query: %s\n\n", q)
|
|
log.Fatalf(err.Error())
|
|
return err
|
|
}
|
|
}
|
|
|
|
_, err = tx.Exec(d.PrepareQuery("insert into migrations(version, upgraded_date) values (?, ?)"), migration.Version, time.Now())
|
|
if err != nil {
|
|
handleRollbackError(tx.Rollback())
|
|
return err
|
|
}
|
|
|
|
switch migration.Version {
|
|
case "2.8.26":
|
|
err = migration_2_8_26{db: d}.Apply(tx)
|
|
case "2.8.42":
|
|
err = migration_2_8_42{db: d}.Apply(tx)
|
|
}
|
|
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
fmt.Println()
|
|
|
|
return tx.Commit()
|
|
}
|
|
|
|
// TryRollbackMigration attempts to rollback the database to an earlier version if a rollback exists
|
|
func (d *SqlDb) TryRollbackMigration(version db.Migration) {
|
|
data, _ := dbAssets.ReadFile(getVersionErrPath(version))
|
|
if len(data) == 0 {
|
|
fmt.Println("Rollback SQL does not exist.")
|
|
fmt.Println()
|
|
return
|
|
}
|
|
|
|
queries := getVersionSQL(getVersionErrPath(version))
|
|
for _, query := range queries {
|
|
fmt.Printf(" [ROLLBACK] > %v\n", query)
|
|
q := d.prepareMigration(query)
|
|
if q == "" {
|
|
continue
|
|
}
|
|
if _, err := d.exec(q); err != nil {
|
|
fmt.Println(" [ROLLBACK] - Stopping")
|
|
return
|
|
}
|
|
}
|
|
}
|