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.

gitea.go 16KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636
  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. "context"
  8. "fmt"
  9. "io"
  10. "net/http"
  11. "net/url"
  12. "os"
  13. "path"
  14. "path/filepath"
  15. "strings"
  16. "sync"
  17. "time"
  18. "code.gitea.io/gitea/models"
  19. "code.gitea.io/gitea/modules/git"
  20. "code.gitea.io/gitea/modules/log"
  21. "code.gitea.io/gitea/modules/migrations/base"
  22. "code.gitea.io/gitea/modules/repository"
  23. repo_module "code.gitea.io/gitea/modules/repository"
  24. "code.gitea.io/gitea/modules/setting"
  25. "code.gitea.io/gitea/modules/structs"
  26. "code.gitea.io/gitea/modules/timeutil"
  27. gouuid "github.com/satori/go.uuid"
  28. )
  29. var (
  30. _ base.Uploader = &GiteaLocalUploader{}
  31. )
  32. // GiteaLocalUploader implements an Uploader to gitea sites
  33. type GiteaLocalUploader struct {
  34. ctx context.Context
  35. doer *models.User
  36. repoOwner string
  37. repoName string
  38. repo *models.Repository
  39. labels sync.Map
  40. milestones sync.Map
  41. issues sync.Map
  42. gitRepo *git.Repository
  43. prHeadCache map[string]struct{}
  44. userMap map[int64]int64 // external user id mapping to user id
  45. gitServiceType structs.GitServiceType
  46. }
  47. // NewGiteaLocalUploader creates an gitea Uploader via gitea API v1
  48. func NewGiteaLocalUploader(ctx context.Context, doer *models.User, repoOwner, repoName string) *GiteaLocalUploader {
  49. return &GiteaLocalUploader{
  50. ctx: ctx,
  51. doer: doer,
  52. repoOwner: repoOwner,
  53. repoName: repoName,
  54. prHeadCache: make(map[string]struct{}),
  55. userMap: make(map[int64]int64),
  56. }
  57. }
  58. // MaxBatchInsertSize returns the table's max batch insert size
  59. func (g *GiteaLocalUploader) MaxBatchInsertSize(tp string) int {
  60. switch tp {
  61. case "issue":
  62. return models.MaxBatchInsertSize(new(models.Issue))
  63. case "comment":
  64. return models.MaxBatchInsertSize(new(models.Comment))
  65. case "milestone":
  66. return models.MaxBatchInsertSize(new(models.Milestone))
  67. case "label":
  68. return models.MaxBatchInsertSize(new(models.Label))
  69. case "release":
  70. return models.MaxBatchInsertSize(new(models.Release))
  71. case "pullrequest":
  72. return models.MaxBatchInsertSize(new(models.PullRequest))
  73. }
  74. return 10
  75. }
  76. // CreateRepo creates a repository
  77. func (g *GiteaLocalUploader) CreateRepo(repo *base.Repository, opts base.MigrateOptions) error {
  78. owner, err := models.GetUserByName(g.repoOwner)
  79. if err != nil {
  80. return err
  81. }
  82. var remoteAddr = repo.CloneURL
  83. if len(opts.AuthUsername) > 0 {
  84. u, err := url.Parse(repo.CloneURL)
  85. if err != nil {
  86. return err
  87. }
  88. u.User = url.UserPassword(opts.AuthUsername, opts.AuthPassword)
  89. remoteAddr = u.String()
  90. }
  91. var r *models.Repository
  92. if opts.MigrateToRepoID <= 0 {
  93. r, err = repo_module.CreateRepository(g.doer, owner, models.CreateRepoOptions{
  94. Name: g.repoName,
  95. Description: repo.Description,
  96. OriginalURL: repo.OriginalURL,
  97. GitServiceType: opts.GitServiceType,
  98. IsPrivate: opts.Private,
  99. IsMirror: opts.Mirror,
  100. Status: models.RepositoryBeingMigrated,
  101. })
  102. } else {
  103. r, err = models.GetRepositoryByID(opts.MigrateToRepoID)
  104. }
  105. if err != nil {
  106. return err
  107. }
  108. r, err = repository.MigrateRepositoryGitData(g.doer, owner, r, structs.MigrateRepoOption{
  109. RepoName: g.repoName,
  110. Description: repo.Description,
  111. OriginalURL: repo.OriginalURL,
  112. GitServiceType: opts.GitServiceType,
  113. Mirror: repo.IsMirror,
  114. CloneAddr: remoteAddr,
  115. Private: repo.IsPrivate,
  116. Wiki: opts.Wiki,
  117. Releases: opts.Releases, // if didn't get releases, then sync them from tags
  118. })
  119. g.repo = r
  120. if err != nil {
  121. return err
  122. }
  123. g.gitRepo, err = git.OpenRepository(r.RepoPath())
  124. return err
  125. }
  126. // Close closes this uploader
  127. func (g *GiteaLocalUploader) Close() {
  128. if g.gitRepo != nil {
  129. g.gitRepo.Close()
  130. }
  131. }
  132. // CreateTopics creates topics
  133. func (g *GiteaLocalUploader) CreateTopics(topics ...string) error {
  134. return models.SaveTopics(g.repo.ID, topics...)
  135. }
  136. // CreateMilestones creates milestones
  137. func (g *GiteaLocalUploader) CreateMilestones(milestones ...*base.Milestone) error {
  138. var mss = make([]*models.Milestone, 0, len(milestones))
  139. for _, milestone := range milestones {
  140. var deadline timeutil.TimeStamp
  141. if milestone.Deadline != nil {
  142. deadline = timeutil.TimeStamp(milestone.Deadline.Unix())
  143. }
  144. if deadline == 0 {
  145. deadline = timeutil.TimeStamp(time.Date(9999, 1, 1, 0, 0, 0, 0, setting.DefaultUILocation).Unix())
  146. }
  147. var ms = models.Milestone{
  148. RepoID: g.repo.ID,
  149. Name: milestone.Title,
  150. Content: milestone.Description,
  151. IsClosed: milestone.State == "closed",
  152. DeadlineUnix: deadline,
  153. }
  154. if ms.IsClosed && milestone.Closed != nil {
  155. ms.ClosedDateUnix = timeutil.TimeStamp(milestone.Closed.Unix())
  156. }
  157. mss = append(mss, &ms)
  158. }
  159. err := models.InsertMilestones(mss...)
  160. if err != nil {
  161. return err
  162. }
  163. for _, ms := range mss {
  164. g.milestones.Store(ms.Name, ms.ID)
  165. }
  166. return nil
  167. }
  168. // CreateLabels creates labels
  169. func (g *GiteaLocalUploader) CreateLabels(labels ...*base.Label) error {
  170. var lbs = make([]*models.Label, 0, len(labels))
  171. for _, label := range labels {
  172. lbs = append(lbs, &models.Label{
  173. RepoID: g.repo.ID,
  174. Name: label.Name,
  175. Description: label.Description,
  176. Color: fmt.Sprintf("#%s", label.Color),
  177. })
  178. }
  179. err := models.NewLabels(lbs...)
  180. if err != nil {
  181. return err
  182. }
  183. for _, lb := range lbs {
  184. g.labels.Store(lb.Name, lb)
  185. }
  186. return nil
  187. }
  188. // CreateReleases creates releases
  189. func (g *GiteaLocalUploader) CreateReleases(releases ...*base.Release) error {
  190. var rels = make([]*models.Release, 0, len(releases))
  191. for _, release := range releases {
  192. var rel = models.Release{
  193. RepoID: g.repo.ID,
  194. TagName: release.TagName,
  195. LowerTagName: strings.ToLower(release.TagName),
  196. Target: release.TargetCommitish,
  197. Title: release.Name,
  198. Sha1: release.TargetCommitish,
  199. Note: release.Body,
  200. IsDraft: release.Draft,
  201. IsPrerelease: release.Prerelease,
  202. IsTag: false,
  203. CreatedUnix: timeutil.TimeStamp(release.Created.Unix()),
  204. }
  205. userid, ok := g.userMap[release.PublisherID]
  206. tp := g.gitServiceType.Name()
  207. if !ok && tp != "" {
  208. var err error
  209. userid, err = models.GetUserIDByExternalUserID(tp, fmt.Sprintf("%v", release.PublisherID))
  210. if err != nil {
  211. log.Error("GetUserIDByExternalUserID: %v", err)
  212. }
  213. if userid > 0 {
  214. g.userMap[release.PublisherID] = userid
  215. }
  216. }
  217. if userid > 0 {
  218. rel.PublisherID = userid
  219. } else {
  220. rel.PublisherID = g.doer.ID
  221. rel.OriginalAuthor = release.PublisherName
  222. rel.OriginalAuthorID = release.PublisherID
  223. }
  224. // calc NumCommits
  225. commit, err := g.gitRepo.GetCommit(rel.TagName)
  226. if err != nil {
  227. return fmt.Errorf("GetCommit: %v", err)
  228. }
  229. rel.NumCommits, err = commit.CommitsCount()
  230. if err != nil {
  231. return fmt.Errorf("CommitsCount: %v", err)
  232. }
  233. for _, asset := range release.Assets {
  234. var attach = models.Attachment{
  235. UUID: gouuid.NewV4().String(),
  236. Name: asset.Name,
  237. DownloadCount: int64(*asset.DownloadCount),
  238. Size: int64(*asset.Size),
  239. CreatedUnix: timeutil.TimeStamp(asset.Created.Unix()),
  240. }
  241. // download attachment
  242. err = func() error {
  243. resp, err := http.Get(asset.URL)
  244. if err != nil {
  245. return err
  246. }
  247. defer resp.Body.Close()
  248. localPath := attach.LocalPath()
  249. if err = os.MkdirAll(path.Dir(localPath), os.ModePerm); err != nil {
  250. return fmt.Errorf("MkdirAll: %v", err)
  251. }
  252. fw, err := os.Create(localPath)
  253. if err != nil {
  254. return fmt.Errorf("Create: %v", err)
  255. }
  256. defer fw.Close()
  257. _, err = io.Copy(fw, resp.Body)
  258. return err
  259. }()
  260. if err != nil {
  261. return err
  262. }
  263. rel.Attachments = append(rel.Attachments, &attach)
  264. }
  265. rels = append(rels, &rel)
  266. }
  267. return models.InsertReleases(rels...)
  268. }
  269. // SyncTags syncs releases with tags in the database
  270. func (g *GiteaLocalUploader) SyncTags() error {
  271. return repository.SyncReleasesWithTags(g.repo, g.gitRepo)
  272. }
  273. // CreateIssues creates issues
  274. func (g *GiteaLocalUploader) CreateIssues(issues ...*base.Issue) error {
  275. var iss = make([]*models.Issue, 0, len(issues))
  276. for _, issue := range issues {
  277. var labels []*models.Label
  278. for _, label := range issue.Labels {
  279. lb, ok := g.labels.Load(label.Name)
  280. if ok {
  281. labels = append(labels, lb.(*models.Label))
  282. }
  283. }
  284. var milestoneID int64
  285. if issue.Milestone != "" {
  286. milestone, ok := g.milestones.Load(issue.Milestone)
  287. if ok {
  288. milestoneID = milestone.(int64)
  289. }
  290. }
  291. var is = models.Issue{
  292. RepoID: g.repo.ID,
  293. Repo: g.repo,
  294. Index: issue.Number,
  295. Title: issue.Title,
  296. Content: issue.Content,
  297. IsClosed: issue.State == "closed",
  298. IsLocked: issue.IsLocked,
  299. MilestoneID: milestoneID,
  300. Labels: labels,
  301. CreatedUnix: timeutil.TimeStamp(issue.Created.Unix()),
  302. }
  303. userid, ok := g.userMap[issue.PosterID]
  304. tp := g.gitServiceType.Name()
  305. if !ok && tp != "" {
  306. var err error
  307. userid, err = models.GetUserIDByExternalUserID(tp, fmt.Sprintf("%v", issue.PosterID))
  308. if err != nil {
  309. log.Error("GetUserIDByExternalUserID: %v", err)
  310. }
  311. if userid > 0 {
  312. g.userMap[issue.PosterID] = userid
  313. }
  314. }
  315. if userid > 0 {
  316. is.PosterID = userid
  317. } else {
  318. is.PosterID = g.doer.ID
  319. is.OriginalAuthor = issue.PosterName
  320. is.OriginalAuthorID = issue.PosterID
  321. }
  322. if issue.Closed != nil {
  323. is.ClosedUnix = timeutil.TimeStamp(issue.Closed.Unix())
  324. }
  325. // TODO: add reactions
  326. iss = append(iss, &is)
  327. }
  328. err := models.InsertIssues(iss...)
  329. if err != nil {
  330. return err
  331. }
  332. for _, is := range iss {
  333. g.issues.Store(is.Index, is.ID)
  334. }
  335. return nil
  336. }
  337. // CreateComments creates comments of issues
  338. func (g *GiteaLocalUploader) CreateComments(comments ...*base.Comment) error {
  339. var cms = make([]*models.Comment, 0, len(comments))
  340. for _, comment := range comments {
  341. var issueID int64
  342. if issueIDStr, ok := g.issues.Load(comment.IssueIndex); !ok {
  343. issue, err := models.GetIssueByIndex(g.repo.ID, comment.IssueIndex)
  344. if err != nil {
  345. return err
  346. }
  347. issueID = issue.ID
  348. g.issues.Store(comment.IssueIndex, issueID)
  349. } else {
  350. issueID = issueIDStr.(int64)
  351. }
  352. userid, ok := g.userMap[comment.PosterID]
  353. tp := g.gitServiceType.Name()
  354. if !ok && tp != "" {
  355. var err error
  356. userid, err = models.GetUserIDByExternalUserID(tp, fmt.Sprintf("%v", comment.PosterID))
  357. if err != nil {
  358. log.Error("GetUserIDByExternalUserID: %v", err)
  359. }
  360. if userid > 0 {
  361. g.userMap[comment.PosterID] = userid
  362. }
  363. }
  364. cm := models.Comment{
  365. IssueID: issueID,
  366. Type: models.CommentTypeComment,
  367. Content: comment.Content,
  368. CreatedUnix: timeutil.TimeStamp(comment.Created.Unix()),
  369. }
  370. if userid > 0 {
  371. cm.PosterID = userid
  372. } else {
  373. cm.PosterID = g.doer.ID
  374. cm.OriginalAuthor = comment.PosterName
  375. cm.OriginalAuthorID = comment.PosterID
  376. }
  377. cms = append(cms, &cm)
  378. // TODO: Reactions
  379. }
  380. return models.InsertIssueComments(cms)
  381. }
  382. // CreatePullRequests creates pull requests
  383. func (g *GiteaLocalUploader) CreatePullRequests(prs ...*base.PullRequest) error {
  384. var gprs = make([]*models.PullRequest, 0, len(prs))
  385. for _, pr := range prs {
  386. gpr, err := g.newPullRequest(pr)
  387. if err != nil {
  388. return err
  389. }
  390. userid, ok := g.userMap[pr.PosterID]
  391. tp := g.gitServiceType.Name()
  392. if !ok && tp != "" {
  393. var err error
  394. userid, err = models.GetUserIDByExternalUserID(tp, fmt.Sprintf("%v", pr.PosterID))
  395. if err != nil {
  396. log.Error("GetUserIDByExternalUserID: %v", err)
  397. }
  398. if userid > 0 {
  399. g.userMap[pr.PosterID] = userid
  400. }
  401. }
  402. if userid > 0 {
  403. gpr.Issue.PosterID = userid
  404. } else {
  405. gpr.Issue.PosterID = g.doer.ID
  406. gpr.Issue.OriginalAuthor = pr.PosterName
  407. gpr.Issue.OriginalAuthorID = pr.PosterID
  408. }
  409. gprs = append(gprs, gpr)
  410. }
  411. if err := models.InsertPullRequests(gprs...); err != nil {
  412. return err
  413. }
  414. for _, pr := range gprs {
  415. g.issues.Store(pr.Issue.Index, pr.Issue.ID)
  416. }
  417. return nil
  418. }
  419. func (g *GiteaLocalUploader) newPullRequest(pr *base.PullRequest) (*models.PullRequest, error) {
  420. var labels []*models.Label
  421. for _, label := range pr.Labels {
  422. lb, ok := g.labels.Load(label.Name)
  423. if ok {
  424. labels = append(labels, lb.(*models.Label))
  425. }
  426. }
  427. var milestoneID int64
  428. if pr.Milestone != "" {
  429. milestone, ok := g.milestones.Load(pr.Milestone)
  430. if ok {
  431. milestoneID = milestone.(int64)
  432. }
  433. }
  434. // download patch file
  435. err := func() error {
  436. resp, err := http.Get(pr.PatchURL)
  437. if err != nil {
  438. return err
  439. }
  440. defer resp.Body.Close()
  441. pullDir := filepath.Join(g.repo.RepoPath(), "pulls")
  442. if err = os.MkdirAll(pullDir, os.ModePerm); err != nil {
  443. return err
  444. }
  445. f, err := os.Create(filepath.Join(pullDir, fmt.Sprintf("%d.patch", pr.Number)))
  446. if err != nil {
  447. return err
  448. }
  449. defer f.Close()
  450. _, err = io.Copy(f, resp.Body)
  451. return err
  452. }()
  453. if err != nil {
  454. return nil, err
  455. }
  456. // set head information
  457. pullHead := filepath.Join(g.repo.RepoPath(), "refs", "pull", fmt.Sprintf("%d", pr.Number))
  458. if err := os.MkdirAll(pullHead, os.ModePerm); err != nil {
  459. return nil, err
  460. }
  461. p, err := os.Create(filepath.Join(pullHead, "head"))
  462. if err != nil {
  463. return nil, err
  464. }
  465. _, err = p.WriteString(pr.Head.SHA)
  466. p.Close()
  467. if err != nil {
  468. return nil, err
  469. }
  470. var head = "unknown repository"
  471. if pr.IsForkPullRequest() && pr.State != "closed" {
  472. if pr.Head.OwnerName != "" {
  473. remote := pr.Head.OwnerName
  474. _, ok := g.prHeadCache[remote]
  475. if !ok {
  476. // git remote add
  477. err := g.gitRepo.AddRemote(remote, pr.Head.CloneURL, true)
  478. if err != nil {
  479. log.Error("AddRemote failed: %s", err)
  480. } else {
  481. g.prHeadCache[remote] = struct{}{}
  482. ok = true
  483. }
  484. }
  485. if ok {
  486. _, err = git.NewCommand("fetch", remote, pr.Head.Ref).RunInDir(g.repo.RepoPath())
  487. if err != nil {
  488. log.Error("Fetch branch from %s failed: %v", pr.Head.CloneURL, err)
  489. } else {
  490. headBranch := filepath.Join(g.repo.RepoPath(), "refs", "heads", pr.Head.OwnerName, pr.Head.Ref)
  491. if err := os.MkdirAll(filepath.Dir(headBranch), os.ModePerm); err != nil {
  492. return nil, err
  493. }
  494. b, err := os.Create(headBranch)
  495. if err != nil {
  496. return nil, err
  497. }
  498. _, err = b.WriteString(pr.Head.SHA)
  499. b.Close()
  500. if err != nil {
  501. return nil, err
  502. }
  503. head = pr.Head.OwnerName + "/" + pr.Head.Ref
  504. }
  505. }
  506. }
  507. } else {
  508. head = pr.Head.Ref
  509. }
  510. var issue = models.Issue{
  511. RepoID: g.repo.ID,
  512. Repo: g.repo,
  513. Title: pr.Title,
  514. Index: pr.Number,
  515. Content: pr.Content,
  516. MilestoneID: milestoneID,
  517. IsPull: true,
  518. IsClosed: pr.State == "closed",
  519. IsLocked: pr.IsLocked,
  520. Labels: labels,
  521. CreatedUnix: timeutil.TimeStamp(pr.Created.Unix()),
  522. }
  523. userid, ok := g.userMap[pr.PosterID]
  524. if !ok {
  525. var err error
  526. userid, err = models.GetUserIDByExternalUserID("github", fmt.Sprintf("%v", pr.PosterID))
  527. if err != nil {
  528. log.Error("GetUserIDByExternalUserID: %v", err)
  529. }
  530. if userid > 0 {
  531. g.userMap[pr.PosterID] = userid
  532. }
  533. }
  534. if userid > 0 {
  535. issue.PosterID = userid
  536. } else {
  537. issue.PosterID = g.doer.ID
  538. issue.OriginalAuthor = pr.PosterName
  539. issue.OriginalAuthorID = pr.PosterID
  540. }
  541. var pullRequest = models.PullRequest{
  542. HeadRepoID: g.repo.ID,
  543. HeadBranch: head,
  544. BaseRepoID: g.repo.ID,
  545. BaseBranch: pr.Base.Ref,
  546. MergeBase: pr.Base.SHA,
  547. Index: pr.Number,
  548. HasMerged: pr.Merged,
  549. Issue: &issue,
  550. }
  551. if pullRequest.Issue.IsClosed && pr.Closed != nil {
  552. pullRequest.Issue.ClosedUnix = timeutil.TimeStamp(pr.Closed.Unix())
  553. }
  554. if pullRequest.HasMerged && pr.MergedTime != nil {
  555. pullRequest.MergedUnix = timeutil.TimeStamp(pr.MergedTime.Unix())
  556. pullRequest.MergedCommitID = pr.MergeCommitSHA
  557. pullRequest.MergerID = g.doer.ID
  558. }
  559. // TODO: reactions
  560. // TODO: assignees
  561. return &pullRequest, nil
  562. }
  563. // Rollback when migrating failed, this will rollback all the changes.
  564. func (g *GiteaLocalUploader) Rollback() error {
  565. if g.repo != nil && g.repo.ID > 0 {
  566. if err := models.DeleteRepository(g.doer, g.repo.OwnerID, g.repo.ID); err != nil {
  567. return err
  568. }
  569. }
  570. return nil
  571. }