2019-08-14 23:32:19 +08:00
// Copyright 2019 The Gitea Authors.
// All rights reserved.
// Use of this source code is governed by a MIT-style
// license that can be found in the LICENSE file.
package pull
import (
2019-11-14 10:57:36 +08:00
"bytes"
"fmt"
"strings"
2019-08-14 23:32:19 +08:00
"code.gitea.io/gitea/models"
2019-11-14 10:57:36 +08:00
"code.gitea.io/gitea/modules/git"
2019-11-05 19:04:08 +08:00
"code.gitea.io/gitea/modules/notification"
2019-11-14 10:57:36 +08:00
"code.gitea.io/gitea/modules/setting"
"code.gitea.io/gitea/services/gitdiff"
2019-08-14 23:32:19 +08:00
)
2019-11-14 10:57:36 +08:00
// CreateCodeComment creates a comment on the code line
2020-01-09 02:47:45 +01:00
func CreateCodeComment ( doer * models . User , gitRepo * git . Repository , issue * models . Issue , line int64 , content string , treePath string , isReview bool , replyReviewID int64 , latestCommitID string ) ( * models . Comment , error ) {
2019-11-24 02:46:16 -03:00
var (
existsReview bool
err error
)
// CreateCodeComment() is used for:
// - Single comments
// - Comments that are part of a review
// - Comments that reply to an existing review
2019-11-14 10:57:36 +08:00
if ! isReview {
2019-11-24 02:46:16 -03:00
// It's not part of a review; maybe a reply to a review comment or a single comment.
// Check if there are reviews for that line already; if there are, this is a reply
if existsReview , err = models . ReviewExists ( issue , treePath , line ) ; err != nil {
return nil , err
}
}
// Comments that are replies don't require a review header to show up in the issue view
if ! isReview && existsReview {
if err = issue . LoadRepo ( ) ; err != nil {
2019-11-14 10:57:36 +08:00
return nil , err
}
comment , err := createCodeComment (
doer ,
issue . Repo ,
issue ,
content ,
treePath ,
line ,
replyReviewID ,
)
if err != nil {
return nil , err
}
notification . NotifyCreateIssueComment ( doer , issue . Repo , issue , comment )
return comment , nil
}
review , err := models . GetCurrentReview ( doer , issue )
if err != nil {
if ! models . IsErrReviewNotExist ( err ) {
return nil , err
}
review , err = models . CreateReview ( models . CreateReviewOptions {
Type : models . ReviewTypePending ,
Reviewer : doer ,
Issue : issue ,
2019-12-04 02:08:56 +01:00
Official : false ,
2020-01-09 02:47:45 +01:00
CommitID : latestCommitID ,
2019-11-14 10:57:36 +08:00
} )
if err != nil {
return nil , err
}
}
comment , err := createCodeComment (
doer ,
issue . Repo ,
issue ,
content ,
treePath ,
line ,
review . ID ,
)
2019-08-14 23:32:19 +08:00
if err != nil {
return nil , err
}
2019-11-24 02:46:16 -03:00
if ! isReview && ! existsReview {
// Submit the review we've just created so the comment shows up in the issue view
2020-01-09 02:47:45 +01:00
if _ , _ , err = SubmitReview ( doer , gitRepo , issue , models . ReviewTypeComment , "" , latestCommitID ) ; err != nil {
2019-11-24 02:46:16 -03:00
return nil , err
}
}
// NOTICE: if it's a pending review the notifications will not be fired until user submit review.
2019-11-14 10:57:36 +08:00
return comment , nil
}
// createCodeComment creates a plain code comment at the specified line / path
func createCodeComment ( doer * models . User , repo * models . Repository , issue * models . Issue , content , treePath string , line , reviewID int64 ) ( * models . Comment , error ) {
var commitID , patch string
if err := issue . LoadPullRequest ( ) ; err != nil {
return nil , fmt . Errorf ( "GetPullRequestByIssueID: %v" , err )
}
pr := issue . PullRequest
if err := pr . GetBaseRepo ( ) ; err != nil {
return nil , fmt . Errorf ( "GetHeadRepo: %v" , err )
}
gitRepo , err := git . OpenRepository ( pr . BaseRepo . RepoPath ( ) )
if err != nil {
return nil , fmt . Errorf ( "OpenRepository: %v" , err )
2019-11-13 15:36:04 -03:00
}
2019-11-14 10:57:36 +08:00
defer gitRepo . Close ( )
2019-11-05 19:04:08 +08:00
2019-11-14 10:57:36 +08:00
// FIXME validate treePath
// Get latest commit referencing the commented line
// No need for get commit for base branch changes
if line > 0 {
commit , err := gitRepo . LineBlame ( pr . GetGitRefName ( ) , gitRepo . Path , treePath , uint ( line ) )
if err == nil {
commitID = commit . ID . String ( )
} else if ! strings . Contains ( err . Error ( ) , "exit status 128 - fatal: no such path" ) {
return nil , fmt . Errorf ( "LineBlame[%s, %s, %s, %d]: %v" , pr . GetGitRefName ( ) , gitRepo . Path , treePath , line , err )
}
}
// Only fetch diff if comment is review comment
if reviewID != 0 {
headCommitID , err := gitRepo . GetRefCommitID ( pr . GetGitRefName ( ) )
if err != nil {
return nil , fmt . Errorf ( "GetRefCommitID[%s]: %v" , pr . GetGitRefName ( ) , err )
}
patchBuf := new ( bytes . Buffer )
if err := gitdiff . GetRawDiffForFile ( gitRepo . Path , pr . MergeBase , headCommitID , gitdiff . RawDiffNormal , treePath , patchBuf ) ; err != nil {
return nil , fmt . Errorf ( "GetRawDiffForLine[%s, %s, %s, %s]: %v" , err , gitRepo . Path , pr . MergeBase , headCommitID , treePath )
}
patch = gitdiff . CutDiffAroundLine ( patchBuf , int64 ( ( & models . Comment { Line : line } ) . UnsignedLine ( ) ) , line < 0 , setting . UI . CodeCommentLines )
}
2019-12-16 11:54:24 +08:00
return models . CreateComment ( & models . CreateCommentOptions {
2019-11-14 10:57:36 +08:00
Type : models . CommentTypeCode ,
Doer : doer ,
Repo : repo ,
Issue : issue ,
Content : content ,
LineNum : line ,
TreePath : treePath ,
CommitSHA : commitID ,
ReviewID : reviewID ,
Patch : patch ,
} )
2019-10-18 03:33:19 -05:00
}
2019-11-14 10:57:36 +08:00
// SubmitReview creates a review out of the existing pending review or creates a new one if no pending review exist
2020-01-09 02:47:45 +01:00
func SubmitReview ( doer * models . User , gitRepo * git . Repository , issue * models . Issue , reviewType models . ReviewType , content , commitID string ) ( * models . Review , * models . Comment , error ) {
pr , err := issue . GetPullRequest ( )
2019-10-18 03:33:19 -05:00
if err != nil {
2019-11-14 10:57:36 +08:00
return nil , nil , err
2019-10-18 03:33:19 -05:00
}
2020-01-09 02:47:45 +01:00
var stale bool
if reviewType != models . ReviewTypeApprove && reviewType != models . ReviewTypeReject {
stale = false
} else {
headCommitID , err := gitRepo . GetRefCommitID ( pr . GetGitRefName ( ) )
if err != nil {
return nil , nil , err
}
if headCommitID == commitID {
stale = false
} else {
stale , err = checkIfPRContentChanged ( pr , commitID , headCommitID )
if err != nil {
return nil , nil , err
}
}
}
review , comm , err := models . SubmitReview ( doer , issue , reviewType , content , commitID , stale )
2019-11-14 10:57:36 +08:00
if err != nil {
return nil , nil , err
}
2020-01-09 02:47:45 +01:00
2019-11-14 10:57:36 +08:00
notification . NotifyPullRequestReview ( pr , review , comm )
2019-08-14 23:32:19 +08:00
2019-11-14 10:57:36 +08:00
return review , comm , nil
2019-08-14 23:32:19 +08:00
}