You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

api_pull_review_test.go 11KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247
  1. // Copyright 2020 The Gitea Authors. All rights reserved.
  2. // Use of this source code is governed by a MIT-style
  3. // license that can be found in the LICENSE file.
  4. package integrations
  5. import (
  6. "fmt"
  7. "net/http"
  8. "testing"
  9. "code.gitea.io/gitea/models"
  10. api "code.gitea.io/gitea/modules/structs"
  11. "github.com/stretchr/testify/assert"
  12. )
  13. func TestAPIPullReview(t *testing.T) {
  14. defer prepareTestEnv(t)()
  15. pullIssue := models.AssertExistsAndLoadBean(t, &models.Issue{ID: 3}).(*models.Issue)
  16. assert.NoError(t, pullIssue.LoadAttributes())
  17. repo := models.AssertExistsAndLoadBean(t, &models.Repository{ID: pullIssue.RepoID}).(*models.Repository)
  18. // test ListPullReviews
  19. session := loginUser(t, "user2")
  20. token := getTokenForLoggedInUser(t, session)
  21. req := NewRequestf(t, http.MethodGet, "/api/v1/repos/%s/%s/pulls/%d/reviews?token=%s", repo.OwnerName, repo.Name, pullIssue.Index, token)
  22. resp := session.MakeRequest(t, req, http.StatusOK)
  23. var reviews []*api.PullReview
  24. DecodeJSON(t, resp, &reviews)
  25. if !assert.Len(t, reviews, 6) {
  26. return
  27. }
  28. for _, r := range reviews {
  29. assert.EqualValues(t, pullIssue.HTMLURL(), r.HTMLPullURL)
  30. }
  31. assert.EqualValues(t, 8, reviews[3].ID)
  32. assert.EqualValues(t, "APPROVED", reviews[3].State)
  33. assert.EqualValues(t, 0, reviews[3].CodeCommentsCount)
  34. assert.EqualValues(t, true, reviews[3].Stale)
  35. assert.EqualValues(t, false, reviews[3].Official)
  36. assert.EqualValues(t, 10, reviews[5].ID)
  37. assert.EqualValues(t, "REQUEST_CHANGES", reviews[5].State)
  38. assert.EqualValues(t, 1, reviews[5].CodeCommentsCount)
  39. assert.EqualValues(t, -1, reviews[5].Reviewer.ID) // ghost user
  40. assert.EqualValues(t, false, reviews[5].Stale)
  41. assert.EqualValues(t, true, reviews[5].Official)
  42. // test GetPullReview
  43. 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)
  44. resp = session.MakeRequest(t, req, http.StatusOK)
  45. var review api.PullReview
  46. DecodeJSON(t, resp, &review)
  47. assert.EqualValues(t, *reviews[3], review)
  48. 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)
  49. resp = session.MakeRequest(t, req, http.StatusOK)
  50. DecodeJSON(t, resp, &review)
  51. assert.EqualValues(t, *reviews[5], review)
  52. // test GetPullReviewComments
  53. comment := models.AssertExistsAndLoadBean(t, &models.Comment{ID: 7}).(*models.Comment)
  54. 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)
  55. resp = session.MakeRequest(t, req, http.StatusOK)
  56. var reviewComments []*api.PullReviewComment
  57. DecodeJSON(t, resp, &reviewComments)
  58. assert.Len(t, reviewComments, 1)
  59. assert.EqualValues(t, "Ghost", reviewComments[0].Poster.UserName)
  60. assert.EqualValues(t, "a review from a deleted user", reviewComments[0].Body)
  61. assert.EqualValues(t, comment.ID, reviewComments[0].ID)
  62. assert.EqualValues(t, comment.UpdatedUnix, reviewComments[0].Updated.Unix())
  63. assert.EqualValues(t, comment.HTMLURL(), reviewComments[0].HTMLURL)
  64. // test CreatePullReview
  65. 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{
  66. Body: "body1",
  67. // Event: "" # will result in PENDING
  68. Comments: []api.CreatePullReviewComment{{
  69. Path: "README.md",
  70. Body: "first new line",
  71. OldLineNum: 0,
  72. NewLineNum: 1,
  73. }, {
  74. Path: "README.md",
  75. Body: "first old line",
  76. OldLineNum: 1,
  77. NewLineNum: 0,
  78. }, {
  79. Path: "iso-8859-1.txt",
  80. Body: "this line contains a non-utf-8 character",
  81. OldLineNum: 0,
  82. NewLineNum: 1,
  83. },
  84. },
  85. })
  86. resp = session.MakeRequest(t, req, http.StatusOK)
  87. DecodeJSON(t, resp, &review)
  88. assert.EqualValues(t, 6, review.ID)
  89. assert.EqualValues(t, "PENDING", review.State)
  90. assert.EqualValues(t, 3, review.CodeCommentsCount)
  91. // test SubmitPullReview
  92. 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{
  93. Event: "APPROVED",
  94. Body: "just two nits",
  95. })
  96. resp = session.MakeRequest(t, req, http.StatusOK)
  97. DecodeJSON(t, resp, &review)
  98. assert.EqualValues(t, 6, review.ID)
  99. assert.EqualValues(t, "APPROVED", review.State)
  100. assert.EqualValues(t, 3, review.CodeCommentsCount)
  101. // test dismiss review
  102. req = NewRequestWithJSON(t, http.MethodPost, fmt.Sprintf("/api/v1/repos/%s/%s/pulls/%d/reviews/%d/dismissals?token=%s", repo.OwnerName, repo.Name, pullIssue.Index, review.ID, token), &api.DismissPullReviewOptions{
  103. Message: "test",
  104. })
  105. resp = session.MakeRequest(t, req, http.StatusOK)
  106. DecodeJSON(t, resp, &review)
  107. assert.EqualValues(t, 6, review.ID)
  108. assert.EqualValues(t, true, review.Dismissed)
  109. // test dismiss review
  110. req = NewRequest(t, http.MethodPost, fmt.Sprintf("/api/v1/repos/%s/%s/pulls/%d/reviews/%d/undismissals?token=%s", repo.OwnerName, repo.Name, pullIssue.Index, review.ID, token))
  111. resp = session.MakeRequest(t, req, http.StatusOK)
  112. DecodeJSON(t, resp, &review)
  113. assert.EqualValues(t, 6, review.ID)
  114. assert.EqualValues(t, false, review.Dismissed)
  115. // test DeletePullReview
  116. 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{
  117. Body: "just a comment",
  118. Event: "COMMENT",
  119. })
  120. resp = session.MakeRequest(t, req, http.StatusOK)
  121. DecodeJSON(t, resp, &review)
  122. assert.EqualValues(t, "COMMENT", review.State)
  123. assert.EqualValues(t, 0, review.CodeCommentsCount)
  124. 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)
  125. resp = session.MakeRequest(t, req, http.StatusNoContent)
  126. // test get review requests
  127. // to make it simple, use same api with get review
  128. pullIssue12 := models.AssertExistsAndLoadBean(t, &models.Issue{ID: 12}).(*models.Issue)
  129. assert.NoError(t, pullIssue12.LoadAttributes())
  130. repo3 := models.AssertExistsAndLoadBean(t, &models.Repository{ID: pullIssue12.RepoID}).(*models.Repository)
  131. req = NewRequestf(t, http.MethodGet, "/api/v1/repos/%s/%s/pulls/%d/reviews?token=%s", repo3.OwnerName, repo3.Name, pullIssue12.Index, token)
  132. resp = session.MakeRequest(t, req, http.StatusOK)
  133. DecodeJSON(t, resp, &reviews)
  134. assert.EqualValues(t, 11, reviews[0].ID)
  135. assert.EqualValues(t, "REQUEST_REVIEW", reviews[0].State)
  136. assert.EqualValues(t, 0, reviews[0].CodeCommentsCount)
  137. assert.EqualValues(t, false, reviews[0].Stale)
  138. assert.EqualValues(t, true, reviews[0].Official)
  139. assert.EqualValues(t, "test_team", reviews[0].ReviewerTeam.Name)
  140. assert.EqualValues(t, 12, reviews[1].ID)
  141. assert.EqualValues(t, "REQUEST_REVIEW", reviews[1].State)
  142. assert.EqualValues(t, 0, reviews[0].CodeCommentsCount)
  143. assert.EqualValues(t, false, reviews[1].Stale)
  144. assert.EqualValues(t, true, reviews[1].Official)
  145. assert.EqualValues(t, 1, reviews[1].Reviewer.ID)
  146. }
  147. func TestAPIPullReviewRequest(t *testing.T) {
  148. defer prepareTestEnv(t)()
  149. pullIssue := models.AssertExistsAndLoadBean(t, &models.Issue{ID: 3}).(*models.Issue)
  150. assert.NoError(t, pullIssue.LoadAttributes())
  151. repo := models.AssertExistsAndLoadBean(t, &models.Repository{ID: pullIssue.RepoID}).(*models.Repository)
  152. // Test add Review Request
  153. session := loginUser(t, "user2")
  154. token := getTokenForLoggedInUser(t, session)
  155. req := NewRequestWithJSON(t, http.MethodPost, fmt.Sprintf("/api/v1/repos/%s/%s/pulls/%d/requested_reviewers?token=%s", repo.OwnerName, repo.Name, pullIssue.Index, token), &api.PullReviewRequestOptions{
  156. Reviewers: []string{"user4@example.com", "user8"},
  157. })
  158. session.MakeRequest(t, req, http.StatusCreated)
  159. // poster of pr can't be reviewer
  160. req = NewRequestWithJSON(t, http.MethodPost, fmt.Sprintf("/api/v1/repos/%s/%s/pulls/%d/requested_reviewers?token=%s", repo.OwnerName, repo.Name, pullIssue.Index, token), &api.PullReviewRequestOptions{
  161. Reviewers: []string{"user1"},
  162. })
  163. session.MakeRequest(t, req, http.StatusUnprocessableEntity)
  164. // test user not exist
  165. req = NewRequestWithJSON(t, http.MethodPost, fmt.Sprintf("/api/v1/repos/%s/%s/pulls/%d/requested_reviewers?token=%s", repo.OwnerName, repo.Name, pullIssue.Index, token), &api.PullReviewRequestOptions{
  166. Reviewers: []string{"testOther"},
  167. })
  168. session.MakeRequest(t, req, http.StatusNotFound)
  169. // Test Remove Review Request
  170. session2 := loginUser(t, "user4")
  171. token2 := getTokenForLoggedInUser(t, session2)
  172. req = NewRequestWithJSON(t, http.MethodDelete, fmt.Sprintf("/api/v1/repos/%s/%s/pulls/%d/requested_reviewers?token=%s", repo.OwnerName, repo.Name, pullIssue.Index, token2), &api.PullReviewRequestOptions{
  173. Reviewers: []string{"user4"},
  174. })
  175. session.MakeRequest(t, req, http.StatusNoContent)
  176. // doer is not admin
  177. req = NewRequestWithJSON(t, http.MethodDelete, fmt.Sprintf("/api/v1/repos/%s/%s/pulls/%d/requested_reviewers?token=%s", repo.OwnerName, repo.Name, pullIssue.Index, token2), &api.PullReviewRequestOptions{
  178. Reviewers: []string{"user8"},
  179. })
  180. session.MakeRequest(t, req, http.StatusUnprocessableEntity)
  181. req = NewRequestWithJSON(t, http.MethodDelete, fmt.Sprintf("/api/v1/repos/%s/%s/pulls/%d/requested_reviewers?token=%s", repo.OwnerName, repo.Name, pullIssue.Index, token), &api.PullReviewRequestOptions{
  182. Reviewers: []string{"user8"},
  183. })
  184. session.MakeRequest(t, req, http.StatusNoContent)
  185. // Test team review request
  186. pullIssue12 := models.AssertExistsAndLoadBean(t, &models.Issue{ID: 12}).(*models.Issue)
  187. assert.NoError(t, pullIssue12.LoadAttributes())
  188. repo3 := models.AssertExistsAndLoadBean(t, &models.Repository{ID: pullIssue12.RepoID}).(*models.Repository)
  189. // Test add Team Review Request
  190. req = NewRequestWithJSON(t, http.MethodPost, fmt.Sprintf("/api/v1/repos/%s/%s/pulls/%d/requested_reviewers?token=%s", repo3.OwnerName, repo3.Name, pullIssue12.Index, token), &api.PullReviewRequestOptions{
  191. TeamReviewers: []string{"team1", "owners"},
  192. })
  193. session.MakeRequest(t, req, http.StatusCreated)
  194. // Test add Team Review Request to not allowned
  195. req = NewRequestWithJSON(t, http.MethodPost, fmt.Sprintf("/api/v1/repos/%s/%s/pulls/%d/requested_reviewers?token=%s", repo3.OwnerName, repo3.Name, pullIssue12.Index, token), &api.PullReviewRequestOptions{
  196. TeamReviewers: []string{"test_team"},
  197. })
  198. session.MakeRequest(t, req, http.StatusUnprocessableEntity)
  199. // Test add Team Review Request to not exist
  200. req = NewRequestWithJSON(t, http.MethodPost, fmt.Sprintf("/api/v1/repos/%s/%s/pulls/%d/requested_reviewers?token=%s", repo3.OwnerName, repo3.Name, pullIssue12.Index, token), &api.PullReviewRequestOptions{
  201. TeamReviewers: []string{"not_exist_team"},
  202. })
  203. session.MakeRequest(t, req, http.StatusNotFound)
  204. // Test Remove team Review Request
  205. req = NewRequestWithJSON(t, http.MethodDelete, fmt.Sprintf("/api/v1/repos/%s/%s/pulls/%d/requested_reviewers?token=%s", repo3.OwnerName, repo3.Name, pullIssue12.Index, token), &api.PullReviewRequestOptions{
  206. TeamReviewers: []string{"team1"},
  207. })
  208. session.MakeRequest(t, req, http.StatusNoContent)
  209. // empty request test
  210. req = NewRequestWithJSON(t, http.MethodPost, fmt.Sprintf("/api/v1/repos/%s/%s/pulls/%d/requested_reviewers?token=%s", repo3.OwnerName, repo3.Name, pullIssue12.Index, token), &api.PullReviewRequestOptions{})
  211. session.MakeRequest(t, req, http.StatusCreated)
  212. req = NewRequestWithJSON(t, http.MethodDelete, fmt.Sprintf("/api/v1/repos/%s/%s/pulls/%d/requested_reviewers?token=%s", repo3.OwnerName, repo3.Name, pullIssue12.Index, token), &api.PullReviewRequestOptions{})
  213. session.MakeRequest(t, req, http.StatusNoContent)
  214. }