2018-08-06 07:43:22 +03:00
// Copyright 2018 The Gitea Authors. All rights reserved.
2022-11-27 13:20:29 -05:00
// SPDX-License-Identifier: MIT
2018-08-06 07:43:22 +03:00
package repo
import (
2022-06-30 23:55:08 +08:00
"errors"
2018-08-06 07:43:22 +03:00
"fmt"
2021-04-05 17:30:52 +02:00
"net/http"
2018-08-06 07:43:22 +03:00
2022-06-13 17:37:59 +08:00
issues_model "code.gitea.io/gitea/models/issues"
2022-05-07 20:28:10 +02:00
pull_model "code.gitea.io/gitea/models/pull"
2021-01-08 22:49:55 +01:00
"code.gitea.io/gitea/modules/base"
2018-08-06 07:43:22 +03:00
"code.gitea.io/gitea/modules/context"
2022-05-07 20:28:10 +02:00
"code.gitea.io/gitea/modules/json"
2018-08-06 07:43:22 +03:00
"code.gitea.io/gitea/modules/log"
2021-06-15 03:12:33 +02:00
"code.gitea.io/gitea/modules/setting"
2021-01-26 23:36:53 +08:00
"code.gitea.io/gitea/modules/web"
2021-04-06 20:44:05 +01:00
"code.gitea.io/gitea/services/forms"
2019-09-27 08:22:36 +08:00
pull_service "code.gitea.io/gitea/services/pull"
2018-08-06 07:43:22 +03:00
)
2021-01-08 22:49:55 +01:00
const (
2024-01-24 04:26:28 +01:00
tplDiffConversation base . TplName = "repo/diff/conversation"
tplTimelineConversation base . TplName = "repo/issue/view_content/conversation"
tplNewComment base . TplName = "repo/diff/new_comment"
2021-01-08 22:49:55 +01:00
)
// RenderNewCodeCommentForm will render the form for creating a new review comment
func RenderNewCodeCommentForm ( ctx * context . Context ) {
issue := GetActionIssue ( ctx )
2023-07-06 02:52:12 +08:00
if ctx . Written ( ) {
return
}
2021-01-08 22:49:55 +01:00
if ! issue . IsPull {
return
}
2022-06-13 17:37:59 +08:00
currentReview , err := issues_model . GetCurrentReview ( ctx , ctx . Doer , issue )
if err != nil && ! issues_model . IsErrReviewNotExist ( err ) {
2021-01-08 22:49:55 +01: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
2021-04-05 17:30:52 +02:00
ctx . HTML ( http . StatusOK , tplNewComment )
2021-01-08 22:49:55 +01:00
}
2018-08-06 07:43:22 +03:00
// CreateCodeComment will create a code comment including an pending review if required
2021-01-26 23:36:53 +08:00
func CreateCodeComment ( ctx * context . Context ) {
2021-04-06 20:44:05 +01:00
form := web . GetForm ( ctx ) . ( * forms . CodeCommentForm )
2018-08-06 07:43:22 +03:00
issue := GetActionIssue ( ctx )
2023-07-06 02:52:12 +08:00
if ctx . Written ( ) {
2018-08-06 07:43:22 +03:00
return
}
2023-07-06 02:52:12 +08: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 10:57:36 +08:00
2018-08-06 07:43:22 +03:00
signedLine := form . Line
if form . Side == "previous" {
signedLine *= - 1
}
2022-01-19 23:26:57 +00:00
comment , err := pull_service . CreateCodeComment ( ctx ,
2022-03-22 08:03:22 +01:00
ctx . Doer ,
2020-01-09 02:47:45 +01:00
ctx . Repo . GitRepo ,
2018-08-06 07:43:22 +03:00
issue ,
2019-11-14 10:57:36 +08:00
signedLine ,
2018-08-06 07:43:22 +03:00
form . Content ,
form . TreePath ,
2023-03-04 15:13:37 +08:00
! form . SingleReview ,
2019-11-14 10:57:36 +08:00
form . Reply ,
2020-01-09 02:47:45 +01:00
form . LatestCommitID ,
2018-08-06 07:43:22 +03:00
)
if err != nil {
ctx . ServerError ( "CreateCodeComment" , err )
return
}
2020-03-30 19:52:45 +01:00
if comment == nil {
log . Trace ( "Comment not created: %-v #%d[%d]" , ctx . Repo . Repository , issue . Index , issue . ID )
2019-11-14 10:57:36 +08:00
ctx . Redirect ( fmt . Sprintf ( "%s/pulls/%d/files" , ctx . Repo . RepoLink , issue . Index ) )
2020-03-30 19:52:45 +01:00
return
2019-11-14 10:57:36 +08:00
}
2020-03-30 19:52:45 +01:00
log . Trace ( "Comment created: %-v #%d[%d] Comment[%d]" , ctx . Repo . Repository , issue . Index , issue . ID , comment . ID )
2021-01-08 22:49:55 +01:00
2024-01-24 04:26:28 +01:00
renderConversation ( ctx , comment , form . Origin )
2018-08-06 07:43:22 +03:00
}
2020-04-18 21:50:25 +08:00
// UpdateResolveConversation add or remove an Conversation resolved mark
func UpdateResolveConversation ( ctx * context . Context ) {
2021-08-11 02:31:13 +02:00
origin := ctx . FormString ( "origin" )
action := ctx . FormString ( "action" )
2021-07-29 09:42:15 +08:00
commentID := ctx . FormInt64 ( "comment_id" )
2020-04-18 21:50:25 +08:00
2022-06-13 17:37:59 +08:00
comment , err := issues_model . GetCommentByID ( ctx , commentID )
2020-04-18 21:50:25 +08:00
if err != nil {
ctx . ServerError ( "GetIssueByID" , err )
return
}
2022-11-19 09:12:33 +01:00
if err = comment . LoadIssue ( ctx ) ; err != nil {
2020-04-18 21:50:25 +08:00
ctx . ServerError ( "comment.LoadIssue" , err )
return
}
2022-06-30 23:55:08 +08: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 21:50:25 +08:00
var permResult bool
2023-09-29 14:12:54 +02:00
if permResult , err = issues_model . CanMarkConversation ( ctx , comment . Issue , ctx . Doer ) ; err != nil {
2020-04-18 21:50:25 +08:00
ctx . ServerError ( "CanMarkConversation" , err )
return
}
if ! permResult {
2021-04-05 17:30:52 +02:00
ctx . Error ( http . StatusForbidden )
2020-04-18 21:50:25 +08:00
return
}
if ! comment . Issue . IsPull {
2021-04-05 17:30:52 +02:00
ctx . Error ( http . StatusBadRequest )
2020-04-18 21:50:25 +08:00
return
}
if action == "Resolve" || action == "UnResolve" {
2023-09-29 14:12:54 +02:00
err = issues_model . MarkConversation ( ctx , comment , ctx . Doer , action == "Resolve" )
2020-04-18 21:50:25 +08:00
if err != nil {
ctx . ServerError ( "MarkConversation" , err )
return
}
} else {
2021-04-05 17:30:52 +02:00
ctx . Error ( http . StatusBadRequest )
2020-04-18 21:50:25 +08:00
return
}
2024-01-24 04:26:28 +01:00
renderConversation ( ctx , comment , origin )
2020-04-18 21:50:25 +08:00
}
2024-01-24 04:26:28 +01:00
func renderConversation ( ctx * context . Context , comment * issues_model . Comment , origin string ) {
2024-02-04 21:05:01 +08:00
ctx . Data [ "PageIsPullFiles" ] = origin == "diff"
2023-06-21 18:08:12 +02:00
comments , err := issues_model . FetchCodeCommentsByLine ( ctx , comment . Issue , ctx . Doer , comment . TreePath , comment . Line , ctx . Data [ "ShowOutdatedComments" ] . ( bool ) )
2021-01-08 22:49:55 +01:00
if err != nil {
ctx . ServerError ( "FetchCodeCommentsByLine" , err )
return
}
2024-02-04 21:05:01 +08:00
if len ( comments ) == 0 {
// if the comments are empty (deleted, outdated, etc), it doesn't need to render anything, just return an empty body to replace "conversation-holder" on the page
ctx . Resp . WriteHeader ( http . StatusOK )
return
}
2021-01-08 22:49:55 +01:00
ctx . Data [ "comments" ] = comments
2024-01-24 04:26:28 +01:00
if ctx . Data [ "CanMarkConversation" ] , err = issues_model . CanMarkConversation ( ctx , comment . Issue , ctx . Doer ) ; err != nil {
ctx . ServerError ( "CanMarkConversation" , err )
return
}
2021-01-08 22:49:55 +01:00
ctx . Data [ "Issue" ] = comment . Issue
2022-11-19 09:12:33 +01:00
if err = comment . Issue . LoadPullRequest ( ctx ) ; err != nil {
2021-01-08 22:49:55 +01: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-01-24 04:26:28 +01:00
if origin == "diff" {
ctx . HTML ( http . StatusOK , tplDiffConversation )
} else if origin == "timeline" {
ctx . HTML ( http . StatusOK , tplTimelineConversation )
2024-02-04 21:05:01 +08:00
} else {
ctx . Error ( http . StatusBadRequest , "Unknown origin: " + origin )
2024-01-24 04:26:28 +01:00
}
2021-01-08 22:49:55 +01: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 23:36:53 +08:00
func SubmitReview ( ctx * context . Context ) {
2021-04-06 20:44:05 +01:00
form := web . GetForm ( ctx ) . ( * forms . SubmitReviewForm )
2018-08-06 07:43:22 +03:00
issue := GetActionIssue ( ctx )
2023-07-06 02:52:12 +08:00
if ctx . Written ( ) {
2018-08-06 07:43:22 +03:00
return
}
2023-07-06 02:52:12 +08: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 16:01:37 +08: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 06:04:01 +01:00
switch reviewType {
2022-06-13 17:37:59 +08:00
case issues_model . ReviewTypeUnknown :
2019-11-14 10:57:36 +08:00
ctx . ServerError ( "ReviewType" , fmt . Errorf ( "unknown ReviewType: %s" , form . Type ) )
2018-08-06 07:43:22 +03:00
return
2018-08-20 06:04:01 +01:00
// can not approve/reject your own PR
2022-06-13 17:37:59 +08:00
case issues_model . ReviewTypeApprove , issues_model . ReviewTypeReject :
2022-03-22 08:03:22 +01:00
if issue . IsPoster ( ctx . Doer . ID ) {
2018-08-20 06:04:01 +01:00
var translated string
2022-06-13 17:37:59 +08:00
if reviewType == issues_model . ReviewTypeApprove {
2018-08-20 06:04:01 +01:00
translated = ctx . Tr ( "repo.issues.review.self.approval" )
} else {
translated = ctx . Tr ( "repo.issues.review.self.rejection" )
}
ctx . Flash . Error ( translated )
2023-06-14 16:01:37 +08:00
ctx . JSONRedirect ( fmt . Sprintf ( "%s/pulls/%d/files" , ctx . Repo . RepoLink , issue . Index ) )
2018-08-20 06:04:01 +01:00
return
}
2018-08-06 07:43:22 +03:00
}
2018-08-07 18:15:41 +01:00
2021-06-15 03:12:33 +02:00
var attachments [ ] string
if setting . Attachment . Enabled {
attachments = form . Files
}
2022-03-22 08:03:22 +01: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 17:37:59 +08:00
if issues_model . IsContentEmptyErr ( err ) {
2019-11-14 10:57:36 +08:00
ctx . Flash . Error ( ctx . Tr ( "repo.issues.review.content.empty" ) )
2023-06-14 16:01:37 +08:00
ctx . JSONRedirect ( fmt . Sprintf ( "%s/pulls/%d/files" , ctx . Repo . RepoLink , issue . Index ) )
2019-11-14 10:57:36 +08:00
} else {
ctx . ServerError ( "SubmitReview" , err )
2018-08-06 07:43:22 +03:00
}
2018-10-18 19:23:05 +08:00
return
}
2023-06-14 16:01:37 +08: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-12 01:32:25 +08:00
// DismissReview dismissing stale review by repo admin
func DismissReview ( ctx * context . Context ) {
2021-04-06 20:44:05 +01:00
form := web . GetForm ( ctx ) . ( * forms . DismissReviewForm )
2022-07-19 15:20:28 +02:00
comm , err := pull_service . DismissReview ( ctx , form . ReviewID , ctx . Repo . Repository . ID , form . Message , ctx . Doer , true , true )
2021-02-12 01:32:25 +08:00
if err != nil {
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 20:28:10 +02: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 11:43:18 +08:00
issue , ok := getPullInfo ( ctx )
if ! ok {
2022-05-07 20:28:10 +02: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 )
}
}