2018-08-06 07:43:22 +03:00
// Copyright 2018 The Gitea Authors. All rights reserved.
2022-11-27 21:20:29 +03:00
// SPDX-License-Identifier: MIT
2018-08-06 07:43:22 +03:00
package repo
import (
2022-06-30 18:55:08 +03:00
"errors"
2018-08-06 07:43:22 +03:00
"fmt"
2021-04-05 18:30:52 +03:00
"net/http"
2018-08-06 07:43:22 +03:00
2022-06-13 12:37:59 +03:00
issues_model "code.gitea.io/gitea/models/issues"
2024-11-11 07:28:54 +03:00
"code.gitea.io/gitea/models/organization"
2022-05-07 21:28:10 +03:00
pull_model "code.gitea.io/gitea/models/pull"
2024-03-06 08:09:38 +03:00
user_model "code.gitea.io/gitea/models/user"
2021-01-09 00:49:55 +03:00
"code.gitea.io/gitea/modules/base"
2022-05-07 21:28:10 +03:00
"code.gitea.io/gitea/modules/json"
2018-08-06 07:43:22 +03:00
"code.gitea.io/gitea/modules/log"
2021-06-15 04:12:33 +03:00
"code.gitea.io/gitea/modules/setting"
2021-01-26 18:36:53 +03:00
"code.gitea.io/gitea/modules/web"
2024-02-27 10:12:22 +03:00
"code.gitea.io/gitea/services/context"
"code.gitea.io/gitea/services/context/upload"
2021-04-06 22:44:05 +03:00
"code.gitea.io/gitea/services/forms"
2024-11-11 07:28:54 +03:00
issue_service "code.gitea.io/gitea/services/issue"
2019-09-27 03:22:36 +03:00
pull_service "code.gitea.io/gitea/services/pull"
2024-03-06 08:09:38 +03:00
user_service "code.gitea.io/gitea/services/user"
2018-08-06 07:43:22 +03:00
)
2021-01-09 00:49:55 +03:00
const (
2024-01-24 06:26:28 +03:00
tplDiffConversation base . TplName = "repo/diff/conversation"
2024-02-08 04:50:48 +03:00
tplConversationOutdated base . TplName = "repo/diff/conversation_outdated"
2024-01-24 06:26:28 +03:00
tplTimelineConversation base . TplName = "repo/issue/view_content/conversation"
tplNewComment base . TplName = "repo/diff/new_comment"
2021-01-09 00:49:55 +03:00
)
// RenderNewCodeCommentForm will render the form for creating a new review comment
func RenderNewCodeCommentForm ( ctx * context . Context ) {
issue := GetActionIssue ( ctx )
2023-07-05 21:52:12 +03:00
if ctx . Written ( ) {
return
}
2021-01-09 00:49:55 +03:00
if ! issue . IsPull {
return
}
2022-06-13 12:37:59 +03:00
currentReview , err := issues_model . GetCurrentReview ( ctx , ctx . Doer , issue )
if err != nil && ! issues_model . IsErrReviewNotExist ( err ) {
2021-01-09 00:49:55 +03:00
ctx . ServerError ( "GetCurrentReview" , err )
return
}
ctx . Data [ "PageIsPullFiles" ] = true
ctx . Data [ "Issue" ] = issue
ctx . Data [ "CurrentReview" ] = currentReview
pullHeadCommitID , err := ctx . Repo . GitRepo . GetRefCommitID ( issue . PullRequest . GetGitRefName ( ) )
if err != nil {
ctx . ServerError ( "GetRefCommitID" , err )
return
}
ctx . Data [ "AfterCommitID" ] = pullHeadCommitID
2024-02-25 09:00:55 +03:00
ctx . Data [ "IsAttachmentEnabled" ] = setting . Attachment . Enabled
upload . AddUploadContext ( ctx , "comment" )
2021-04-05 18:30:52 +03:00
ctx . HTML ( http . StatusOK , tplNewComment )
2021-01-09 00:49:55 +03:00
}
2018-08-06 07:43:22 +03:00
// CreateCodeComment will create a code comment including an pending review if required
2021-01-26 18:36:53 +03:00
func CreateCodeComment ( ctx * context . Context ) {
2021-04-06 22:44:05 +03:00
form := web . GetForm ( ctx ) . ( * forms . CodeCommentForm )
2018-08-06 07:43:22 +03:00
issue := GetActionIssue ( ctx )
2023-07-05 21:52:12 +03:00
if ctx . Written ( ) {
2018-08-06 07:43:22 +03:00
return
}
2023-07-05 21:52:12 +03:00
if ! issue . IsPull {
2018-08-06 07:43:22 +03:00
return
}
if ctx . HasError ( ) {
ctx . Flash . Error ( ctx . Data [ "ErrorMsg" ] . ( string ) )
ctx . Redirect ( fmt . Sprintf ( "%s/pulls/%d/files" , ctx . Repo . RepoLink , issue . Index ) )
return
}
2019-11-14 05:57:36 +03:00
2018-08-06 07:43:22 +03:00
signedLine := form . Line
if form . Side == "previous" {
signedLine *= - 1
}
2024-02-25 09:00:55 +03:00
var attachments [ ] string
if setting . Attachment . Enabled {
attachments = form . Files
}
2022-01-20 02:26:57 +03:00
comment , err := pull_service . CreateCodeComment ( ctx ,
2022-03-22 10:03:22 +03:00
ctx . Doer ,
2020-01-09 04:47:45 +03:00
ctx . Repo . GitRepo ,
2018-08-06 07:43:22 +03:00
issue ,
2019-11-14 05:57:36 +03:00
signedLine ,
2018-08-06 07:43:22 +03:00
form . Content ,
form . TreePath ,
2023-03-04 10:13:37 +03:00
! form . SingleReview ,
2019-11-14 05:57:36 +03:00
form . Reply ,
2020-01-09 04:47:45 +03:00
form . LatestCommitID ,
2024-02-25 09:00:55 +03:00
attachments ,
2018-08-06 07:43:22 +03:00
)
if err != nil {
ctx . ServerError ( "CreateCodeComment" , err )
return
}
2020-03-30 21:52:45 +03:00
if comment == nil {
log . Trace ( "Comment not created: %-v #%d[%d]" , ctx . Repo . Repository , issue . Index , issue . ID )
2019-11-14 05:57:36 +03:00
ctx . Redirect ( fmt . Sprintf ( "%s/pulls/%d/files" , ctx . Repo . RepoLink , issue . Index ) )
2020-03-30 21:52:45 +03:00
return
2019-11-14 05:57:36 +03:00
}
2020-03-30 21:52:45 +03:00
log . Trace ( "Comment created: %-v #%d[%d] Comment[%d]" , ctx . Repo . Repository , issue . Index , issue . ID , comment . ID )
2021-01-09 00:49:55 +03:00
2024-01-24 06:26:28 +03:00
renderConversation ( ctx , comment , form . Origin )
2018-08-06 07:43:22 +03:00
}
2020-04-18 16:50:25 +03:00
// UpdateResolveConversation add or remove an Conversation resolved mark
func UpdateResolveConversation ( ctx * context . Context ) {
2021-08-11 03:31:13 +03:00
origin := ctx . FormString ( "origin" )
action := ctx . FormString ( "action" )
2021-07-29 04:42:15 +03:00
commentID := ctx . FormInt64 ( "comment_id" )
2020-04-18 16:50:25 +03:00
2022-06-13 12:37:59 +03:00
comment , err := issues_model . GetCommentByID ( ctx , commentID )
2020-04-18 16:50:25 +03:00
if err != nil {
ctx . ServerError ( "GetIssueByID" , err )
return
}
2022-11-19 11:12:33 +03:00
if err = comment . LoadIssue ( ctx ) ; err != nil {
2020-04-18 16:50:25 +03:00
ctx . ServerError ( "comment.LoadIssue" , err )
return
}
2022-06-30 18:55:08 +03:00
if comment . Issue . RepoID != ctx . Repo . Repository . ID {
ctx . NotFound ( "comment's repoID is incorrect" , errors . New ( "comment's repoID is incorrect" ) )
return
}
2020-04-18 16:50:25 +03:00
var permResult bool
2023-09-29 15:12:54 +03:00
if permResult , err = issues_model . CanMarkConversation ( ctx , comment . Issue , ctx . Doer ) ; err != nil {
2020-04-18 16:50:25 +03:00
ctx . ServerError ( "CanMarkConversation" , err )
return
}
if ! permResult {
2021-04-05 18:30:52 +03:00
ctx . Error ( http . StatusForbidden )
2020-04-18 16:50:25 +03:00
return
}
if ! comment . Issue . IsPull {
2021-04-05 18:30:52 +03:00
ctx . Error ( http . StatusBadRequest )
2020-04-18 16:50:25 +03:00
return
}
if action == "Resolve" || action == "UnResolve" {
2023-09-29 15:12:54 +03:00
err = issues_model . MarkConversation ( ctx , comment , ctx . Doer , action == "Resolve" )
2020-04-18 16:50:25 +03:00
if err != nil {
ctx . ServerError ( "MarkConversation" , err )
return
}
} else {
2021-04-05 18:30:52 +03:00
ctx . Error ( http . StatusBadRequest )
2020-04-18 16:50:25 +03:00
return
}
2024-01-24 06:26:28 +03:00
renderConversation ( ctx , comment , origin )
2020-04-18 16:50:25 +03:00
}
2024-01-24 06:26:28 +03:00
func renderConversation ( ctx * context . Context , comment * issues_model . Comment , origin string ) {
2024-02-04 16:05:01 +03:00
ctx . Data [ "PageIsPullFiles" ] = origin == "diff"
2024-02-17 17:07:56 +03:00
showOutdatedComments := origin == "timeline" || ctx . Data [ "ShowOutdatedComments" ] . ( bool )
comments , err := issues_model . FetchCodeCommentsByLine ( ctx , comment . Issue , ctx . Doer , comment . TreePath , comment . Line , showOutdatedComments )
2021-01-09 00:49:55 +03:00
if err != nil {
ctx . ServerError ( "FetchCodeCommentsByLine" , err )
return
}
2024-02-04 16:05:01 +03:00
if len ( comments ) == 0 {
2024-02-08 04:50:48 +03:00
// if the comments are empty (deleted, outdated, etc), it's better to tell the users that it is outdated
ctx . HTML ( http . StatusOK , tplConversationOutdated )
2024-02-04 16:05:01 +03:00
return
}
2024-03-12 10:23:44 +03:00
if err := comments . LoadAttachments ( ctx ) ; err != nil {
ctx . ServerError ( "LoadAttachments" , err )
return
2024-02-25 09:00:55 +03:00
}
ctx . Data [ "IsAttachmentEnabled" ] = setting . Attachment . Enabled
upload . AddUploadContext ( ctx , "comment" )
2021-01-09 00:49:55 +03:00
ctx . Data [ "comments" ] = comments
2024-01-24 06:26:28 +03:00
if ctx . Data [ "CanMarkConversation" ] , err = issues_model . CanMarkConversation ( ctx , comment . Issue , ctx . Doer ) ; err != nil {
ctx . ServerError ( "CanMarkConversation" , err )
return
}
2021-01-09 00:49:55 +03:00
ctx . Data [ "Issue" ] = comment . Issue
2022-11-19 11:12:33 +03:00
if err = comment . Issue . LoadPullRequest ( ctx ) ; err != nil {
2021-01-09 00:49:55 +03:00
ctx . ServerError ( "comment.Issue.LoadPullRequest" , err )
return
}
pullHeadCommitID , err := ctx . Repo . GitRepo . GetRefCommitID ( comment . Issue . PullRequest . GetGitRefName ( ) )
if err != nil {
ctx . ServerError ( "GetRefCommitID" , err )
return
}
ctx . Data [ "AfterCommitID" ] = pullHeadCommitID
2024-03-06 08:09:38 +03:00
ctx . Data [ "CanBlockUser" ] = func ( blocker , blockee * user_model . User ) bool {
return user_service . CanBlockUser ( ctx , ctx . Doer , blocker , blockee )
}
2024-01-24 06:26:28 +03:00
if origin == "diff" {
ctx . HTML ( http . StatusOK , tplDiffConversation )
} else if origin == "timeline" {
ctx . HTML ( http . StatusOK , tplTimelineConversation )
2024-02-04 16:05:01 +03:00
} else {
ctx . Error ( http . StatusBadRequest , "Unknown origin: " + origin )
2024-01-24 06:26:28 +03:00
}
2021-01-09 00:49:55 +03:00
}
2018-08-06 07:43:22 +03:00
// SubmitReview creates a review out of the existing pending review or creates a new one if no pending review exist
2021-01-26 18:36:53 +03:00
func SubmitReview ( ctx * context . Context ) {
2021-04-06 22:44:05 +03:00
form := web . GetForm ( ctx ) . ( * forms . SubmitReviewForm )
2018-08-06 07:43:22 +03:00
issue := GetActionIssue ( ctx )
2023-07-05 21:52:12 +03:00
if ctx . Written ( ) {
2018-08-06 07:43:22 +03:00
return
}
2023-07-05 21:52:12 +03:00
if ! issue . IsPull {
2018-08-06 07:43:22 +03:00
return
}
if ctx . HasError ( ) {
ctx . Flash . Error ( ctx . Data [ "ErrorMsg" ] . ( string ) )
2023-06-14 11:01:37 +03:00
ctx . JSONRedirect ( fmt . Sprintf ( "%s/pulls/%d/files" , ctx . Repo . RepoLink , issue . Index ) )
2018-08-06 07:43:22 +03:00
return
}
reviewType := form . ReviewType ( )
2018-08-20 08:04:01 +03:00
switch reviewType {
2022-06-13 12:37:59 +03:00
case issues_model . ReviewTypeUnknown :
2019-11-14 05:57:36 +03:00
ctx . ServerError ( "ReviewType" , fmt . Errorf ( "unknown ReviewType: %s" , form . Type ) )
2018-08-06 07:43:22 +03:00
return
2018-08-20 08:04:01 +03:00
// can not approve/reject your own PR
2022-06-13 12:37:59 +03:00
case issues_model . ReviewTypeApprove , issues_model . ReviewTypeReject :
2022-03-22 10:03:22 +03:00
if issue . IsPoster ( ctx . Doer . ID ) {
2018-08-20 08:04:01 +03:00
var translated string
2022-06-13 12:37:59 +03:00
if reviewType == issues_model . ReviewTypeApprove {
2024-02-15 00:48:45 +03:00
translated = ctx . Locale . TrString ( "repo.issues.review.self.approval" )
2018-08-20 08:04:01 +03:00
} else {
2024-02-15 00:48:45 +03:00
translated = ctx . Locale . TrString ( "repo.issues.review.self.rejection" )
2018-08-20 08:04:01 +03:00
}
ctx . Flash . Error ( translated )
2023-06-14 11:01:37 +03:00
ctx . JSONRedirect ( fmt . Sprintf ( "%s/pulls/%d/files" , ctx . Repo . RepoLink , issue . Index ) )
2018-08-20 08:04:01 +03:00
return
}
2018-08-06 07:43:22 +03:00
}
2018-08-07 20:15:41 +03:00
2021-06-15 04:12:33 +03:00
var attachments [ ] string
if setting . Attachment . Enabled {
attachments = form . Files
}
2022-03-22 10:03:22 +03:00
_ , comm , err := pull_service . SubmitReview ( ctx , ctx . Doer , ctx . Repo . GitRepo , issue , reviewType , form . Content , form . CommitID , attachments )
2018-08-06 07:43:22 +03:00
if err != nil {
2022-06-13 12:37:59 +03:00
if issues_model . IsContentEmptyErr ( err ) {
2019-11-14 05:57:36 +03:00
ctx . Flash . Error ( ctx . Tr ( "repo.issues.review.content.empty" ) )
2023-06-14 11:01:37 +03:00
ctx . JSONRedirect ( fmt . Sprintf ( "%s/pulls/%d/files" , ctx . Repo . RepoLink , issue . Index ) )
2024-04-27 14:55:03 +03:00
} else if errors . Is ( err , pull_service . ErrSubmitReviewOnClosedPR ) {
ctx . Status ( http . StatusUnprocessableEntity )
2019-11-14 05:57:36 +03:00
} else {
ctx . ServerError ( "SubmitReview" , err )
2018-08-06 07:43:22 +03:00
}
2018-10-18 14:23:05 +03:00
return
}
2023-06-14 11:01:37 +03:00
ctx . JSONRedirect ( fmt . Sprintf ( "%s/pulls/%d#%s" , ctx . Repo . RepoLink , issue . Index , comm . HashTag ( ) ) )
2018-08-06 07:43:22 +03:00
}
2021-02-11 20:32:25 +03:00
// DismissReview dismissing stale review by repo admin
func DismissReview ( ctx * context . Context ) {
2021-04-06 22:44:05 +03:00
form := web . GetForm ( ctx ) . ( * forms . DismissReviewForm )
2022-07-19 16:20:28 +03:00
comm , err := pull_service . DismissReview ( ctx , form . ReviewID , ctx . Repo . Repository . ID , form . Message , ctx . Doer , true , true )
2021-02-11 20:32:25 +03:00
if err != nil {
2024-03-28 18:19:24 +03:00
if pull_service . IsErrDismissRequestOnClosedPR ( err ) {
ctx . Status ( http . StatusForbidden )
return
}
2021-02-11 20:32:25 +03:00
ctx . ServerError ( "pull_service.DismissReview" , err )
return
}
ctx . Redirect ( fmt . Sprintf ( "%s/pulls/%d#%s" , ctx . Repo . RepoLink , comm . Issue . Index , comm . HashTag ( ) ) )
}
2022-05-07 21:28:10 +03:00
// viewedFilesUpdate Struct to parse the body of a request to update the reviewed files of a PR
// If you want to implement an API to update the review, simply move this struct into modules.
type viewedFilesUpdate struct {
Files map [ string ] bool ` json:"files" `
HeadCommitSHA string ` json:"headCommitSHA" `
}
func UpdateViewedFiles ( ctx * context . Context ) {
// Find corresponding PR
2023-08-07 06:43:18 +03:00
issue , ok := getPullInfo ( ctx )
if ! ok {
2022-05-07 21:28:10 +03:00
return
}
pull := issue . PullRequest
var data * viewedFilesUpdate
err := json . NewDecoder ( ctx . Req . Body ) . Decode ( & data )
if err != nil {
log . Warn ( "Attempted to update a review but could not parse request body: %v" , err )
ctx . Resp . WriteHeader ( http . StatusBadRequest )
return
}
// Expect the review to have been now if no head commit was supplied
if data . HeadCommitSHA == "" {
data . HeadCommitSHA = pull . HeadCommitID
}
updatedFiles := make ( map [ string ] pull_model . ViewedState , len ( data . Files ) )
for file , viewed := range data . Files {
// Only unviewed and viewed are possible, has-changed can not be set from the outside
state := pull_model . Unviewed
if viewed {
state = pull_model . Viewed
}
updatedFiles [ file ] = state
}
if err := pull_model . UpdateReviewState ( ctx , ctx . Doer . ID , pull . ID , data . HeadCommitSHA , updatedFiles ) ; err != nil {
ctx . ServerError ( "UpdateReview" , err )
}
}
2024-11-11 07:28:54 +03:00
// UpdatePullReviewRequest add or remove review request
func UpdatePullReviewRequest ( ctx * context . Context ) {
issues := getActionIssues ( ctx )
if ctx . Written ( ) {
return
}
reviewID := ctx . FormInt64 ( "id" )
action := ctx . FormString ( "action" )
// TODO: Not support 'clear' now
if action != "attach" && action != "detach" {
ctx . Status ( http . StatusForbidden )
return
}
for _ , issue := range issues {
if err := issue . LoadRepo ( ctx ) ; err != nil {
ctx . ServerError ( "issue.LoadRepo" , err )
return
}
if ! issue . IsPull {
log . Warn (
"UpdatePullReviewRequest: refusing to add review request for non-PR issue %-v#%d" ,
issue . Repo , issue . Index ,
)
ctx . Status ( http . StatusForbidden )
return
}
if reviewID < 0 {
// negative reviewIDs represent team requests
if err := issue . Repo . LoadOwner ( ctx ) ; err != nil {
ctx . ServerError ( "issue.Repo.LoadOwner" , err )
return
}
if ! issue . Repo . Owner . IsOrganization ( ) {
log . Warn (
"UpdatePullReviewRequest: refusing to add team review request for %s#%d owned by non organization UID[%d]" ,
issue . Repo . FullName ( ) , issue . Index , issue . Repo . ID ,
)
ctx . Status ( http . StatusForbidden )
return
}
team , err := organization . GetTeamByID ( ctx , - reviewID )
if err != nil {
ctx . ServerError ( "GetTeamByID" , err )
return
}
if team . OrgID != issue . Repo . OwnerID {
log . Warn (
"UpdatePullReviewRequest: refusing to add team review request for UID[%d] team %s to %s#%d owned by UID[%d]" ,
team . OrgID , team . Name , issue . Repo . FullName ( ) , issue . Index , issue . Repo . ID )
ctx . Status ( http . StatusForbidden )
return
}
_ , err = issue_service . TeamReviewRequest ( ctx , issue , ctx . Doer , team , action == "attach" )
if err != nil {
if issues_model . IsErrNotValidReviewRequest ( err ) {
log . Warn (
"UpdatePullReviewRequest: refusing to add invalid team review request for UID[%d] team %s to %s#%d owned by UID[%d]: Error: %v" ,
team . OrgID , team . Name , issue . Repo . FullName ( ) , issue . Index , issue . Repo . ID ,
err ,
)
ctx . Status ( http . StatusForbidden )
return
}
ctx . ServerError ( "TeamReviewRequest" , err )
return
}
continue
}
reviewer , err := user_model . GetUserByID ( ctx , reviewID )
if err != nil {
if user_model . IsErrUserNotExist ( err ) {
log . Warn (
"UpdatePullReviewRequest: requested reviewer [%d] for %-v to %-v#%d is not exist: Error: %v" ,
reviewID , issue . Repo , issue . Index ,
err ,
)
ctx . Status ( http . StatusForbidden )
return
}
ctx . ServerError ( "GetUserByID" , err )
return
}
_ , err = issue_service . ReviewRequest ( ctx , issue , ctx . Doer , & ctx . Repo . Permission , reviewer , action == "attach" )
if err != nil {
if issues_model . IsErrNotValidReviewRequest ( err ) {
log . Warn (
"UpdatePullReviewRequest: refusing to add invalid review request for %-v to %-v#%d: Error: %v" ,
reviewer , issue . Repo , issue . Index ,
err ,
)
ctx . Status ( http . StatusForbidden )
return
}
if issues_model . IsErrReviewRequestOnClosedPR ( err ) {
ctx . Status ( http . StatusForbidden )
return
}
ctx . ServerError ( "ReviewRequest" , err )
return
}
}
ctx . JSONOK ( )
}