2019-05-07 04:12:51 +03:00
// Copyright 2019 The Gitea Authors. All rights reserved.
// Copyright 2018 Jonas Franz. All rights reserved.
// Use of this source code is governed by a MIT-style
// license that can be found in the LICENSE file.
package migrations
import (
"context"
"fmt"
"net/http"
"net/url"
"strings"
2019-12-17 07:16:54 +03:00
"time"
2019-05-07 04:12:51 +03:00
"code.gitea.io/gitea/modules/log"
"code.gitea.io/gitea/modules/migrations/base"
2019-10-14 09:10:42 +03:00
"code.gitea.io/gitea/modules/structs"
2020-05-08 18:46:05 +03:00
"code.gitea.io/gitea/modules/util"
2019-05-07 04:12:51 +03:00
2020-07-31 17:22:34 +03:00
"github.com/google/go-github/v32/github"
2019-05-07 04:12:51 +03:00
"golang.org/x/oauth2"
)
var (
_ base . Downloader = & GithubDownloaderV3 { }
_ base . DownloaderFactory = & GithubDownloaderV3Factory { }
2020-01-16 18:15:44 +03:00
// GithubLimitRateRemaining limit to wait for new rate to apply
GithubLimitRateRemaining = 0
2019-05-07 04:12:51 +03:00
)
func init ( ) {
RegisterDownloaderFactory ( & GithubDownloaderV3Factory { } )
}
// GithubDownloaderV3Factory defines a github downloader v3 factory
type GithubDownloaderV3Factory struct {
}
// Match returns ture if the migration remote URL matched this downloader factory
func ( f * GithubDownloaderV3Factory ) Match ( opts base . MigrateOptions ) ( bool , error ) {
2019-10-13 16:23:14 +03:00
u , err := url . Parse ( opts . CloneAddr )
2019-05-07 04:12:51 +03:00
if err != nil {
return false , err
}
2019-10-14 09:10:42 +03:00
return strings . EqualFold ( u . Host , "github.com" ) && opts . AuthUsername != "" , nil
2019-05-07 04:12:51 +03:00
}
// New returns a Downloader related to this factory according MigrateOptions
func ( f * GithubDownloaderV3Factory ) New ( opts base . MigrateOptions ) ( base . Downloader , error ) {
2019-10-13 16:23:14 +03:00
u , err := url . Parse ( opts . CloneAddr )
2019-05-07 04:12:51 +03:00
if err != nil {
return nil , err
}
fields := strings . Split ( u . Path , "/" )
oldOwner := fields [ 1 ]
oldName := strings . TrimSuffix ( fields [ 2 ] , ".git" )
log . Trace ( "Create github downloader: %s/%s" , oldOwner , oldName )
return NewGithubDownloaderV3 ( opts . AuthUsername , opts . AuthPassword , oldOwner , oldName ) , nil
}
2019-10-14 09:10:42 +03:00
// GitServiceType returns the type of git service
func ( f * GithubDownloaderV3Factory ) GitServiceType ( ) structs . GitServiceType {
return structs . GithubService
}
2019-05-07 04:12:51 +03:00
// GithubDownloaderV3 implements a Downloader interface to get repository informations
// from github via APIv3
type GithubDownloaderV3 struct {
ctx context . Context
client * github . Client
repoOwner string
repoName string
userName string
password string
2019-12-17 07:16:54 +03:00
rate * github . Rate
2019-05-07 04:12:51 +03:00
}
// NewGithubDownloaderV3 creates a github Downloader via github v3 API
func NewGithubDownloaderV3 ( userName , password , repoOwner , repoName string ) * GithubDownloaderV3 {
var downloader = GithubDownloaderV3 {
userName : userName ,
password : password ,
ctx : context . Background ( ) ,
repoOwner : repoOwner ,
repoName : repoName ,
}
var client * http . Client
if userName != "" {
if password == "" {
ts := oauth2 . StaticTokenSource (
& oauth2 . Token { AccessToken : userName } ,
)
client = oauth2 . NewClient ( downloader . ctx , ts )
} else {
client = & http . Client {
Transport : & http . Transport {
Proxy : func ( req * http . Request ) ( * url . URL , error ) {
req . SetBasicAuth ( userName , password )
return nil , nil
} ,
} ,
}
}
}
downloader . client = github . NewClient ( client )
return & downloader
}
2019-12-17 07:16:54 +03:00
// SetContext set context
func ( g * GithubDownloaderV3 ) SetContext ( ctx context . Context ) {
g . ctx = ctx
}
func ( g * GithubDownloaderV3 ) sleep ( ) {
2020-01-16 18:15:44 +03:00
for g . rate != nil && g . rate . Remaining <= GithubLimitRateRemaining {
2019-12-17 07:16:54 +03:00
timer := time . NewTimer ( time . Until ( g . rate . Reset . Time ) )
select {
case <- g . ctx . Done ( ) :
2020-05-08 18:46:05 +03:00
util . StopTimer ( timer )
2019-12-17 07:16:54 +03:00
return
case <- timer . C :
}
2020-01-16 18:15:44 +03:00
err := g . RefreshRate ( )
2019-12-17 07:16:54 +03:00
if err != nil {
log . Error ( "g.client.RateLimits: %s" , err )
}
2020-01-16 18:15:44 +03:00
}
}
2019-12-17 07:16:54 +03:00
2020-01-16 18:15:44 +03:00
// RefreshRate update the current rate (doesn't count in rate limit)
func ( g * GithubDownloaderV3 ) RefreshRate ( ) error {
rates , _ , err := g . client . RateLimits ( g . ctx )
if err != nil {
return err
2019-12-17 07:16:54 +03:00
}
2020-01-16 18:15:44 +03:00
g . rate = rates . GetCore ( )
return nil
2019-12-17 07:16:54 +03:00
}
2019-05-07 04:12:51 +03:00
// GetRepoInfo returns a repository information
func ( g * GithubDownloaderV3 ) GetRepoInfo ( ) ( * base . Repository , error ) {
2019-12-17 07:16:54 +03:00
g . sleep ( )
gr , resp , err := g . client . Repositories . Get ( g . ctx , g . repoOwner , g . repoName )
2019-05-07 04:12:51 +03:00
if err != nil {
return nil , err
}
2019-12-17 07:16:54 +03:00
g . rate = & resp . Rate
2019-05-07 04:12:51 +03:00
// convert github repo to stand Repo
return & base . Repository {
Owner : g . repoOwner ,
Name : gr . GetName ( ) ,
IsPrivate : * gr . Private ,
Description : gr . GetDescription ( ) ,
2019-07-08 05:14:12 +03:00
OriginalURL : gr . GetHTMLURL ( ) ,
2019-05-07 04:12:51 +03:00
CloneURL : gr . GetCloneURL ( ) ,
} , nil
}
2019-08-14 09:16:12 +03:00
// GetTopics return github topics
func ( g * GithubDownloaderV3 ) GetTopics ( ) ( [ ] string , error ) {
2019-12-17 07:16:54 +03:00
g . sleep ( )
r , resp , err := g . client . Repositories . Get ( g . ctx , g . repoOwner , g . repoName )
if err != nil {
return nil , err
}
g . rate = & resp . Rate
return r . Topics , nil
2019-08-14 09:16:12 +03:00
}
2019-05-07 04:12:51 +03:00
// GetMilestones returns milestones
func ( g * GithubDownloaderV3 ) GetMilestones ( ) ( [ ] * base . Milestone , error ) {
var perPage = 100
var milestones = make ( [ ] * base . Milestone , 0 , perPage )
for i := 1 ; ; i ++ {
2019-12-17 07:16:54 +03:00
g . sleep ( )
ms , resp , err := g . client . Issues . ListMilestones ( g . ctx , g . repoOwner , g . repoName ,
2019-05-07 04:12:51 +03:00
& github . MilestoneListOptions {
State : "all" ,
ListOptions : github . ListOptions {
Page : i ,
PerPage : perPage ,
} } )
if err != nil {
return nil , err
}
2019-12-17 07:16:54 +03:00
g . rate = & resp . Rate
2019-05-07 04:12:51 +03:00
for _ , m := range ms {
var desc string
if m . Description != nil {
desc = * m . Description
}
var state = "open"
if m . State != nil {
state = * m . State
}
milestones = append ( milestones , & base . Milestone {
Title : * m . Title ,
Description : desc ,
Deadline : m . DueOn ,
State : state ,
Created : * m . CreatedAt ,
Updated : m . UpdatedAt ,
Closed : m . ClosedAt ,
} )
}
if len ( ms ) < perPage {
break
}
}
return milestones , nil
}
func convertGithubLabel ( label * github . Label ) * base . Label {
var desc string
if label . Description != nil {
desc = * label . Description
}
return & base . Label {
Name : * label . Name ,
Color : * label . Color ,
Description : desc ,
}
}
// GetLabels returns labels
func ( g * GithubDownloaderV3 ) GetLabels ( ) ( [ ] * base . Label , error ) {
var perPage = 100
var labels = make ( [ ] * base . Label , 0 , perPage )
for i := 1 ; ; i ++ {
2019-12-17 07:16:54 +03:00
g . sleep ( )
ls , resp , err := g . client . Issues . ListLabels ( g . ctx , g . repoOwner , g . repoName ,
2019-05-07 04:12:51 +03:00
& github . ListOptions {
Page : i ,
PerPage : perPage ,
} )
if err != nil {
return nil , err
}
2019-12-17 07:16:54 +03:00
g . rate = & resp . Rate
2019-05-07 04:12:51 +03:00
for _ , label := range ls {
labels = append ( labels , convertGithubLabel ( label ) )
}
if len ( ls ) < perPage {
break
}
}
return labels , nil
}
func ( g * GithubDownloaderV3 ) convertGithubRelease ( rel * github . RepositoryRelease ) * base . Release {
var (
name string
desc string
)
if rel . Body != nil {
desc = * rel . Body
}
if rel . Name != nil {
name = * rel . Name
}
2019-10-05 14:09:27 +03:00
var email string
if rel . Author . Email != nil {
email = * rel . Author . Email
}
2019-05-07 04:12:51 +03:00
r := & base . Release {
TagName : * rel . TagName ,
TargetCommitish : * rel . TargetCommitish ,
Name : name ,
Body : desc ,
Draft : * rel . Draft ,
Prerelease : * rel . Prerelease ,
Created : rel . CreatedAt . Time ,
2019-10-05 14:09:27 +03:00
PublisherID : * rel . Author . ID ,
PublisherName : * rel . Author . Login ,
PublisherEmail : email ,
2019-05-07 04:12:51 +03:00
Published : rel . PublishedAt . Time ,
}
for _ , asset := range rel . Assets {
u , _ := url . Parse ( * asset . BrowserDownloadURL )
u . User = url . UserPassword ( g . userName , g . password )
r . Assets = append ( r . Assets , base . ReleaseAsset {
URL : u . String ( ) ,
Name : * asset . Name ,
ContentType : asset . ContentType ,
Size : asset . Size ,
DownloadCount : asset . DownloadCount ,
Created : asset . CreatedAt . Time ,
Updated : asset . UpdatedAt . Time ,
} )
}
return r
}
// GetReleases returns releases
func ( g * GithubDownloaderV3 ) GetReleases ( ) ( [ ] * base . Release , error ) {
var perPage = 100
var releases = make ( [ ] * base . Release , 0 , perPage )
for i := 1 ; ; i ++ {
2019-12-17 07:16:54 +03:00
g . sleep ( )
ls , resp , err := g . client . Repositories . ListReleases ( g . ctx , g . repoOwner , g . repoName ,
2019-05-07 04:12:51 +03:00
& github . ListOptions {
Page : i ,
PerPage : perPage ,
} )
if err != nil {
return nil , err
}
2019-12-17 07:16:54 +03:00
g . rate = & resp . Rate
2019-05-07 04:12:51 +03:00
for _ , release := range ls {
releases = append ( releases , g . convertGithubRelease ( release ) )
}
if len ( ls ) < perPage {
break
}
}
return releases , nil
}
// GetIssues returns issues according start and limit
2019-05-30 23:26:57 +03:00
func ( g * GithubDownloaderV3 ) GetIssues ( page , perPage int ) ( [ ] * base . Issue , bool , error ) {
2019-05-07 04:12:51 +03:00
opt := & github . IssueListByRepoOptions {
Sort : "created" ,
Direction : "asc" ,
State : "all" ,
ListOptions : github . ListOptions {
PerPage : perPage ,
2019-05-30 23:26:57 +03:00
Page : page ,
2019-05-07 04:12:51 +03:00
} ,
}
2019-05-30 23:26:57 +03:00
var allIssues = make ( [ ] * base . Issue , 0 , perPage )
2019-12-17 07:16:54 +03:00
g . sleep ( )
issues , resp , err := g . client . Issues . ListByRepo ( g . ctx , g . repoOwner , g . repoName , opt )
2019-05-30 23:26:57 +03:00
if err != nil {
return nil , false , fmt . Errorf ( "error while listing repos: %v" , err )
}
2019-12-17 07:16:54 +03:00
g . rate = & resp . Rate
2019-05-30 23:26:57 +03:00
for _ , issue := range issues {
if issue . IsPullRequest ( ) {
continue
2019-05-07 04:12:51 +03:00
}
2019-05-30 23:26:57 +03:00
var body string
if issue . Body != nil {
body = * issue . Body
2019-05-07 04:12:51 +03:00
}
2019-05-30 23:26:57 +03:00
var milestone string
if issue . Milestone != nil {
milestone = * issue . Milestone . Title
}
var labels = make ( [ ] * base . Label , 0 , len ( issue . Labels ) )
for _ , l := range issue . Labels {
2020-07-31 17:22:34 +03:00
labels = append ( labels , convertGithubLabel ( l ) )
2019-05-30 23:26:57 +03:00
}
var email string
if issue . User . Email != nil {
email = * issue . User . Email
}
2020-01-15 14:14:07 +03:00
// get reactions
var reactions [ ] * base . Reaction
for i := 1 ; ; i ++ {
g . sleep ( )
res , resp , err := g . client . Reactions . ListIssueReactions ( g . ctx , g . repoOwner , g . repoName , issue . GetNumber ( ) , & github . ListOptions {
Page : i ,
PerPage : perPage ,
} )
if err != nil {
return nil , false , err
}
g . rate = & resp . Rate
if len ( res ) == 0 {
break
}
for _ , reaction := range res {
reactions = append ( reactions , & base . Reaction {
UserID : reaction . User . GetID ( ) ,
UserName : reaction . User . GetLogin ( ) ,
Content : reaction . GetContent ( ) ,
} )
}
}
2019-05-30 23:26:57 +03:00
allIssues = append ( allIssues , & base . Issue {
Title : * issue . Title ,
Number : int64 ( * issue . Number ) ,
2019-07-08 05:14:12 +03:00
PosterID : * issue . User . ID ,
2019-05-30 23:26:57 +03:00
PosterName : * issue . User . Login ,
PosterEmail : email ,
Content : body ,
Milestone : milestone ,
State : * issue . State ,
Created : * issue . CreatedAt ,
2020-01-14 13:29:22 +03:00
Updated : * issue . UpdatedAt ,
2019-05-30 23:26:57 +03:00
Labels : labels ,
Reactions : reactions ,
Closed : issue . ClosedAt ,
IsLocked : * issue . Locked ,
} )
2019-05-07 04:12:51 +03:00
}
2019-05-30 23:26:57 +03:00
return allIssues , len ( issues ) < perPage , nil
2019-05-07 04:12:51 +03:00
}
// GetComments returns comments according issueNumber
func ( g * GithubDownloaderV3 ) GetComments ( issueNumber int64 ) ( [ ] * base . Comment , error ) {
2020-01-23 20:28:15 +03:00
var (
allComments = make ( [ ] * base . Comment , 0 , 100 )
created = "created"
asc = "asc"
)
2019-05-07 04:12:51 +03:00
opt := & github . IssueListCommentsOptions {
2020-07-31 17:22:34 +03:00
Sort : & created ,
Direction : & asc ,
2019-05-07 04:12:51 +03:00
ListOptions : github . ListOptions {
PerPage : 100 ,
} ,
}
for {
2019-12-17 07:16:54 +03:00
g . sleep ( )
2019-05-07 04:12:51 +03:00
comments , resp , err := g . client . Issues . ListComments ( g . ctx , g . repoOwner , g . repoName , int ( issueNumber ) , opt )
if err != nil {
return nil , fmt . Errorf ( "error while listing repos: %v" , err )
}
2019-12-17 07:16:54 +03:00
g . rate = & resp . Rate
2019-05-07 04:12:51 +03:00
for _ , comment := range comments {
var email string
if comment . User . Email != nil {
email = * comment . User . Email
}
2020-01-15 14:14:07 +03:00
// get reactions
var reactions [ ] * base . Reaction
for i := 1 ; ; i ++ {
g . sleep ( )
res , resp , err := g . client . Reactions . ListIssueCommentReactions ( g . ctx , g . repoOwner , g . repoName , comment . GetID ( ) , & github . ListOptions {
Page : i ,
PerPage : 100 ,
} )
if err != nil {
return nil , err
}
g . rate = & resp . Rate
if len ( res ) == 0 {
break
}
for _ , reaction := range res {
reactions = append ( reactions , & base . Reaction {
UserID : reaction . User . GetID ( ) ,
UserName : reaction . User . GetLogin ( ) ,
Content : reaction . GetContent ( ) ,
} )
}
2019-05-07 04:12:51 +03:00
}
allComments = append ( allComments , & base . Comment {
2019-06-29 16:38:22 +03:00
IssueIndex : issueNumber ,
2019-07-08 05:14:12 +03:00
PosterID : * comment . User . ID ,
2019-05-07 04:12:51 +03:00
PosterName : * comment . User . Login ,
PosterEmail : email ,
Content : * comment . Body ,
Created : * comment . CreatedAt ,
2020-01-14 13:29:22 +03:00
Updated : * comment . UpdatedAt ,
2019-05-07 04:12:51 +03:00
Reactions : reactions ,
} )
}
if resp . NextPage == 0 {
break
}
opt . Page = resp . NextPage
}
return allComments , nil
}
2019-05-30 23:26:57 +03:00
// GetPullRequests returns pull requests according page and perPage
func ( g * GithubDownloaderV3 ) GetPullRequests ( page , perPage int ) ( [ ] * base . PullRequest , error ) {
2019-05-07 04:12:51 +03:00
opt := & github . PullRequestListOptions {
Sort : "created" ,
Direction : "asc" ,
State : "all" ,
ListOptions : github . ListOptions {
2019-05-30 23:26:57 +03:00
PerPage : perPage ,
Page : page ,
2019-05-07 04:12:51 +03:00
} ,
}
2019-05-30 23:26:57 +03:00
var allPRs = make ( [ ] * base . PullRequest , 0 , perPage )
2019-12-17 07:16:54 +03:00
g . sleep ( )
prs , resp , err := g . client . PullRequests . List ( g . ctx , g . repoOwner , g . repoName , opt )
2019-05-30 23:26:57 +03:00
if err != nil {
return nil , fmt . Errorf ( "error while listing repos: %v" , err )
}
2019-12-17 07:16:54 +03:00
g . rate = & resp . Rate
2019-05-30 23:26:57 +03:00
for _ , pr := range prs {
var body string
if pr . Body != nil {
body = * pr . Body
}
var milestone string
if pr . Milestone != nil {
milestone = * pr . Milestone . Title
}
var labels = make ( [ ] * base . Label , 0 , len ( pr . Labels ) )
for _ , l := range pr . Labels {
labels = append ( labels , convertGithubLabel ( l ) )
}
2019-05-07 04:12:51 +03:00
2019-05-30 23:26:57 +03:00
var email string
if pr . User . Email != nil {
email = * pr . User . Email
}
var merged bool
// pr.Merged is not valid, so use MergedAt to test if it's merged
if pr . MergedAt != nil {
merged = true
}
2019-05-07 04:12:51 +03:00
2019-06-18 19:15:39 +03:00
var (
headRepoName string
cloneURL string
headRef string
headSHA string
)
2019-05-30 23:26:57 +03:00
if pr . Head . Repo != nil {
2019-06-18 19:15:39 +03:00
if pr . Head . Repo . Name != nil {
headRepoName = * pr . Head . Repo . Name
}
if pr . Head . Repo . CloneURL != nil {
cloneURL = * pr . Head . Repo . CloneURL
}
}
if pr . Head . Ref != nil {
headRef = * pr . Head . Ref
}
if pr . Head . SHA != nil {
headSHA = * pr . Head . SHA
2019-05-07 04:12:51 +03:00
}
2019-05-30 23:26:57 +03:00
var mergeCommitSHA string
if pr . MergeCommitSHA != nil {
mergeCommitSHA = * pr . MergeCommitSHA
2019-05-07 04:12:51 +03:00
}
2019-05-30 23:26:57 +03:00
2019-06-18 19:15:39 +03:00
var headUserName string
if pr . Head . User != nil && pr . Head . User . Login != nil {
headUserName = * pr . Head . User . Login
}
2020-01-15 14:14:07 +03:00
// get reactions
var reactions [ ] * base . Reaction
for i := 1 ; ; i ++ {
g . sleep ( )
res , resp , err := g . client . Reactions . ListIssueReactions ( g . ctx , g . repoOwner , g . repoName , pr . GetNumber ( ) , & github . ListOptions {
Page : i ,
PerPage : perPage ,
} )
if err != nil {
return nil , err
}
g . rate = & resp . Rate
if len ( res ) == 0 {
break
}
for _ , reaction := range res {
reactions = append ( reactions , & base . Reaction {
UserID : reaction . User . GetID ( ) ,
UserName : reaction . User . GetLogin ( ) ,
Content : reaction . GetContent ( ) ,
} )
}
}
2019-05-30 23:26:57 +03:00
allPRs = append ( allPRs , & base . PullRequest {
Title : * pr . Title ,
Number : int64 ( * pr . Number ) ,
PosterName : * pr . User . Login ,
2019-07-08 05:14:12 +03:00
PosterID : * pr . User . ID ,
2019-05-30 23:26:57 +03:00
PosterEmail : email ,
Content : body ,
Milestone : milestone ,
State : * pr . State ,
Created : * pr . CreatedAt ,
2020-01-14 13:29:22 +03:00
Updated : * pr . UpdatedAt ,
2019-05-30 23:26:57 +03:00
Closed : pr . ClosedAt ,
Labels : labels ,
Merged : merged ,
MergeCommitSHA : mergeCommitSHA ,
MergedTime : pr . MergedAt ,
IsLocked : pr . ActiveLockReason != nil ,
Head : base . PullRequestBranch {
2019-06-18 19:15:39 +03:00
Ref : headRef ,
SHA : headSHA ,
2019-05-30 23:26:57 +03:00
RepoName : headRepoName ,
2019-06-18 19:15:39 +03:00
OwnerName : headUserName ,
2019-05-30 23:26:57 +03:00
CloneURL : cloneURL ,
} ,
Base : base . PullRequestBranch {
Ref : * pr . Base . Ref ,
SHA : * pr . Base . SHA ,
RepoName : * pr . Base . Repo . Name ,
OwnerName : * pr . Base . User . Login ,
} ,
2020-01-15 14:14:07 +03:00
PatchURL : * pr . PatchURL ,
Reactions : reactions ,
2019-05-30 23:26:57 +03:00
} )
2019-05-07 04:12:51 +03:00
}
2019-05-30 23:26:57 +03:00
2019-05-07 04:12:51 +03:00
return allPRs , nil
}
2020-01-23 20:28:15 +03:00
func convertGithubReview ( r * github . PullRequestReview ) * base . Review {
return & base . Review {
ID : r . GetID ( ) ,
ReviewerID : r . GetUser ( ) . GetID ( ) ,
ReviewerName : r . GetUser ( ) . GetLogin ( ) ,
CommitID : r . GetCommitID ( ) ,
Content : r . GetBody ( ) ,
CreatedAt : r . GetSubmittedAt ( ) ,
State : r . GetState ( ) ,
}
}
func ( g * GithubDownloaderV3 ) convertGithubReviewComments ( cs [ ] * github . PullRequestComment ) ( [ ] * base . ReviewComment , error ) {
var rcs = make ( [ ] * base . ReviewComment , 0 , len ( cs ) )
for _ , c := range cs {
// get reactions
var reactions [ ] * base . Reaction
for i := 1 ; ; i ++ {
g . sleep ( )
res , resp , err := g . client . Reactions . ListPullRequestCommentReactions ( g . ctx , g . repoOwner , g . repoName , c . GetID ( ) , & github . ListOptions {
Page : i ,
PerPage : 100 ,
} )
if err != nil {
return nil , err
}
g . rate = & resp . Rate
if len ( res ) == 0 {
break
}
for _ , reaction := range res {
reactions = append ( reactions , & base . Reaction {
UserID : reaction . User . GetID ( ) ,
UserName : reaction . User . GetLogin ( ) ,
Content : reaction . GetContent ( ) ,
} )
}
}
rcs = append ( rcs , & base . ReviewComment {
ID : c . GetID ( ) ,
InReplyTo : c . GetInReplyTo ( ) ,
Content : c . GetBody ( ) ,
TreePath : c . GetPath ( ) ,
DiffHunk : c . GetDiffHunk ( ) ,
Position : c . GetPosition ( ) ,
CommitID : c . GetCommitID ( ) ,
PosterID : c . GetUser ( ) . GetID ( ) ,
Reactions : reactions ,
CreatedAt : c . GetCreatedAt ( ) ,
UpdatedAt : c . GetUpdatedAt ( ) ,
} )
}
return rcs , nil
}
// GetReviews returns pull requests review
func ( g * GithubDownloaderV3 ) GetReviews ( pullRequestNumber int64 ) ( [ ] * base . Review , error ) {
var allReviews = make ( [ ] * base . Review , 0 , 100 )
opt := & github . ListOptions {
PerPage : 100 ,
}
for {
g . sleep ( )
reviews , resp , err := g . client . PullRequests . ListReviews ( g . ctx , g . repoOwner , g . repoName , int ( pullRequestNumber ) , opt )
if err != nil {
return nil , fmt . Errorf ( "error while listing repos: %v" , err )
}
g . rate = & resp . Rate
for _ , review := range reviews {
r := convertGithubReview ( review )
r . IssueIndex = pullRequestNumber
// retrieve all review comments
opt2 := & github . ListOptions {
PerPage : 100 ,
}
for {
g . sleep ( )
reviewComments , resp , err := g . client . PullRequests . ListReviewComments ( g . ctx , g . repoOwner , g . repoName , int ( pullRequestNumber ) , review . GetID ( ) , opt2 )
if err != nil {
return nil , fmt . Errorf ( "error while listing repos: %v" , err )
}
g . rate = & resp . Rate
cs , err := g . convertGithubReviewComments ( reviewComments )
if err != nil {
return nil , err
}
r . Comments = append ( r . Comments , cs ... )
if resp . NextPage == 0 {
break
}
opt2 . Page = resp . NextPage
}
allReviews = append ( allReviews , r )
}
if resp . NextPage == 0 {
break
}
opt . Page = resp . NextPage
}
return allReviews , nil
}