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

migrate.go 5.7 kB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241
  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. "code.gitea.io/gitea/models"
  8. "code.gitea.io/gitea/modules/log"
  9. "code.gitea.io/gitea/modules/migrations/base"
  10. )
  11. // MigrateOptions is equal to base.MigrateOptions
  12. type MigrateOptions = base.MigrateOptions
  13. var (
  14. factories []base.DownloaderFactory
  15. )
  16. // RegisterDownloaderFactory registers a downloader factory
  17. func RegisterDownloaderFactory(factory base.DownloaderFactory) {
  18. factories = append(factories, factory)
  19. }
  20. // MigrateRepository migrate repository according MigrateOptions
  21. func MigrateRepository(doer *models.User, ownerName string, opts base.MigrateOptions) (*models.Repository, error) {
  22. var (
  23. downloader base.Downloader
  24. uploader = NewGiteaLocalUploader(doer, ownerName, opts.Name)
  25. )
  26. for _, factory := range factories {
  27. if match, err := factory.Match(opts); err != nil {
  28. return nil, err
  29. } else if match {
  30. downloader, err = factory.New(opts)
  31. if err != nil {
  32. return nil, err
  33. }
  34. break
  35. }
  36. }
  37. if downloader == nil {
  38. opts.Wiki = true
  39. opts.Milestones = false
  40. opts.Labels = false
  41. opts.Releases = false
  42. opts.Comments = false
  43. opts.Issues = false
  44. opts.PullRequests = false
  45. downloader = NewPlainGitDownloader(ownerName, opts.Name, opts.RemoteURL)
  46. log.Trace("Will migrate from git: %s", opts.RemoteURL)
  47. }
  48. if err := migrateRepository(downloader, uploader, opts); err != nil {
  49. if err1 := uploader.Rollback(); err1 != nil {
  50. log.Error("rollback failed: %v", err1)
  51. }
  52. return nil, err
  53. }
  54. return uploader.repo, nil
  55. }
  56. // migrateRepository will download informations and upload to Uploader, this is a simple
  57. // process for small repository. For a big repository, save all the data to disk
  58. // before upload is better
  59. func migrateRepository(downloader base.Downloader, uploader base.Uploader, opts base.MigrateOptions) error {
  60. repo, err := downloader.GetRepoInfo()
  61. if err != nil {
  62. return err
  63. }
  64. repo.IsPrivate = opts.Private
  65. repo.IsMirror = opts.Mirror
  66. if opts.Description != "" {
  67. repo.Description = opts.Description
  68. }
  69. log.Trace("migrating git data")
  70. if err := uploader.CreateRepo(repo, opts); err != nil {
  71. return err
  72. }
  73. if opts.Milestones {
  74. log.Trace("migrating milestones")
  75. milestones, err := downloader.GetMilestones()
  76. if err != nil {
  77. return err
  78. }
  79. msBatchSize := uploader.MaxBatchInsertSize("milestone")
  80. for len(milestones) > 0 {
  81. if len(milestones) < msBatchSize {
  82. msBatchSize = len(milestones)
  83. }
  84. if err := uploader.CreateMilestones(milestones...); err != nil {
  85. return err
  86. }
  87. milestones = milestones[msBatchSize:]
  88. }
  89. }
  90. if opts.Labels {
  91. log.Trace("migrating labels")
  92. labels, err := downloader.GetLabels()
  93. if err != nil {
  94. return err
  95. }
  96. lbBatchSize := uploader.MaxBatchInsertSize("label")
  97. for len(labels) > 0 {
  98. if len(labels) < lbBatchSize {
  99. lbBatchSize = len(labels)
  100. }
  101. if err := uploader.CreateLabels(labels...); err != nil {
  102. return err
  103. }
  104. labels = labels[lbBatchSize:]
  105. }
  106. }
  107. if opts.Releases {
  108. log.Trace("migrating releases")
  109. releases, err := downloader.GetReleases()
  110. if err != nil {
  111. return err
  112. }
  113. relBatchSize := uploader.MaxBatchInsertSize("release")
  114. for len(releases) > 0 {
  115. if len(releases) < relBatchSize {
  116. relBatchSize = len(releases)
  117. }
  118. if err := uploader.CreateReleases(releases[:relBatchSize]...); err != nil {
  119. return err
  120. }
  121. releases = releases[relBatchSize:]
  122. }
  123. }
  124. var commentBatchSize = uploader.MaxBatchInsertSize("comment")
  125. if opts.Issues {
  126. log.Trace("migrating issues and comments")
  127. var issueBatchSize = uploader.MaxBatchInsertSize("issue")
  128. for i := 1; ; i++ {
  129. issues, isEnd, err := downloader.GetIssues(i, issueBatchSize)
  130. if err != nil {
  131. return err
  132. }
  133. if err := uploader.CreateIssues(issues...); err != nil {
  134. return err
  135. }
  136. if !opts.Comments {
  137. continue
  138. }
  139. var allComments = make([]*base.Comment, 0, commentBatchSize)
  140. for _, issue := range issues {
  141. comments, err := downloader.GetComments(issue.Number)
  142. if err != nil {
  143. return err
  144. }
  145. allComments = append(allComments, comments...)
  146. if len(allComments) >= commentBatchSize {
  147. if err := uploader.CreateComments(allComments[:commentBatchSize]...); err != nil {
  148. return err
  149. }
  150. allComments = allComments[commentBatchSize:]
  151. }
  152. }
  153. if len(allComments) > 0 {
  154. if err := uploader.CreateComments(allComments...); err != nil {
  155. return err
  156. }
  157. }
  158. if isEnd {
  159. break
  160. }
  161. }
  162. }
  163. if opts.PullRequests {
  164. log.Trace("migrating pull requests and comments")
  165. var prBatchSize = uploader.MaxBatchInsertSize("pullrequest")
  166. for i := 1; ; i++ {
  167. prs, err := downloader.GetPullRequests(i, prBatchSize)
  168. if err != nil {
  169. return err
  170. }
  171. if err := uploader.CreatePullRequests(prs...); err != nil {
  172. return err
  173. }
  174. if !opts.Comments {
  175. continue
  176. }
  177. var allComments = make([]*base.Comment, 0, commentBatchSize)
  178. for _, pr := range prs {
  179. comments, err := downloader.GetComments(pr.Number)
  180. if err != nil {
  181. return err
  182. }
  183. allComments = append(allComments, comments...)
  184. if len(allComments) >= commentBatchSize {
  185. if err := uploader.CreateComments(allComments[:commentBatchSize]...); err != nil {
  186. return err
  187. }
  188. allComments = allComments[commentBatchSize:]
  189. }
  190. }
  191. if len(allComments) > 0 {
  192. if err := uploader.CreateComments(allComments...); err != nil {
  193. return err
  194. }
  195. }
  196. if len(prs) < prBatchSize {
  197. break
  198. }
  199. }
  200. }
  201. return nil
  202. }