You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

migrate.go 3.3KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118
  1. // Copyright 2019 Gitea. All rights reserved.
  2. // Use of this source code is governed by a MIT-style
  3. // license that can be found in the LICENSE file.
  4. package task
  5. import (
  6. "bytes"
  7. "errors"
  8. "fmt"
  9. "strings"
  10. "code.gitea.io/gitea/models"
  11. "code.gitea.io/gitea/modules/log"
  12. "code.gitea.io/gitea/modules/migrations"
  13. "code.gitea.io/gitea/modules/notification"
  14. "code.gitea.io/gitea/modules/structs"
  15. "code.gitea.io/gitea/modules/timeutil"
  16. "code.gitea.io/gitea/modules/util"
  17. )
  18. func handleCreateError(owner *models.User, err error, name string) error {
  19. switch {
  20. case models.IsErrReachLimitOfRepo(err):
  21. return fmt.Errorf("You have already reached your limit of %d repositories", owner.MaxCreationLimit())
  22. case models.IsErrRepoAlreadyExist(err):
  23. return errors.New("The repository name is already used")
  24. case models.IsErrNameReserved(err):
  25. return fmt.Errorf("The repository name '%s' is reserved", err.(models.ErrNameReserved).Name)
  26. case models.IsErrNamePatternNotAllowed(err):
  27. return fmt.Errorf("The pattern '%s' is not allowed in a repository name", err.(models.ErrNamePatternNotAllowed).Pattern)
  28. default:
  29. return err
  30. }
  31. }
  32. func runMigrateTask(t *models.Task) (err error) {
  33. defer func() {
  34. if e := recover(); e != nil {
  35. var buf bytes.Buffer
  36. fmt.Fprintf(&buf, "Handler crashed with error: %v", log.Stack(2))
  37. err = errors.New(buf.String())
  38. }
  39. if err == nil {
  40. err = models.FinishMigrateTask(t)
  41. if err == nil {
  42. notification.NotifyMigrateRepository(t.Doer, t.Owner, t.Repo)
  43. return
  44. }
  45. log.Error("FinishMigrateTask failed: %s", err.Error())
  46. }
  47. t.EndTime = timeutil.TimeStampNow()
  48. t.Status = structs.TaskStatusFailed
  49. t.Errors = err.Error()
  50. if err := t.UpdateCols("status", "errors", "end_time"); err != nil {
  51. log.Error("Task UpdateCols failed: %s", err.Error())
  52. }
  53. if t.Repo != nil {
  54. if errDelete := models.DeleteRepository(t.Doer, t.OwnerID, t.Repo.ID); errDelete != nil {
  55. log.Error("DeleteRepository: %v", errDelete)
  56. }
  57. }
  58. }()
  59. if err := t.LoadRepo(); err != nil {
  60. return err
  61. }
  62. // if repository is ready, then just finsih the task
  63. if t.Repo.Status == models.RepositoryReady {
  64. return nil
  65. }
  66. if err := t.LoadDoer(); err != nil {
  67. return err
  68. }
  69. if err := t.LoadOwner(); err != nil {
  70. return err
  71. }
  72. t.StartTime = timeutil.TimeStampNow()
  73. t.Status = structs.TaskStatusRunning
  74. if err := t.UpdateCols("start_time", "status"); err != nil {
  75. return err
  76. }
  77. var opts *structs.MigrateRepoOption
  78. opts, err = t.MigrateConfig()
  79. if err != nil {
  80. return err
  81. }
  82. opts.MigrateToRepoID = t.RepoID
  83. repo, err := migrations.MigrateRepository(t.Doer, t.Owner.Name, *opts)
  84. if err == nil {
  85. log.Trace("Repository migrated [%d]: %s/%s", repo.ID, t.Owner.Name, repo.Name)
  86. return nil
  87. }
  88. if models.IsErrRepoAlreadyExist(err) {
  89. return errors.New("The repository name is already used")
  90. }
  91. // remoteAddr may contain credentials, so we sanitize it
  92. err = util.URLSanitizedError(err, opts.CloneAddr)
  93. if strings.Contains(err.Error(), "Authentication failed") ||
  94. strings.Contains(err.Error(), "could not read Username") {
  95. return fmt.Errorf("Authentication failed: %v", err.Error())
  96. } else if strings.Contains(err.Error(), "fatal:") {
  97. return fmt.Errorf("Migration failed: %v", err.Error())
  98. }
  99. return handleCreateError(t.Owner, err, "MigratePost")
  100. }