Migrate to Xormigrate (#2711)

Co-authored-by: Anbraten <anton@ju60.de>
This commit is contained in:
qwerty287 2023-11-28 10:31:54 +01:00 committed by GitHub
parent 8dd74acdee
commit 7bacbd5699
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
40 changed files with 305 additions and 347 deletions

View file

@ -52,8 +52,6 @@ import (
) )
func setupStore(c *cli.Context) (store.Store, error) { func setupStore(c *cli.Context) (store.Store, error) {
// TODO: find a better way than global var to pass down to allow long migrations
server.Config.Server.Migrations.AllowLong = c.Bool("migrations-allow-long")
datasource := c.String("datasource") datasource := c.String("datasource")
driver := c.String("driver") driver := c.String("driver")
xorm := store.XORM{ xorm := store.XORM{
@ -90,7 +88,7 @@ func setupStore(c *cli.Context) (store.Store, error) {
log.Fatal().Err(err).Msg("could not open datastore") log.Fatal().Err(err).Msg("could not open datastore")
} }
if err := store.Migrate(); err != nil { if err := store.Migrate(c.Bool("migrations-allow-long")); err != nil {
log.Fatal().Err(err).Msg("could not migrate datastore") log.Fatal().Err(err).Msg("could not migrate datastore")
} }

1
go.mod
View file

@ -61,6 +61,7 @@ require (
k8s.io/api v0.28.4 k8s.io/api v0.28.4
k8s.io/apimachinery v0.28.4 k8s.io/apimachinery v0.28.4
k8s.io/client-go v0.28.4 k8s.io/client-go v0.28.4
src.techknowlogick.com/xormigrate v1.7.1
xorm.io/builder v0.3.13 xorm.io/builder v0.3.13
xorm.io/xorm v1.3.4 xorm.io/xorm v1.3.4
) )

7
go.sum
View file

@ -62,6 +62,7 @@ github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
github.com/davidmz/go-pageant v1.0.2 h1:bPblRCh5jGU+Uptpz6LgMZGD5hJoOt7otgT454WvHn0= github.com/davidmz/go-pageant v1.0.2 h1:bPblRCh5jGU+Uptpz6LgMZGD5hJoOt7otgT454WvHn0=
github.com/davidmz/go-pageant v1.0.2/go.mod h1:P2EDDnMqIwG5Rrp05dTRITj9z2zpGcD9efWSkTNKLIE= github.com/davidmz/go-pageant v1.0.2/go.mod h1:P2EDDnMqIwG5Rrp05dTRITj9z2zpGcD9efWSkTNKLIE=
github.com/denisenkom/go-mssqldb v0.12.3 h1:pBSGx9Tq67pBOTLmxNuirNTeB8Vjmf886Kx+8Y+8shw=
github.com/denisenkom/go-mssqldb v0.12.3/go.mod h1:k0mtMFOnU+AihqFxPMiF05rtiDrorD1Vrm1KEz5hxDo= github.com/denisenkom/go-mssqldb v0.12.3/go.mod h1:k0mtMFOnU+AihqFxPMiF05rtiDrorD1Vrm1KEz5hxDo=
github.com/distribution/reference v0.5.0 h1:/FUIFXtfc/x2gpa5/VGfiGLuOIdYa1t65IKK2OFGvA0= github.com/distribution/reference v0.5.0 h1:/FUIFXtfc/x2gpa5/VGfiGLuOIdYa1t65IKK2OFGvA0=
github.com/distribution/reference v0.5.0/go.mod h1:BbU0aIcezP1/5jX/8MP0YiH4SdvB5Y4f/wlDRiLyi3E= github.com/distribution/reference v0.5.0/go.mod h1:BbU0aIcezP1/5jX/8MP0YiH4SdvB5Y4f/wlDRiLyi3E=
@ -145,7 +146,9 @@ github.com/gogo/protobuf v1.3.2 h1:Ov1cvc58UF3b5XjBnZv7+opcTcQFZebYjWzi34vdm4Q=
github.com/gogo/protobuf v1.3.2/go.mod h1:P1XiOD3dCwIKUDQYPy72D8LYyHL2YPYrpS2s69NZV8Q= github.com/gogo/protobuf v1.3.2/go.mod h1:P1XiOD3dCwIKUDQYPy72D8LYyHL2YPYrpS2s69NZV8Q=
github.com/golang-jwt/jwt/v5 v5.1.0 h1:UGKbA/IPjtS6zLcdB7i5TyACMgSbOTiR8qzXgw8HWQU= github.com/golang-jwt/jwt/v5 v5.1.0 h1:UGKbA/IPjtS6zLcdB7i5TyACMgSbOTiR8qzXgw8HWQU=
github.com/golang-jwt/jwt/v5 v5.1.0/go.mod h1:pqrtFR0X4osieyHYxtmOUWsAWrfe1Q5UVIyoH402zdk= github.com/golang-jwt/jwt/v5 v5.1.0/go.mod h1:pqrtFR0X4osieyHYxtmOUWsAWrfe1Q5UVIyoH402zdk=
github.com/golang-sql/civil v0.0.0-20190719163853-cb61b32ac6fe h1:lXe2qZdvpiX5WZkZR4hgp4KJVfY3nMkvmwbVkpv1rVY=
github.com/golang-sql/civil v0.0.0-20190719163853-cb61b32ac6fe/go.mod h1:8vg3r2VgvsThLBIFL93Qb5yWzgyZWhEmBwUJWevAkK0= github.com/golang-sql/civil v0.0.0-20190719163853-cb61b32ac6fe/go.mod h1:8vg3r2VgvsThLBIFL93Qb5yWzgyZWhEmBwUJWevAkK0=
github.com/golang-sql/sqlexp v0.1.0 h1:ZCD6MBpcuOVfGVqsEmY5/4FtYiKz6tSyUv9LPEDei6A=
github.com/golang-sql/sqlexp v0.1.0/go.mod h1:J4ad9Vo8ZCWQ2GMrC4UCQy1JpCbwU9m3EOqtpKwwwHI= github.com/golang-sql/sqlexp v0.1.0/go.mod h1:J4ad9Vo8ZCWQ2GMrC4UCQy1JpCbwU9m3EOqtpKwwwHI=
github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
github.com/golang/protobuf v1.5.0/go.mod h1:FsONVRAS9T7sI+LIUmWTfcYkHO4aIWwzhcaSAoJOfIk= github.com/golang/protobuf v1.5.0/go.mod h1:FsONVRAS9T7sI+LIUmWTfcYkHO4aIWwzhcaSAoJOfIk=
@ -236,6 +239,7 @@ github.com/jackc/puddle v1.1.3/go.mod h1:m4B5Dj62Y0fbyuIc15OsIqK0+JU8nkqQjsgx7dv
github.com/jackc/puddle v1.3.0/go.mod h1:m4B5Dj62Y0fbyuIc15OsIqK0+JU8nkqQjsgx7dvjSWk= github.com/jackc/puddle v1.3.0/go.mod h1:m4B5Dj62Y0fbyuIc15OsIqK0+JU8nkqQjsgx7dvjSWk=
github.com/jellydator/ttlcache/v3 v3.1.0 h1:0gPFG0IHHP6xyUyXq+JaD8fwkDCqgqwohXNJBcYE71g= github.com/jellydator/ttlcache/v3 v3.1.0 h1:0gPFG0IHHP6xyUyXq+JaD8fwkDCqgqwohXNJBcYE71g=
github.com/jellydator/ttlcache/v3 v3.1.0/go.mod h1:hi7MGFdMAwZna5n2tuvh63DvFLzVKySzCVW6+0gA2n4= github.com/jellydator/ttlcache/v3 v3.1.0/go.mod h1:hi7MGFdMAwZna5n2tuvh63DvFLzVKySzCVW6+0gA2n4=
github.com/joho/godotenv v1.3.0/go.mod h1:7hK45KPybAkOC6peb+G5yklZfMxEjkZhHbwpqxOKXbg=
github.com/joho/godotenv v1.5.1 h1:7eLL/+HRGLY0ldzfGMeQkb7vMd0as4CfYvUVzLqw0N0= github.com/joho/godotenv v1.5.1 h1:7eLL/+HRGLY0ldzfGMeQkb7vMd0as4CfYvUVzLqw0N0=
github.com/joho/godotenv v1.5.1/go.mod h1:f4LDr5Voq0i2e/R5DDNOoa2zzDfwtkZa6DnEwAbqwq4= github.com/joho/godotenv v1.5.1/go.mod h1:f4LDr5Voq0i2e/R5DDNOoa2zzDfwtkZa6DnEwAbqwq4=
github.com/josharian/intern v1.0.0 h1:vlS4z54oSdjm0bgjRigI+G1HpF+tI+9rE5LLzOg8HmY= github.com/josharian/intern v1.0.0 h1:vlS4z54oSdjm0bgjRigI+G1HpF+tI+9rE5LLzOg8HmY=
@ -671,8 +675,11 @@ sigs.k8s.io/structured-merge-diff/v4 v4.2.3 h1:PRbqxJClWWYMNV1dhaG4NsibJbArud9kF
sigs.k8s.io/structured-merge-diff/v4 v4.2.3/go.mod h1:qjx8mGObPmV2aSZepjQjbmb2ihdVs8cGKBraizNC69E= sigs.k8s.io/structured-merge-diff/v4 v4.2.3/go.mod h1:qjx8mGObPmV2aSZepjQjbmb2ihdVs8cGKBraizNC69E=
sigs.k8s.io/yaml v1.3.0 h1:a2VclLzOGrwOHDiV8EfBGhvjHvP46CtW5j6POvhYGGo= sigs.k8s.io/yaml v1.3.0 h1:a2VclLzOGrwOHDiV8EfBGhvjHvP46CtW5j6POvhYGGo=
sigs.k8s.io/yaml v1.3.0/go.mod h1:GeOyir5tyXNByN85N/dRIT9es5UQNerPYEKK56eTBm8= sigs.k8s.io/yaml v1.3.0/go.mod h1:GeOyir5tyXNByN85N/dRIT9es5UQNerPYEKK56eTBm8=
src.techknowlogick.com/xormigrate v1.7.1 h1:RKGLLUAqJ+zO8iZ7eOc7oLH7f0cs2gfXSZSvBRBHnlY=
src.techknowlogick.com/xormigrate v1.7.1/go.mod h1:YGNBdj8prENlySwIKmfoEXp7ILGjAltyKFXD0qLgD7U=
xorm.io/builder v0.3.11-0.20220531020008-1bd24a7dc978/go.mod h1:aUW0S9eb9VCaPohFCH3j7czOx1PMW3i1HrSzbLYGBSE= xorm.io/builder v0.3.11-0.20220531020008-1bd24a7dc978/go.mod h1:aUW0S9eb9VCaPohFCH3j7czOx1PMW3i1HrSzbLYGBSE=
xorm.io/builder v0.3.13 h1:a3jmiVVL19psGeXx8GIurTp7p0IIgqeDmwhcR6BAOAo= xorm.io/builder v0.3.13 h1:a3jmiVVL19psGeXx8GIurTp7p0IIgqeDmwhcR6BAOAo=
xorm.io/builder v0.3.13/go.mod h1:aUW0S9eb9VCaPohFCH3j7czOx1PMW3i1HrSzbLYGBSE= xorm.io/builder v0.3.13/go.mod h1:aUW0S9eb9VCaPohFCH3j7czOx1PMW3i1HrSzbLYGBSE=
xorm.io/xorm v1.3.3/go.mod h1:qFJGFoVYbbIdnz2vaL5OxSQ2raleMpyRRalnq3n9OJo=
xorm.io/xorm v1.3.4 h1:vWFKzR3DhGUDl5b4srhUjhDwjxkZAc4C7BFszpu0swI= xorm.io/xorm v1.3.4 h1:vWFKzR3DhGUDl5b4srhUjhDwjxkZAc4C7BFszpu0swI=
xorm.io/xorm v1.3.4/go.mod h1:qFJGFoVYbbIdnz2vaL5OxSQ2raleMpyRRalnq3n9OJo= xorm.io/xorm v1.3.4/go.mod h1:qFJGFoVYbbIdnz2vaL5OxSQ2raleMpyRRalnq3n9OJo=

View file

@ -70,9 +70,6 @@ var Config = struct {
RootPath string RootPath string
CustomCSSFile string CustomCSSFile string
CustomJsFile string CustomJsFile string
Migrations struct {
AllowLong bool
}
EnableSwagger bool EnableSwagger bool
// Open bool // Open bool
// Orgs map[string]struct{} // Orgs map[string]struct{}

View file

@ -16,6 +16,7 @@ package datastore
import ( import (
"github.com/rs/zerolog" "github.com/rs/zerolog"
"go.woodpecker-ci.org/woodpecker/server/store" "go.woodpecker-ci.org/woodpecker/server/store"
"go.woodpecker-ci.org/woodpecker/server/store/datastore/migration" "go.woodpecker-ci.org/woodpecker/server/store/datastore/migration"
@ -54,8 +55,8 @@ func (s storage) Ping() error {
} }
// Migrate old storage or init new one // Migrate old storage or init new one
func (s storage) Migrate() error { func (s storage) Migrate(allowLong bool) error {
return migration.Migrate(s.engine) return migration.Migrate(s.engine, allowLong)
} }
func (s storage) Close() error { func (s storage) Close() error {

View file

@ -0,0 +1,33 @@
package migration
import (
"src.techknowlogick.com/xormigrate"
"xorm.io/xorm"
)
type v000Migrations struct {
Name string `xorm:"UNIQUE"`
}
func (m *v000Migrations) TableName() string {
return "migrations"
}
var legacyToXormigrate = xormigrate.Migration{
ID: "legacy-to-xormigrate",
MigrateSession: func(sess *xorm.Session) error {
var mig []*v000Migrations
if err := sess.Find(&mig); err != nil {
return err
}
for _, m := range mig {
if _, err := sess.Insert(&xormigrate.Migration{
ID: m.Name,
}); err != nil {
return err
}
}
return sess.DropTable("migrations")
},
}

View file

@ -18,14 +18,14 @@ import (
"fmt" "fmt"
"github.com/rs/zerolog/log" "github.com/rs/zerolog/log"
"src.techknowlogick.com/xormigrate"
"xorm.io/xorm" "xorm.io/xorm"
"xorm.io/xorm/schemas" "xorm.io/xorm/schemas"
) )
var legacy2Xorm = task{ var legacy2Xorm = xormigrate.Migration{
name: "xorm", ID: "xorm",
required: true, MigrateSession: func(sess *xorm.Session) error {
fn: func(sess *xorm.Session) error {
// make sure we have required migrations - else fail and point to last major version // make sure we have required migrations - else fail and point to last major version
for _, mig := range []string{ for _, mig := range []string{
// users // users
@ -74,7 +74,7 @@ var legacy2Xorm = task{
"create-table-build-config", "create-table-build-config",
"populate-build-config", "populate-build-config",
} { } {
exist, err := sess.Exist(&migrations{mig}) exist, err := sess.Exist(&xormigrate.Migration{ID: mig})
if err != nil { if err != nil {
return fmt.Errorf("test migration existence: %w", err) return fmt.Errorf("test migration existence: %w", err)
} }

View file

@ -15,12 +15,13 @@
package migration package migration
import ( import (
"src.techknowlogick.com/xormigrate"
"xorm.io/xorm" "xorm.io/xorm"
) )
var alterTableReposDropFallback = task{ var alterTableReposDropFallback = xormigrate.Migration{
name: "alter-table-drop-repo-fallback", ID: "alter-table-drop-repo-fallback",
fn: func(sess *xorm.Session) error { MigrateSession: func(sess *xorm.Session) error {
return dropTableColumns(sess, "repos", "repo_fallback") return dropTableColumns(sess, "repos", "repo_fallback")
}, },
} }

View file

@ -15,12 +15,13 @@
package migration package migration
import ( import (
"src.techknowlogick.com/xormigrate"
"xorm.io/xorm" "xorm.io/xorm"
) )
var alterTableReposDropAllowDeploysAllowTags = task{ var alterTableReposDropAllowDeploysAllowTags = xormigrate.Migration{
name: "drop-allow-push-tags-deploys-columns", ID: "drop-allow-push-tags-deploys-columns",
fn: func(sess *xorm.Session) error { MigrateSession: func(sess *xorm.Session) error {
return dropTableColumns(sess, "repos", return dropTableColumns(sess, "repos",
"repo_allow_deploys", "repo_allow_deploys",
"repo_allow_tags", "repo_allow_tags",

View file

@ -15,14 +15,15 @@
package migration package migration
import ( import (
"src.techknowlogick.com/xormigrate"
"xorm.io/xorm" "xorm.io/xorm"
"go.woodpecker-ci.org/woodpecker/server/model" "go.woodpecker-ci.org/woodpecker/server/model"
) )
var fixPRSecretEventName = task{ var fixPRSecretEventName = xormigrate.Migration{
name: "fix-pr-secret-event-name", ID: "fix-pr-secret-event-name",
fn: func(sess *xorm.Session) error { MigrateSession: func(sess *xorm.Session) error {
const batchSize = 100 const batchSize = 100
for start := 0; ; start += batchSize { for start := 0; ; start += batchSize {
secrets := make([]*model.Secret, 0, batchSize) secrets := make([]*model.Secret, 0, batchSize)

View file

@ -15,12 +15,13 @@
package migration package migration
import ( import (
"src.techknowlogick.com/xormigrate"
"xorm.io/xorm" "xorm.io/xorm"
) )
var alterTableReposDropCounter = task{ var alterTableReposDropCounter = xormigrate.Migration{
name: "alter-table-drop-counter", ID: "alter-table-drop-counter",
fn: func(sess *xorm.Session) error { MigrateSession: func(sess *xorm.Session) error {
return dropTableColumns(sess, "repos", "repo_counter") return dropTableColumns(sess, "repos", "repo_counter")
}, },
} }

View file

@ -15,12 +15,13 @@
package migration package migration
import ( import (
"src.techknowlogick.com/xormigrate"
"xorm.io/xorm" "xorm.io/xorm"
) )
var dropSenders = task{ var dropSenders = xormigrate.Migration{
name: "drop-senders", ID: "drop-senders",
fn: func(sess *xorm.Session) error { MigrateSession: func(sess *xorm.Session) error {
return sess.DropTable("senders") return sess.DropTable("senders")
}, },
} }

View file

@ -15,13 +15,14 @@
package migration package migration
import ( import (
"src.techknowlogick.com/xormigrate"
"xorm.io/xorm" "xorm.io/xorm"
"xorm.io/xorm/schemas" "xorm.io/xorm/schemas"
) )
var alterTableLogUpdateColumnLogDataType = task{ var alterTableLogUpdateColumnLogDataType = xormigrate.Migration{
name: "alter-table-logs-update-type-of-data", ID: "alter-table-logs-update-type-of-data",
fn: func(sess *xorm.Session) (err error) { MigrateSession: func(sess *xorm.Session) (err error) {
dialect := sess.Engine().Dialect().URI().DBType dialect := sess.Engine().Dialect().URI().DBType
switch dialect { switch dialect {

View file

@ -15,24 +15,25 @@
package migration package migration
import ( import (
"src.techknowlogick.com/xormigrate"
"xorm.io/xorm" "xorm.io/xorm"
) )
type SecretV007 struct { type SecretV008 struct {
Owner string `json:"-" xorm:"NOT NULL DEFAULT '' UNIQUE(s) INDEX 'secret_owner'"` Owner string `json:"-" xorm:"NOT NULL DEFAULT '' UNIQUE(s) INDEX 'secret_owner'"`
RepoID int64 `json:"-" xorm:"NOT NULL DEFAULT 0 UNIQUE(s) INDEX 'secret_repo_id'"` RepoID int64 `json:"-" xorm:"NOT NULL DEFAULT 0 UNIQUE(s) INDEX 'secret_repo_id'"`
Name string `json:"name" xorm:"NOT NULL UNIQUE(s) INDEX 'secret_name'"` Name string `json:"name" xorm:"NOT NULL UNIQUE(s) INDEX 'secret_name'"`
} }
// TableName return database table name for xorm // TableName return database table name for xorm
func (SecretV007) TableName() string { func (SecretV008) TableName() string {
return "secrets" return "secrets"
} }
var alterTableSecretsAddUserCol = task{ var alterTableSecretsAddUserCol = xormigrate.Migration{
name: "alter-table-add-secrets-user-id", ID: "alter-table-add-secrets-user-id",
fn: func(sess *xorm.Session) error { MigrateSession: func(sess *xorm.Session) error {
if err := sess.Sync(new(SecretV007)); err != nil { if err := sess.Sync(new(SecretV008)); err != nil {
return err return err
} }
if err := alterColumnDefault(sess, "secrets", "secret_repo_id", "0"); err != nil { if err := alterColumnDefault(sess, "secrets", "secret_repo_id", "0"); err != nil {

View file

@ -15,12 +15,13 @@
package migration package migration
import ( import (
"src.techknowlogick.com/xormigrate"
"xorm.io/xorm" "xorm.io/xorm"
) )
var lowercaseSecretNames = task{ var lowercaseSecretNames = xormigrate.Migration{
name: "lowercase-secret-names", ID: "lowercase-secret-names",
fn: func(sess *xorm.Session) (err error) { MigrateSession: func(sess *xorm.Session) (err error) {
_, err = sess.Exec("UPDATE secrets SET secret_name = LOWER(secret_name);") _, err = sess.Exec("UPDATE secrets SET secret_name = LOWER(secret_name);")
return err return err
}, },

View file

@ -15,14 +15,15 @@
package migration package migration
import ( import (
"src.techknowlogick.com/xormigrate"
"xorm.io/xorm" "xorm.io/xorm"
"go.woodpecker-ci.org/woodpecker/server/model" "go.woodpecker-ci.org/woodpecker/server/model"
) )
var recreateAgentsTable = task{ var recreateAgentsTable = xormigrate.Migration{
name: "recreate-agents-table", ID: "recreate-agents-table",
fn: func(sess *xorm.Session) error { MigrateSession: func(sess *xorm.Session) error {
if err := sess.DropTable("agents"); err != nil { if err := sess.DropTable("agents"); err != nil {
return err return err
} }

View file

@ -15,13 +15,13 @@
package migration package migration
import ( import (
"src.techknowlogick.com/xormigrate"
"xorm.io/xorm" "xorm.io/xorm"
) )
var renameBuildsToPipeline = task{ var renameBuildsToPipeline = xormigrate.Migration{
name: "rename-builds-to-pipeline", ID: "rename-builds-to-pipeline",
required: true, MigrateSession: func(sess *xorm.Session) error {
fn: func(sess *xorm.Session) error {
err := renameTable(sess, "builds", "pipelines") err := renameTable(sess, "builds", "pipelines")
if err != nil { if err != nil {
return err return err

View file

@ -17,6 +17,7 @@ package migration
import ( import (
"strings" "strings"
"src.techknowlogick.com/xormigrate"
"xorm.io/xorm" "xorm.io/xorm"
) )
@ -25,10 +26,9 @@ type oldTable struct {
columns []string columns []string
} }
var renameColumnsBuildsToPipeline = task{ var renameColumnsBuildsToPipeline = xormigrate.Migration{
name: "rename-columns-builds-to-pipeline", ID: "rename-columns-builds-to-pipeline",
required: true, MigrateSession: func(sess *xorm.Session) error {
fn: func(sess *xorm.Session) error {
var oldColumns []*oldTable var oldColumns []*oldTable
oldColumns = append(oldColumns, &oldTable{ oldColumns = append(oldColumns, &oldTable{

View file

@ -17,13 +17,13 @@ package migration
import ( import (
"strings" "strings"
"src.techknowlogick.com/xormigrate"
"xorm.io/xorm" "xorm.io/xorm"
) )
var renameTableProcsToSteps = task{ var renameTableProcsToSteps = xormigrate.Migration{
name: "rename-procs-to-steps", ID: "rename-procs-to-steps",
required: true, MigrateSession: func(sess *xorm.Session) error {
fn: func(sess *xorm.Session) error {
err := renameTable(sess, "procs", "steps") err := renameTable(sess, "procs", "steps")
if err != nil { if err != nil {
return err return err

View file

@ -15,28 +15,28 @@
package migration package migration
import ( import (
"src.techknowlogick.com/xormigrate"
"xorm.io/xorm" "xorm.io/xorm"
) )
type oldRepo012 struct { type oldRepo013 struct {
ID int64 `xorm:"pk autoincr 'repo_id'"` ID int64 `xorm:"pk autoincr 'repo_id'"`
RemoteID string `xorm:"remote_id"` RemoteID string `xorm:"remote_id"`
} }
func (oldRepo012) TableName() string { func (oldRepo013) TableName() string {
return "repos" return "repos"
} }
var renameRemoteToForge = task{ var renameRemoteToForge = xormigrate.Migration{
name: "rename-remote-to-forge", ID: "rename-remote-to-forge",
required: true, MigrateSession: func(sess *xorm.Session) error {
fn: func(sess *xorm.Session) error {
if err := renameColumn(sess, "pipelines", "pipeline_remote", "pipeline_clone_url"); err != nil { if err := renameColumn(sess, "pipelines", "pipeline_remote", "pipeline_clone_url"); err != nil {
return err return err
} }
// make sure the column exist before rename it // make sure the column exist before rename it
if err := sess.Sync(new(oldRepo012)); err != nil { if err := sess.Sync(new(oldRepo013)); err != nil {
return err return err
} }

View file

@ -15,13 +15,13 @@
package migration package migration
import ( import (
"src.techknowlogick.com/xormigrate"
"xorm.io/xorm" "xorm.io/xorm"
) )
var renameForgeIDToForgeRemoteID = task{ var renameForgeIDToForgeRemoteID = xormigrate.Migration{
name: "rename-forge-id-to-forge-remote-id", ID: "rename-forge-id-to-forge-remote-id",
required: true, MigrateSession: func(sess *xorm.Session) error {
fn: func(sess *xorm.Session) error {
return renameColumn(sess, "repos", "forge_id", "forge_remote_id") return renameColumn(sess, "repos", "forge_id", "forge_remote_id")
}, },
} }

View file

@ -15,13 +15,13 @@
package migration package migration
import ( import (
"src.techknowlogick.com/xormigrate"
"xorm.io/xorm" "xorm.io/xorm"
) )
var removeActiveFromUsers = task{ var removeActiveFromUsers = xormigrate.Migration{
name: "remove-active-from-users", ID: "remove-active-from-users",
required: true, MigrateSession: func(sess *xorm.Session) error {
fn: func(sess *xorm.Session) error {
return dropTableColumns(sess, "users", "user_active") return dropTableColumns(sess, "users", "user_active")
}, },
} }

View file

@ -15,13 +15,13 @@
package migration package migration
import ( import (
"src.techknowlogick.com/xormigrate"
"xorm.io/xorm" "xorm.io/xorm"
) )
var removeInactiveRepos = task{ var removeInactiveRepos = xormigrate.Migration{
name: "remove-inactive-repos", ID: "remove-inactive-repos",
required: true, MigrateSession: func(sess *xorm.Session) error {
fn: func(sess *xorm.Session) error {
// If the timeout is 0, the repo was never activated, so we remove it. // If the timeout is 0, the repo was never activated, so we remove it.
_, err := sess.Table("repos").Where("repo_active = ?", false).And("repo_timeout = ?", 0).Delete() _, err := sess.Table("repos").Where("repo_active = ?", false).And("repo_timeout = ?", 0).Delete()
if err != nil { if err != nil {

View file

@ -15,12 +15,13 @@
package migration package migration
import ( import (
"src.techknowlogick.com/xormigrate"
"xorm.io/xorm" "xorm.io/xorm"
) )
var dropFiles = task{ var dropFiles = xormigrate.Migration{
name: "drop-files", ID: "drop-files",
fn: func(sess *xorm.Session) error { MigrateSession: func(sess *xorm.Session) error {
return sess.DropTable("files") return sess.DropTable("files")
}, },
} }

View file

@ -15,23 +15,24 @@
package migration package migration
import ( import (
"src.techknowlogick.com/xormigrate"
"xorm.io/xorm" "xorm.io/xorm"
) )
type oldStep017 struct { type oldStep018 struct {
ID int64 `xorm:"pk autoincr 'step_id'"` ID int64 `xorm:"pk autoincr 'step_id'"`
Machine string `xorm:"step_machine"` Machine string `xorm:"step_machine"`
} }
func (oldStep017) TableName() string { func (oldStep018) TableName() string {
return "steps" return "steps"
} }
var removeMachineCol = task{ var removeMachineCol = xormigrate.Migration{
name: "remove-machine-col", ID: "remove-machine-col",
fn: func(sess *xorm.Session) error { MigrateSession: func(sess *xorm.Session) error {
// make sure step_machine column exists // make sure step_machine column exists
if err := sess.Sync(new(oldStep017)); err != nil { if err := sess.Sync(new(oldStep018)); err != nil {
return err return err
} }
return dropTableColumns(sess, "steps", "step_machine") return dropTableColumns(sess, "steps", "step_machine")

View file

@ -15,24 +15,25 @@
package migration package migration
import ( import (
"src.techknowlogick.com/xormigrate"
"xorm.io/xorm" "xorm.io/xorm"
) )
type oldPipeline018 struct { type oldPipeline019 struct {
ID int64 `xorm:"pk autoincr 'pipeline_id'"` ID int64 `xorm:"pk autoincr 'pipeline_id'"`
Signed bool `xorm:"pipeline_signed"` Signed bool `xorm:"pipeline_signed"`
Verified bool `xorm:"pipeline_verified"` Verified bool `xorm:"pipeline_verified"`
} }
func (oldPipeline018) TableName() string { func (oldPipeline019) TableName() string {
return "pipelines" return "pipelines"
} }
var dropOldCols = task{ var dropOldCols = xormigrate.Migration{
name: "drop-old-col", ID: "drop-old-col",
fn: func(sess *xorm.Session) error { MigrateSession: func(sess *xorm.Session) error {
// make sure columns on pipelines exist // make sure columns on pipelines exist
if err := sess.Sync(new(oldPipeline018)); err != nil { if err := sess.Sync(new(oldPipeline019)); err != nil {
return err return err
} }
if err := dropTableColumns(sess, "steps", "step_pgid"); err != nil { if err := dropTableColumns(sess, "steps", "step_pgid"); err != nil {

View file

@ -22,29 +22,26 @@ import (
"github.com/rs/zerolog/log" "github.com/rs/zerolog/log"
"github.com/tevino/abool/v2" "github.com/tevino/abool/v2"
"src.techknowlogick.com/xormigrate"
"xorm.io/xorm" "xorm.io/xorm"
"go.woodpecker-ci.org/woodpecker/server"
"go.woodpecker-ci.org/woodpecker/shared/utils" "go.woodpecker-ci.org/woodpecker/shared/utils"
) )
// maxDefaultSqliteItems set the threshold at witch point the migration will fail by default // perPage020 sets the size of the slice to read per page
var maxDefaultSqliteItems019 = 5000 var perPage020 = 100
// perPage019 set the size of the slice to read per page type oldLogs020 struct {
var perPage019 = 100
type oldLogs019 struct {
ID int64 `xorm:"pk autoincr 'log_id'"` ID int64 `xorm:"pk autoincr 'log_id'"`
StepID int64 `xorm:"UNIQUE 'log_step_id'"` StepID int64 `xorm:"UNIQUE 'log_step_id'"`
Data []byte `xorm:"LONGBLOB 'log_data'"` Data []byte `xorm:"LONGBLOB 'log_data'"`
} }
func (oldLogs019) TableName() string { func (oldLogs020) TableName() string {
return "logs" return "logs"
} }
type oldLogEntry019 struct { type oldLogEntry020 struct {
Step string `json:"step,omitempty"` Step string `json:"step,omitempty"`
Time int64 `json:"time,omitempty"` Time int64 `json:"time,omitempty"`
Type int `json:"type,omitempty"` Type int `json:"type,omitempty"`
@ -52,7 +49,7 @@ type oldLogEntry019 struct {
Out string `json:"out,omitempty"` Out string `json:"out,omitempty"`
} }
type newLogEntry019 struct { type newLogEntry020 struct {
ID int64 `xorm:"pk autoincr 'id'"` ID int64 `xorm:"pk autoincr 'id'"`
StepID int64 `xorm:"'step_id'"` StepID int64 `xorm:"'step_id'"`
Time int64 Time int64
@ -62,38 +59,27 @@ type newLogEntry019 struct {
Type int Type int
} }
func (newLogEntry019) TableName() string { func (newLogEntry020) TableName() string {
return "log_entries" return "log_entries"
} }
var initLogsEntriesTable = task{ var initLogsEntriesTable = xormigrate.Migration{
name: "init-log_entries", ID: "init-log_entries",
required: true, MigrateSession: func(sess *xorm.Session) error {
fn: func(sess *xorm.Session) error { return sess.Sync(new(newLogEntry020))
return sess.Sync(new(newLogEntry019))
}, },
} }
var migrateLogs2LogEntries = task{ var migrateLogs2LogEntries = xormigrate.Migration{
name: "migrate-logs-to-log_entries", ID: "migrate-logs-to-log_entries",
required: false, Long: true,
engineFn: func(e *xorm.Engine) error { Migrate: func(e *xorm.Engine) error {
// make sure old logs table exists // make sure old logs table exists
if exist, err := e.IsTableExist(new(oldLogs019)); !exist || err != nil { if exist, err := e.IsTableExist(new(oldLogs020)); !exist || err != nil {
return err return err
} }
// first we check if we have just 1000 entries to migrate if err := e.Sync(new(oldLogs020)); err != nil {
toMigrate, err := e.Count(new(oldLogs019))
if err != nil {
return err
}
if toMigrate > int64(maxDefaultSqliteItems019) && !server.Config.Server.Migrations.AllowLong {
return fmt.Errorf("Migrating logs to log_entries is skipped, as we have %d entries to convert. Set 'WOODPECKER_MIGRATIONS_ALLOW_LONG' to 'true' to migrate anyway", toMigrate)
}
if err := e.Sync(new(oldLogs019)); err != nil {
return err return err
} }
@ -101,8 +87,8 @@ var migrateLogs2LogEntries = task{
page := 0 page := 0
offset := 0 offset := 0
logs := make([]*oldLogs019, 0, perPage019) logs := make([]*oldLogs020, 0, perPage020)
logEntries := make([]*oldLogEntry019, 0, 50) logEntries := make([]*oldLogEntry020, 0, 50)
sigterm := abool.New() sigterm := abool.New()
ctx, cancelCtx := context.WithCancelCause(context.Background()) ctx, cancelCtx := context.WithCancelCause(context.Background())
@ -124,7 +110,7 @@ var migrateLogs2LogEntries = task{
} }
logs = logs[:0] logs = logs[:0]
err := sess.Limit(perPage019, offset).Find(&logs) err := sess.Limit(perPage020, offset).Find(&logs)
if err != nil { if err != nil {
return err return err
} }
@ -146,7 +132,7 @@ var migrateLogs2LogEntries = task{
time = logEntry.Time time = logEntry.Time
} }
log := &newLogEntry019{ log := &newLogEntry020{
StepID: l.StepID, StepID: l.StepID,
Data: []byte(logEntry.Out), Data: []byte(logEntry.Out),
Line: logEntry.Pos, Line: logEntry.Pos,
@ -168,7 +154,7 @@ var migrateLogs2LogEntries = task{
return err return err
} }
if len(logs) < perPage019 { if len(logs) < perPage020 {
break break
} }

View file

@ -15,12 +15,13 @@
package migration package migration
import ( import (
"src.techknowlogick.com/xormigrate"
"xorm.io/xorm" "xorm.io/xorm"
"go.woodpecker-ci.org/woodpecker/server/model" "go.woodpecker-ci.org/woodpecker/server/model"
) )
type oldStep020 struct { type oldStep021 struct {
ID int64 `xorm:"pk autoincr 'step_id'"` ID int64 `xorm:"pk autoincr 'step_id'"`
PipelineID int64 `xorm:"UNIQUE(s) INDEX 'step_pipeline_id'"` PipelineID int64 `xorm:"UNIQUE(s) INDEX 'step_pipeline_id'"`
PID int `xorm:"UNIQUE(s) 'step_pid'"` PID int `xorm:"UNIQUE(s) 'step_pid'"`
@ -35,23 +36,22 @@ type oldStep020 struct {
Environ map[string]string `xorm:"json 'step_environ'"` Environ map[string]string `xorm:"json 'step_environ'"`
} }
func (oldStep020) TableName() string { func (oldStep021) TableName() string {
return "steps" return "steps"
} }
var parentStepsToWorkflows = task{ var parentStepsToWorkflows = xormigrate.Migration{
name: "parent-steps-to-workflows", ID: "parent-steps-to-workflows",
required: true, MigrateSession: func(sess *xorm.Session) error {
fn: func(sess *xorm.Session) error {
if err := sess.Sync(new(model.Workflow)); err != nil { if err := sess.Sync(new(model.Workflow)); err != nil {
return err return err
} }
// make sure the columns exist before removing them // make sure the columns exist before removing them
if err := sess.Sync(new(oldStep020)); err != nil { if err := sess.Sync(new(oldStep021)); err != nil {
return err return err
} }
var parentSteps []*oldStep020 var parentSteps []*oldStep021
err := sess.Where("step_ppid = ?", 0).Find(&parentSteps) err := sess.Where("step_ppid = ?", 0).Find(&parentSteps)
if err != nil { if err != nil {
return err return err
@ -76,7 +76,7 @@ var parentStepsToWorkflows = task{
return err return err
} }
_, err = sess.Delete(&oldStep020{ID: p.ID}) _, err = sess.Delete(&oldStep021{ID: p.ID})
if err != nil { if err != nil {
return err return err
} }

View file

@ -18,13 +18,14 @@ import (
"fmt" "fmt"
"strings" "strings"
"src.techknowlogick.com/xormigrate"
"xorm.io/builder" "xorm.io/builder"
"xorm.io/xorm" "xorm.io/xorm"
"go.woodpecker-ci.org/woodpecker/server/model" "go.woodpecker-ci.org/woodpecker/server/model"
) )
type oldSecret021 struct { type oldSecret022 struct {
ID int64 `xorm:"pk autoincr 'secret_id'"` ID int64 `xorm:"pk autoincr 'secret_id'"`
Owner string `xorm:"'secret_owner'"` Owner string `xorm:"'secret_owner'"`
OrgID int64 `xorm:"NOT NULL DEFAULT 0 'secret_org_id'"` OrgID int64 `xorm:"NOT NULL DEFAULT 0 'secret_org_id'"`
@ -32,51 +33,50 @@ type oldSecret021 struct {
Name string `xorm:"NOT NULL INDEX 'secret_name'"` Name string `xorm:"NOT NULL INDEX 'secret_name'"`
} }
func (oldSecret021) TableName() string { func (oldSecret022) TableName() string {
return "secrets" return "secrets"
} }
type syncRepo021 struct { type syncRepo022 struct {
OrgID int64 `json:"org_id" xorm:"repo_org_id"` OrgID int64 `json:"org_id" xorm:"repo_org_id"`
} }
// TableName return database table name for xorm // TableName return database table name for xorm
func (syncRepo021) TableName() string { func (syncRepo022) TableName() string {
return "repos" return "repos"
} }
type repo021 struct { type repo022 struct {
ID int64 `json:"id,omitempty" xorm:"pk autoincr 'repo_id'"` ID int64 `json:"id,omitempty" xorm:"pk autoincr 'repo_id'"`
OrgID int64 `json:"org_id" xorm:"repo_org_id"` OrgID int64 `json:"org_id" xorm:"repo_org_id"`
Owner string `json:"owner" xorm:"UNIQUE(name) 'repo_owner'"` Owner string `json:"owner" xorm:"UNIQUE(name) 'repo_owner'"`
} }
// TableName return database table name for xorm // TableName return database table name for xorm
func (repo021) TableName() string { func (repo022) TableName() string {
return "repos" return "repos"
} }
var addOrgs = task{ var addOrgs = xormigrate.Migration{
name: "add-orgs", ID: "add-orgs",
required: true, MigrateSession: func(sess *xorm.Session) error {
fn: func(sess *xorm.Session) error {
if exist, err := sess.IsTableExist("orgs"); exist && err == nil { if exist, err := sess.IsTableExist("orgs"); exist && err == nil {
if err := sess.DropTable("orgs"); err != nil { if err := sess.DropTable("orgs"); err != nil {
return fmt.Errorf("drop old orgs table failed: %w", err) return fmt.Errorf("drop old orgs table failed: %w", err)
} }
} }
if err := sess.Sync(new(model.Org), new(syncRepo021), new(model.User)); err != nil { if err := sess.Sync(new(model.Org), new(syncRepo022), new(model.User)); err != nil {
return fmt.Errorf("sync new models failed: %w", err) return fmt.Errorf("sync new models failed: %w", err)
} }
// make sure the columns exist before removing them // make sure the columns exist before removing them
if _, err := sess.SyncWithOptions(xorm.SyncOptions{IgnoreConstrains: true, IgnoreIndices: true}, new(oldSecret021)); err != nil { if _, err := sess.SyncWithOptions(xorm.SyncOptions{IgnoreConstrains: true, IgnoreIndices: true}, new(oldSecret022)); err != nil {
return fmt.Errorf("sync old secrets models failed: %w", err) return fmt.Errorf("sync old secrets models failed: %w", err)
} }
// get all org names from repos // get all org names from repos
var repos []*repo021 var repos []*repo022
if err := sess.Find(&repos); err != nil { if err := sess.Find(&repos); err != nil {
return fmt.Errorf("find all repos failed: %w", err) return fmt.Errorf("find all repos failed: %w", err)
} }
@ -107,7 +107,7 @@ var addOrgs = task{
orgs[orgName] = org orgs[orgName] = org
// update org secrets // update org secrets
var secrets []*oldSecret021 var secrets []*oldSecret022
if err := sess.Where(builder.Eq{"secret_owner": orgName, "secret_repo_id": 0}).Find(&secrets); err != nil { if err := sess.Where(builder.Eq{"secret_owner": orgName, "secret_repo_id": 0}).Find(&secrets); err != nil {
return fmt.Errorf("get org secrets failed: %w", err) return fmt.Errorf("get org secrets failed: %w", err)
} }

View file

@ -17,15 +17,15 @@ package migration
import ( import (
"fmt" "fmt"
"src.techknowlogick.com/xormigrate"
"xorm.io/xorm" "xorm.io/xorm"
"go.woodpecker-ci.org/woodpecker/server/model" "go.woodpecker-ci.org/woodpecker/server/model"
) )
var addOrgID = task{ var addOrgID = xormigrate.Migration{
name: "add-org-id", ID: "add-org-id",
required: true, MigrateSession: func(sess *xorm.Session) error {
fn: func(sess *xorm.Session) error {
if err := sess.Sync(new(model.User)); err != nil { if err := sess.Sync(new(model.User)); err != nil {
return fmt.Errorf("sync new models failed: %w", err) return fmt.Errorf("sync new models failed: %w", err)
} }

View file

@ -15,13 +15,14 @@
package migration package migration
import ( import (
"src.techknowlogick.com/xormigrate"
"xorm.io/xorm" "xorm.io/xorm"
"xorm.io/xorm/schemas" "xorm.io/xorm/schemas"
) )
var alterTableTasksUpdateColumnTaskDataType = task{ var alterTableTasksUpdateColumnTaskDataType = xormigrate.Migration{
name: "alter-table-tasks-update-type-of-task-data", ID: "alter-table-tasks-update-type-of-task-data",
fn: func(sess *xorm.Session) (err error) { MigrateSession: func(sess *xorm.Session) (err error) {
dialect := sess.Engine().Dialect().URI().DBType dialect := sess.Engine().Dialect().URI().DBType
switch dialect { switch dialect {

View file

@ -15,13 +15,14 @@
package migration package migration
import ( import (
"src.techknowlogick.com/xormigrate"
"xorm.io/xorm" "xorm.io/xorm"
"xorm.io/xorm/schemas" "xorm.io/xorm/schemas"
) )
var alterTableConfigUpdateColumnConfigDataType = task{ var alterTableConfigUpdateColumnConfigDataType = xormigrate.Migration{
name: "alter-table-config-update-type-of-config-data", ID: "alter-table-config-update-type-of-config-data",
fn: func(sess *xorm.Session) (err error) { MigrateSession: func(sess *xorm.Session) (err error) {
dialect := sess.Engine().Dialect().URI().DBType dialect := sess.Engine().Dialect().URI().DBType
switch dialect { switch dialect {

View file

@ -15,10 +15,11 @@
package migration package migration
import ( import (
"src.techknowlogick.com/xormigrate"
"xorm.io/xorm" "xorm.io/xorm"
) )
type oldSecret025 struct { type oldSecret026 struct {
ID int64 `json:"id" xorm:"pk autoincr 'secret_id'"` ID int64 `json:"id" xorm:"pk autoincr 'secret_id'"`
PluginsOnly bool `json:"plugins_only" xorm:"secret_plugins_only"` PluginsOnly bool `json:"plugins_only" xorm:"secret_plugins_only"`
SkipVerify bool `json:"-" xorm:"secret_skip_verify"` SkipVerify bool `json:"-" xorm:"secret_skip_verify"`
@ -26,15 +27,15 @@ type oldSecret025 struct {
Images []string `json:"images" xorm:"json 'secret_images'"` Images []string `json:"images" xorm:"json 'secret_images'"`
} }
func (oldSecret025) TableName() string { func (oldSecret026) TableName() string {
return "secrets" return "secrets"
} }
var removePluginOnlyOptionFromSecretsTable = task{ var removePluginOnlyOptionFromSecretsTable = xormigrate.Migration{
name: "remove-plugin-only-option-from-secrets-table", ID: "remove-plugin-only-option-from-secrets-table",
fn: func(sess *xorm.Session) (err error) { MigrateSession: func(sess *xorm.Session) (err error) {
// make sure plugin_only column exists // make sure plugin_only column exists
if err := sess.Sync(new(oldSecret025)); err != nil { if err := sess.Sync(new(oldSecret026)); err != nil {
return err return err
} }

View file

@ -15,53 +15,54 @@
package migration package migration
import ( import (
"src.techknowlogick.com/xormigrate"
"xorm.io/xorm" "xorm.io/xorm"
"go.woodpecker-ci.org/woodpecker/pipeline/errors" "go.woodpecker-ci.org/woodpecker/pipeline/errors"
) )
// perPage026 set the size of the slice to read per page // perPage027 set the size of the slice to read per page
var perPage026 = 100 var perPage027 = 100
type pipeline026 struct { type pipeline027 struct {
ID int64 `json:"id" xorm:"pk autoincr 'pipeline_id'"` ID int64 `json:"id" xorm:"pk autoincr 'pipeline_id'"`
Error string `json:"error" xorm:"LONGTEXT 'pipeline_error'"` // old error format Error string `json:"error" xorm:"LONGTEXT 'pipeline_error'"` // old error format
Errors []*errors.PipelineError `json:"errors" xorm:"json 'pipeline_errors'"` // new error format Errors []*errors.PipelineError `json:"errors" xorm:"json 'pipeline_errors'"` // new error format
} }
func (pipeline026) TableName() string { func (pipeline027) TableName() string {
return "pipelines" return "pipelines"
} }
type PipelineError026 struct { type PipelineError027 struct {
Type string `json:"type"` Type string `json:"type"`
Message string `json:"message"` Message string `json:"message"`
IsWarning bool `json:"is_warning"` IsWarning bool `json:"is_warning"`
Data any `json:"data"` Data any `json:"data"`
} }
var convertToNewPipelineErrorFormat = task{ var convertToNewPipelineErrorFormat = xormigrate.Migration{
name: "convert-to-new-pipeline-error-format", ID: "convert-to-new-pipeline-error-format",
required: true, Long: true,
fn: func(sess *xorm.Session) (err error) { MigrateSession: func(sess *xorm.Session) (err error) {
// make sure pipeline_error column exists // make sure pipeline_error column exists
if err := sess.Sync(new(pipeline026)); err != nil { if err := sess.Sync(new(pipeline027)); err != nil {
return err return err
} }
page := 0 page := 0
oldPipelines := make([]*pipeline026, 0, perPage026) oldPipelines := make([]*pipeline027, 0, perPage027)
for { for {
oldPipelines = oldPipelines[:0] oldPipelines = oldPipelines[:0]
err := sess.Limit(perPage026, page*perPage026).Cols("pipeline_id", "pipeline_error").Where("pipeline_error != ''").Find(&oldPipelines) err := sess.Limit(perPage027, page*perPage027).Cols("pipeline_id", "pipeline_error").Where("pipeline_error != ''").Find(&oldPipelines)
if err != nil { if err != nil {
return err return err
} }
for _, oldPipeline := range oldPipelines { for _, oldPipeline := range oldPipelines {
var newPipeline pipeline026 var newPipeline pipeline027
newPipeline.ID = oldPipeline.ID newPipeline.ID = oldPipeline.ID
newPipeline.Errors = []*errors.PipelineError{{ newPipeline.Errors = []*errors.PipelineError{{
Type: "generic", Type: "generic",
@ -73,7 +74,7 @@ var convertToNewPipelineErrorFormat = task{
} }
} }
if len(oldPipelines) < perPage026 { if len(oldPipelines) < perPage027 {
break break
} }

View file

@ -15,13 +15,13 @@
package migration package migration
import ( import (
"src.techknowlogick.com/xormigrate"
"xorm.io/xorm" "xorm.io/xorm"
) )
var renameLinkToURL = task{ var renameLinkToURL = xormigrate.Migration{
name: "rename-link-to-url", ID: "rename-link-to-url",
required: true, MigrateSession: func(sess *xorm.Session) (err error) {
fn: func(sess *xorm.Session) (err error) {
if err := renameColumn(sess, "pipelines", "pipeline_link", "pipeline_forge_url"); err != nil { if err := renameColumn(sess, "pipelines", "pipeline_link", "pipeline_forge_url"); err != nil {
return err return err
} }

View file

@ -0,0 +1,55 @@
// Copyright 2023 Woodpecker Authors
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package migration
import (
"fmt"
"github.com/rs/zerolog/log"
)
type xormigrateLogger struct{}
func (l *xormigrateLogger) Debug(v ...interface{}) {
log.Debug().Msg(fmt.Sprint(v...))
}
func (l *xormigrateLogger) Debugf(format string, v ...interface{}) {
log.Debug().Msgf(format, v...)
}
func (l *xormigrateLogger) Info(v ...interface{}) {
log.Info().Msg(fmt.Sprint(v...))
}
func (l *xormigrateLogger) Infof(format string, v ...interface{}) {
log.Info().Msgf(format, v...)
}
func (l *xormigrateLogger) Warn(v ...interface{}) {
log.Warn().Msg(fmt.Sprint(v...))
}
func (l *xormigrateLogger) Warnf(format string, v ...interface{}) {
log.Warn().Msgf(format, v...)
}
func (l *xormigrateLogger) Error(v ...interface{}) {
log.Error().Msg(fmt.Sprint(v...))
}
func (l *xormigrateLogger) Errorf(format string, v ...interface{}) {
log.Error().Msgf(format, v...)
}

View file

@ -15,21 +15,19 @@
package migration package migration
import ( import (
"context"
"errors"
"fmt" "fmt"
"reflect" "reflect"
"time"
"github.com/rs/zerolog/log" "src.techknowlogick.com/xormigrate"
"xorm.io/xorm" "xorm.io/xorm"
"go.woodpecker-ci.org/woodpecker/server/model" "go.woodpecker-ci.org/woodpecker/server/model"
) )
// APPEND NEW MIGRATIONS // APPEND NEW MIGRATIONS
// they are executed in order and if one fails woodpecker will try to rollback that specific one and quits // they are executed in order and if one fails Xormigrate will try to rollback that specific one and quits
var migrationTasks = []*task{ var migrationTasks = []*xormigrate.Migration{
&legacyToXormigrate,
&legacy2Xorm, &legacy2Xorm,
&alterTableReposDropFallback, &alterTableReposDropFallback,
&alterTableReposDropAllowDeploysAllowTags, &alterTableReposDropAllowDeploysAllowTags,
@ -82,70 +80,25 @@ var allBeans = []any{
new(model.Org), new(model.Org),
} }
type migrations struct { func Migrate(e *xorm.Engine, allowLong bool) error {
Name string `xorm:"UNIQUE"`
}
type task struct {
name string
required bool
fn func(sess *xorm.Session) error
// engineFn does manage session on it's own. only use it if you really need to
engineFn func(e *xorm.Engine) error
}
// initNew create tables for new instance
func initNew(sess *xorm.Session) error {
if err := syncAll(sess); err != nil {
return err
}
// dummy run migrations
for _, task := range migrationTasks {
if _, err := sess.Insert(&migrations{task.name}); err != nil {
return err
}
}
return nil
}
func Migrate(e *xorm.Engine) error {
e.SetDisableGlobalCache(true) e.SetDisableGlobalCache(true)
if err := e.Sync(new(migrations)); err != nil { m := xormigrate.New(e, migrationTasks)
return fmt.Errorf("error to create migrations table: %w", err) m.AllowLong(allowLong)
} oldCount, err := e.Table("migrations").Count()
if oldCount < 1 || err != nil {
sess := e.NewSession() // allow new schema initialization if old migrations table is empty or it does not exist (err != nil)
defer sess.Close() // schema initialization will always run if we call `InitSchema`
if err := sess.Begin(); err != nil { m.InitSchema(func(engine *xorm.Engine) error {
return fmt.Errorf("could not create initial migration session: %w", err) // do nothing on schema init, models are synced in any case below
}
// check if we have a fresh installation or need to check for migrations
c, err := sess.Count(new(migrations))
if err != nil {
return fmt.Errorf("could not count migrations: %w", err)
}
if c == 0 {
if err := initNew(sess); err != nil {
return fmt.Errorf("could not init a new database: %w", err)
}
if err := sess.Commit(); err != nil {
return fmt.Errorf("could not commit initial migration session: %w", err)
}
return nil return nil
})
} }
if err := sess.Commit(); err != nil { m.SetLogger(&xormigrateLogger{})
return fmt.Errorf("could not commit initial migration session: %w", err)
}
if err := runTasks(e, migrationTasks); err != nil { if err := m.Migrate(); err != nil {
return fmt.Errorf("run tasks failed: %w", err) return err
} }
e.SetDisableGlobalCache(false) e.SetDisableGlobalCache(false)
@ -157,74 +110,7 @@ func Migrate(e *xorm.Engine) error {
return nil return nil
} }
func runTasks(e *xorm.Engine, tasks []*task) error { func syncAll(sess *xorm.Engine) error {
// cache migrations in db
migCache := make(map[string]bool)
var migList []*migrations
if err := e.Find(&migList); err != nil {
return err
}
for i := range migList {
migCache[migList[i].Name] = true
}
for _, task := range tasks {
if migCache[task.name] {
log.Trace().Msgf("migration task '%s' already applied", task.name)
continue
}
log.Trace().Msgf("start migration task '%s'", task.name)
aliveMsgCancel := showBeAliveSign(task.name)
defer aliveMsgCancel(nil)
var taskErr error
if task.fn != nil {
sess := e.NewSession().NoCache()
defer sess.Close()
if err := sess.Begin(); err != nil {
return fmt.Errorf("could not begin session for '%s': %w", task.name, err)
}
if taskErr = task.fn(sess); taskErr != nil {
aliveMsgCancel(nil)
if err := sess.Rollback(); err != nil {
taskErr = errors.Join(taskErr, err)
}
} else if err := sess.Commit(); err != nil {
return fmt.Errorf("could not commit session for '%s': %w", task.name, err)
}
} else if task.engineFn != nil {
taskErr = task.engineFn(e)
} else {
log.Trace().Msgf("skip migration task '%s'", task.name)
aliveMsgCancel(nil)
continue
}
aliveMsgCancel(nil)
if taskErr != nil {
if task.required {
return fmt.Errorf("migration task '%s' failed: %w", task.name, taskErr)
}
log.Error().Err(taskErr).Msgf("migration task '%s' failed but is not required", task.name)
continue
}
log.Debug().Msgf("migration task '%s' done", task.name)
if _, err := e.Insert(&migrations{task.name}); err != nil {
return fmt.Errorf("migration task '%s' could not be marked as finished: %w", task.name, err)
}
migCache[task.name] = true
}
return nil
}
type syncEngine interface {
Sync(beans ...any) error
}
func syncAll(sess syncEngine) error {
for _, bean := range allBeans { for _, bean := range allBeans {
if err := sess.Sync(bean); err != nil { if err := sess.Sync(bean); err != nil {
return fmt.Errorf("Sync error '%s': %w", reflect.TypeOf(bean), err) return fmt.Errorf("Sync error '%s': %w", reflect.TypeOf(bean), err)
@ -232,20 +118,3 @@ func syncAll(sess syncEngine) error {
} }
return nil return nil
} }
var showBeAliveSignDelay = time.Second * 20
func showBeAliveSign(taskName string) context.CancelCauseFunc {
ctx, cancel := context.WithCancelCause(context.Background())
go func() {
for {
select {
case <-ctx.Done():
return
case <-time.After(showBeAliveSignDelay):
log.Info().Msgf("Migration '%s' is still running, please be patient", taskName)
}
}
}()
return cancel
}

View file

@ -94,14 +94,9 @@ func testDB(t *testing.T, new bool) (engine *xorm.Engine, closeDB func()) {
} }
func TestMigrate(t *testing.T) { func TestMigrate(t *testing.T) {
// make all tasks required for tests
for _, task := range migrationTasks {
task.required = true
}
// init new db // init new db
engine, closeDB := testDB(t, true) engine, closeDB := testDB(t, true)
assert.NoError(t, Migrate(engine)) assert.NoError(t, Migrate(engine, true))
closeDB() closeDB()
dbType := engine.Dialect().URI().DBType dbType := engine.Dialect().URI().DBType
@ -112,6 +107,6 @@ func TestMigrate(t *testing.T) {
// migrate old db // migrate old db
engine, closeDB = testDB(t, false) engine, closeDB = testDB(t, false)
assert.NoError(t, Migrate(engine)) assert.NoError(t, Migrate(engine, true))
closeDB() closeDB()
} }

View file

@ -1155,13 +1155,13 @@ func (_m *Store) LogSave(_a0 *model.Step, _a1 []*model.LogEntry) error {
return r0 return r0
} }
// Migrate provides a mock function with given fields: // Migrate provides a mock function with given fields: _a0
func (_m *Store) Migrate() error { func (_m *Store) Migrate(_a0 bool) error {
ret := _m.Called() ret := _m.Called(_a0)
var r0 error var r0 error
if rf, ok := ret.Get(0).(func() error); ok { if rf, ok := ret.Get(0).(func(bool) error); ok {
r0 = rf() r0 = rf(_a0)
} else { } else {
r0 = ret.Error(0) r0 = ret.Error(0)
} }

View file

@ -197,5 +197,5 @@ type Store interface {
// Store operations // Store operations
Ping() error Ping() error
Close() error Close() error
Migrate() error Migrate(bool) error
} }