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.

onedev.go 15KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634
  1. // Copyright 2021 The Gitea Authors. All rights reserved.
  2. // SPDX-License-Identifier: MIT
  3. package migrations
  4. import (
  5. "context"
  6. "fmt"
  7. "net/http"
  8. "net/url"
  9. "strconv"
  10. "strings"
  11. "time"
  12. "code.gitea.io/gitea/modules/json"
  13. "code.gitea.io/gitea/modules/log"
  14. base "code.gitea.io/gitea/modules/migration"
  15. "code.gitea.io/gitea/modules/structs"
  16. )
  17. var (
  18. _ base.Downloader = &OneDevDownloader{}
  19. _ base.DownloaderFactory = &OneDevDownloaderFactory{}
  20. )
  21. func init() {
  22. RegisterDownloaderFactory(&OneDevDownloaderFactory{})
  23. }
  24. // OneDevDownloaderFactory defines a downloader factory
  25. type OneDevDownloaderFactory struct{}
  26. // New returns a downloader related to this factory according MigrateOptions
  27. func (f *OneDevDownloaderFactory) New(ctx context.Context, opts base.MigrateOptions) (base.Downloader, error) {
  28. u, err := url.Parse(opts.CloneAddr)
  29. if err != nil {
  30. return nil, err
  31. }
  32. var repoName string
  33. fields := strings.Split(strings.Trim(u.Path, "/"), "/")
  34. if len(fields) == 2 && fields[0] == "projects" {
  35. repoName = fields[1]
  36. } else if len(fields) == 1 {
  37. repoName = fields[0]
  38. } else {
  39. return nil, fmt.Errorf("invalid path: %s", u.Path)
  40. }
  41. u.Path = ""
  42. u.Fragment = ""
  43. log.Trace("Create onedev downloader. BaseURL: %v RepoName: %s", u, repoName)
  44. return NewOneDevDownloader(ctx, u, opts.AuthUsername, opts.AuthPassword, repoName), nil
  45. }
  46. // GitServiceType returns the type of git service
  47. func (f *OneDevDownloaderFactory) GitServiceType() structs.GitServiceType {
  48. return structs.OneDevService
  49. }
  50. type onedevUser struct {
  51. ID int64 `json:"id"`
  52. Name string `json:"name"`
  53. Email string `json:"email"`
  54. }
  55. // OneDevDownloader implements a Downloader interface to get repository information
  56. // from OneDev
  57. type OneDevDownloader struct {
  58. base.NullDownloader
  59. ctx context.Context
  60. client *http.Client
  61. baseURL *url.URL
  62. repoName string
  63. repoID int64
  64. maxIssueIndex int64
  65. userMap map[int64]*onedevUser
  66. milestoneMap map[int64]string
  67. }
  68. // SetContext set context
  69. func (d *OneDevDownloader) SetContext(ctx context.Context) {
  70. d.ctx = ctx
  71. }
  72. // NewOneDevDownloader creates a new downloader
  73. func NewOneDevDownloader(ctx context.Context, baseURL *url.URL, username, password, repoName string) *OneDevDownloader {
  74. downloader := &OneDevDownloader{
  75. ctx: ctx,
  76. baseURL: baseURL,
  77. repoName: repoName,
  78. client: &http.Client{
  79. Transport: &http.Transport{
  80. Proxy: func(req *http.Request) (*url.URL, error) {
  81. if len(username) > 0 && len(password) > 0 {
  82. req.SetBasicAuth(username, password)
  83. }
  84. return nil, nil
  85. },
  86. },
  87. },
  88. userMap: make(map[int64]*onedevUser),
  89. milestoneMap: make(map[int64]string),
  90. }
  91. return downloader
  92. }
  93. // String implements Stringer
  94. func (d *OneDevDownloader) String() string {
  95. return fmt.Sprintf("migration from oneDev server %s [%d]/%s", d.baseURL, d.repoID, d.repoName)
  96. }
  97. func (d *OneDevDownloader) LogString() string {
  98. if d == nil {
  99. return "<OneDevDownloader nil>"
  100. }
  101. return fmt.Sprintf("<OneDevDownloader %s [%d]/%s>", d.baseURL, d.repoID, d.repoName)
  102. }
  103. func (d *OneDevDownloader) callAPI(endpoint string, parameter map[string]string, result any) error {
  104. u, err := d.baseURL.Parse(endpoint)
  105. if err != nil {
  106. return err
  107. }
  108. if parameter != nil {
  109. query := u.Query()
  110. for k, v := range parameter {
  111. query.Set(k, v)
  112. }
  113. u.RawQuery = query.Encode()
  114. }
  115. req, err := http.NewRequestWithContext(d.ctx, "GET", u.String(), nil)
  116. if err != nil {
  117. return err
  118. }
  119. resp, err := d.client.Do(req)
  120. if err != nil {
  121. return err
  122. }
  123. defer resp.Body.Close()
  124. decoder := json.NewDecoder(resp.Body)
  125. return decoder.Decode(&result)
  126. }
  127. // GetRepoInfo returns repository information
  128. func (d *OneDevDownloader) GetRepoInfo() (*base.Repository, error) {
  129. info := make([]struct {
  130. ID int64 `json:"id"`
  131. Name string `json:"name"`
  132. Description string `json:"description"`
  133. }, 0, 1)
  134. err := d.callAPI(
  135. "/api/projects",
  136. map[string]string{
  137. "query": `"Name" is "` + d.repoName + `"`,
  138. "offset": "0",
  139. "count": "1",
  140. },
  141. &info,
  142. )
  143. if err != nil {
  144. return nil, err
  145. }
  146. if len(info) != 1 {
  147. return nil, fmt.Errorf("Project %s not found", d.repoName)
  148. }
  149. d.repoID = info[0].ID
  150. cloneURL, err := d.baseURL.Parse(info[0].Name)
  151. if err != nil {
  152. return nil, err
  153. }
  154. originalURL, err := d.baseURL.Parse("/projects/" + info[0].Name)
  155. if err != nil {
  156. return nil, err
  157. }
  158. return &base.Repository{
  159. Name: info[0].Name,
  160. Description: info[0].Description,
  161. CloneURL: cloneURL.String(),
  162. OriginalURL: originalURL.String(),
  163. }, nil
  164. }
  165. // GetMilestones returns milestones
  166. func (d *OneDevDownloader) GetMilestones() ([]*base.Milestone, error) {
  167. rawMilestones := make([]struct {
  168. ID int64 `json:"id"`
  169. Name string `json:"name"`
  170. Description string `json:"description"`
  171. DueDate *time.Time `json:"dueDate"`
  172. Closed bool `json:"closed"`
  173. }, 0, 100)
  174. endpoint := fmt.Sprintf("/api/projects/%d/milestones", d.repoID)
  175. milestones := make([]*base.Milestone, 0, 100)
  176. offset := 0
  177. for {
  178. err := d.callAPI(
  179. endpoint,
  180. map[string]string{
  181. "offset": strconv.Itoa(offset),
  182. "count": "100",
  183. },
  184. &rawMilestones,
  185. )
  186. if err != nil {
  187. return nil, err
  188. }
  189. if len(rawMilestones) == 0 {
  190. break
  191. }
  192. offset += 100
  193. for _, milestone := range rawMilestones {
  194. d.milestoneMap[milestone.ID] = milestone.Name
  195. closed := milestone.DueDate
  196. if !milestone.Closed {
  197. closed = nil
  198. }
  199. milestones = append(milestones, &base.Milestone{
  200. Title: milestone.Name,
  201. Description: milestone.Description,
  202. Deadline: milestone.DueDate,
  203. Closed: closed,
  204. })
  205. }
  206. }
  207. return milestones, nil
  208. }
  209. // GetLabels returns labels
  210. func (d *OneDevDownloader) GetLabels() ([]*base.Label, error) {
  211. return []*base.Label{
  212. {
  213. Name: "Bug",
  214. Color: "f64e60",
  215. },
  216. {
  217. Name: "Build Failure",
  218. Color: "f64e60",
  219. },
  220. {
  221. Name: "Discussion",
  222. Color: "8950fc",
  223. },
  224. {
  225. Name: "Improvement",
  226. Color: "1bc5bd",
  227. },
  228. {
  229. Name: "New Feature",
  230. Color: "1bc5bd",
  231. },
  232. {
  233. Name: "Support Request",
  234. Color: "8950fc",
  235. },
  236. }, nil
  237. }
  238. type onedevIssueContext struct {
  239. IsPullRequest bool
  240. }
  241. // GetIssues returns issues
  242. func (d *OneDevDownloader) GetIssues(page, perPage int) ([]*base.Issue, bool, error) {
  243. rawIssues := make([]struct {
  244. ID int64 `json:"id"`
  245. Number int64 `json:"number"`
  246. State string `json:"state"`
  247. Title string `json:"title"`
  248. Description string `json:"description"`
  249. SubmitterID int64 `json:"submitterId"`
  250. SubmitDate time.Time `json:"submitDate"`
  251. }, 0, perPage)
  252. err := d.callAPI(
  253. "/api/issues",
  254. map[string]string{
  255. "query": `"Project" is "` + d.repoName + `"`,
  256. "offset": strconv.Itoa((page - 1) * perPage),
  257. "count": strconv.Itoa(perPage),
  258. },
  259. &rawIssues,
  260. )
  261. if err != nil {
  262. return nil, false, err
  263. }
  264. issues := make([]*base.Issue, 0, len(rawIssues))
  265. for _, issue := range rawIssues {
  266. fields := make([]struct {
  267. Name string `json:"name"`
  268. Value string `json:"value"`
  269. }, 0, 10)
  270. err := d.callAPI(
  271. fmt.Sprintf("/api/issues/%d/fields", issue.ID),
  272. nil,
  273. &fields,
  274. )
  275. if err != nil {
  276. return nil, false, err
  277. }
  278. var label *base.Label
  279. for _, field := range fields {
  280. if field.Name == "Type" {
  281. label = &base.Label{Name: field.Value}
  282. break
  283. }
  284. }
  285. milestones := make([]struct {
  286. ID int64 `json:"id"`
  287. Name string `json:"name"`
  288. }, 0, 10)
  289. err = d.callAPI(
  290. fmt.Sprintf("/api/issues/%d/milestones", issue.ID),
  291. nil,
  292. &milestones,
  293. )
  294. if err != nil {
  295. return nil, false, err
  296. }
  297. milestoneID := int64(0)
  298. if len(milestones) > 0 {
  299. milestoneID = milestones[0].ID
  300. }
  301. state := strings.ToLower(issue.State)
  302. if state == "released" {
  303. state = "closed"
  304. }
  305. poster := d.tryGetUser(issue.SubmitterID)
  306. issues = append(issues, &base.Issue{
  307. Title: issue.Title,
  308. Number: issue.Number,
  309. PosterName: poster.Name,
  310. PosterEmail: poster.Email,
  311. Content: issue.Description,
  312. Milestone: d.milestoneMap[milestoneID],
  313. State: state,
  314. Created: issue.SubmitDate,
  315. Updated: issue.SubmitDate,
  316. Labels: []*base.Label{label},
  317. ForeignIndex: issue.ID,
  318. Context: onedevIssueContext{IsPullRequest: false},
  319. })
  320. if d.maxIssueIndex < issue.Number {
  321. d.maxIssueIndex = issue.Number
  322. }
  323. }
  324. return issues, len(issues) == 0, nil
  325. }
  326. // GetComments returns comments
  327. func (d *OneDevDownloader) GetComments(commentable base.Commentable) ([]*base.Comment, bool, error) {
  328. context, ok := commentable.GetContext().(onedevIssueContext)
  329. if !ok {
  330. return nil, false, fmt.Errorf("unexpected context: %+v", commentable.GetContext())
  331. }
  332. rawComments := make([]struct {
  333. ID int64 `json:"id"`
  334. Date time.Time `json:"date"`
  335. UserID int64 `json:"userId"`
  336. Content string `json:"content"`
  337. }, 0, 100)
  338. var endpoint string
  339. if context.IsPullRequest {
  340. endpoint = fmt.Sprintf("/api/pull-requests/%d/comments", commentable.GetForeignIndex())
  341. } else {
  342. endpoint = fmt.Sprintf("/api/issues/%d/comments", commentable.GetForeignIndex())
  343. }
  344. err := d.callAPI(
  345. endpoint,
  346. nil,
  347. &rawComments,
  348. )
  349. if err != nil {
  350. return nil, false, err
  351. }
  352. rawChanges := make([]struct {
  353. Date time.Time `json:"date"`
  354. UserID int64 `json:"userId"`
  355. Data map[string]any `json:"data"`
  356. }, 0, 100)
  357. if context.IsPullRequest {
  358. endpoint = fmt.Sprintf("/api/pull-requests/%d/changes", commentable.GetForeignIndex())
  359. } else {
  360. endpoint = fmt.Sprintf("/api/issues/%d/changes", commentable.GetForeignIndex())
  361. }
  362. err = d.callAPI(
  363. endpoint,
  364. nil,
  365. &rawChanges,
  366. )
  367. if err != nil {
  368. return nil, false, err
  369. }
  370. comments := make([]*base.Comment, 0, len(rawComments)+len(rawChanges))
  371. for _, comment := range rawComments {
  372. if len(comment.Content) == 0 {
  373. continue
  374. }
  375. poster := d.tryGetUser(comment.UserID)
  376. comments = append(comments, &base.Comment{
  377. IssueIndex: commentable.GetLocalIndex(),
  378. Index: comment.ID,
  379. PosterID: poster.ID,
  380. PosterName: poster.Name,
  381. PosterEmail: poster.Email,
  382. Content: comment.Content,
  383. Created: comment.Date,
  384. Updated: comment.Date,
  385. })
  386. }
  387. for _, change := range rawChanges {
  388. contentV, ok := change.Data["content"]
  389. if !ok {
  390. contentV, ok = change.Data["comment"]
  391. if !ok {
  392. continue
  393. }
  394. }
  395. content, ok := contentV.(string)
  396. if !ok || len(content) == 0 {
  397. continue
  398. }
  399. poster := d.tryGetUser(change.UserID)
  400. comments = append(comments, &base.Comment{
  401. IssueIndex: commentable.GetLocalIndex(),
  402. PosterID: poster.ID,
  403. PosterName: poster.Name,
  404. PosterEmail: poster.Email,
  405. Content: content,
  406. Created: change.Date,
  407. Updated: change.Date,
  408. })
  409. }
  410. return comments, true, nil
  411. }
  412. // GetPullRequests returns pull requests
  413. func (d *OneDevDownloader) GetPullRequests(page, perPage int) ([]*base.PullRequest, bool, error) {
  414. rawPullRequests := make([]struct {
  415. ID int64 `json:"id"`
  416. Number int64 `json:"number"`
  417. Title string `json:"title"`
  418. SubmitterID int64 `json:"submitterId"`
  419. SubmitDate time.Time `json:"submitDate"`
  420. Description string `json:"description"`
  421. TargetBranch string `json:"targetBranch"`
  422. SourceBranch string `json:"sourceBranch"`
  423. BaseCommitHash string `json:"baseCommitHash"`
  424. CloseInfo *struct {
  425. Date *time.Time `json:"date"`
  426. Status string `json:"status"`
  427. }
  428. }, 0, perPage)
  429. err := d.callAPI(
  430. "/api/pull-requests",
  431. map[string]string{
  432. "query": `"Target Project" is "` + d.repoName + `"`,
  433. "offset": strconv.Itoa((page - 1) * perPage),
  434. "count": strconv.Itoa(perPage),
  435. },
  436. &rawPullRequests,
  437. )
  438. if err != nil {
  439. return nil, false, err
  440. }
  441. pullRequests := make([]*base.PullRequest, 0, len(rawPullRequests))
  442. for _, pr := range rawPullRequests {
  443. var mergePreview struct {
  444. TargetHeadCommitHash string `json:"targetHeadCommitHash"`
  445. HeadCommitHash string `json:"headCommitHash"`
  446. MergeStrategy string `json:"mergeStrategy"`
  447. MergeCommitHash string `json:"mergeCommitHash"`
  448. }
  449. err := d.callAPI(
  450. fmt.Sprintf("/api/pull-requests/%d/merge-preview", pr.ID),
  451. nil,
  452. &mergePreview,
  453. )
  454. if err != nil {
  455. return nil, false, err
  456. }
  457. state := "open"
  458. merged := false
  459. var closeTime *time.Time
  460. var mergedTime *time.Time
  461. if pr.CloseInfo != nil {
  462. state = "closed"
  463. closeTime = pr.CloseInfo.Date
  464. if pr.CloseInfo.Status == "MERGED" { // "DISCARDED"
  465. merged = true
  466. mergedTime = pr.CloseInfo.Date
  467. }
  468. }
  469. poster := d.tryGetUser(pr.SubmitterID)
  470. number := pr.Number + d.maxIssueIndex
  471. pullRequests = append(pullRequests, &base.PullRequest{
  472. Title: pr.Title,
  473. Number: number,
  474. PosterName: poster.Name,
  475. PosterID: poster.ID,
  476. Content: pr.Description,
  477. State: state,
  478. Created: pr.SubmitDate,
  479. Updated: pr.SubmitDate,
  480. Closed: closeTime,
  481. Merged: merged,
  482. MergedTime: mergedTime,
  483. Head: base.PullRequestBranch{
  484. Ref: pr.SourceBranch,
  485. SHA: mergePreview.HeadCommitHash,
  486. RepoName: d.repoName,
  487. },
  488. Base: base.PullRequestBranch{
  489. Ref: pr.TargetBranch,
  490. SHA: mergePreview.TargetHeadCommitHash,
  491. RepoName: d.repoName,
  492. },
  493. ForeignIndex: pr.ID,
  494. Context: onedevIssueContext{IsPullRequest: true},
  495. })
  496. // SECURITY: Ensure that the PR is safe
  497. _ = CheckAndEnsureSafePR(pullRequests[len(pullRequests)-1], d.baseURL.String(), d)
  498. }
  499. return pullRequests, len(pullRequests) == 0, nil
  500. }
  501. // GetReviews returns pull requests reviews
  502. func (d *OneDevDownloader) GetReviews(reviewable base.Reviewable) ([]*base.Review, error) {
  503. rawReviews := make([]struct {
  504. ID int64 `json:"id"`
  505. UserID int64 `json:"userId"`
  506. Result *struct {
  507. Commit string `json:"commit"`
  508. Approved bool `json:"approved"`
  509. Comment string `json:"comment"`
  510. }
  511. }, 0, 100)
  512. err := d.callAPI(
  513. fmt.Sprintf("/api/pull-requests/%d/reviews", reviewable.GetForeignIndex()),
  514. nil,
  515. &rawReviews,
  516. )
  517. if err != nil {
  518. return nil, err
  519. }
  520. reviews := make([]*base.Review, 0, len(rawReviews))
  521. for _, review := range rawReviews {
  522. state := base.ReviewStatePending
  523. content := ""
  524. if review.Result != nil {
  525. if len(review.Result.Comment) > 0 {
  526. state = base.ReviewStateCommented
  527. content = review.Result.Comment
  528. }
  529. if review.Result.Approved {
  530. state = base.ReviewStateApproved
  531. }
  532. }
  533. poster := d.tryGetUser(review.UserID)
  534. reviews = append(reviews, &base.Review{
  535. IssueIndex: reviewable.GetLocalIndex(),
  536. ReviewerID: poster.ID,
  537. ReviewerName: poster.Name,
  538. Content: content,
  539. State: state,
  540. })
  541. }
  542. return reviews, nil
  543. }
  544. // GetTopics return repository topics
  545. func (d *OneDevDownloader) GetTopics() ([]string, error) {
  546. return []string{}, nil
  547. }
  548. func (d *OneDevDownloader) tryGetUser(userID int64) *onedevUser {
  549. user, ok := d.userMap[userID]
  550. if !ok {
  551. err := d.callAPI(
  552. fmt.Sprintf("/api/users/%d", userID),
  553. nil,
  554. &user,
  555. )
  556. if err != nil {
  557. user = &onedevUser{
  558. Name: fmt.Sprintf("User %d", userID),
  559. }
  560. }
  561. d.userMap[userID] = user
  562. }
  563. return user
  564. }