mirror of
				https://gitee.com/gitea/gitea
				synced 2025-11-04 16:40:24 +08:00 
			
		
		
		
	API: Add pull review endpoints (#11224)
* API: Added pull review read only endpoints * Update Structs, move Conversion, Refactor * refactor * lint & co * fix lint + refactor * add new Review state, rm unessesary, refacotr loadAttributes, convert patch to diff * add DeletePullReview * add paggination * draft1: Create & submit review * fix lint * fix lint * impruve test * DONT use GhostUser for loadReviewer * expose comments_count of a PullReview * infent GetCodeCommentsCount() * fixes * fix+impruve * some nits * Handle Ghosts 👻 * add TEST for GET apis * complete TESTS * add HTMLURL to PullReview responce * code format as per @lafriks * update swagger definition * Update routers/api/v1/repo/pull_review.go Co-authored-by: David Svantesson <davidsvantesson@gmail.com> * add comments Co-authored-by: Thomas Berger <loki@lokis-chaos.de> Co-authored-by: David Svantesson <davidsvantesson@gmail.com>
This commit is contained in:
		
							
								
								
									
										120
									
								
								integrations/api_pull_review_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										120
									
								
								integrations/api_pull_review_test.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,120 @@
 | 
				
			|||||||
 | 
					// Copyright 2020 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 integrations
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"fmt"
 | 
				
			||||||
 | 
						"net/http"
 | 
				
			||||||
 | 
						"testing"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"code.gitea.io/gitea/models"
 | 
				
			||||||
 | 
						api "code.gitea.io/gitea/modules/structs"
 | 
				
			||||||
 | 
						"github.com/stretchr/testify/assert"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func TestAPIPullReview(t *testing.T) {
 | 
				
			||||||
 | 
						defer prepareTestEnv(t)()
 | 
				
			||||||
 | 
						pullIssue := models.AssertExistsAndLoadBean(t, &models.Issue{ID: 3}).(*models.Issue)
 | 
				
			||||||
 | 
						assert.NoError(t, pullIssue.LoadAttributes())
 | 
				
			||||||
 | 
						repo := models.AssertExistsAndLoadBean(t, &models.Repository{ID: pullIssue.RepoID}).(*models.Repository)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// test ListPullReviews
 | 
				
			||||||
 | 
						session := loginUser(t, "user2")
 | 
				
			||||||
 | 
						token := getTokenForLoggedInUser(t, session)
 | 
				
			||||||
 | 
						req := NewRequestf(t, http.MethodGet, "/api/v1/repos/%s/%s/pulls/%d/reviews?token=%s", repo.OwnerName, repo.Name, pullIssue.Index, token)
 | 
				
			||||||
 | 
						resp := session.MakeRequest(t, req, http.StatusOK)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						var reviews []*api.PullReview
 | 
				
			||||||
 | 
						DecodeJSON(t, resp, &reviews)
 | 
				
			||||||
 | 
						if !assert.Len(t, reviews, 6) {
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						for _, r := range reviews {
 | 
				
			||||||
 | 
							assert.EqualValues(t, pullIssue.HTMLURL(), r.HTMLPullURL)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						assert.EqualValues(t, 8, reviews[3].ID)
 | 
				
			||||||
 | 
						assert.EqualValues(t, "APPROVED", reviews[3].State)
 | 
				
			||||||
 | 
						assert.EqualValues(t, 0, reviews[3].CodeCommentsCount)
 | 
				
			||||||
 | 
						assert.EqualValues(t, true, reviews[3].Stale)
 | 
				
			||||||
 | 
						assert.EqualValues(t, false, reviews[3].Official)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						assert.EqualValues(t, 10, reviews[5].ID)
 | 
				
			||||||
 | 
						assert.EqualValues(t, "REQUEST_CHANGES", reviews[5].State)
 | 
				
			||||||
 | 
						assert.EqualValues(t, 1, reviews[5].CodeCommentsCount)
 | 
				
			||||||
 | 
						assert.EqualValues(t, 0, reviews[5].Reviewer.ID) // ghost user
 | 
				
			||||||
 | 
						assert.EqualValues(t, false, reviews[5].Stale)
 | 
				
			||||||
 | 
						assert.EqualValues(t, true, reviews[5].Official)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// test GetPullReview
 | 
				
			||||||
 | 
						req = NewRequestf(t, http.MethodGet, "/api/v1/repos/%s/%s/pulls/%d/reviews/%d?token=%s", repo.OwnerName, repo.Name, pullIssue.Index, reviews[3].ID, token)
 | 
				
			||||||
 | 
						resp = session.MakeRequest(t, req, http.StatusOK)
 | 
				
			||||||
 | 
						var review api.PullReview
 | 
				
			||||||
 | 
						DecodeJSON(t, resp, &review)
 | 
				
			||||||
 | 
						assert.EqualValues(t, *reviews[3], review)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						req = NewRequestf(t, "GET", "/api/v1/repos/%s/%s/pulls/%d/reviews/%d?token=%s", repo.OwnerName, repo.Name, pullIssue.Index, reviews[5].ID, token)
 | 
				
			||||||
 | 
						resp = session.MakeRequest(t, req, http.StatusOK)
 | 
				
			||||||
 | 
						DecodeJSON(t, resp, &review)
 | 
				
			||||||
 | 
						assert.EqualValues(t, *reviews[5], review)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// test GetPullReviewComments
 | 
				
			||||||
 | 
						comment := models.AssertExistsAndLoadBean(t, &models.Comment{ID: 7}).(*models.Comment)
 | 
				
			||||||
 | 
						req = NewRequestf(t, http.MethodGet, "/api/v1/repos/%s/%s/pulls/%d/reviews/%d/comments?token=%s", repo.OwnerName, repo.Name, pullIssue.Index, 10, token)
 | 
				
			||||||
 | 
						resp = session.MakeRequest(t, req, http.StatusOK)
 | 
				
			||||||
 | 
						var reviewComments []*api.PullReviewComment
 | 
				
			||||||
 | 
						DecodeJSON(t, resp, &reviewComments)
 | 
				
			||||||
 | 
						assert.Len(t, reviewComments, 1)
 | 
				
			||||||
 | 
						assert.EqualValues(t, "Ghost", reviewComments[0].Reviewer.UserName)
 | 
				
			||||||
 | 
						assert.EqualValues(t, "a review from a deleted user", reviewComments[0].Body)
 | 
				
			||||||
 | 
						assert.EqualValues(t, comment.ID, reviewComments[0].ID)
 | 
				
			||||||
 | 
						assert.EqualValues(t, comment.UpdatedUnix, reviewComments[0].Updated.Unix())
 | 
				
			||||||
 | 
						assert.EqualValues(t, comment.HTMLURL(), reviewComments[0].HTMLURL)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// test CreatePullReview
 | 
				
			||||||
 | 
						req = NewRequestWithJSON(t, http.MethodPost, fmt.Sprintf("/api/v1/repos/%s/%s/pulls/%d/reviews?token=%s", repo.OwnerName, repo.Name, pullIssue.Index, token), &api.CreatePullReviewOptions{
 | 
				
			||||||
 | 
							Body: "body1",
 | 
				
			||||||
 | 
							// Event: "" # will result in PENDING
 | 
				
			||||||
 | 
							Comments: []api.CreatePullReviewComment{{
 | 
				
			||||||
 | 
								Path:       "README.md",
 | 
				
			||||||
 | 
								Body:       "first new line",
 | 
				
			||||||
 | 
								OldLineNum: 0,
 | 
				
			||||||
 | 
								NewLineNum: 1,
 | 
				
			||||||
 | 
							}, {
 | 
				
			||||||
 | 
								Path:       "README.md",
 | 
				
			||||||
 | 
								Body:       "first old line",
 | 
				
			||||||
 | 
								OldLineNum: 1,
 | 
				
			||||||
 | 
								NewLineNum: 0,
 | 
				
			||||||
 | 
							},
 | 
				
			||||||
 | 
							},
 | 
				
			||||||
 | 
						})
 | 
				
			||||||
 | 
						resp = session.MakeRequest(t, req, http.StatusOK)
 | 
				
			||||||
 | 
						DecodeJSON(t, resp, &review)
 | 
				
			||||||
 | 
						assert.EqualValues(t, 6, review.ID)
 | 
				
			||||||
 | 
						assert.EqualValues(t, "PENDING", review.State)
 | 
				
			||||||
 | 
						assert.EqualValues(t, 2, review.CodeCommentsCount)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// test SubmitPullReview
 | 
				
			||||||
 | 
						req = NewRequestWithJSON(t, http.MethodPost, fmt.Sprintf("/api/v1/repos/%s/%s/pulls/%d/reviews/%d?token=%s", repo.OwnerName, repo.Name, pullIssue.Index, review.ID, token), &api.SubmitPullReviewOptions{
 | 
				
			||||||
 | 
							Event: "APPROVED",
 | 
				
			||||||
 | 
							Body:  "just two nits",
 | 
				
			||||||
 | 
						})
 | 
				
			||||||
 | 
						resp = session.MakeRequest(t, req, http.StatusOK)
 | 
				
			||||||
 | 
						DecodeJSON(t, resp, &review)
 | 
				
			||||||
 | 
						assert.EqualValues(t, 6, review.ID)
 | 
				
			||||||
 | 
						assert.EqualValues(t, "APPROVED", review.State)
 | 
				
			||||||
 | 
						assert.EqualValues(t, 2, review.CodeCommentsCount)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// test DeletePullReview
 | 
				
			||||||
 | 
						req = NewRequestWithJSON(t, http.MethodPost, fmt.Sprintf("/api/v1/repos/%s/%s/pulls/%d/reviews?token=%s", repo.OwnerName, repo.Name, pullIssue.Index, token), &api.CreatePullReviewOptions{
 | 
				
			||||||
 | 
							Body:  "just a comment",
 | 
				
			||||||
 | 
							Event: "COMMENT",
 | 
				
			||||||
 | 
						})
 | 
				
			||||||
 | 
						resp = session.MakeRequest(t, req, http.StatusOK)
 | 
				
			||||||
 | 
						DecodeJSON(t, resp, &review)
 | 
				
			||||||
 | 
						assert.EqualValues(t, "COMMENT", review.State)
 | 
				
			||||||
 | 
						assert.EqualValues(t, 0, review.CodeCommentsCount)
 | 
				
			||||||
 | 
						req = NewRequestf(t, http.MethodDelete, "/api/v1/repos/%s/%s/pulls/%d/reviews/%d?token=%s", repo.OwnerName, repo.Name, pullIssue.Index, review.ID, token)
 | 
				
			||||||
 | 
						resp = session.MakeRequest(t, req, http.StatusNoContent)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
@@ -8,7 +8,7 @@
 | 
				
			|||||||
  base_repo_id: 1
 | 
					  base_repo_id: 1
 | 
				
			||||||
  head_branch: branch1
 | 
					  head_branch: branch1
 | 
				
			||||||
  base_branch: master
 | 
					  base_branch: master
 | 
				
			||||||
  merge_base: 1234567890abcdef
 | 
					  merge_base: 4a357436d925b5c974181ff12a994538ddc5a269
 | 
				
			||||||
  has_merged: true
 | 
					  has_merged: true
 | 
				
			||||||
  merger_id: 2
 | 
					  merger_id: 2
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -22,7 +22,7 @@
 | 
				
			|||||||
  base_repo_id: 1
 | 
					  base_repo_id: 1
 | 
				
			||||||
  head_branch: branch2
 | 
					  head_branch: branch2
 | 
				
			||||||
  base_branch: master
 | 
					  base_branch: master
 | 
				
			||||||
  merge_base: fedcba9876543210
 | 
					  merge_base: 4a357436d925b5c974181ff12a994538ddc5a269
 | 
				
			||||||
  has_merged: false
 | 
					  has_merged: false
 | 
				
			||||||
 | 
					
 | 
				
			||||||
-
 | 
					-
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -60,6 +60,8 @@
 | 
				
			|||||||
  reviewer_id: 4
 | 
					  reviewer_id: 4
 | 
				
			||||||
  issue_id: 3
 | 
					  issue_id: 3
 | 
				
			||||||
  content: "New review 5"
 | 
					  content: "New review 5"
 | 
				
			||||||
 | 
					  commit_id: 8091a55037cd59e47293aca02981b5a67076b364
 | 
				
			||||||
 | 
					  stale: true
 | 
				
			||||||
  updated_unix: 946684813
 | 
					  updated_unix: 946684813
 | 
				
			||||||
  created_unix: 946684813
 | 
					  created_unix: 946684813
 | 
				
			||||||
-
 | 
					-
 | 
				
			||||||
@@ -77,5 +79,6 @@
 | 
				
			|||||||
  reviewer_id: 100
 | 
					  reviewer_id: 100
 | 
				
			||||||
  issue_id: 3
 | 
					  issue_id: 3
 | 
				
			||||||
  content: "a deleted user's review"
 | 
					  content: "a deleted user's review"
 | 
				
			||||||
 | 
					  official: true
 | 
				
			||||||
  updated_unix: 946684815
 | 
					  updated_unix: 946684815
 | 
				
			||||||
  created_unix: 946684815
 | 
					  created_unix: 946684815
 | 
				
			||||||
@@ -74,9 +74,13 @@ type Review struct {
 | 
				
			|||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func (r *Review) loadCodeComments(e Engine) (err error) {
 | 
					func (r *Review) loadCodeComments(e Engine) (err error) {
 | 
				
			||||||
	if r.CodeComments == nil {
 | 
						if r.CodeComments != nil {
 | 
				
			||||||
		r.CodeComments, err = fetchCodeCommentsByReview(e, r.Issue, nil, r)
 | 
							return
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
						if err = r.loadIssue(e); err != nil {
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						r.CodeComments, err = fetchCodeCommentsByReview(e, r.Issue, nil, r)
 | 
				
			||||||
	return
 | 
						return
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -86,12 +90,15 @@ func (r *Review) LoadCodeComments() error {
 | 
				
			|||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func (r *Review) loadIssue(e Engine) (err error) {
 | 
					func (r *Review) loadIssue(e Engine) (err error) {
 | 
				
			||||||
 | 
						if r.Issue != nil {
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
	r.Issue, err = getIssueByID(e, r.IssueID)
 | 
						r.Issue, err = getIssueByID(e, r.IssueID)
 | 
				
			||||||
	return
 | 
						return
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func (r *Review) loadReviewer(e Engine) (err error) {
 | 
					func (r *Review) loadReviewer(e Engine) (err error) {
 | 
				
			||||||
	if r.ReviewerID == 0 {
 | 
						if r.Reviewer != nil || r.ReviewerID == 0 {
 | 
				
			||||||
		return nil
 | 
							return nil
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
	r.Reviewer, err = getUserByID(e, r.ReviewerID)
 | 
						r.Reviewer, err = getUserByID(e, r.ReviewerID)
 | 
				
			||||||
@@ -104,10 +111,13 @@ func (r *Review) LoadReviewer() error {
 | 
				
			|||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func (r *Review) loadAttributes(e Engine) (err error) {
 | 
					func (r *Review) loadAttributes(e Engine) (err error) {
 | 
				
			||||||
	if err = r.loadReviewer(e); err != nil {
 | 
						if err = r.loadIssue(e); err != nil {
 | 
				
			||||||
		return
 | 
							return
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
	if err = r.loadIssue(e); err != nil {
 | 
						if err = r.loadCodeComments(e); err != nil {
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						if err = r.loadReviewer(e); err != nil {
 | 
				
			||||||
		return
 | 
							return
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
	return
 | 
						return
 | 
				
			||||||
@@ -136,6 +146,7 @@ func GetReviewByID(id int64) (*Review, error) {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
// FindReviewOptions represent possible filters to find reviews
 | 
					// FindReviewOptions represent possible filters to find reviews
 | 
				
			||||||
type FindReviewOptions struct {
 | 
					type FindReviewOptions struct {
 | 
				
			||||||
 | 
						ListOptions
 | 
				
			||||||
	Type         ReviewType
 | 
						Type         ReviewType
 | 
				
			||||||
	IssueID      int64
 | 
						IssueID      int64
 | 
				
			||||||
	ReviewerID   int64
 | 
						ReviewerID   int64
 | 
				
			||||||
@@ -162,6 +173,9 @@ func (opts *FindReviewOptions) toCond() builder.Cond {
 | 
				
			|||||||
func findReviews(e Engine, opts FindReviewOptions) ([]*Review, error) {
 | 
					func findReviews(e Engine, opts FindReviewOptions) ([]*Review, error) {
 | 
				
			||||||
	reviews := make([]*Review, 0, 10)
 | 
						reviews := make([]*Review, 0, 10)
 | 
				
			||||||
	sess := e.Where(opts.toCond())
 | 
						sess := e.Where(opts.toCond())
 | 
				
			||||||
 | 
						if opts.Page > 0 {
 | 
				
			||||||
 | 
							sess = opts.ListOptions.setSessionPagination(sess)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
	return reviews, sess.
 | 
						return reviews, sess.
 | 
				
			||||||
		Asc("created_unix").
 | 
							Asc("created_unix").
 | 
				
			||||||
		Asc("id").
 | 
							Asc("id").
 | 
				
			||||||
@@ -656,3 +670,77 @@ func CanMarkConversation(issue *Issue, doer *User) (permResult bool, err error)
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
	return true, nil
 | 
						return true, nil
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// DeleteReview delete a review and it's code comments
 | 
				
			||||||
 | 
					func DeleteReview(r *Review) error {
 | 
				
			||||||
 | 
						sess := x.NewSession()
 | 
				
			||||||
 | 
						defer sess.Close()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if err := sess.Begin(); err != nil {
 | 
				
			||||||
 | 
							return err
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if r.ID == 0 {
 | 
				
			||||||
 | 
							return fmt.Errorf("review is not allowed to be 0")
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						opts := FindCommentsOptions{
 | 
				
			||||||
 | 
							Type:     CommentTypeCode,
 | 
				
			||||||
 | 
							IssueID:  r.IssueID,
 | 
				
			||||||
 | 
							ReviewID: r.ID,
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if _, err := sess.Where(opts.toConds()).Delete(new(Comment)); err != nil {
 | 
				
			||||||
 | 
							return err
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						opts = FindCommentsOptions{
 | 
				
			||||||
 | 
							Type:     CommentTypeReview,
 | 
				
			||||||
 | 
							IssueID:  r.IssueID,
 | 
				
			||||||
 | 
							ReviewID: r.ID,
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if _, err := sess.Where(opts.toConds()).Delete(new(Comment)); err != nil {
 | 
				
			||||||
 | 
							return err
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if _, err := sess.ID(r.ID).Delete(new(Review)); err != nil {
 | 
				
			||||||
 | 
							return err
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return sess.Commit()
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// GetCodeCommentsCount return count of CodeComments a Review has
 | 
				
			||||||
 | 
					func (r *Review) GetCodeCommentsCount() int {
 | 
				
			||||||
 | 
						opts := FindCommentsOptions{
 | 
				
			||||||
 | 
							Type:     CommentTypeCode,
 | 
				
			||||||
 | 
							IssueID:  r.IssueID,
 | 
				
			||||||
 | 
							ReviewID: r.ID,
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						conds := opts.toConds()
 | 
				
			||||||
 | 
						if r.ID == 0 {
 | 
				
			||||||
 | 
							conds = conds.And(builder.Eq{"invalidated": false})
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						count, err := x.Where(conds).Count(new(Comment))
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							return 0
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return int(count)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// HTMLURL formats a URL-string to the related review issue-comment
 | 
				
			||||||
 | 
					func (r *Review) HTMLURL() string {
 | 
				
			||||||
 | 
						opts := FindCommentsOptions{
 | 
				
			||||||
 | 
							Type:     CommentTypeReview,
 | 
				
			||||||
 | 
							IssueID:  r.IssueID,
 | 
				
			||||||
 | 
							ReviewID: r.ID,
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						comment := new(Comment)
 | 
				
			||||||
 | 
						has, err := x.Where(opts.toConds()).Get(comment)
 | 
				
			||||||
 | 
						if err != nil || !has {
 | 
				
			||||||
 | 
							return ""
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return comment.HTMLURL()
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -131,9 +131,11 @@ func TestGetReviewersByIssueID(t *testing.T) {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
	allReviews, err := GetReviewersByIssueID(issue.ID)
 | 
						allReviews, err := GetReviewersByIssueID(issue.ID)
 | 
				
			||||||
	assert.NoError(t, err)
 | 
						assert.NoError(t, err)
 | 
				
			||||||
	for i, review := range allReviews {
 | 
						if assert.Len(t, allReviews, 3) {
 | 
				
			||||||
		assert.Equal(t, expectedReviews[i].Reviewer, review.Reviewer)
 | 
							for i, review := range allReviews {
 | 
				
			||||||
		assert.Equal(t, expectedReviews[i].Type, review.Type)
 | 
								assert.Equal(t, expectedReviews[i].Reviewer, review.Reviewer)
 | 
				
			||||||
		assert.Equal(t, expectedReviews[i].UpdatedUnix, review.UpdatedUnix)
 | 
								assert.Equal(t, expectedReviews[i].Type, review.Type)
 | 
				
			||||||
 | 
								assert.Equal(t, expectedReviews[i].UpdatedUnix, review.UpdatedUnix)
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										127
									
								
								modules/convert/pull_review.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										127
									
								
								modules/convert/pull_review.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,127 @@
 | 
				
			|||||||
 | 
					// Copyright 2020 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 convert
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"strings"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"code.gitea.io/gitea/models"
 | 
				
			||||||
 | 
						api "code.gitea.io/gitea/modules/structs"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// ToPullReview convert a review to api format
 | 
				
			||||||
 | 
					func ToPullReview(r *models.Review, doer *models.User) (*api.PullReview, error) {
 | 
				
			||||||
 | 
						if err := r.LoadAttributes(); err != nil {
 | 
				
			||||||
 | 
							if !models.IsErrUserNotExist(err) {
 | 
				
			||||||
 | 
								return nil, err
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							r.Reviewer = models.NewGhostUser()
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						auth := false
 | 
				
			||||||
 | 
						if doer != nil {
 | 
				
			||||||
 | 
							auth = doer.IsAdmin || doer.ID == r.ReviewerID
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						result := &api.PullReview{
 | 
				
			||||||
 | 
							ID:                r.ID,
 | 
				
			||||||
 | 
							Reviewer:          ToUser(r.Reviewer, doer != nil, auth),
 | 
				
			||||||
 | 
							State:             api.ReviewStateUnknown,
 | 
				
			||||||
 | 
							Body:              r.Content,
 | 
				
			||||||
 | 
							CommitID:          r.CommitID,
 | 
				
			||||||
 | 
							Stale:             r.Stale,
 | 
				
			||||||
 | 
							Official:          r.Official,
 | 
				
			||||||
 | 
							CodeCommentsCount: r.GetCodeCommentsCount(),
 | 
				
			||||||
 | 
							Submitted:         r.CreatedUnix.AsTime(),
 | 
				
			||||||
 | 
							HTMLURL:           r.HTMLURL(),
 | 
				
			||||||
 | 
							HTMLPullURL:       r.Issue.HTMLURL(),
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						switch r.Type {
 | 
				
			||||||
 | 
						case models.ReviewTypeApprove:
 | 
				
			||||||
 | 
							result.State = api.ReviewStateApproved
 | 
				
			||||||
 | 
						case models.ReviewTypeReject:
 | 
				
			||||||
 | 
							result.State = api.ReviewStateRequestChanges
 | 
				
			||||||
 | 
						case models.ReviewTypeComment:
 | 
				
			||||||
 | 
							result.State = api.ReviewStateComment
 | 
				
			||||||
 | 
						case models.ReviewTypePending:
 | 
				
			||||||
 | 
							result.State = api.ReviewStatePending
 | 
				
			||||||
 | 
						case models.ReviewTypeRequest:
 | 
				
			||||||
 | 
							result.State = api.ReviewStateRequestReview
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return result, nil
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// ToPullReviewList convert a list of review to it's api format
 | 
				
			||||||
 | 
					func ToPullReviewList(rl []*models.Review, doer *models.User) ([]*api.PullReview, error) {
 | 
				
			||||||
 | 
						result := make([]*api.PullReview, 0, len(rl))
 | 
				
			||||||
 | 
						for i := range rl {
 | 
				
			||||||
 | 
							// show pending reviews only for the user who created them
 | 
				
			||||||
 | 
							if rl[i].Type == models.ReviewTypePending && !(doer.IsAdmin || doer.ID == rl[i].ReviewerID) {
 | 
				
			||||||
 | 
								continue
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							r, err := ToPullReview(rl[i], doer)
 | 
				
			||||||
 | 
							if err != nil {
 | 
				
			||||||
 | 
								return nil, err
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							result = append(result, r)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return result, nil
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// ToPullReviewCommentList convert the CodeComments of an review to it's api format
 | 
				
			||||||
 | 
					func ToPullReviewCommentList(review *models.Review, doer *models.User) ([]*api.PullReviewComment, error) {
 | 
				
			||||||
 | 
						if err := review.LoadAttributes(); err != nil {
 | 
				
			||||||
 | 
							if !models.IsErrUserNotExist(err) {
 | 
				
			||||||
 | 
								return nil, err
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							review.Reviewer = models.NewGhostUser()
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						apiComments := make([]*api.PullReviewComment, 0, len(review.CodeComments))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						auth := false
 | 
				
			||||||
 | 
						if doer != nil {
 | 
				
			||||||
 | 
							auth = doer.IsAdmin || doer.ID == review.ReviewerID
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						for _, lines := range review.CodeComments {
 | 
				
			||||||
 | 
							for _, comments := range lines {
 | 
				
			||||||
 | 
								for _, comment := range comments {
 | 
				
			||||||
 | 
									apiComment := &api.PullReviewComment{
 | 
				
			||||||
 | 
										ID:           comment.ID,
 | 
				
			||||||
 | 
										Body:         comment.Content,
 | 
				
			||||||
 | 
										Reviewer:     ToUser(review.Reviewer, doer != nil, auth),
 | 
				
			||||||
 | 
										ReviewID:     review.ID,
 | 
				
			||||||
 | 
										Created:      comment.CreatedUnix.AsTime(),
 | 
				
			||||||
 | 
										Updated:      comment.UpdatedUnix.AsTime(),
 | 
				
			||||||
 | 
										Path:         comment.TreePath,
 | 
				
			||||||
 | 
										CommitID:     comment.CommitSHA,
 | 
				
			||||||
 | 
										OrigCommitID: comment.OldRef,
 | 
				
			||||||
 | 
										DiffHunk:     patch2diff(comment.Patch),
 | 
				
			||||||
 | 
										HTMLURL:      comment.HTMLURL(),
 | 
				
			||||||
 | 
										HTMLPullURL:  review.Issue.APIURL(),
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
									if comment.Line < 0 {
 | 
				
			||||||
 | 
										apiComment.OldLineNum = comment.UnsignedLine()
 | 
				
			||||||
 | 
									} else {
 | 
				
			||||||
 | 
										apiComment.LineNum = comment.UnsignedLine()
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
 | 
									apiComments = append(apiComments, apiComment)
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return apiComments, nil
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func patch2diff(patch string) string {
 | 
				
			||||||
 | 
						split := strings.Split(patch, "\n@@")
 | 
				
			||||||
 | 
						if len(split) == 2 {
 | 
				
			||||||
 | 
							return "@@" + split[1]
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return ""
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										92
									
								
								modules/structs/pull_review.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										92
									
								
								modules/structs/pull_review.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,92 @@
 | 
				
			|||||||
 | 
					// Copyright 2020 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 structs
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"time"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// ReviewStateType review state type
 | 
				
			||||||
 | 
					type ReviewStateType string
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const (
 | 
				
			||||||
 | 
						// ReviewStateApproved pr is approved
 | 
				
			||||||
 | 
						ReviewStateApproved ReviewStateType = "APPROVED"
 | 
				
			||||||
 | 
						// ReviewStatePending pr state is pending
 | 
				
			||||||
 | 
						ReviewStatePending ReviewStateType = "PENDING"
 | 
				
			||||||
 | 
						// ReviewStateComment is a comment review
 | 
				
			||||||
 | 
						ReviewStateComment ReviewStateType = "COMMENT"
 | 
				
			||||||
 | 
						// ReviewStateRequestChanges changes for pr are requested
 | 
				
			||||||
 | 
						ReviewStateRequestChanges ReviewStateType = "REQUEST_CHANGES"
 | 
				
			||||||
 | 
						// ReviewStateRequestReview review is requested from user
 | 
				
			||||||
 | 
						ReviewStateRequestReview ReviewStateType = "REQUEST_REVIEW"
 | 
				
			||||||
 | 
						// ReviewStateUnknown state of pr is unknown
 | 
				
			||||||
 | 
						ReviewStateUnknown ReviewStateType = ""
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// PullReview represents a pull request review
 | 
				
			||||||
 | 
					type PullReview struct {
 | 
				
			||||||
 | 
						ID                int64           `json:"id"`
 | 
				
			||||||
 | 
						Reviewer          *User           `json:"user"`
 | 
				
			||||||
 | 
						State             ReviewStateType `json:"state"`
 | 
				
			||||||
 | 
						Body              string          `json:"body"`
 | 
				
			||||||
 | 
						CommitID          string          `json:"commit_id"`
 | 
				
			||||||
 | 
						Stale             bool            `json:"stale"`
 | 
				
			||||||
 | 
						Official          bool            `json:"official"`
 | 
				
			||||||
 | 
						CodeCommentsCount int             `json:"comments_count"`
 | 
				
			||||||
 | 
						// swagger:strfmt date-time
 | 
				
			||||||
 | 
						Submitted time.Time `json:"submitted_at"`
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						HTMLURL     string `json:"html_url"`
 | 
				
			||||||
 | 
						HTMLPullURL string `json:"pull_request_url"`
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// PullReviewComment represents a comment on a pull request review
 | 
				
			||||||
 | 
					type PullReviewComment struct {
 | 
				
			||||||
 | 
						ID       int64  `json:"id"`
 | 
				
			||||||
 | 
						Body     string `json:"body"`
 | 
				
			||||||
 | 
						Reviewer *User  `json:"user"`
 | 
				
			||||||
 | 
						ReviewID int64  `json:"pull_request_review_id"`
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// swagger:strfmt date-time
 | 
				
			||||||
 | 
						Created time.Time `json:"created_at"`
 | 
				
			||||||
 | 
						// swagger:strfmt date-time
 | 
				
			||||||
 | 
						Updated time.Time `json:"updated_at"`
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						Path         string `json:"path"`
 | 
				
			||||||
 | 
						CommitID     string `json:"commit_id"`
 | 
				
			||||||
 | 
						OrigCommitID string `json:"original_commit_id"`
 | 
				
			||||||
 | 
						DiffHunk     string `json:"diff_hunk"`
 | 
				
			||||||
 | 
						LineNum      uint64 `json:"position"`
 | 
				
			||||||
 | 
						OldLineNum   uint64 `json:"original_position"`
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						HTMLURL     string `json:"html_url"`
 | 
				
			||||||
 | 
						HTMLPullURL string `json:"pull_request_url"`
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// CreatePullReviewOptions are options to create a pull review
 | 
				
			||||||
 | 
					type CreatePullReviewOptions struct {
 | 
				
			||||||
 | 
						Event    ReviewStateType           `json:"event"`
 | 
				
			||||||
 | 
						Body     string                    `json:"body"`
 | 
				
			||||||
 | 
						CommitID string                    `json:"commit_id"`
 | 
				
			||||||
 | 
						Comments []CreatePullReviewComment `json:"comments"`
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// CreatePullReviewComment represent a review comment for creation api
 | 
				
			||||||
 | 
					type CreatePullReviewComment struct {
 | 
				
			||||||
 | 
						// the tree path
 | 
				
			||||||
 | 
						Path string `json:"path"`
 | 
				
			||||||
 | 
						Body string `json:"body"`
 | 
				
			||||||
 | 
						// if comment to old file line or 0
 | 
				
			||||||
 | 
						OldLineNum int64 `json:"old_position"`
 | 
				
			||||||
 | 
						// if comment to new file line or 0
 | 
				
			||||||
 | 
						NewLineNum int64 `json:"new_position"`
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// SubmitPullReviewOptions are options to submit a pending pull review
 | 
				
			||||||
 | 
					type SubmitPullReviewOptions struct {
 | 
				
			||||||
 | 
						Event ReviewStateType `json:"event"`
 | 
				
			||||||
 | 
						Body  string          `json:"body"`
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
@@ -500,7 +500,7 @@ func RegisterRoutes(m *macaron.Macaron) {
 | 
				
			|||||||
	bind := binding.Bind
 | 
						bind := binding.Bind
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if setting.API.EnableSwagger {
 | 
						if setting.API.EnableSwagger {
 | 
				
			||||||
		m.Get("/swagger", misc.Swagger) //Render V1 by default
 | 
							m.Get("/swagger", misc.Swagger) // Render V1 by default
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	m.Group("/v1", func() {
 | 
						m.Group("/v1", func() {
 | 
				
			||||||
@@ -794,6 +794,20 @@ func RegisterRoutes(m *macaron.Macaron) {
 | 
				
			|||||||
							Patch(reqToken(), reqRepoWriter(models.UnitTypePullRequests), bind(api.EditPullRequestOption{}), repo.EditPullRequest)
 | 
												Patch(reqToken(), reqRepoWriter(models.UnitTypePullRequests), bind(api.EditPullRequestOption{}), repo.EditPullRequest)
 | 
				
			||||||
						m.Combo("/merge").Get(repo.IsPullRequestMerged).
 | 
											m.Combo("/merge").Get(repo.IsPullRequestMerged).
 | 
				
			||||||
							Post(reqToken(), mustNotBeArchived, bind(auth.MergePullRequestForm{}), repo.MergePullRequest)
 | 
												Post(reqToken(), mustNotBeArchived, bind(auth.MergePullRequestForm{}), repo.MergePullRequest)
 | 
				
			||||||
 | 
											m.Group("/reviews", func() {
 | 
				
			||||||
 | 
												m.Combo("").
 | 
				
			||||||
 | 
													Get(repo.ListPullReviews).
 | 
				
			||||||
 | 
													Post(reqToken(), bind(api.CreatePullReviewOptions{}), repo.CreatePullReview)
 | 
				
			||||||
 | 
												m.Group("/:id", func() {
 | 
				
			||||||
 | 
													m.Combo("").
 | 
				
			||||||
 | 
														Get(repo.GetPullReview).
 | 
				
			||||||
 | 
														Delete(reqToken(), repo.DeletePullReview).
 | 
				
			||||||
 | 
														Post(reqToken(), bind(api.SubmitPullReviewOptions{}), repo.SubmitPullReview)
 | 
				
			||||||
 | 
													m.Combo("/comments").
 | 
				
			||||||
 | 
														Get(repo.GetPullReviewComments)
 | 
				
			||||||
 | 
												})
 | 
				
			||||||
 | 
											})
 | 
				
			||||||
 | 
					
 | 
				
			||||||
					})
 | 
										})
 | 
				
			||||||
				}, mustAllowPulls, reqRepoReader(models.UnitTypeCode), context.ReferencesGitRepo(false))
 | 
									}, mustAllowPulls, reqRepoReader(models.UnitTypeCode), context.ReferencesGitRepo(false))
 | 
				
			||||||
				m.Group("/statuses", func() {
 | 
									m.Group("/statuses", func() {
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										522
									
								
								routers/api/v1/repo/pull_review.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										522
									
								
								routers/api/v1/repo/pull_review.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,522 @@
 | 
				
			|||||||
 | 
					// Copyright 2020 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 repo
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"fmt"
 | 
				
			||||||
 | 
						"net/http"
 | 
				
			||||||
 | 
						"strings"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"code.gitea.io/gitea/models"
 | 
				
			||||||
 | 
						"code.gitea.io/gitea/modules/context"
 | 
				
			||||||
 | 
						"code.gitea.io/gitea/modules/convert"
 | 
				
			||||||
 | 
						api "code.gitea.io/gitea/modules/structs"
 | 
				
			||||||
 | 
						"code.gitea.io/gitea/routers/api/v1/utils"
 | 
				
			||||||
 | 
						pull_service "code.gitea.io/gitea/services/pull"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// ListPullReviews lists all reviews of a pull request
 | 
				
			||||||
 | 
					func ListPullReviews(ctx *context.APIContext) {
 | 
				
			||||||
 | 
						// swagger:operation GET /repos/{owner}/{repo}/pulls/{index}/reviews repository repoListPullReviews
 | 
				
			||||||
 | 
						// ---
 | 
				
			||||||
 | 
						// summary: List all reviews for a pull request
 | 
				
			||||||
 | 
						// produces:
 | 
				
			||||||
 | 
						// - application/json
 | 
				
			||||||
 | 
						// parameters:
 | 
				
			||||||
 | 
						// - name: owner
 | 
				
			||||||
 | 
						//   in: path
 | 
				
			||||||
 | 
						//   description: owner of the repo
 | 
				
			||||||
 | 
						//   type: string
 | 
				
			||||||
 | 
						//   required: true
 | 
				
			||||||
 | 
						// - name: repo
 | 
				
			||||||
 | 
						//   in: path
 | 
				
			||||||
 | 
						//   description: name of the repo
 | 
				
			||||||
 | 
						//   type: string
 | 
				
			||||||
 | 
						//   required: true
 | 
				
			||||||
 | 
						// - name: index
 | 
				
			||||||
 | 
						//   in: path
 | 
				
			||||||
 | 
						//   description: index of the pull request
 | 
				
			||||||
 | 
						//   type: integer
 | 
				
			||||||
 | 
						//   format: int64
 | 
				
			||||||
 | 
						//   required: true
 | 
				
			||||||
 | 
						// - name: page
 | 
				
			||||||
 | 
						//   in: query
 | 
				
			||||||
 | 
						//   description: page number of results to return (1-based)
 | 
				
			||||||
 | 
						//   type: integer
 | 
				
			||||||
 | 
						// - name: limit
 | 
				
			||||||
 | 
						//   in: query
 | 
				
			||||||
 | 
						//   description: page size of results, maximum page size is 50
 | 
				
			||||||
 | 
						//   type: integer
 | 
				
			||||||
 | 
						// responses:
 | 
				
			||||||
 | 
						//   "200":
 | 
				
			||||||
 | 
						//     "$ref": "#/responses/PullReviewList"
 | 
				
			||||||
 | 
						//   "404":
 | 
				
			||||||
 | 
						//     "$ref": "#/responses/notFound"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						pr, err := models.GetPullRequestByIndex(ctx.Repo.Repository.ID, ctx.ParamsInt64(":index"))
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							if models.IsErrPullRequestNotExist(err) {
 | 
				
			||||||
 | 
								ctx.NotFound("GetPullRequestByIndex", err)
 | 
				
			||||||
 | 
							} else {
 | 
				
			||||||
 | 
								ctx.Error(http.StatusInternalServerError, "GetPullRequestByIndex", err)
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if err = pr.LoadIssue(); err != nil {
 | 
				
			||||||
 | 
							ctx.Error(http.StatusInternalServerError, "LoadIssue", err)
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if err = pr.Issue.LoadRepo(); err != nil {
 | 
				
			||||||
 | 
							ctx.Error(http.StatusInternalServerError, "LoadRepo", err)
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						allReviews, err := models.FindReviews(models.FindReviewOptions{
 | 
				
			||||||
 | 
							ListOptions: utils.GetListOptions(ctx),
 | 
				
			||||||
 | 
							Type:        models.ReviewTypeUnknown,
 | 
				
			||||||
 | 
							IssueID:     pr.IssueID,
 | 
				
			||||||
 | 
						})
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							ctx.Error(http.StatusInternalServerError, "FindReviews", err)
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						apiReviews, err := convert.ToPullReviewList(allReviews, ctx.User)
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							ctx.Error(http.StatusInternalServerError, "convertToPullReviewList", err)
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						ctx.JSON(http.StatusOK, &apiReviews)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// GetPullReview gets a specific review of a pull request
 | 
				
			||||||
 | 
					func GetPullReview(ctx *context.APIContext) {
 | 
				
			||||||
 | 
						// swagger:operation GET /repos/{owner}/{repo}/pulls/{index}/reviews/{id} repository repoGetPullReview
 | 
				
			||||||
 | 
						// ---
 | 
				
			||||||
 | 
						// summary: Get a specific review for a pull request
 | 
				
			||||||
 | 
						// produces:
 | 
				
			||||||
 | 
						// - application/json
 | 
				
			||||||
 | 
						// parameters:
 | 
				
			||||||
 | 
						// - name: owner
 | 
				
			||||||
 | 
						//   in: path
 | 
				
			||||||
 | 
						//   description: owner of the repo
 | 
				
			||||||
 | 
						//   type: string
 | 
				
			||||||
 | 
						//   required: true
 | 
				
			||||||
 | 
						// - name: repo
 | 
				
			||||||
 | 
						//   in: path
 | 
				
			||||||
 | 
						//   description: name of the repo
 | 
				
			||||||
 | 
						//   type: string
 | 
				
			||||||
 | 
						//   required: true
 | 
				
			||||||
 | 
						// - name: index
 | 
				
			||||||
 | 
						//   in: path
 | 
				
			||||||
 | 
						//   description: index of the pull request
 | 
				
			||||||
 | 
						//   type: integer
 | 
				
			||||||
 | 
						//   format: int64
 | 
				
			||||||
 | 
						//   required: true
 | 
				
			||||||
 | 
						// - name: id
 | 
				
			||||||
 | 
						//   in: path
 | 
				
			||||||
 | 
						//   description: id of the review
 | 
				
			||||||
 | 
						//   type: integer
 | 
				
			||||||
 | 
						//   format: int64
 | 
				
			||||||
 | 
						//   required: true
 | 
				
			||||||
 | 
						// responses:
 | 
				
			||||||
 | 
						//   "200":
 | 
				
			||||||
 | 
						//     "$ref": "#/responses/PullReview"
 | 
				
			||||||
 | 
						//   "404":
 | 
				
			||||||
 | 
						//     "$ref": "#/responses/notFound"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						review, _, statusSet := prepareSingleReview(ctx)
 | 
				
			||||||
 | 
						if statusSet {
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						apiReview, err := convert.ToPullReview(review, ctx.User)
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							ctx.Error(http.StatusInternalServerError, "convertToPullReview", err)
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						ctx.JSON(http.StatusOK, apiReview)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// GetPullReviewComments lists all comments of a pull request review
 | 
				
			||||||
 | 
					func GetPullReviewComments(ctx *context.APIContext) {
 | 
				
			||||||
 | 
						// swagger:operation GET /repos/{owner}/{repo}/pulls/{index}/reviews/{id}/comments repository repoGetPullReviewComments
 | 
				
			||||||
 | 
						// ---
 | 
				
			||||||
 | 
						// summary: Get a specific review for a pull request
 | 
				
			||||||
 | 
						// produces:
 | 
				
			||||||
 | 
						// - application/json
 | 
				
			||||||
 | 
						// parameters:
 | 
				
			||||||
 | 
						// - name: owner
 | 
				
			||||||
 | 
						//   in: path
 | 
				
			||||||
 | 
						//   description: owner of the repo
 | 
				
			||||||
 | 
						//   type: string
 | 
				
			||||||
 | 
						//   required: true
 | 
				
			||||||
 | 
						// - name: repo
 | 
				
			||||||
 | 
						//   in: path
 | 
				
			||||||
 | 
						//   description: name of the repo
 | 
				
			||||||
 | 
						//   type: string
 | 
				
			||||||
 | 
						//   required: true
 | 
				
			||||||
 | 
						// - name: index
 | 
				
			||||||
 | 
						//   in: path
 | 
				
			||||||
 | 
						//   description: index of the pull request
 | 
				
			||||||
 | 
						//   type: integer
 | 
				
			||||||
 | 
						//   format: int64
 | 
				
			||||||
 | 
						//   required: true
 | 
				
			||||||
 | 
						// - name: id
 | 
				
			||||||
 | 
						//   in: path
 | 
				
			||||||
 | 
						//   description: id of the review
 | 
				
			||||||
 | 
						//   type: integer
 | 
				
			||||||
 | 
						//   format: int64
 | 
				
			||||||
 | 
						//   required: true
 | 
				
			||||||
 | 
						// responses:
 | 
				
			||||||
 | 
						//   "200":
 | 
				
			||||||
 | 
						//     "$ref": "#/responses/PullReviewCommentList"
 | 
				
			||||||
 | 
						//   "404":
 | 
				
			||||||
 | 
						//     "$ref": "#/responses/notFound"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						review, _, statusSet := prepareSingleReview(ctx)
 | 
				
			||||||
 | 
						if statusSet {
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						apiComments, err := convert.ToPullReviewCommentList(review, ctx.User)
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							ctx.Error(http.StatusInternalServerError, "convertToPullReviewCommentList", err)
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						ctx.JSON(http.StatusOK, apiComments)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// DeletePullReview delete a specific review from a pull request
 | 
				
			||||||
 | 
					func DeletePullReview(ctx *context.APIContext) {
 | 
				
			||||||
 | 
						// swagger:operation DELETE /repos/{owner}/{repo}/pulls/{index}/reviews/{id} repository repoDeletePullReview
 | 
				
			||||||
 | 
						// ---
 | 
				
			||||||
 | 
						// summary: Delete a specific review from a pull request
 | 
				
			||||||
 | 
						// produces:
 | 
				
			||||||
 | 
						// - application/json
 | 
				
			||||||
 | 
						// parameters:
 | 
				
			||||||
 | 
						// - name: owner
 | 
				
			||||||
 | 
						//   in: path
 | 
				
			||||||
 | 
						//   description: owner of the repo
 | 
				
			||||||
 | 
						//   type: string
 | 
				
			||||||
 | 
						//   required: true
 | 
				
			||||||
 | 
						// - name: repo
 | 
				
			||||||
 | 
						//   in: path
 | 
				
			||||||
 | 
						//   description: name of the repo
 | 
				
			||||||
 | 
						//   type: string
 | 
				
			||||||
 | 
						//   required: true
 | 
				
			||||||
 | 
						// - name: index
 | 
				
			||||||
 | 
						//   in: path
 | 
				
			||||||
 | 
						//   description: index of the pull request
 | 
				
			||||||
 | 
						//   type: integer
 | 
				
			||||||
 | 
						//   format: int64
 | 
				
			||||||
 | 
						//   required: true
 | 
				
			||||||
 | 
						// - name: id
 | 
				
			||||||
 | 
						//   in: path
 | 
				
			||||||
 | 
						//   description: id of the review
 | 
				
			||||||
 | 
						//   type: integer
 | 
				
			||||||
 | 
						//   format: int64
 | 
				
			||||||
 | 
						//   required: true
 | 
				
			||||||
 | 
						// responses:
 | 
				
			||||||
 | 
						//   "204":
 | 
				
			||||||
 | 
						//     "$ref": "#/responses/empty"
 | 
				
			||||||
 | 
						//   "403":
 | 
				
			||||||
 | 
						//     "$ref": "#/responses/forbidden"
 | 
				
			||||||
 | 
						//   "404":
 | 
				
			||||||
 | 
						//     "$ref": "#/responses/notFound"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						review, _, statusSet := prepareSingleReview(ctx)
 | 
				
			||||||
 | 
						if statusSet {
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if ctx.User == nil {
 | 
				
			||||||
 | 
							ctx.NotFound()
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						if !ctx.User.IsAdmin && ctx.User.ID != review.ReviewerID {
 | 
				
			||||||
 | 
							ctx.Error(http.StatusForbidden, "only admin and user itself can delete a review", nil)
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if err := models.DeleteReview(review); err != nil {
 | 
				
			||||||
 | 
							ctx.Error(http.StatusInternalServerError, "DeleteReview", fmt.Errorf("can not delete ReviewID: %d", review.ID))
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						ctx.Status(http.StatusNoContent)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// CreatePullReview create a review to an pull request
 | 
				
			||||||
 | 
					func CreatePullReview(ctx *context.APIContext, opts api.CreatePullReviewOptions) {
 | 
				
			||||||
 | 
						// swagger:operation POST /repos/{owner}/{repo}/pulls/{index}/reviews repository repoCreatePullReview
 | 
				
			||||||
 | 
						// ---
 | 
				
			||||||
 | 
						// summary: Create a review to an pull request
 | 
				
			||||||
 | 
						// produces:
 | 
				
			||||||
 | 
						// - application/json
 | 
				
			||||||
 | 
						// parameters:
 | 
				
			||||||
 | 
						// - name: owner
 | 
				
			||||||
 | 
						//   in: path
 | 
				
			||||||
 | 
						//   description: owner of the repo
 | 
				
			||||||
 | 
						//   type: string
 | 
				
			||||||
 | 
						//   required: true
 | 
				
			||||||
 | 
						// - name: repo
 | 
				
			||||||
 | 
						//   in: path
 | 
				
			||||||
 | 
						//   description: name of the repo
 | 
				
			||||||
 | 
						//   type: string
 | 
				
			||||||
 | 
						//   required: true
 | 
				
			||||||
 | 
						// - name: index
 | 
				
			||||||
 | 
						//   in: path
 | 
				
			||||||
 | 
						//   description: index of the pull request
 | 
				
			||||||
 | 
						//   type: integer
 | 
				
			||||||
 | 
						//   format: int64
 | 
				
			||||||
 | 
						//   required: true
 | 
				
			||||||
 | 
						// - name: body
 | 
				
			||||||
 | 
						//   in: body
 | 
				
			||||||
 | 
						//   required: true
 | 
				
			||||||
 | 
						//   schema:
 | 
				
			||||||
 | 
						//     "$ref": "#/definitions/CreatePullReviewOptions"
 | 
				
			||||||
 | 
						// responses:
 | 
				
			||||||
 | 
						//   "200":
 | 
				
			||||||
 | 
						//     "$ref": "#/responses/PullReview"
 | 
				
			||||||
 | 
						//   "404":
 | 
				
			||||||
 | 
						//     "$ref": "#/responses/notFound"
 | 
				
			||||||
 | 
						//   "422":
 | 
				
			||||||
 | 
						//     "$ref": "#/responses/validationError"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						pr, err := models.GetPullRequestByIndex(ctx.Repo.Repository.ID, ctx.ParamsInt64(":index"))
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							if models.IsErrPullRequestNotExist(err) {
 | 
				
			||||||
 | 
								ctx.NotFound("GetPullRequestByIndex", err)
 | 
				
			||||||
 | 
							} else {
 | 
				
			||||||
 | 
								ctx.Error(http.StatusInternalServerError, "GetPullRequestByIndex", err)
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// determine review type
 | 
				
			||||||
 | 
						reviewType, isWrong := preparePullReviewType(ctx, pr, opts.Event, opts.Body)
 | 
				
			||||||
 | 
						if isWrong {
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if err := pr.Issue.LoadRepo(); err != nil {
 | 
				
			||||||
 | 
							ctx.Error(http.StatusInternalServerError, "pr.Issue.LoadRepo", err)
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// create review comments
 | 
				
			||||||
 | 
						for _, c := range opts.Comments {
 | 
				
			||||||
 | 
							line := c.NewLineNum
 | 
				
			||||||
 | 
							if c.OldLineNum > 0 {
 | 
				
			||||||
 | 
								line = c.OldLineNum * -1
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							if _, err := pull_service.CreateCodeComment(
 | 
				
			||||||
 | 
								ctx.User,
 | 
				
			||||||
 | 
								ctx.Repo.GitRepo,
 | 
				
			||||||
 | 
								pr.Issue,
 | 
				
			||||||
 | 
								line,
 | 
				
			||||||
 | 
								c.Body,
 | 
				
			||||||
 | 
								c.Path,
 | 
				
			||||||
 | 
								true, // is review
 | 
				
			||||||
 | 
								0,    // no reply
 | 
				
			||||||
 | 
								opts.CommitID,
 | 
				
			||||||
 | 
							); err != nil {
 | 
				
			||||||
 | 
								ctx.ServerError("CreateCodeComment", err)
 | 
				
			||||||
 | 
								return
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// create review and associate all pending review comments
 | 
				
			||||||
 | 
						review, _, err := pull_service.SubmitReview(ctx.User, ctx.Repo.GitRepo, pr.Issue, reviewType, opts.Body, opts.CommitID)
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							ctx.Error(http.StatusInternalServerError, "SubmitReview", err)
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// convert response
 | 
				
			||||||
 | 
						apiReview, err := convert.ToPullReview(review, ctx.User)
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							ctx.Error(http.StatusInternalServerError, "convertToPullReview", err)
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						ctx.JSON(http.StatusOK, apiReview)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// SubmitPullReview submit a pending review to an pull request
 | 
				
			||||||
 | 
					func SubmitPullReview(ctx *context.APIContext, opts api.SubmitPullReviewOptions) {
 | 
				
			||||||
 | 
						// swagger:operation POST /repos/{owner}/{repo}/pulls/{index}/reviews/{id} repository repoSubmitPullReview
 | 
				
			||||||
 | 
						// ---
 | 
				
			||||||
 | 
						// summary: Submit a pending review to an pull request
 | 
				
			||||||
 | 
						// produces:
 | 
				
			||||||
 | 
						// - application/json
 | 
				
			||||||
 | 
						// parameters:
 | 
				
			||||||
 | 
						// - name: owner
 | 
				
			||||||
 | 
						//   in: path
 | 
				
			||||||
 | 
						//   description: owner of the repo
 | 
				
			||||||
 | 
						//   type: string
 | 
				
			||||||
 | 
						//   required: true
 | 
				
			||||||
 | 
						// - name: repo
 | 
				
			||||||
 | 
						//   in: path
 | 
				
			||||||
 | 
						//   description: name of the repo
 | 
				
			||||||
 | 
						//   type: string
 | 
				
			||||||
 | 
						//   required: true
 | 
				
			||||||
 | 
						// - name: index
 | 
				
			||||||
 | 
						//   in: path
 | 
				
			||||||
 | 
						//   description: index of the pull request
 | 
				
			||||||
 | 
						//   type: integer
 | 
				
			||||||
 | 
						//   format: int64
 | 
				
			||||||
 | 
						//   required: true
 | 
				
			||||||
 | 
						// - name: id
 | 
				
			||||||
 | 
						//   in: path
 | 
				
			||||||
 | 
						//   description: id of the review
 | 
				
			||||||
 | 
						//   type: integer
 | 
				
			||||||
 | 
						//   format: int64
 | 
				
			||||||
 | 
						//   required: true
 | 
				
			||||||
 | 
						// - name: body
 | 
				
			||||||
 | 
						//   in: body
 | 
				
			||||||
 | 
						//   required: true
 | 
				
			||||||
 | 
						//   schema:
 | 
				
			||||||
 | 
						//     "$ref": "#/definitions/SubmitPullReviewOptions"
 | 
				
			||||||
 | 
						// responses:
 | 
				
			||||||
 | 
						//   "200":
 | 
				
			||||||
 | 
						//     "$ref": "#/responses/PullReview"
 | 
				
			||||||
 | 
						//   "404":
 | 
				
			||||||
 | 
						//     "$ref": "#/responses/notFound"
 | 
				
			||||||
 | 
						//   "422":
 | 
				
			||||||
 | 
						//     "$ref": "#/responses/validationError"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						review, pr, isWrong := prepareSingleReview(ctx)
 | 
				
			||||||
 | 
						if isWrong {
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if review.Type != models.ReviewTypePending {
 | 
				
			||||||
 | 
							ctx.Error(http.StatusUnprocessableEntity, "", fmt.Errorf("only a pending review can be submitted"))
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// determine review type
 | 
				
			||||||
 | 
						reviewType, isWrong := preparePullReviewType(ctx, pr, opts.Event, opts.Body)
 | 
				
			||||||
 | 
						if isWrong {
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// if review stay pending return
 | 
				
			||||||
 | 
						if reviewType == models.ReviewTypePending {
 | 
				
			||||||
 | 
							ctx.Error(http.StatusUnprocessableEntity, "", fmt.Errorf("review stay pending"))
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						headCommitID, err := ctx.Repo.GitRepo.GetRefCommitID(pr.GetGitRefName())
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							ctx.Error(http.StatusInternalServerError, "GitRepo: GetRefCommitID", err)
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// create review and associate all pending review comments
 | 
				
			||||||
 | 
						review, _, err = pull_service.SubmitReview(ctx.User, ctx.Repo.GitRepo, pr.Issue, reviewType, opts.Body, headCommitID)
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							ctx.Error(http.StatusInternalServerError, "SubmitReview", err)
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// convert response
 | 
				
			||||||
 | 
						apiReview, err := convert.ToPullReview(review, ctx.User)
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							ctx.Error(http.StatusInternalServerError, "convertToPullReview", err)
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						ctx.JSON(http.StatusOK, apiReview)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// preparePullReviewType return ReviewType and false or nil and true if an error happen
 | 
				
			||||||
 | 
					func preparePullReviewType(ctx *context.APIContext, pr *models.PullRequest, event api.ReviewStateType, body string) (models.ReviewType, bool) {
 | 
				
			||||||
 | 
						if err := pr.LoadIssue(); err != nil {
 | 
				
			||||||
 | 
							ctx.Error(http.StatusInternalServerError, "LoadIssue", err)
 | 
				
			||||||
 | 
							return -1, true
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						var reviewType models.ReviewType
 | 
				
			||||||
 | 
						switch event {
 | 
				
			||||||
 | 
						case api.ReviewStateApproved:
 | 
				
			||||||
 | 
							// can not approve your own PR
 | 
				
			||||||
 | 
							if pr.Issue.IsPoster(ctx.User.ID) {
 | 
				
			||||||
 | 
								ctx.Error(http.StatusUnprocessableEntity, "", fmt.Errorf("approve your own pull is not allowed"))
 | 
				
			||||||
 | 
								return -1, true
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							reviewType = models.ReviewTypeApprove
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						case api.ReviewStateRequestChanges:
 | 
				
			||||||
 | 
							// can not reject your own PR
 | 
				
			||||||
 | 
							if pr.Issue.IsPoster(ctx.User.ID) {
 | 
				
			||||||
 | 
								ctx.Error(http.StatusUnprocessableEntity, "", fmt.Errorf("reject your own pull is not allowed"))
 | 
				
			||||||
 | 
								return -1, true
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							reviewType = models.ReviewTypeReject
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						case api.ReviewStateComment:
 | 
				
			||||||
 | 
							reviewType = models.ReviewTypeComment
 | 
				
			||||||
 | 
						default:
 | 
				
			||||||
 | 
							reviewType = models.ReviewTypePending
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// reject reviews with empty body if not approve type
 | 
				
			||||||
 | 
						if reviewType != models.ReviewTypeApprove && len(strings.TrimSpace(body)) == 0 {
 | 
				
			||||||
 | 
							ctx.Error(http.StatusUnprocessableEntity, "", fmt.Errorf("review event %s need body", event))
 | 
				
			||||||
 | 
							return -1, true
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return reviewType, false
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// prepareSingleReview return review, related pull and false or nil, nil and true if an error happen
 | 
				
			||||||
 | 
					func prepareSingleReview(ctx *context.APIContext) (*models.Review, *models.PullRequest, bool) {
 | 
				
			||||||
 | 
						pr, err := models.GetPullRequestByIndex(ctx.Repo.Repository.ID, ctx.ParamsInt64(":index"))
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							if models.IsErrPullRequestNotExist(err) {
 | 
				
			||||||
 | 
								ctx.NotFound("GetPullRequestByIndex", err)
 | 
				
			||||||
 | 
							} else {
 | 
				
			||||||
 | 
								ctx.Error(http.StatusInternalServerError, "GetPullRequestByIndex", err)
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							return nil, nil, true
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						review, err := models.GetReviewByID(ctx.ParamsInt64(":id"))
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							if models.IsErrReviewNotExist(err) {
 | 
				
			||||||
 | 
								ctx.NotFound("GetReviewByID", err)
 | 
				
			||||||
 | 
							} else {
 | 
				
			||||||
 | 
								ctx.Error(http.StatusInternalServerError, "GetReviewByID", err)
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							return nil, nil, true
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// validate the the review is for the given PR
 | 
				
			||||||
 | 
						if review.IssueID != pr.IssueID {
 | 
				
			||||||
 | 
							ctx.NotFound("ReviewNotInPR")
 | 
				
			||||||
 | 
							return nil, nil, true
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// make sure that the user has access to this review if it is pending
 | 
				
			||||||
 | 
						if review.Type == models.ReviewTypePending && review.ReviewerID != ctx.User.ID && !ctx.User.IsAdmin {
 | 
				
			||||||
 | 
							ctx.NotFound("GetReviewByID")
 | 
				
			||||||
 | 
							return nil, nil, true
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if err := review.LoadAttributes(); err != nil && !models.IsErrUserNotExist(err) {
 | 
				
			||||||
 | 
							ctx.Error(http.StatusInternalServerError, "ReviewLoadAttributes", err)
 | 
				
			||||||
 | 
							return nil, nil, true
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return review, pr, false
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
@@ -137,4 +137,13 @@ type swaggerParameterBodies struct {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
	// in:body
 | 
						// in:body
 | 
				
			||||||
	CreateOAuth2ApplicationOptions api.CreateOAuth2ApplicationOptions
 | 
						CreateOAuth2ApplicationOptions api.CreateOAuth2ApplicationOptions
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// in:body
 | 
				
			||||||
 | 
						CreatePullReviewOptions api.CreatePullReviewOptions
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// in:body
 | 
				
			||||||
 | 
						CreatePullReviewComment api.CreatePullReviewComment
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// in:body
 | 
				
			||||||
 | 
						SubmitPullReviewOptions api.SubmitPullReviewOptions
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -141,6 +141,34 @@ type swaggerResponsePullRequestList struct {
 | 
				
			|||||||
	Body []api.PullRequest `json:"body"`
 | 
						Body []api.PullRequest `json:"body"`
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// PullReview
 | 
				
			||||||
 | 
					// swagger:response PullReview
 | 
				
			||||||
 | 
					type swaggerResponsePullReview struct {
 | 
				
			||||||
 | 
						// in:body
 | 
				
			||||||
 | 
						Body api.PullReview `json:"body"`
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// PullReviewList
 | 
				
			||||||
 | 
					// swagger:response PullReviewList
 | 
				
			||||||
 | 
					type swaggerResponsePullReviewList struct {
 | 
				
			||||||
 | 
						// in:body
 | 
				
			||||||
 | 
						Body []api.PullReview `json:"body"`
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// PullComment
 | 
				
			||||||
 | 
					// swagger:response PullReviewComment
 | 
				
			||||||
 | 
					type swaggerPullReviewComment struct {
 | 
				
			||||||
 | 
						// in:body
 | 
				
			||||||
 | 
						Body api.PullReviewComment `json:"body"`
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// PullCommentList
 | 
				
			||||||
 | 
					// swagger:response PullReviewCommentList
 | 
				
			||||||
 | 
					type swaggerResponsePullReviewCommentList struct {
 | 
				
			||||||
 | 
						// in:body
 | 
				
			||||||
 | 
						Body []api.PullReviewComment `json:"body"`
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// Status
 | 
					// Status
 | 
				
			||||||
// swagger:response Status
 | 
					// swagger:response Status
 | 
				
			||||||
type swaggerResponseStatus struct {
 | 
					type swaggerResponseStatus struct {
 | 
				
			||||||
@@ -172,35 +200,35 @@ type swaggerResponseSearchResults struct {
 | 
				
			|||||||
// AttachmentList
 | 
					// AttachmentList
 | 
				
			||||||
// swagger:response AttachmentList
 | 
					// swagger:response AttachmentList
 | 
				
			||||||
type swaggerResponseAttachmentList struct {
 | 
					type swaggerResponseAttachmentList struct {
 | 
				
			||||||
	//in: body
 | 
						// in: body
 | 
				
			||||||
	Body []api.Attachment `json:"body"`
 | 
						Body []api.Attachment `json:"body"`
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// Attachment
 | 
					// Attachment
 | 
				
			||||||
// swagger:response Attachment
 | 
					// swagger:response Attachment
 | 
				
			||||||
type swaggerResponseAttachment struct {
 | 
					type swaggerResponseAttachment struct {
 | 
				
			||||||
	//in: body
 | 
						// in: body
 | 
				
			||||||
	Body api.Attachment `json:"body"`
 | 
						Body api.Attachment `json:"body"`
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// GitTreeResponse
 | 
					// GitTreeResponse
 | 
				
			||||||
// swagger:response GitTreeResponse
 | 
					// swagger:response GitTreeResponse
 | 
				
			||||||
type swaggerGitTreeResponse struct {
 | 
					type swaggerGitTreeResponse struct {
 | 
				
			||||||
	//in: body
 | 
						// in: body
 | 
				
			||||||
	Body api.GitTreeResponse `json:"body"`
 | 
						Body api.GitTreeResponse `json:"body"`
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// GitBlobResponse
 | 
					// GitBlobResponse
 | 
				
			||||||
// swagger:response GitBlobResponse
 | 
					// swagger:response GitBlobResponse
 | 
				
			||||||
type swaggerGitBlobResponse struct {
 | 
					type swaggerGitBlobResponse struct {
 | 
				
			||||||
	//in: body
 | 
						// in: body
 | 
				
			||||||
	Body api.GitBlobResponse `json:"body"`
 | 
						Body api.GitBlobResponse `json:"body"`
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// Commit
 | 
					// Commit
 | 
				
			||||||
// swagger:response Commit
 | 
					// swagger:response Commit
 | 
				
			||||||
type swaggerCommit struct {
 | 
					type swaggerCommit struct {
 | 
				
			||||||
	//in: body
 | 
						// in: body
 | 
				
			||||||
	Body api.Commit `json:"body"`
 | 
						Body api.Commit `json:"body"`
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -222,28 +250,28 @@ type swaggerCommitList struct {
 | 
				
			|||||||
	// True if there is another page
 | 
						// True if there is another page
 | 
				
			||||||
	HasMore bool `json:"X-HasMore"`
 | 
						HasMore bool `json:"X-HasMore"`
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	//in: body
 | 
						// in: body
 | 
				
			||||||
	Body []api.Commit `json:"body"`
 | 
						Body []api.Commit `json:"body"`
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// EmptyRepository
 | 
					// EmptyRepository
 | 
				
			||||||
// swagger:response EmptyRepository
 | 
					// swagger:response EmptyRepository
 | 
				
			||||||
type swaggerEmptyRepository struct {
 | 
					type swaggerEmptyRepository struct {
 | 
				
			||||||
	//in: body
 | 
						// in: body
 | 
				
			||||||
	Body api.APIError `json:"body"`
 | 
						Body api.APIError `json:"body"`
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// FileResponse
 | 
					// FileResponse
 | 
				
			||||||
// swagger:response FileResponse
 | 
					// swagger:response FileResponse
 | 
				
			||||||
type swaggerFileResponse struct {
 | 
					type swaggerFileResponse struct {
 | 
				
			||||||
	//in: body
 | 
						// in: body
 | 
				
			||||||
	Body api.FileResponse `json:"body"`
 | 
						Body api.FileResponse `json:"body"`
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// ContentsResponse
 | 
					// ContentsResponse
 | 
				
			||||||
// swagger:response ContentsResponse
 | 
					// swagger:response ContentsResponse
 | 
				
			||||||
type swaggerContentsResponse struct {
 | 
					type swaggerContentsResponse struct {
 | 
				
			||||||
	//in: body
 | 
						// in: body
 | 
				
			||||||
	Body api.ContentsResponse `json:"body"`
 | 
						Body api.ContentsResponse `json:"body"`
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -257,20 +285,20 @@ type swaggerContentsListResponse struct {
 | 
				
			|||||||
// FileDeleteResponse
 | 
					// FileDeleteResponse
 | 
				
			||||||
// swagger:response FileDeleteResponse
 | 
					// swagger:response FileDeleteResponse
 | 
				
			||||||
type swaggerFileDeleteResponse struct {
 | 
					type swaggerFileDeleteResponse struct {
 | 
				
			||||||
	//in: body
 | 
						// in: body
 | 
				
			||||||
	Body api.FileDeleteResponse `json:"body"`
 | 
						Body api.FileDeleteResponse `json:"body"`
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// TopicListResponse
 | 
					// TopicListResponse
 | 
				
			||||||
// swagger:response TopicListResponse
 | 
					// swagger:response TopicListResponse
 | 
				
			||||||
type swaggerTopicListResponse struct {
 | 
					type swaggerTopicListResponse struct {
 | 
				
			||||||
	//in: body
 | 
						// in: body
 | 
				
			||||||
	Body []api.TopicResponse `json:"body"`
 | 
						Body []api.TopicResponse `json:"body"`
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// TopicNames
 | 
					// TopicNames
 | 
				
			||||||
// swagger:response TopicNames
 | 
					// swagger:response TopicNames
 | 
				
			||||||
type swaggerTopicNames struct {
 | 
					type swaggerTopicNames struct {
 | 
				
			||||||
	//in: body
 | 
						// in: body
 | 
				
			||||||
	Body api.TopicName `json:"body"`
 | 
						Body api.TopicName `json:"body"`
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -6714,6 +6714,333 @@
 | 
				
			|||||||
        }
 | 
					        }
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
 | 
					    "/repos/{owner}/{repo}/pulls/{index}/reviews": {
 | 
				
			||||||
 | 
					      "get": {
 | 
				
			||||||
 | 
					        "produces": [
 | 
				
			||||||
 | 
					          "application/json"
 | 
				
			||||||
 | 
					        ],
 | 
				
			||||||
 | 
					        "tags": [
 | 
				
			||||||
 | 
					          "repository"
 | 
				
			||||||
 | 
					        ],
 | 
				
			||||||
 | 
					        "summary": "List all reviews for a pull request",
 | 
				
			||||||
 | 
					        "operationId": "repoListPullReviews",
 | 
				
			||||||
 | 
					        "parameters": [
 | 
				
			||||||
 | 
					          {
 | 
				
			||||||
 | 
					            "type": "string",
 | 
				
			||||||
 | 
					            "description": "owner of the repo",
 | 
				
			||||||
 | 
					            "name": "owner",
 | 
				
			||||||
 | 
					            "in": "path",
 | 
				
			||||||
 | 
					            "required": true
 | 
				
			||||||
 | 
					          },
 | 
				
			||||||
 | 
					          {
 | 
				
			||||||
 | 
					            "type": "string",
 | 
				
			||||||
 | 
					            "description": "name of the repo",
 | 
				
			||||||
 | 
					            "name": "repo",
 | 
				
			||||||
 | 
					            "in": "path",
 | 
				
			||||||
 | 
					            "required": true
 | 
				
			||||||
 | 
					          },
 | 
				
			||||||
 | 
					          {
 | 
				
			||||||
 | 
					            "type": "integer",
 | 
				
			||||||
 | 
					            "format": "int64",
 | 
				
			||||||
 | 
					            "description": "index of the pull request",
 | 
				
			||||||
 | 
					            "name": "index",
 | 
				
			||||||
 | 
					            "in": "path",
 | 
				
			||||||
 | 
					            "required": true
 | 
				
			||||||
 | 
					          },
 | 
				
			||||||
 | 
					          {
 | 
				
			||||||
 | 
					            "type": "integer",
 | 
				
			||||||
 | 
					            "description": "page number of results to return (1-based)",
 | 
				
			||||||
 | 
					            "name": "page",
 | 
				
			||||||
 | 
					            "in": "query"
 | 
				
			||||||
 | 
					          },
 | 
				
			||||||
 | 
					          {
 | 
				
			||||||
 | 
					            "type": "integer",
 | 
				
			||||||
 | 
					            "description": "page size of results, maximum page size is 50",
 | 
				
			||||||
 | 
					            "name": "limit",
 | 
				
			||||||
 | 
					            "in": "query"
 | 
				
			||||||
 | 
					          }
 | 
				
			||||||
 | 
					        ],
 | 
				
			||||||
 | 
					        "responses": {
 | 
				
			||||||
 | 
					          "200": {
 | 
				
			||||||
 | 
					            "$ref": "#/responses/PullReviewList"
 | 
				
			||||||
 | 
					          },
 | 
				
			||||||
 | 
					          "404": {
 | 
				
			||||||
 | 
					            "$ref": "#/responses/notFound"
 | 
				
			||||||
 | 
					          }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					      },
 | 
				
			||||||
 | 
					      "post": {
 | 
				
			||||||
 | 
					        "produces": [
 | 
				
			||||||
 | 
					          "application/json"
 | 
				
			||||||
 | 
					        ],
 | 
				
			||||||
 | 
					        "tags": [
 | 
				
			||||||
 | 
					          "repository"
 | 
				
			||||||
 | 
					        ],
 | 
				
			||||||
 | 
					        "summary": "Create a review to an pull request",
 | 
				
			||||||
 | 
					        "operationId": "repoCreatePullReview",
 | 
				
			||||||
 | 
					        "parameters": [
 | 
				
			||||||
 | 
					          {
 | 
				
			||||||
 | 
					            "type": "string",
 | 
				
			||||||
 | 
					            "description": "owner of the repo",
 | 
				
			||||||
 | 
					            "name": "owner",
 | 
				
			||||||
 | 
					            "in": "path",
 | 
				
			||||||
 | 
					            "required": true
 | 
				
			||||||
 | 
					          },
 | 
				
			||||||
 | 
					          {
 | 
				
			||||||
 | 
					            "type": "string",
 | 
				
			||||||
 | 
					            "description": "name of the repo",
 | 
				
			||||||
 | 
					            "name": "repo",
 | 
				
			||||||
 | 
					            "in": "path",
 | 
				
			||||||
 | 
					            "required": true
 | 
				
			||||||
 | 
					          },
 | 
				
			||||||
 | 
					          {
 | 
				
			||||||
 | 
					            "type": "integer",
 | 
				
			||||||
 | 
					            "format": "int64",
 | 
				
			||||||
 | 
					            "description": "index of the pull request",
 | 
				
			||||||
 | 
					            "name": "index",
 | 
				
			||||||
 | 
					            "in": "path",
 | 
				
			||||||
 | 
					            "required": true
 | 
				
			||||||
 | 
					          },
 | 
				
			||||||
 | 
					          {
 | 
				
			||||||
 | 
					            "name": "body",
 | 
				
			||||||
 | 
					            "in": "body",
 | 
				
			||||||
 | 
					            "required": true,
 | 
				
			||||||
 | 
					            "schema": {
 | 
				
			||||||
 | 
					              "$ref": "#/definitions/CreatePullReviewOptions"
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					          }
 | 
				
			||||||
 | 
					        ],
 | 
				
			||||||
 | 
					        "responses": {
 | 
				
			||||||
 | 
					          "200": {
 | 
				
			||||||
 | 
					            "$ref": "#/responses/PullReview"
 | 
				
			||||||
 | 
					          },
 | 
				
			||||||
 | 
					          "404": {
 | 
				
			||||||
 | 
					            "$ref": "#/responses/notFound"
 | 
				
			||||||
 | 
					          },
 | 
				
			||||||
 | 
					          "422": {
 | 
				
			||||||
 | 
					            "$ref": "#/responses/validationError"
 | 
				
			||||||
 | 
					          }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    "/repos/{owner}/{repo}/pulls/{index}/reviews/{id}": {
 | 
				
			||||||
 | 
					      "get": {
 | 
				
			||||||
 | 
					        "produces": [
 | 
				
			||||||
 | 
					          "application/json"
 | 
				
			||||||
 | 
					        ],
 | 
				
			||||||
 | 
					        "tags": [
 | 
				
			||||||
 | 
					          "repository"
 | 
				
			||||||
 | 
					        ],
 | 
				
			||||||
 | 
					        "summary": "Get a specific review for a pull request",
 | 
				
			||||||
 | 
					        "operationId": "repoGetPullReview",
 | 
				
			||||||
 | 
					        "parameters": [
 | 
				
			||||||
 | 
					          {
 | 
				
			||||||
 | 
					            "type": "string",
 | 
				
			||||||
 | 
					            "description": "owner of the repo",
 | 
				
			||||||
 | 
					            "name": "owner",
 | 
				
			||||||
 | 
					            "in": "path",
 | 
				
			||||||
 | 
					            "required": true
 | 
				
			||||||
 | 
					          },
 | 
				
			||||||
 | 
					          {
 | 
				
			||||||
 | 
					            "type": "string",
 | 
				
			||||||
 | 
					            "description": "name of the repo",
 | 
				
			||||||
 | 
					            "name": "repo",
 | 
				
			||||||
 | 
					            "in": "path",
 | 
				
			||||||
 | 
					            "required": true
 | 
				
			||||||
 | 
					          },
 | 
				
			||||||
 | 
					          {
 | 
				
			||||||
 | 
					            "type": "integer",
 | 
				
			||||||
 | 
					            "format": "int64",
 | 
				
			||||||
 | 
					            "description": "index of the pull request",
 | 
				
			||||||
 | 
					            "name": "index",
 | 
				
			||||||
 | 
					            "in": "path",
 | 
				
			||||||
 | 
					            "required": true
 | 
				
			||||||
 | 
					          },
 | 
				
			||||||
 | 
					          {
 | 
				
			||||||
 | 
					            "type": "integer",
 | 
				
			||||||
 | 
					            "format": "int64",
 | 
				
			||||||
 | 
					            "description": "id of the review",
 | 
				
			||||||
 | 
					            "name": "id",
 | 
				
			||||||
 | 
					            "in": "path",
 | 
				
			||||||
 | 
					            "required": true
 | 
				
			||||||
 | 
					          }
 | 
				
			||||||
 | 
					        ],
 | 
				
			||||||
 | 
					        "responses": {
 | 
				
			||||||
 | 
					          "200": {
 | 
				
			||||||
 | 
					            "$ref": "#/responses/PullReview"
 | 
				
			||||||
 | 
					          },
 | 
				
			||||||
 | 
					          "404": {
 | 
				
			||||||
 | 
					            "$ref": "#/responses/notFound"
 | 
				
			||||||
 | 
					          }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					      },
 | 
				
			||||||
 | 
					      "post": {
 | 
				
			||||||
 | 
					        "produces": [
 | 
				
			||||||
 | 
					          "application/json"
 | 
				
			||||||
 | 
					        ],
 | 
				
			||||||
 | 
					        "tags": [
 | 
				
			||||||
 | 
					          "repository"
 | 
				
			||||||
 | 
					        ],
 | 
				
			||||||
 | 
					        "summary": "Submit a pending review to an pull request",
 | 
				
			||||||
 | 
					        "operationId": "repoSubmitPullReview",
 | 
				
			||||||
 | 
					        "parameters": [
 | 
				
			||||||
 | 
					          {
 | 
				
			||||||
 | 
					            "type": "string",
 | 
				
			||||||
 | 
					            "description": "owner of the repo",
 | 
				
			||||||
 | 
					            "name": "owner",
 | 
				
			||||||
 | 
					            "in": "path",
 | 
				
			||||||
 | 
					            "required": true
 | 
				
			||||||
 | 
					          },
 | 
				
			||||||
 | 
					          {
 | 
				
			||||||
 | 
					            "type": "string",
 | 
				
			||||||
 | 
					            "description": "name of the repo",
 | 
				
			||||||
 | 
					            "name": "repo",
 | 
				
			||||||
 | 
					            "in": "path",
 | 
				
			||||||
 | 
					            "required": true
 | 
				
			||||||
 | 
					          },
 | 
				
			||||||
 | 
					          {
 | 
				
			||||||
 | 
					            "type": "integer",
 | 
				
			||||||
 | 
					            "format": "int64",
 | 
				
			||||||
 | 
					            "description": "index of the pull request",
 | 
				
			||||||
 | 
					            "name": "index",
 | 
				
			||||||
 | 
					            "in": "path",
 | 
				
			||||||
 | 
					            "required": true
 | 
				
			||||||
 | 
					          },
 | 
				
			||||||
 | 
					          {
 | 
				
			||||||
 | 
					            "type": "integer",
 | 
				
			||||||
 | 
					            "format": "int64",
 | 
				
			||||||
 | 
					            "description": "id of the review",
 | 
				
			||||||
 | 
					            "name": "id",
 | 
				
			||||||
 | 
					            "in": "path",
 | 
				
			||||||
 | 
					            "required": true
 | 
				
			||||||
 | 
					          },
 | 
				
			||||||
 | 
					          {
 | 
				
			||||||
 | 
					            "name": "body",
 | 
				
			||||||
 | 
					            "in": "body",
 | 
				
			||||||
 | 
					            "required": true,
 | 
				
			||||||
 | 
					            "schema": {
 | 
				
			||||||
 | 
					              "$ref": "#/definitions/SubmitPullReviewOptions"
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					          }
 | 
				
			||||||
 | 
					        ],
 | 
				
			||||||
 | 
					        "responses": {
 | 
				
			||||||
 | 
					          "200": {
 | 
				
			||||||
 | 
					            "$ref": "#/responses/PullReview"
 | 
				
			||||||
 | 
					          },
 | 
				
			||||||
 | 
					          "404": {
 | 
				
			||||||
 | 
					            "$ref": "#/responses/notFound"
 | 
				
			||||||
 | 
					          },
 | 
				
			||||||
 | 
					          "422": {
 | 
				
			||||||
 | 
					            "$ref": "#/responses/validationError"
 | 
				
			||||||
 | 
					          }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					      },
 | 
				
			||||||
 | 
					      "delete": {
 | 
				
			||||||
 | 
					        "produces": [
 | 
				
			||||||
 | 
					          "application/json"
 | 
				
			||||||
 | 
					        ],
 | 
				
			||||||
 | 
					        "tags": [
 | 
				
			||||||
 | 
					          "repository"
 | 
				
			||||||
 | 
					        ],
 | 
				
			||||||
 | 
					        "summary": "Delete a specific review from a pull request",
 | 
				
			||||||
 | 
					        "operationId": "repoDeletePullReview",
 | 
				
			||||||
 | 
					        "parameters": [
 | 
				
			||||||
 | 
					          {
 | 
				
			||||||
 | 
					            "type": "string",
 | 
				
			||||||
 | 
					            "description": "owner of the repo",
 | 
				
			||||||
 | 
					            "name": "owner",
 | 
				
			||||||
 | 
					            "in": "path",
 | 
				
			||||||
 | 
					            "required": true
 | 
				
			||||||
 | 
					          },
 | 
				
			||||||
 | 
					          {
 | 
				
			||||||
 | 
					            "type": "string",
 | 
				
			||||||
 | 
					            "description": "name of the repo",
 | 
				
			||||||
 | 
					            "name": "repo",
 | 
				
			||||||
 | 
					            "in": "path",
 | 
				
			||||||
 | 
					            "required": true
 | 
				
			||||||
 | 
					          },
 | 
				
			||||||
 | 
					          {
 | 
				
			||||||
 | 
					            "type": "integer",
 | 
				
			||||||
 | 
					            "format": "int64",
 | 
				
			||||||
 | 
					            "description": "index of the pull request",
 | 
				
			||||||
 | 
					            "name": "index",
 | 
				
			||||||
 | 
					            "in": "path",
 | 
				
			||||||
 | 
					            "required": true
 | 
				
			||||||
 | 
					          },
 | 
				
			||||||
 | 
					          {
 | 
				
			||||||
 | 
					            "type": "integer",
 | 
				
			||||||
 | 
					            "format": "int64",
 | 
				
			||||||
 | 
					            "description": "id of the review",
 | 
				
			||||||
 | 
					            "name": "id",
 | 
				
			||||||
 | 
					            "in": "path",
 | 
				
			||||||
 | 
					            "required": true
 | 
				
			||||||
 | 
					          }
 | 
				
			||||||
 | 
					        ],
 | 
				
			||||||
 | 
					        "responses": {
 | 
				
			||||||
 | 
					          "204": {
 | 
				
			||||||
 | 
					            "$ref": "#/responses/empty"
 | 
				
			||||||
 | 
					          },
 | 
				
			||||||
 | 
					          "403": {
 | 
				
			||||||
 | 
					            "$ref": "#/responses/forbidden"
 | 
				
			||||||
 | 
					          },
 | 
				
			||||||
 | 
					          "404": {
 | 
				
			||||||
 | 
					            "$ref": "#/responses/notFound"
 | 
				
			||||||
 | 
					          }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    "/repos/{owner}/{repo}/pulls/{index}/reviews/{id}/comments": {
 | 
				
			||||||
 | 
					      "get": {
 | 
				
			||||||
 | 
					        "produces": [
 | 
				
			||||||
 | 
					          "application/json"
 | 
				
			||||||
 | 
					        ],
 | 
				
			||||||
 | 
					        "tags": [
 | 
				
			||||||
 | 
					          "repository"
 | 
				
			||||||
 | 
					        ],
 | 
				
			||||||
 | 
					        "summary": "Get a specific review for a pull request",
 | 
				
			||||||
 | 
					        "operationId": "repoGetPullReviewComments",
 | 
				
			||||||
 | 
					        "parameters": [
 | 
				
			||||||
 | 
					          {
 | 
				
			||||||
 | 
					            "type": "string",
 | 
				
			||||||
 | 
					            "description": "owner of the repo",
 | 
				
			||||||
 | 
					            "name": "owner",
 | 
				
			||||||
 | 
					            "in": "path",
 | 
				
			||||||
 | 
					            "required": true
 | 
				
			||||||
 | 
					          },
 | 
				
			||||||
 | 
					          {
 | 
				
			||||||
 | 
					            "type": "string",
 | 
				
			||||||
 | 
					            "description": "name of the repo",
 | 
				
			||||||
 | 
					            "name": "repo",
 | 
				
			||||||
 | 
					            "in": "path",
 | 
				
			||||||
 | 
					            "required": true
 | 
				
			||||||
 | 
					          },
 | 
				
			||||||
 | 
					          {
 | 
				
			||||||
 | 
					            "type": "integer",
 | 
				
			||||||
 | 
					            "format": "int64",
 | 
				
			||||||
 | 
					            "description": "index of the pull request",
 | 
				
			||||||
 | 
					            "name": "index",
 | 
				
			||||||
 | 
					            "in": "path",
 | 
				
			||||||
 | 
					            "required": true
 | 
				
			||||||
 | 
					          },
 | 
				
			||||||
 | 
					          {
 | 
				
			||||||
 | 
					            "type": "integer",
 | 
				
			||||||
 | 
					            "format": "int64",
 | 
				
			||||||
 | 
					            "description": "id of the review",
 | 
				
			||||||
 | 
					            "name": "id",
 | 
				
			||||||
 | 
					            "in": "path",
 | 
				
			||||||
 | 
					            "required": true
 | 
				
			||||||
 | 
					          }
 | 
				
			||||||
 | 
					        ],
 | 
				
			||||||
 | 
					        "responses": {
 | 
				
			||||||
 | 
					          "200": {
 | 
				
			||||||
 | 
					            "$ref": "#/responses/PullReviewCommentList"
 | 
				
			||||||
 | 
					          },
 | 
				
			||||||
 | 
					          "404": {
 | 
				
			||||||
 | 
					            "$ref": "#/responses/notFound"
 | 
				
			||||||
 | 
					          }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
    "/repos/{owner}/{repo}/raw/{filepath}": {
 | 
					    "/repos/{owner}/{repo}/raw/{filepath}": {
 | 
				
			||||||
      "get": {
 | 
					      "get": {
 | 
				
			||||||
        "produces": [
 | 
					        "produces": [
 | 
				
			||||||
@@ -10975,6 +11302,59 @@
 | 
				
			|||||||
      },
 | 
					      },
 | 
				
			||||||
      "x-go-package": "code.gitea.io/gitea/modules/structs"
 | 
					      "x-go-package": "code.gitea.io/gitea/modules/structs"
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
 | 
					    "CreatePullReviewComment": {
 | 
				
			||||||
 | 
					      "description": "CreatePullReviewComment represent a review comment for creation api",
 | 
				
			||||||
 | 
					      "type": "object",
 | 
				
			||||||
 | 
					      "properties": {
 | 
				
			||||||
 | 
					        "body": {
 | 
				
			||||||
 | 
					          "type": "string",
 | 
				
			||||||
 | 
					          "x-go-name": "Body"
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "new_position": {
 | 
				
			||||||
 | 
					          "description": "if comment to new file line or 0",
 | 
				
			||||||
 | 
					          "type": "integer",
 | 
				
			||||||
 | 
					          "format": "int64",
 | 
				
			||||||
 | 
					          "x-go-name": "NewLineNum"
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "old_position": {
 | 
				
			||||||
 | 
					          "description": "if comment to old file line or 0",
 | 
				
			||||||
 | 
					          "type": "integer",
 | 
				
			||||||
 | 
					          "format": "int64",
 | 
				
			||||||
 | 
					          "x-go-name": "OldLineNum"
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "path": {
 | 
				
			||||||
 | 
					          "description": "the tree path",
 | 
				
			||||||
 | 
					          "type": "string",
 | 
				
			||||||
 | 
					          "x-go-name": "Path"
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					      },
 | 
				
			||||||
 | 
					      "x-go-package": "code.gitea.io/gitea/modules/structs"
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    "CreatePullReviewOptions": {
 | 
				
			||||||
 | 
					      "description": "CreatePullReviewOptions are options to create a pull review",
 | 
				
			||||||
 | 
					      "type": "object",
 | 
				
			||||||
 | 
					      "properties": {
 | 
				
			||||||
 | 
					        "body": {
 | 
				
			||||||
 | 
					          "type": "string",
 | 
				
			||||||
 | 
					          "x-go-name": "Body"
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "comments": {
 | 
				
			||||||
 | 
					          "type": "array",
 | 
				
			||||||
 | 
					          "items": {
 | 
				
			||||||
 | 
					            "$ref": "#/definitions/CreatePullReviewComment"
 | 
				
			||||||
 | 
					          },
 | 
				
			||||||
 | 
					          "x-go-name": "Comments"
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "commit_id": {
 | 
				
			||||||
 | 
					          "type": "string",
 | 
				
			||||||
 | 
					          "x-go-name": "CommitID"
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "event": {
 | 
				
			||||||
 | 
					          "$ref": "#/definitions/ReviewStateType"
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					      },
 | 
				
			||||||
 | 
					      "x-go-package": "code.gitea.io/gitea/modules/structs"
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
    "CreateReleaseOption": {
 | 
					    "CreateReleaseOption": {
 | 
				
			||||||
      "description": "CreateReleaseOption options when creating a release",
 | 
					      "description": "CreateReleaseOption options when creating a release",
 | 
				
			||||||
      "type": "object",
 | 
					      "type": "object",
 | 
				
			||||||
@@ -13143,6 +13523,126 @@
 | 
				
			|||||||
      },
 | 
					      },
 | 
				
			||||||
      "x-go-package": "code.gitea.io/gitea/modules/structs"
 | 
					      "x-go-package": "code.gitea.io/gitea/modules/structs"
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
 | 
					    "PullReview": {
 | 
				
			||||||
 | 
					      "description": "PullReview represents a pull request review",
 | 
				
			||||||
 | 
					      "type": "object",
 | 
				
			||||||
 | 
					      "properties": {
 | 
				
			||||||
 | 
					        "body": {
 | 
				
			||||||
 | 
					          "type": "string",
 | 
				
			||||||
 | 
					          "x-go-name": "Body"
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "comments_count": {
 | 
				
			||||||
 | 
					          "type": "integer",
 | 
				
			||||||
 | 
					          "format": "int64",
 | 
				
			||||||
 | 
					          "x-go-name": "CodeCommentsCount"
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "commit_id": {
 | 
				
			||||||
 | 
					          "type": "string",
 | 
				
			||||||
 | 
					          "x-go-name": "CommitID"
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "html_url": {
 | 
				
			||||||
 | 
					          "type": "string",
 | 
				
			||||||
 | 
					          "x-go-name": "HTMLURL"
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "id": {
 | 
				
			||||||
 | 
					          "type": "integer",
 | 
				
			||||||
 | 
					          "format": "int64",
 | 
				
			||||||
 | 
					          "x-go-name": "ID"
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "official": {
 | 
				
			||||||
 | 
					          "type": "boolean",
 | 
				
			||||||
 | 
					          "x-go-name": "Official"
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "pull_request_url": {
 | 
				
			||||||
 | 
					          "type": "string",
 | 
				
			||||||
 | 
					          "x-go-name": "HTMLPullURL"
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "stale": {
 | 
				
			||||||
 | 
					          "type": "boolean",
 | 
				
			||||||
 | 
					          "x-go-name": "Stale"
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "state": {
 | 
				
			||||||
 | 
					          "$ref": "#/definitions/ReviewStateType"
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "submitted_at": {
 | 
				
			||||||
 | 
					          "type": "string",
 | 
				
			||||||
 | 
					          "format": "date-time",
 | 
				
			||||||
 | 
					          "x-go-name": "Submitted"
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "user": {
 | 
				
			||||||
 | 
					          "$ref": "#/definitions/User"
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					      },
 | 
				
			||||||
 | 
					      "x-go-package": "code.gitea.io/gitea/modules/structs"
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    "PullReviewComment": {
 | 
				
			||||||
 | 
					      "description": "PullReviewComment represents a comment on a pull request review",
 | 
				
			||||||
 | 
					      "type": "object",
 | 
				
			||||||
 | 
					      "properties": {
 | 
				
			||||||
 | 
					        "body": {
 | 
				
			||||||
 | 
					          "type": "string",
 | 
				
			||||||
 | 
					          "x-go-name": "Body"
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "commit_id": {
 | 
				
			||||||
 | 
					          "type": "string",
 | 
				
			||||||
 | 
					          "x-go-name": "CommitID"
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "created_at": {
 | 
				
			||||||
 | 
					          "type": "string",
 | 
				
			||||||
 | 
					          "format": "date-time",
 | 
				
			||||||
 | 
					          "x-go-name": "Created"
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "diff_hunk": {
 | 
				
			||||||
 | 
					          "type": "string",
 | 
				
			||||||
 | 
					          "x-go-name": "DiffHunk"
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "html_url": {
 | 
				
			||||||
 | 
					          "type": "string",
 | 
				
			||||||
 | 
					          "x-go-name": "HTMLURL"
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "id": {
 | 
				
			||||||
 | 
					          "type": "integer",
 | 
				
			||||||
 | 
					          "format": "int64",
 | 
				
			||||||
 | 
					          "x-go-name": "ID"
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "original_commit_id": {
 | 
				
			||||||
 | 
					          "type": "string",
 | 
				
			||||||
 | 
					          "x-go-name": "OrigCommitID"
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "original_position": {
 | 
				
			||||||
 | 
					          "type": "integer",
 | 
				
			||||||
 | 
					          "format": "uint64",
 | 
				
			||||||
 | 
					          "x-go-name": "OldLineNum"
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "path": {
 | 
				
			||||||
 | 
					          "type": "string",
 | 
				
			||||||
 | 
					          "x-go-name": "Path"
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "position": {
 | 
				
			||||||
 | 
					          "type": "integer",
 | 
				
			||||||
 | 
					          "format": "uint64",
 | 
				
			||||||
 | 
					          "x-go-name": "LineNum"
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "pull_request_review_id": {
 | 
				
			||||||
 | 
					          "type": "integer",
 | 
				
			||||||
 | 
					          "format": "int64",
 | 
				
			||||||
 | 
					          "x-go-name": "ReviewID"
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "pull_request_url": {
 | 
				
			||||||
 | 
					          "type": "string",
 | 
				
			||||||
 | 
					          "x-go-name": "HTMLPullURL"
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "updated_at": {
 | 
				
			||||||
 | 
					          "type": "string",
 | 
				
			||||||
 | 
					          "format": "date-time",
 | 
				
			||||||
 | 
					          "x-go-name": "Updated"
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "user": {
 | 
				
			||||||
 | 
					          "$ref": "#/definitions/User"
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					      },
 | 
				
			||||||
 | 
					      "x-go-package": "code.gitea.io/gitea/modules/structs"
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
    "Reaction": {
 | 
					    "Reaction": {
 | 
				
			||||||
      "description": "Reaction contain one reaction",
 | 
					      "description": "Reaction contain one reaction",
 | 
				
			||||||
      "type": "object",
 | 
					      "type": "object",
 | 
				
			||||||
@@ -13486,6 +13986,11 @@
 | 
				
			|||||||
      },
 | 
					      },
 | 
				
			||||||
      "x-go-package": "code.gitea.io/gitea/modules/structs"
 | 
					      "x-go-package": "code.gitea.io/gitea/modules/structs"
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
 | 
					    "ReviewStateType": {
 | 
				
			||||||
 | 
					      "description": "ReviewStateType review state type",
 | 
				
			||||||
 | 
					      "type": "string",
 | 
				
			||||||
 | 
					      "x-go-package": "code.gitea.io/gitea/modules/structs"
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
    "SearchResults": {
 | 
					    "SearchResults": {
 | 
				
			||||||
      "description": "SearchResults results of a successful search",
 | 
					      "description": "SearchResults results of a successful search",
 | 
				
			||||||
      "type": "object",
 | 
					      "type": "object",
 | 
				
			||||||
@@ -13586,6 +14091,20 @@
 | 
				
			|||||||
      },
 | 
					      },
 | 
				
			||||||
      "x-go-package": "code.gitea.io/gitea/modules/structs"
 | 
					      "x-go-package": "code.gitea.io/gitea/modules/structs"
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
 | 
					    "SubmitPullReviewOptions": {
 | 
				
			||||||
 | 
					      "description": "SubmitPullReviewOptions are options to submit a pending pull review",
 | 
				
			||||||
 | 
					      "type": "object",
 | 
				
			||||||
 | 
					      "properties": {
 | 
				
			||||||
 | 
					        "body": {
 | 
				
			||||||
 | 
					          "type": "string",
 | 
				
			||||||
 | 
					          "x-go-name": "Body"
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "event": {
 | 
				
			||||||
 | 
					          "$ref": "#/definitions/ReviewStateType"
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					      },
 | 
				
			||||||
 | 
					      "x-go-package": "code.gitea.io/gitea/modules/structs"
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
    "Tag": {
 | 
					    "Tag": {
 | 
				
			||||||
      "description": "Tag represents a repository tag",
 | 
					      "description": "Tag represents a repository tag",
 | 
				
			||||||
      "type": "object",
 | 
					      "type": "object",
 | 
				
			||||||
@@ -14324,6 +14843,36 @@
 | 
				
			|||||||
        }
 | 
					        }
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
 | 
					    "PullReview": {
 | 
				
			||||||
 | 
					      "description": "PullReview",
 | 
				
			||||||
 | 
					      "schema": {
 | 
				
			||||||
 | 
					        "$ref": "#/definitions/PullReview"
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    "PullReviewComment": {
 | 
				
			||||||
 | 
					      "description": "PullComment",
 | 
				
			||||||
 | 
					      "schema": {
 | 
				
			||||||
 | 
					        "$ref": "#/definitions/PullReviewComment"
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    "PullReviewCommentList": {
 | 
				
			||||||
 | 
					      "description": "PullCommentList",
 | 
				
			||||||
 | 
					      "schema": {
 | 
				
			||||||
 | 
					        "type": "array",
 | 
				
			||||||
 | 
					        "items": {
 | 
				
			||||||
 | 
					          "$ref": "#/definitions/PullReviewComment"
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    "PullReviewList": {
 | 
				
			||||||
 | 
					      "description": "PullReviewList",
 | 
				
			||||||
 | 
					      "schema": {
 | 
				
			||||||
 | 
					        "type": "array",
 | 
				
			||||||
 | 
					        "items": {
 | 
				
			||||||
 | 
					          "$ref": "#/definitions/PullReview"
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
    "Reaction": {
 | 
					    "Reaction": {
 | 
				
			||||||
      "description": "Reaction",
 | 
					      "description": "Reaction",
 | 
				
			||||||
      "schema": {
 | 
					      "schema": {
 | 
				
			||||||
@@ -14561,7 +15110,7 @@
 | 
				
			|||||||
    "parameterBodies": {
 | 
					    "parameterBodies": {
 | 
				
			||||||
      "description": "parameterBodies",
 | 
					      "description": "parameterBodies",
 | 
				
			||||||
      "schema": {
 | 
					      "schema": {
 | 
				
			||||||
        "$ref": "#/definitions/CreateOAuth2ApplicationOptions"
 | 
					        "$ref": "#/definitions/SubmitPullReviewOptions"
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
    "redirect": {
 | 
					    "redirect": {
 | 
				
			||||||
 
 | 
				
			|||||||
		Reference in New Issue
	
	Block a user