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.

transfer.go 4.2KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132
  1. // Copyright 2019 The Gitea Authors. All rights reserved.
  2. // SPDX-License-Identifier: MIT
  3. package repository
  4. import (
  5. "context"
  6. "fmt"
  7. "code.gitea.io/gitea/models"
  8. "code.gitea.io/gitea/models/organization"
  9. "code.gitea.io/gitea/models/perm"
  10. access_model "code.gitea.io/gitea/models/perm/access"
  11. repo_model "code.gitea.io/gitea/models/repo"
  12. user_model "code.gitea.io/gitea/models/user"
  13. "code.gitea.io/gitea/modules/log"
  14. repo_module "code.gitea.io/gitea/modules/repository"
  15. "code.gitea.io/gitea/modules/sync"
  16. notify_service "code.gitea.io/gitea/services/notify"
  17. )
  18. // repoWorkingPool represents a working pool to order the parallel changes to the same repository
  19. // TODO: use clustered lock (unique queue? or *abuse* cache)
  20. var repoWorkingPool = sync.NewExclusivePool()
  21. // TransferOwnership transfers all corresponding setting from old user to new one.
  22. func TransferOwnership(ctx context.Context, doer, newOwner *user_model.User, repo *repo_model.Repository, teams []*organization.Team) error {
  23. if err := repo.LoadOwner(ctx); err != nil {
  24. return err
  25. }
  26. for _, team := range teams {
  27. if newOwner.ID != team.OrgID {
  28. return fmt.Errorf("team %d does not belong to organization", team.ID)
  29. }
  30. }
  31. oldOwner := repo.Owner
  32. repoWorkingPool.CheckIn(fmt.Sprint(repo.ID))
  33. if err := models.TransferOwnership(ctx, doer, newOwner.Name, repo); err != nil {
  34. repoWorkingPool.CheckOut(fmt.Sprint(repo.ID))
  35. return err
  36. }
  37. repoWorkingPool.CheckOut(fmt.Sprint(repo.ID))
  38. newRepo, err := repo_model.GetRepositoryByID(ctx, repo.ID)
  39. if err != nil {
  40. return err
  41. }
  42. for _, team := range teams {
  43. if err := models.AddRepository(ctx, team, newRepo); err != nil {
  44. return err
  45. }
  46. }
  47. notify_service.TransferRepository(ctx, doer, repo, oldOwner.Name)
  48. return nil
  49. }
  50. // ChangeRepositoryName changes all corresponding setting from old repository name to new one.
  51. func ChangeRepositoryName(ctx context.Context, doer *user_model.User, repo *repo_model.Repository, newRepoName string) error {
  52. log.Trace("ChangeRepositoryName: %s/%s -> %s", doer.Name, repo.Name, newRepoName)
  53. oldRepoName := repo.Name
  54. // Change repository directory name. We must lock the local copy of the
  55. // repo so that we can atomically rename the repo path and updates the
  56. // local copy's origin accordingly.
  57. repoWorkingPool.CheckIn(fmt.Sprint(repo.ID))
  58. if err := repo_model.ChangeRepositoryName(ctx, doer, repo, newRepoName); err != nil {
  59. repoWorkingPool.CheckOut(fmt.Sprint(repo.ID))
  60. return err
  61. }
  62. repoWorkingPool.CheckOut(fmt.Sprint(repo.ID))
  63. repo.Name = newRepoName
  64. notify_service.RenameRepository(ctx, doer, repo, oldRepoName)
  65. return nil
  66. }
  67. // StartRepositoryTransfer transfer a repo from one owner to a new one.
  68. // it make repository into pending transfer state, if doer can not create repo for new owner.
  69. func StartRepositoryTransfer(ctx context.Context, doer, newOwner *user_model.User, repo *repo_model.Repository, teams []*organization.Team) error {
  70. if err := models.TestRepositoryReadyForTransfer(repo.Status); err != nil {
  71. return err
  72. }
  73. // Admin is always allowed to transfer || user transfer repo back to his account
  74. if doer.IsAdmin || doer.ID == newOwner.ID {
  75. return TransferOwnership(ctx, doer, newOwner, repo, teams)
  76. }
  77. // If new owner is an org and user can create repos he can transfer directly too
  78. if newOwner.IsOrganization() {
  79. allowed, err := organization.CanCreateOrgRepo(ctx, newOwner.ID, doer.ID)
  80. if err != nil {
  81. return err
  82. }
  83. if allowed {
  84. return TransferOwnership(ctx, doer, newOwner, repo, teams)
  85. }
  86. }
  87. // In case the new owner would not have sufficient access to the repo, give access rights for read
  88. hasAccess, err := access_model.HasAccess(ctx, newOwner.ID, repo)
  89. if err != nil {
  90. return err
  91. }
  92. if !hasAccess {
  93. if err := repo_module.AddCollaborator(ctx, repo, newOwner); err != nil {
  94. return err
  95. }
  96. if err := repo_model.ChangeCollaborationAccessMode(ctx, repo, newOwner.ID, perm.AccessModeRead); err != nil {
  97. return err
  98. }
  99. }
  100. // Make repo as pending for transfer
  101. repo.Status = repo_model.RepositoryPendingTransfer
  102. if err := models.CreatePendingRepositoryTransfer(ctx, doer, newOwner, repo.ID, teams); err != nil {
  103. return err
  104. }
  105. // notify users who are able to accept / reject transfer
  106. notify_service.RepoPendingTransfer(ctx, doer, newOwner, repo)
  107. return nil
  108. }