2019-05-07 09:12:51 +08: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"
2020-08-27 20:36:37 -05:00
"io"
2019-05-07 09:12:51 +08:00
"net/http"
"net/url"
2021-06-30 15:23:49 +08:00
"strconv"
2019-05-07 09:12:51 +08:00
"strings"
2019-12-17 12:16:54 +08:00
"time"
2019-05-07 09:12:51 +08:00
"code.gitea.io/gitea/modules/log"
"code.gitea.io/gitea/modules/migrations/base"
2019-10-14 14:10:42 +08:00
"code.gitea.io/gitea/modules/structs"
2020-05-08 16:46:05 +01:00
"code.gitea.io/gitea/modules/util"
2019-05-07 09:12:51 +08:00
2020-07-31 16:22:34 +02:00
"github.com/google/go-github/v32/github"
2019-05-07 09:12:51 +08:00
"golang.org/x/oauth2"
)
var (
_ base . Downloader = & GithubDownloaderV3 { }
_ base . DownloaderFactory = & GithubDownloaderV3Factory { }
2020-01-16 16:15:44 +01:00
// GithubLimitRateRemaining limit to wait for new rate to apply
GithubLimitRateRemaining = 0
2019-05-07 09:12:51 +08:00
)
func init ( ) {
RegisterDownloaderFactory ( & GithubDownloaderV3Factory { } )
}
// GithubDownloaderV3Factory defines a github downloader v3 factory
type GithubDownloaderV3Factory struct {
}
// New returns a Downloader related to this factory according MigrateOptions
2020-09-02 18:49:25 +01:00
func ( f * GithubDownloaderV3Factory ) New ( ctx context . Context , opts base . MigrateOptions ) ( base . Downloader , error ) {
2019-10-13 21:23:14 +08:00
u , err := url . Parse ( opts . CloneAddr )
2019-05-07 09:12:51 +08:00
if err != nil {
return nil , err
}
2020-09-21 09:36:51 -05:00
baseURL := u . Scheme + "://" + u . Host
2019-05-07 09:12:51 +08:00
fields := strings . Split ( u . Path , "/" )
oldOwner := fields [ 1 ]
oldName := strings . TrimSuffix ( fields [ 2 ] , ".git" )
log . Trace ( "Create github downloader: %s/%s" , oldOwner , oldName )
2020-09-21 09:36:51 -05:00
return NewGithubDownloaderV3 ( ctx , baseURL , opts . AuthUsername , opts . AuthPassword , opts . AuthToken , oldOwner , oldName ) , nil
2019-05-07 09:12:51 +08:00
}
2019-10-14 14:10:42 +08:00
// GitServiceType returns the type of git service
func ( f * GithubDownloaderV3Factory ) GitServiceType ( ) structs . GitServiceType {
return structs . GithubService
}
2021-07-08 07:38:13 -04:00
// GithubDownloaderV3 implements a Downloader interface to get repository information
2019-05-07 09:12:51 +08:00
// from github via APIv3
type GithubDownloaderV3 struct {
2021-01-21 20:33:58 +01:00
base . NullDownloader
2020-10-25 13:11:03 +08:00
ctx context . Context
client * github . Client
repoOwner string
repoName string
userName string
password string
rate * github . Rate
maxPerPage int
2019-05-07 09:12:51 +08:00
}
// NewGithubDownloaderV3 creates a github Downloader via github v3 API
2020-09-21 09:36:51 -05:00
func NewGithubDownloaderV3 ( ctx context . Context , baseURL , userName , password , token , repoOwner , repoName string ) * GithubDownloaderV3 {
2019-05-07 09:12:51 +08:00
var downloader = GithubDownloaderV3 {
2020-10-25 13:11:03 +08:00
userName : userName ,
password : password ,
ctx : ctx ,
repoOwner : repoOwner ,
repoName : repoName ,
maxPerPage : 100 ,
2019-05-07 09:12:51 +08:00
}
2020-08-27 20:36:37 -05:00
client := & http . Client {
Transport : & http . Transport {
Proxy : func ( req * http . Request ) ( * url . URL , error ) {
req . SetBasicAuth ( userName , password )
return nil , nil
} ,
} ,
}
if token != "" {
ts := oauth2 . StaticTokenSource (
& oauth2 . Token { AccessToken : token } ,
)
client = oauth2 . NewClient ( downloader . ctx , ts )
2019-05-07 09:12:51 +08:00
}
downloader . client = github . NewClient ( client )
2020-09-21 09:36:51 -05:00
if baseURL != "https://github.com" {
downloader . client , _ = github . NewEnterpriseClient ( baseURL , baseURL , client )
}
2019-05-07 09:12:51 +08:00
return & downloader
}
2019-12-17 12:16:54 +08:00
// SetContext set context
func ( g * GithubDownloaderV3 ) SetContext ( ctx context . Context ) {
g . ctx = ctx
}
func ( g * GithubDownloaderV3 ) sleep ( ) {
2020-01-16 16:15:44 +01:00
for g . rate != nil && g . rate . Remaining <= GithubLimitRateRemaining {
2019-12-17 12:16:54 +08:00
timer := time . NewTimer ( time . Until ( g . rate . Reset . Time ) )
select {
case <- g . ctx . Done ( ) :
2020-05-08 16:46:05 +01:00
util . StopTimer ( timer )
2019-12-17 12:16:54 +08:00
return
case <- timer . C :
}
2020-01-16 16:15:44 +01:00
err := g . RefreshRate ( )
2019-12-17 12:16:54 +08:00
if err != nil {
log . Error ( "g.client.RateLimits: %s" , err )
}
2020-01-16 16:15:44 +01:00
}
}
2019-12-17 12:16:54 +08:00
2020-01-16 16:15:44 +01: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 {
2021-04-15 15:34:22 +02:00
// if rate limit is not enabled, ignore it
if strings . Contains ( err . Error ( ) , "404" ) {
g . rate = nil
return nil
}
2020-01-16 16:15:44 +01:00
return err
2019-12-17 12:16:54 +08:00
}
2020-01-16 16:15:44 +01:00
g . rate = rates . GetCore ( )
return nil
2019-12-17 12:16:54 +08:00
}
2019-05-07 09:12:51 +08:00
// GetRepoInfo returns a repository information
func ( g * GithubDownloaderV3 ) GetRepoInfo ( ) ( * base . Repository , error ) {
2019-12-17 12:16:54 +08:00
g . sleep ( )
gr , resp , err := g . client . Repositories . Get ( g . ctx , g . repoOwner , g . repoName )
2019-05-07 09:12:51 +08:00
if err != nil {
return nil , err
}
2019-12-17 12:16:54 +08:00
g . rate = & resp . Rate
2020-09-15 16:37:44 +02:00
defaultBranch := ""
if gr . DefaultBranch != nil {
defaultBranch = * gr . DefaultBranch
}
2019-05-07 09:12:51 +08:00
// convert github repo to stand Repo
return & base . Repository {
2020-09-15 16:37:44 +02:00
Owner : g . repoOwner ,
Name : gr . GetName ( ) ,
IsPrivate : * gr . Private ,
Description : gr . GetDescription ( ) ,
OriginalURL : gr . GetHTMLURL ( ) ,
CloneURL : gr . GetCloneURL ( ) ,
DefaultBranch : defaultBranch ,
2019-05-07 09:12:51 +08:00
} , nil
}
2019-08-14 08:16:12 +02:00
// GetTopics return github topics
func ( g * GithubDownloaderV3 ) GetTopics ( ) ( [ ] string , error ) {
2019-12-17 12:16:54 +08: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 08:16:12 +02:00
}
2019-05-07 09:12:51 +08:00
// GetMilestones returns milestones
func ( g * GithubDownloaderV3 ) GetMilestones ( ) ( [ ] * base . Milestone , error ) {
2020-10-25 13:11:03 +08:00
var perPage = g . maxPerPage
2019-05-07 09:12:51 +08:00
var milestones = make ( [ ] * base . Milestone , 0 , perPage )
for i := 1 ; ; i ++ {
2019-12-17 12:16:54 +08:00
g . sleep ( )
ms , resp , err := g . client . Issues . ListMilestones ( g . ctx , g . repoOwner , g . repoName ,
2019-05-07 09:12:51 +08:00
& github . MilestoneListOptions {
State : "all" ,
ListOptions : github . ListOptions {
Page : i ,
PerPage : perPage ,
} } )
if err != nil {
return nil , err
}
2019-12-17 12:16:54 +08:00
g . rate = & resp . Rate
2019-05-07 09:12:51 +08: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 ) {
2020-10-25 13:11:03 +08:00
var perPage = g . maxPerPage
2019-05-07 09:12:51 +08:00
var labels = make ( [ ] * base . Label , 0 , perPage )
for i := 1 ; ; i ++ {
2019-12-17 12:16:54 +08:00
g . sleep ( )
ls , resp , err := g . client . Issues . ListLabels ( g . ctx , g . repoOwner , g . repoName ,
2019-05-07 09:12:51 +08:00
& github . ListOptions {
Page : i ,
PerPage : perPage ,
} )
if err != nil {
return nil , err
}
2019-12-17 12:16:54 +08:00
g . rate = & resp . Rate
2019-05-07 09:12:51 +08: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 {
r := & base . Release {
TagName : * rel . TagName ,
TargetCommitish : * rel . TargetCommitish ,
Draft : * rel . Draft ,
Prerelease : * rel . Prerelease ,
Created : rel . CreatedAt . Time ,
2019-10-05 19:09:27 +08:00
PublisherID : * rel . Author . ID ,
PublisherName : * rel . Author . Login ,
2021-05-16 00:37:17 +02:00
}
if rel . Body != nil {
r . Body = * rel . Body
}
if rel . Name != nil {
r . Name = * rel . Name
}
if rel . Author . Email != nil {
r . PublisherEmail = * rel . Author . Email
}
if rel . PublishedAt != nil {
r . Published = rel . PublishedAt . Time
2019-05-07 09:12:51 +08:00
}
for _ , asset := range rel . Assets {
2021-02-18 08:36:49 +08:00
var assetID = * asset . ID // Don't optimize this, for closure we need a local variable
2020-12-27 11:34:19 +08:00
r . Assets = append ( r . Assets , & base . ReleaseAsset {
2020-08-27 20:36:37 -05:00
ID : * asset . ID ,
2019-05-07 09:12:51 +08:00
Name : * asset . Name ,
ContentType : asset . ContentType ,
Size : asset . Size ,
DownloadCount : asset . DownloadCount ,
Created : asset . CreatedAt . Time ,
Updated : asset . UpdatedAt . Time ,
2020-12-27 11:34:19 +08:00
DownloadFunc : func ( ) ( io . ReadCloser , error ) {
2021-02-07 23:56:11 +08:00
g . sleep ( )
2021-05-16 00:37:17 +02:00
asset , redirectURL , err := g . client . Repositories . DownloadReleaseAsset ( g . ctx , g . repoOwner , g . repoName , assetID , nil )
2020-12-27 11:34:19 +08:00
if err != nil {
return nil , err
}
2021-05-16 00:37:17 +02:00
if err := g . RefreshRate ( ) ; err != nil {
2021-02-07 23:56:11 +08:00
log . Error ( "g.client.RateLimits: %s" , err )
}
2020-12-27 11:34:19 +08:00
if asset == nil {
2021-05-16 00:37:17 +02:00
if redirectURL != "" {
2021-02-18 08:36:49 +08:00
g . sleep ( )
2021-05-16 00:37:17 +02:00
req , err := http . NewRequestWithContext ( g . ctx , "GET" , redirectURL , nil )
2021-02-18 08:36:49 +08:00
if err != nil {
return nil , err
}
resp , err := http . DefaultClient . Do ( req )
err1 := g . RefreshRate ( )
if err1 != nil {
log . Error ( "g.client.RateLimits: %s" , err1 )
}
if err != nil {
return nil , err
}
return resp . Body , nil
}
return nil , fmt . Errorf ( "No release asset found for %d" , assetID )
2020-12-27 11:34:19 +08:00
}
return asset , nil
} ,
2019-05-07 09:12:51 +08:00
} )
}
return r
}
// GetReleases returns releases
func ( g * GithubDownloaderV3 ) GetReleases ( ) ( [ ] * base . Release , error ) {
2020-10-25 13:11:03 +08:00
var perPage = g . maxPerPage
2019-05-07 09:12:51 +08:00
var releases = make ( [ ] * base . Release , 0 , perPage )
for i := 1 ; ; i ++ {
2019-12-17 12:16:54 +08:00
g . sleep ( )
ls , resp , err := g . client . Repositories . ListReleases ( g . ctx , g . repoOwner , g . repoName ,
2019-05-07 09:12:51 +08:00
& github . ListOptions {
Page : i ,
PerPage : perPage ,
} )
if err != nil {
return nil , err
}
2019-12-17 12:16:54 +08:00
g . rate = & resp . Rate
2019-05-07 09:12:51 +08: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-31 04:26:57 +08:00
func ( g * GithubDownloaderV3 ) GetIssues ( page , perPage int ) ( [ ] * base . Issue , bool , error ) {
2020-10-25 13:11:03 +08:00
if perPage > g . maxPerPage {
perPage = g . maxPerPage
}
2019-05-07 09:12:51 +08:00
opt := & github . IssueListByRepoOptions {
Sort : "created" ,
Direction : "asc" ,
State : "all" ,
ListOptions : github . ListOptions {
PerPage : perPage ,
2019-05-31 04:26:57 +08:00
Page : page ,
2019-05-07 09:12:51 +08:00
} ,
}
2019-05-31 04:26:57 +08:00
var allIssues = make ( [ ] * base . Issue , 0 , perPage )
2019-12-17 12:16:54 +08:00
g . sleep ( )
issues , resp , err := g . client . Issues . ListByRepo ( g . ctx , g . repoOwner , g . repoName , opt )
2019-05-31 04:26:57 +08:00
if err != nil {
return nil , false , fmt . Errorf ( "error while listing repos: %v" , err )
}
2020-12-27 11:34:19 +08:00
log . Trace ( "Request get issues %d/%d, but in fact get %d" , perPage , page , len ( issues ) )
2019-12-17 12:16:54 +08:00
g . rate = & resp . Rate
2019-05-31 04:26:57 +08:00
for _ , issue := range issues {
if issue . IsPullRequest ( ) {
continue
2019-05-07 09:12:51 +08:00
}
2019-05-31 04:26:57 +08:00
var body string
if issue . Body != nil {
body = * issue . Body
2019-05-07 09:12:51 +08:00
}
2019-05-31 04:26:57 +08: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 16:22:34 +02:00
labels = append ( labels , convertGithubLabel ( l ) )
2019-05-31 04:26:57 +08:00
}
var email string
if issue . User . Email != nil {
email = * issue . User . Email
}
2020-01-15 19:14:07 +08: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-31 04:26:57 +08:00
allIssues = append ( allIssues , & base . Issue {
Title : * issue . Title ,
Number : int64 ( * issue . Number ) ,
2019-07-07 22:14:12 -04:00
PosterID : * issue . User . ID ,
2019-05-31 04:26:57 +08:00
PosterName : * issue . User . Login ,
PosterEmail : email ,
Content : body ,
Milestone : milestone ,
State : * issue . State ,
Created : * issue . CreatedAt ,
2020-01-14 18:29:22 +08:00
Updated : * issue . UpdatedAt ,
2019-05-31 04:26:57 +08:00
Labels : labels ,
Reactions : reactions ,
Closed : issue . ClosedAt ,
IsLocked : * issue . Locked ,
} )
2019-05-07 09:12:51 +08:00
}
2019-05-31 04:26:57 +08:00
return allIssues , len ( issues ) < perPage , nil
2019-05-07 09:12:51 +08:00
}
2021-06-30 15:23:49 +08:00
// SupportGetRepoComments return true if it supports get repo comments
func ( g * GithubDownloaderV3 ) SupportGetRepoComments ( ) bool {
return true
}
2019-05-07 09:12:51 +08:00
// GetComments returns comments according issueNumber
2021-06-30 15:23:49 +08:00
func ( g * GithubDownloaderV3 ) GetComments ( opts base . GetCommentOptions ) ( [ ] * base . Comment , bool , error ) {
if opts . IssueNumber > 0 {
comments , err := g . getComments ( opts . IssueNumber )
return comments , false , err
}
return g . GetAllComments ( opts . Page , opts . PageSize )
}
func ( g * GithubDownloaderV3 ) getComments ( issueNumber int64 ) ( [ ] * base . Comment , error ) {
2020-01-24 01:28:15 +08:00
var (
2020-10-25 13:11:03 +08:00
allComments = make ( [ ] * base . Comment , 0 , g . maxPerPage )
2020-01-24 01:28:15 +08:00
created = "created"
asc = "asc"
)
2019-05-07 09:12:51 +08:00
opt := & github . IssueListCommentsOptions {
2020-07-31 16:22:34 +02:00
Sort : & created ,
Direction : & asc ,
2019-05-07 09:12:51 +08:00
ListOptions : github . ListOptions {
2020-10-25 13:11:03 +08:00
PerPage : g . maxPerPage ,
2019-05-07 09:12:51 +08:00
} ,
}
for {
2019-12-17 12:16:54 +08:00
g . sleep ( )
2019-05-07 09:12:51 +08: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 12:16:54 +08:00
g . rate = & resp . Rate
2019-05-07 09:12:51 +08:00
for _ , comment := range comments {
var email string
if comment . User . Email != nil {
email = * comment . User . Email
}
2020-01-15 19:14:07 +08: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 ,
2020-10-25 13:11:03 +08:00
PerPage : g . maxPerPage ,
2020-01-15 19:14:07 +08:00
} )
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 09:12:51 +08:00
}
allComments = append ( allComments , & base . Comment {
2019-06-29 21:38:22 +08:00
IssueIndex : issueNumber ,
2019-07-07 22:14:12 -04:00
PosterID : * comment . User . ID ,
2019-05-07 09:12:51 +08:00
PosterName : * comment . User . Login ,
PosterEmail : email ,
Content : * comment . Body ,
Created : * comment . CreatedAt ,
2020-01-14 18:29:22 +08:00
Updated : * comment . UpdatedAt ,
2019-05-07 09:12:51 +08:00
Reactions : reactions ,
} )
}
if resp . NextPage == 0 {
break
}
opt . Page = resp . NextPage
}
return allComments , nil
}
2021-06-30 15:23:49 +08:00
// GetAllComments returns repository comments according page and perPageSize
func ( g * GithubDownloaderV3 ) GetAllComments ( page , perPage int ) ( [ ] * base . Comment , bool , error ) {
var (
allComments = make ( [ ] * base . Comment , 0 , perPage )
created = "created"
asc = "asc"
)
opt := & github . IssueListCommentsOptions {
Sort : & created ,
Direction : & asc ,
ListOptions : github . ListOptions {
Page : page ,
PerPage : perPage ,
} ,
}
g . sleep ( )
comments , resp , err := g . client . Issues . ListComments ( g . ctx , g . repoOwner , g . repoName , 0 , opt )
if err != nil {
return nil , false , fmt . Errorf ( "error while listing repos: %v" , err )
}
log . Trace ( "Request get comments %d/%d, but in fact get %d" , perPage , page , len ( comments ) )
g . rate = & resp . Rate
for _ , comment := range comments {
var email string
if comment . User . Email != nil {
email = * comment . User . Email
}
// 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 : g . maxPerPage ,
} )
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 ( ) ,
} )
}
}
idx := strings . LastIndex ( * comment . IssueURL , "/" )
issueIndex , _ := strconv . ParseInt ( ( * comment . IssueURL ) [ idx + 1 : ] , 10 , 64 )
allComments = append ( allComments , & base . Comment {
IssueIndex : issueIndex ,
PosterID : * comment . User . ID ,
PosterName : * comment . User . Login ,
PosterEmail : email ,
Content : * comment . Body ,
Created : * comment . CreatedAt ,
Updated : * comment . UpdatedAt ,
Reactions : reactions ,
} )
}
return allComments , len ( allComments ) < perPage , nil
}
2019-05-31 04:26:57 +08:00
// GetPullRequests returns pull requests according page and perPage
2020-10-14 06:06:00 +02:00
func ( g * GithubDownloaderV3 ) GetPullRequests ( page , perPage int ) ( [ ] * base . PullRequest , bool , error ) {
2020-10-25 13:11:03 +08:00
if perPage > g . maxPerPage {
perPage = g . maxPerPage
}
2019-05-07 09:12:51 +08:00
opt := & github . PullRequestListOptions {
Sort : "created" ,
Direction : "asc" ,
State : "all" ,
ListOptions : github . ListOptions {
2019-05-31 04:26:57 +08:00
PerPage : perPage ,
Page : page ,
2019-05-07 09:12:51 +08:00
} ,
}
2019-05-31 04:26:57 +08:00
var allPRs = make ( [ ] * base . PullRequest , 0 , perPage )
2019-12-17 12:16:54 +08:00
g . sleep ( )
prs , resp , err := g . client . PullRequests . List ( g . ctx , g . repoOwner , g . repoName , opt )
2019-05-31 04:26:57 +08:00
if err != nil {
2020-10-14 06:06:00 +02:00
return nil , false , fmt . Errorf ( "error while listing repos: %v" , err )
2019-05-31 04:26:57 +08:00
}
2021-06-30 15:23:49 +08:00
log . Trace ( "Request get pull requests %d/%d, but in fact get %d" , perPage , page , len ( prs ) )
2019-12-17 12:16:54 +08:00
g . rate = & resp . Rate
2019-05-31 04:26:57 +08: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 09:12:51 +08:00
2019-05-31 04:26:57 +08: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 09:12:51 +08:00
2019-06-19 00:15:39 +08:00
var (
headRepoName string
cloneURL string
headRef string
headSHA string
)
2019-05-31 04:26:57 +08:00
if pr . Head . Repo != nil {
2019-06-19 00:15:39 +08: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 09:12:51 +08:00
}
2019-05-31 04:26:57 +08:00
var mergeCommitSHA string
if pr . MergeCommitSHA != nil {
mergeCommitSHA = * pr . MergeCommitSHA
2019-05-07 09:12:51 +08:00
}
2019-05-31 04:26:57 +08:00
2019-06-19 00:15:39 +08:00
var headUserName string
if pr . Head . User != nil && pr . Head . User . Login != nil {
headUserName = * pr . Head . User . Login
}
2020-01-15 19:14:07 +08: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 {
2020-10-14 06:06:00 +02:00
return nil , false , err
2020-01-15 19:14:07 +08:00
}
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-31 04:26:57 +08:00
allPRs = append ( allPRs , & base . PullRequest {
Title : * pr . Title ,
Number : int64 ( * pr . Number ) ,
PosterName : * pr . User . Login ,
2019-07-07 22:14:12 -04:00
PosterID : * pr . User . ID ,
2019-05-31 04:26:57 +08:00
PosterEmail : email ,
Content : body ,
Milestone : milestone ,
State : * pr . State ,
Created : * pr . CreatedAt ,
2020-01-14 18:29:22 +08:00
Updated : * pr . UpdatedAt ,
2019-05-31 04:26:57 +08:00
Closed : pr . ClosedAt ,
Labels : labels ,
Merged : merged ,
MergeCommitSHA : mergeCommitSHA ,
MergedTime : pr . MergedAt ,
IsLocked : pr . ActiveLockReason != nil ,
Head : base . PullRequestBranch {
2019-06-19 00:15:39 +08:00
Ref : headRef ,
SHA : headSHA ,
2019-05-31 04:26:57 +08:00
RepoName : headRepoName ,
2019-06-19 00:15:39 +08:00
OwnerName : headUserName ,
2019-05-31 04:26:57 +08: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 19:14:07 +08:00
PatchURL : * pr . PatchURL ,
Reactions : reactions ,
2019-05-31 04:26:57 +08:00
} )
2019-05-07 09:12:51 +08:00
}
2019-05-31 04:26:57 +08:00
2020-10-14 06:06:00 +02:00
return allPRs , len ( prs ) < perPage , nil
2019-05-07 09:12:51 +08:00
}
2020-01-24 01:28:15 +08: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 ,
2020-10-25 13:11:03 +08:00
PerPage : g . maxPerPage ,
2020-01-24 01:28:15 +08:00
} )
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 ) {
2020-10-25 13:11:03 +08:00
var allReviews = make ( [ ] * base . Review , 0 , g . maxPerPage )
2020-01-24 01:28:15 +08:00
opt := & github . ListOptions {
2020-10-25 13:11:03 +08:00
PerPage : g . maxPerPage ,
2020-01-24 01:28:15 +08:00
}
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 {
2020-10-25 13:11:03 +08:00
PerPage : g . maxPerPage ,
2020-01-24 01:28:15 +08:00
}
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
}