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 6.5KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271
  1. // Copyright 2019 The Gitea Authors. All rights reserved.
  2. // Copyright 2018 Jonas Franz. All rights reserved.
  3. // Use of this source code is governed by a MIT-style
  4. // license that can be found in the LICENSE file.
  5. package migrations
  6. import (
  7. "fmt"
  8. "code.gitea.io/gitea/models"
  9. "code.gitea.io/gitea/modules/log"
  10. "code.gitea.io/gitea/modules/migrations/base"
  11. "code.gitea.io/gitea/modules/structs"
  12. )
  13. // MigrateOptions is equal to base.MigrateOptions
  14. type MigrateOptions = base.MigrateOptions
  15. var (
  16. factories []base.DownloaderFactory
  17. )
  18. // RegisterDownloaderFactory registers a downloader factory
  19. func RegisterDownloaderFactory(factory base.DownloaderFactory) {
  20. factories = append(factories, factory)
  21. }
  22. // MigrateRepository migrate repository according MigrateOptions
  23. func MigrateRepository(doer *models.User, ownerName string, opts base.MigrateOptions) (*models.Repository, error) {
  24. var (
  25. downloader base.Downloader
  26. uploader = NewGiteaLocalUploader(doer, ownerName, opts.RepoName)
  27. theFactory base.DownloaderFactory
  28. )
  29. for _, factory := range factories {
  30. if match, err := factory.Match(opts); err != nil {
  31. return nil, err
  32. } else if match {
  33. downloader, err = factory.New(opts)
  34. if err != nil {
  35. return nil, err
  36. }
  37. theFactory = factory
  38. break
  39. }
  40. }
  41. if downloader == nil {
  42. opts.Wiki = true
  43. opts.Milestones = false
  44. opts.Labels = false
  45. opts.Releases = false
  46. opts.Comments = false
  47. opts.Issues = false
  48. opts.PullRequests = false
  49. opts.GitServiceType = structs.PlainGitService
  50. downloader = NewPlainGitDownloader(ownerName, opts.RepoName, opts.CloneAddr)
  51. log.Trace("Will migrate from git: %s", opts.CloneAddr)
  52. } else if opts.GitServiceType == structs.NotMigrated {
  53. opts.GitServiceType = theFactory.GitServiceType()
  54. }
  55. uploader.gitServiceType = opts.GitServiceType
  56. if err := migrateRepository(downloader, uploader, opts); err != nil {
  57. if err1 := uploader.Rollback(); err1 != nil {
  58. log.Error("rollback failed: %v", err1)
  59. }
  60. if err2 := models.CreateRepositoryNotice(fmt.Sprintf("Migrate repository from %s failed: %v", opts.CloneAddr, err)); err2 != nil {
  61. log.Error("create respotiry notice failed: ", err2)
  62. }
  63. return nil, err
  64. }
  65. return uploader.repo, nil
  66. }
  67. // migrateRepository will download informations and upload to Uploader, this is a simple
  68. // process for small repository. For a big repository, save all the data to disk
  69. // before upload is better
  70. func migrateRepository(downloader base.Downloader, uploader base.Uploader, opts base.MigrateOptions) error {
  71. repo, err := downloader.GetRepoInfo()
  72. if err != nil {
  73. return err
  74. }
  75. repo.IsPrivate = opts.Private
  76. repo.IsMirror = opts.Mirror
  77. if opts.Description != "" {
  78. repo.Description = opts.Description
  79. }
  80. log.Trace("migrating git data")
  81. if err := uploader.CreateRepo(repo, opts); err != nil {
  82. return err
  83. }
  84. defer uploader.Close()
  85. log.Trace("migrating topics")
  86. topics, err := downloader.GetTopics()
  87. if err != nil {
  88. return err
  89. }
  90. if len(topics) > 0 {
  91. if err := uploader.CreateTopics(topics...); err != nil {
  92. return err
  93. }
  94. }
  95. if opts.Milestones {
  96. log.Trace("migrating milestones")
  97. milestones, err := downloader.GetMilestones()
  98. if err != nil {
  99. return err
  100. }
  101. msBatchSize := uploader.MaxBatchInsertSize("milestone")
  102. for len(milestones) > 0 {
  103. if len(milestones) < msBatchSize {
  104. msBatchSize = len(milestones)
  105. }
  106. if err := uploader.CreateMilestones(milestones...); err != nil {
  107. return err
  108. }
  109. milestones = milestones[msBatchSize:]
  110. }
  111. }
  112. if opts.Labels {
  113. log.Trace("migrating labels")
  114. labels, err := downloader.GetLabels()
  115. if err != nil {
  116. return err
  117. }
  118. lbBatchSize := uploader.MaxBatchInsertSize("label")
  119. for len(labels) > 0 {
  120. if len(labels) < lbBatchSize {
  121. lbBatchSize = len(labels)
  122. }
  123. if err := uploader.CreateLabels(labels...); err != nil {
  124. return err
  125. }
  126. labels = labels[lbBatchSize:]
  127. }
  128. }
  129. if opts.Releases {
  130. log.Trace("migrating releases")
  131. releases, err := downloader.GetReleases()
  132. if err != nil {
  133. return err
  134. }
  135. relBatchSize := uploader.MaxBatchInsertSize("release")
  136. for len(releases) > 0 {
  137. if len(releases) < relBatchSize {
  138. relBatchSize = len(releases)
  139. }
  140. if err := uploader.CreateReleases(releases[:relBatchSize]...); err != nil {
  141. return err
  142. }
  143. releases = releases[relBatchSize:]
  144. }
  145. // Once all releases (if any) are inserted, sync any remaining non-release tags
  146. if err := uploader.SyncTags(); err != nil {
  147. return err
  148. }
  149. }
  150. var commentBatchSize = uploader.MaxBatchInsertSize("comment")
  151. if opts.Issues {
  152. log.Trace("migrating issues and comments")
  153. var issueBatchSize = uploader.MaxBatchInsertSize("issue")
  154. for i := 1; ; i++ {
  155. issues, isEnd, err := downloader.GetIssues(i, issueBatchSize)
  156. if err != nil {
  157. return err
  158. }
  159. if err := uploader.CreateIssues(issues...); err != nil {
  160. return err
  161. }
  162. if !opts.Comments {
  163. continue
  164. }
  165. var allComments = make([]*base.Comment, 0, commentBatchSize)
  166. for _, issue := range issues {
  167. comments, err := downloader.GetComments(issue.Number)
  168. if err != nil {
  169. return err
  170. }
  171. allComments = append(allComments, comments...)
  172. if len(allComments) >= commentBatchSize {
  173. if err := uploader.CreateComments(allComments[:commentBatchSize]...); err != nil {
  174. return err
  175. }
  176. allComments = allComments[commentBatchSize:]
  177. }
  178. }
  179. if len(allComments) > 0 {
  180. if err := uploader.CreateComments(allComments...); err != nil {
  181. return err
  182. }
  183. }
  184. if isEnd {
  185. break
  186. }
  187. }
  188. }
  189. if opts.PullRequests {
  190. log.Trace("migrating pull requests and comments")
  191. var prBatchSize = uploader.MaxBatchInsertSize("pullrequest")
  192. for i := 1; ; i++ {
  193. prs, err := downloader.GetPullRequests(i, prBatchSize)
  194. if err != nil {
  195. return err
  196. }
  197. if err := uploader.CreatePullRequests(prs...); err != nil {
  198. return err
  199. }
  200. if !opts.Comments {
  201. continue
  202. }
  203. var allComments = make([]*base.Comment, 0, commentBatchSize)
  204. for _, pr := range prs {
  205. comments, err := downloader.GetComments(pr.Number)
  206. if err != nil {
  207. return err
  208. }
  209. allComments = append(allComments, comments...)
  210. if len(allComments) >= commentBatchSize {
  211. if err := uploader.CreateComments(allComments[:commentBatchSize]...); err != nil {
  212. return err
  213. }
  214. allComments = allComments[commentBatchSize:]
  215. }
  216. }
  217. if len(allComments) > 0 {
  218. if err := uploader.CreateComments(allComments...); err != nil {
  219. return err
  220. }
  221. }
  222. if len(prs) < prBatchSize {
  223. break
  224. }
  225. }
  226. }
  227. return nil
  228. }