2019-08-14 18:32:19 +03: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 (
2022-01-20 02:26:57 +03:00
"context"
2019-11-14 05:57:36 +03:00
"fmt"
2021-02-27 21:46:14 +03:00
"io"
2020-10-06 03:18:55 +03:00
"regexp"
2019-11-14 05:57:36 +03:00
"strings"
2021-09-19 14:49:59 +03:00
"code.gitea.io/gitea/models/db"
2022-06-13 12:37:59 +03:00
issues_model "code.gitea.io/gitea/models/issues"
2021-12-10 04:27:50 +03:00
repo_model "code.gitea.io/gitea/models/repo"
2021-11-24 12:49:20 +03:00
user_model "code.gitea.io/gitea/models/user"
2019-11-14 05:57:36 +03:00
"code.gitea.io/gitea/modules/git"
2021-02-27 21:46:14 +03:00
"code.gitea.io/gitea/modules/log"
2019-11-05 14:04:08 +03:00
"code.gitea.io/gitea/modules/notification"
2019-11-14 05:57:36 +03:00
"code.gitea.io/gitea/modules/setting"
2022-07-19 16:20:28 +03:00
"code.gitea.io/gitea/modules/util"
2019-08-14 18:32:19 +03:00
)
2019-11-14 05:57:36 +03:00
// CreateCodeComment creates a comment on the code line
2022-06-13 12:37:59 +03:00
func CreateCodeComment ( ctx context . Context , doer * user_model . User , gitRepo * git . Repository , issue * issues_model . Issue , line int64 , content , treePath string , isReview bool , replyReviewID int64 , latestCommitID string ) ( * issues_model . Comment , error ) {
2019-11-24 08: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
2020-08-21 10:53:14 +03:00
if ! isReview && replyReviewID != 0 {
2019-11-24 08: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
2022-06-13 12:37:59 +03:00
if existsReview , err = issues_model . ReviewExists ( issue , treePath , line ) ; err != nil {
2019-11-24 08:46:16 +03:00
return nil , err
}
}
// Comments that are replies don't require a review header to show up in the issue view
if ! isReview && existsReview {
2022-04-08 12:11:15 +03:00
if err = issue . LoadRepo ( ctx ) ; err != nil {
2019-11-14 05:57:36 +03:00
return nil , err
}
2022-01-20 02:26:57 +03:00
comment , err := createCodeComment ( ctx ,
2019-11-14 05:57:36 +03:00
doer ,
issue . Repo ,
issue ,
content ,
treePath ,
line ,
replyReviewID ,
)
if err != nil {
return nil , err
}
2022-06-13 12:37:59 +03:00
mentions , err := issues_model . FindAndUpdateIssueMentions ( ctx , issue , doer , comment . Content )
2021-01-02 20:04:02 +03:00
if err != nil {
return nil , err
}
notification . NotifyCreateIssueComment ( doer , issue . Repo , issue , comment , mentions )
2019-11-14 05:57:36 +03:00
return comment , nil
}
2022-06-13 12:37:59 +03:00
review , err := issues_model . GetCurrentReview ( ctx , doer , issue )
2019-11-14 05:57:36 +03:00
if err != nil {
2022-06-13 12:37:59 +03:00
if ! issues_model . IsErrReviewNotExist ( err ) {
2019-11-14 05:57:36 +03:00
return nil , err
}
2022-06-13 12:37:59 +03:00
if review , err = issues_model . CreateReview ( ctx , issues_model . CreateReviewOptions {
Type : issues_model . ReviewTypePending ,
2019-11-14 05:57:36 +03:00
Reviewer : doer ,
Issue : issue ,
2019-12-04 04:08:56 +03:00
Official : false ,
2020-01-09 04:47:45 +03:00
CommitID : latestCommitID ,
2020-10-12 22:55:13 +03:00
} ) ; err != nil {
2019-11-14 05:57:36 +03:00
return nil , err
}
}
2022-01-20 02:26:57 +03:00
comment , err := createCodeComment ( ctx ,
2019-11-14 05:57:36 +03:00
doer ,
issue . Repo ,
issue ,
content ,
treePath ,
line ,
review . ID ,
)
2019-08-14 18:32:19 +03:00
if err != nil {
return nil , err
}
2019-11-24 08:46:16 +03:00
if ! isReview && ! existsReview {
// Submit the review we've just created so the comment shows up in the issue view
2022-06-13 12:37:59 +03:00
if _ , _ , err = SubmitReview ( ctx , doer , gitRepo , issue , issues_model . ReviewTypeComment , "" , latestCommitID , nil ) ; err != nil {
2019-11-24 08: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 05:57:36 +03:00
return comment , nil
}
2020-10-06 03:18:55 +03:00
var notEnoughLines = regexp . MustCompile ( ` exit status 128 - fatal: file .* has only \d+ lines? ` )
2019-11-14 05:57:36 +03:00
// createCodeComment creates a plain code comment at the specified line / path
2022-06-13 12:37:59 +03:00
func createCodeComment ( ctx context . Context , doer * user_model . User , repo * repo_model . Repository , issue * issues_model . Issue , content , treePath string , line , reviewID int64 ) ( * issues_model . Comment , error ) {
2019-11-14 05:57:36 +03:00
var commitID , patch string
if err := issue . LoadPullRequest ( ) ; err != nil {
return nil , fmt . Errorf ( "GetPullRequestByIssueID: %v" , err )
}
pr := issue . PullRequest
2022-04-28 14:48:48 +03:00
if err := pr . LoadBaseRepoCtx ( ctx ) ; err != nil {
2020-03-03 01:31:55 +03:00
return nil , fmt . Errorf ( "LoadHeadRepo: %v" , err )
2019-11-14 05:57:36 +03:00
}
2022-01-20 02:26:57 +03:00
gitRepo , closer , err := git . RepositoryFromContextOrOpen ( ctx , pr . BaseRepo . RepoPath ( ) )
2019-11-14 05:57:36 +03:00
if err != nil {
2022-03-29 22:13:41 +03:00
return nil , fmt . Errorf ( "RepositoryFromContextOrOpen: %v" , err )
2019-11-13 21:36:04 +03:00
}
2022-01-20 02:26:57 +03:00
defer closer . Close ( )
2019-11-05 14:04:08 +03:00
2020-11-09 09:15:09 +03:00
invalidated := false
head := pr . GetGitRefName ( )
2019-11-14 05:57:36 +03:00
if line > 0 {
2020-11-09 09:15:09 +03:00
if reviewID != 0 {
2022-06-13 12:37:59 +03:00
first , err := issues_model . FindComments ( ctx , & issues_model . FindCommentsOptions {
2020-11-09 09:15:09 +03:00
ReviewID : reviewID ,
Line : line ,
TreePath : treePath ,
2022-06-13 12:37:59 +03:00
Type : issues_model . CommentTypeCode ,
2021-09-24 14:32:56 +03:00
ListOptions : db . ListOptions {
2020-11-09 09:15:09 +03:00
PageSize : 1 ,
Page : 1 ,
} ,
} )
if err == nil && len ( first ) > 0 {
commitID = first [ 0 ] . CommitSHA
invalidated = first [ 0 ] . Invalidated
patch = first [ 0 ] . Patch
2022-06-13 12:37:59 +03:00
} else if err != nil && ! issues_model . IsErrCommentNotExist ( err ) {
2020-11-09 09:15:09 +03:00
return nil , fmt . Errorf ( "Find first comment for %d line %d path %s. Error: %v" , reviewID , line , treePath , err )
} else {
2022-06-13 12:37:59 +03:00
review , err := issues_model . GetReviewByID ( ctx , reviewID )
2020-11-09 09:15:09 +03:00
if err == nil && len ( review . CommitID ) > 0 {
head = review . CommitID
2022-06-13 12:37:59 +03:00
} else if err != nil && ! issues_model . IsErrReviewNotExist ( err ) {
2020-11-09 09:15:09 +03:00
return nil , fmt . Errorf ( "GetReviewByID %d. Error: %v" , reviewID , err )
}
}
}
if len ( commitID ) == 0 {
// FIXME validate treePath
// Get latest commit referencing the commented line
// No need for get commit for base branch changes
commit , err := gitRepo . LineBlame ( head , 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" ) || notEnoughLines . MatchString ( err . Error ( ) ) ) {
return nil , fmt . Errorf ( "LineBlame[%s, %s, %s, %d]: %v" , pr . GetGitRefName ( ) , gitRepo . Path , treePath , line , err )
}
2019-11-14 05:57:36 +03:00
}
}
// Only fetch diff if comment is review comment
2020-11-09 09:15:09 +03:00
if len ( patch ) == 0 && reviewID != 0 {
2020-12-16 12:54:58 +03:00
headCommitID , err := gitRepo . GetRefCommitID ( pr . GetGitRefName ( ) )
if err != nil {
return nil , fmt . Errorf ( "GetRefCommitID[%s]: %v" , pr . GetGitRefName ( ) , err )
}
2020-11-09 09:15:09 +03:00
if len ( commitID ) == 0 {
2020-12-16 12:54:58 +03:00
commitID = headCommitID
2019-11-14 05:57:36 +03:00
}
2021-02-27 21:46:14 +03:00
reader , writer := io . Pipe ( )
defer func ( ) {
_ = reader . Close ( )
_ = writer . Close ( )
} ( )
go func ( ) {
if err := git . GetRepoRawDiffForFile ( gitRepo , pr . MergeBase , headCommitID , git . RawDiffNormal , treePath , writer ) ; err != nil {
_ = writer . CloseWithError ( fmt . Errorf ( "GetRawDiffForLine[%s, %s, %s, %s]: %v" , gitRepo . Path , pr . MergeBase , headCommitID , treePath , err ) )
return
}
_ = writer . Close ( )
} ( )
2022-06-13 12:37:59 +03:00
patch , err = git . CutDiffAroundLine ( reader , int64 ( ( & issues_model . Comment { Line : line } ) . UnsignedLine ( ) ) , line < 0 , setting . UI . CodeCommentLines )
2021-02-27 21:46:14 +03:00
if err != nil {
log . Error ( "Error whilst generating patch: %v" , err )
return nil , err
2019-11-14 05:57:36 +03:00
}
}
2022-06-13 12:37:59 +03:00
return issues_model . CreateComment ( & issues_model . CreateCommentOptions {
Type : issues_model . CommentTypeCode ,
2020-11-09 09:15:09 +03:00
Doer : doer ,
Repo : repo ,
Issue : issue ,
Content : content ,
LineNum : line ,
TreePath : treePath ,
CommitSHA : commitID ,
ReviewID : reviewID ,
Patch : patch ,
Invalidated : invalidated ,
2019-11-14 05:57:36 +03:00
} )
2019-10-18 11:33:19 +03:00
}
2019-11-14 05:57:36 +03:00
// SubmitReview creates a review out of the existing pending review or creates a new one if no pending review exist
2022-06-13 12:37:59 +03:00
func SubmitReview ( ctx context . Context , doer * user_model . User , gitRepo * git . Repository , issue * issues_model . Issue , reviewType issues_model . ReviewType , content , commitID string , attachmentUUIDs [ ] string ) ( * issues_model . Review , * issues_model . Comment , error ) {
2020-01-09 04:47:45 +03:00
pr , err := issue . GetPullRequest ( )
2019-10-18 11:33:19 +03:00
if err != nil {
2019-11-14 05:57:36 +03:00
return nil , nil , err
2019-10-18 11:33:19 +03:00
}
2020-01-09 04:47:45 +03:00
var stale bool
2022-06-13 12:37:59 +03:00
if reviewType != issues_model . ReviewTypeApprove && reviewType != issues_model . ReviewTypeReject {
2020-01-09 04:47:45 +03:00
stale = false
} else {
headCommitID , err := gitRepo . GetRefCommitID ( pr . GetGitRefName ( ) )
if err != nil {
return nil , nil , err
}
if headCommitID == commitID {
stale = false
} else {
2022-01-20 02:26:57 +03:00
stale , err = checkIfPRContentChanged ( ctx , pr , commitID , headCommitID )
2020-01-09 04:47:45 +03:00
if err != nil {
return nil , nil , err
}
}
}
2022-06-13 12:37:59 +03:00
review , comm , err := issues_model . SubmitReview ( doer , issue , reviewType , content , commitID , stale , attachmentUUIDs )
2019-11-14 05:57:36 +03:00
if err != nil {
return nil , nil , err
}
2020-01-09 04:47:45 +03:00
2022-06-13 12:37:59 +03:00
mentions , err := issues_model . FindAndUpdateIssueMentions ( ctx , issue , doer , comm . Content )
2021-01-02 20:04:02 +03:00
if err != nil {
return nil , nil , err
}
notification . NotifyPullRequestReview ( pr , review , comm , mentions )
for _ , lines := range review . CodeComments {
for _ , comments := range lines {
for _ , codeComment := range comments {
2022-06-13 12:37:59 +03:00
mentions , err := issues_model . FindAndUpdateIssueMentions ( ctx , issue , doer , codeComment . Content )
2021-01-02 20:04:02 +03:00
if err != nil {
return nil , nil , err
}
notification . NotifyPullRequestCodeComment ( pr , codeComment , mentions )
}
}
}
2019-08-14 18:32:19 +03:00
2019-11-14 05:57:36 +03:00
return review , comm , nil
2019-08-14 18:32:19 +03:00
}
2021-02-11 20:32:25 +03:00
// DismissReview dismissing stale review by repo admin
2022-07-19 16:20:28 +03:00
func DismissReview ( ctx context . Context , reviewID , repoID int64 , message string , doer * user_model . User , isDismiss , dismissPriors bool ) ( comment * issues_model . Comment , err error ) {
2022-06-13 12:37:59 +03:00
review , err := issues_model . GetReviewByID ( ctx , reviewID )
2021-02-11 20:32:25 +03:00
if err != nil {
return
}
2022-06-13 12:37:59 +03:00
if review . Type != issues_model . ReviewTypeApprove && review . Type != issues_model . ReviewTypeReject {
2021-02-11 20:32:25 +03:00
return nil , fmt . Errorf ( "not need to dismiss this review because it's type is not Approve or change request" )
}
2022-06-30 18:55:08 +03:00
// load data for notify
if err = review . LoadAttributes ( ctx ) ; err != nil {
return nil , err
}
// Check if the review's repoID is the one we're currently expecting.
if review . Issue . RepoID != repoID {
return nil , fmt . Errorf ( "reviews's repository is not the same as the one we expect" )
}
2022-06-13 12:37:59 +03:00
if err = issues_model . DismissReview ( review , isDismiss ) ; err != nil {
2021-02-11 20:32:25 +03:00
return
}
2022-07-19 16:20:28 +03:00
if dismissPriors {
reviews , err := issues_model . GetReviews ( ctx , & issues_model . GetReviewOptions {
IssueID : review . IssueID ,
ReviewerID : review . ReviewerID ,
Dismissed : util . OptionalBoolFalse ,
} )
if err != nil {
return nil , err
}
for _ , oldReview := range reviews {
if err = issues_model . DismissReview ( oldReview , true ) ; err != nil {
return nil , err
}
}
}
2021-02-11 20:32:25 +03:00
if ! isDismiss {
return nil , nil
}
if err = review . Issue . LoadPullRequest ( ) ; err != nil {
return
}
2022-06-13 12:37:59 +03:00
if err = review . Issue . LoadAttributes ( ctx ) ; err != nil {
2021-02-11 20:32:25 +03:00
return
}
2022-06-13 12:37:59 +03:00
comment , err = issues_model . CreateComment ( & issues_model . CreateCommentOptions {
2021-02-11 20:32:25 +03:00
Doer : doer ,
Content : message ,
2022-06-13 12:37:59 +03:00
Type : issues_model . CommentTypeDismissReview ,
2021-02-11 20:32:25 +03:00
ReviewID : review . ID ,
Issue : review . Issue ,
Repo : review . Issue . Repo ,
} )
if err != nil {
return
}
comment . Review = review
comment . Poster = doer
comment . Issue = review . Issue
notification . NotifyPullRevieweDismiss ( doer , review , comment )
2022-06-20 13:02:49 +03:00
return comment , err
2021-02-11 20:32:25 +03:00
}