123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149 |
- // Copyright 2015 The Gogs Authors. All rights reserved.
- // Use of this source code is governed by a MIT-style
- // license that can be found in the LICENSE file.
- package migrations
- import (
- "github.com/pkg/errors"
- "gorm.io/gorm"
- log "unknwon.dev/clog/v2"
- )
- const minDBVersion = 19
- type Migration interface {
- Description() string
- Migrate(*gorm.DB) error
- }
- type migration struct {
- description string
- migrate func(*gorm.DB) error
- }
- func NewMigration(desc string, fn func(*gorm.DB) error) Migration {
- return &migration{desc, fn}
- }
- func (m *migration) Description() string {
- return m.description
- }
- func (m *migration) Migrate(db *gorm.DB) error {
- return m.migrate(db)
- }
- // Version represents the version table. It should have only one row with `id == 1`.
- type Version struct {
- ID int64
- Version int64
- }
- // This is a sequence of migrations. Add new migrations to the bottom of the list.
- // If you want to "retire" a migration, remove it from the top of the list and
- // update _MIN_VER_DB accordingly
- var migrations = []Migration{
- // v0 -> v4 : before 0.6.0 -> last support 0.7.33
- // v4 -> v10: before 0.7.0 -> last support 0.9.141
- // v10 -> v19: before 0.11.55 -> last support 0.12.0
- // Add new migration here, example:
- // v18 -> v19:v0.11.55
- // NewMigration("clean unlinked webhook and hook_tasks", cleanUnlinkedWebhookAndHookTasks),
- // v19 -> v20:v0.13.0
- NewMigration("migrate access tokens to store SHA56", migrateAccessTokenToSHA256),
- // v20 -> v21:v0.13.0
- NewMigration("add index to action.user_id", addIndexToActionUserID),
- // v21 -> v22:v0.13.0
- //
- // NOTE: There was a bug in calculating the value of the `version.version`
- // column after a migration is done, thus some instances are on v21 but some are
- // on v22. Let's make a noop v22 to make sure every instance will not miss a
- // real future migration.
- NewMigration("noop", func(*gorm.DB) error { return nil }),
- }
- var errMigrationSkipped = errors.New("the migration has been skipped")
- // Migrate migrates the database schema and/or data to the current version.
- func Migrate(db *gorm.DB) error {
- // NOTE: GORM has problem migrating tables that happen to have columns with the
- // same name, see https://github.com/gogs/gogs/issues/7056.
- if !db.Migrator().HasTable(new(Version)) {
- err := db.AutoMigrate(new(Version))
- if err != nil {
- return errors.Wrap(err, `auto migrate "version" table`)
- }
- }
- var current Version
- err := db.Where("id = ?", 1).First(¤t).Error
- if err == gorm.ErrRecordNotFound {
- err = db.Create(
- &Version{
- ID: 1,
- Version: int64(minDBVersion + len(migrations)),
- },
- ).Error
- if err != nil {
- return errors.Wrap(err, "create the version record")
- }
- return nil
- } else if err != nil {
- return errors.Wrap(err, "get the version record")
- }
- if minDBVersion > current.Version {
- log.Fatal(`
- Hi there, thank you for using Gogs for so long!
- However, Gogs has stopped supporting auto-migration from your previously installed version.
- But the good news is, it's very easy to fix this problem!
- You can migrate your older database using a previous release, then you can upgrade to the newest version.
- Please save following instructions to somewhere and start working:
- - If you were using below 0.6.0 (e.g. 0.5.x), download last supported archive from following link:
- https://gogs.io/gogs/releases/tag/v0.7.33
- - If you were using below 0.7.0 (e.g. 0.6.x), download last supported archive from following link:
- https://gogs.io/gogs/releases/tag/v0.9.141
- - If you were using below 0.11.55 (e.g. 0.9.141), download last supported archive from following link:
- https://gogs.io/gogs/releases/tag/v0.12.0
- Once finished downloading:
- 1. Extract the archive and to upgrade steps as usual.
- 2. Run it once. To verify, you should see some migration traces.
- 3. Once it starts web server successfully, stop it.
- 4. Now it's time to put back the release archive you originally intent to upgrade.
- 5. Enjoy!
- In case you're stilling getting this notice, go through instructions again until it disappears.`)
- return nil
- }
- if int(current.Version-minDBVersion) > len(migrations) {
- // User downgraded Gogs.
- current.Version = int64(len(migrations) + minDBVersion)
- return db.Where("id = ?", current.ID).Updates(current).Error
- }
- for _, m := range migrations[current.Version-minDBVersion:] {
- log.Info("Migration: %s", m.Description())
- if err = m.Migrate(db); err != nil {
- if err != errMigrationSkipped {
- return errors.Wrap(err, "do migrate")
- }
- log.Trace("The migration %q has been skipped", m.Description())
- }
- current.Version++
- err = db.Where("id = ?", current.ID).Updates(current).Error
- if err != nil {
- return errors.Wrap(err, "update the version record")
- }
- }
- return nil
- }
|