mirror of
https://codeberg.org/forgejo/forgejo.git
synced 2025-01-02 14:28:52 -05: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:
parent
4ed7d2a2bb
commit
c97494a4f4
12 changed files with 1580 additions and 26 deletions
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": {
|
||||||
|
|
Loading…
Reference in a new issue