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 14KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306
  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/db"
  10. issues_model "code.gitea.io/gitea/models/issues"
  11. repo_model "code.gitea.io/gitea/models/repo"
  12. "code.gitea.io/gitea/models/unittest"
  13. "code.gitea.io/gitea/modules/json"
  14. api "code.gitea.io/gitea/modules/structs"
  15. "github.com/stretchr/testify/assert"
  16. )
  17. func TestAPIPullReview(t *testing.T) {
  18. defer prepareTestEnv(t)()
  19. pullIssue := unittest.AssertExistsAndLoadBean(t, &issues_model.Issue{ID: 3}).(*issues_model.Issue)
  20. assert.NoError(t, pullIssue.LoadAttributes(db.DefaultContext))
  21. repo := unittest.AssertExistsAndLoadBean(t, &repo_model.Repository{ID: pullIssue.RepoID}).(*repo_model.Repository)
  22. // test ListPullReviews
  23. session := loginUser(t, "user2")
  24. token := getTokenForLoggedInUser(t, session)
  25. req := NewRequestf(t, http.MethodGet, "/api/v1/repos/%s/%s/pulls/%d/reviews?token=%s", repo.OwnerName, repo.Name, pullIssue.Index, token)
  26. resp := session.MakeRequest(t, req, http.StatusOK)
  27. var reviews []*api.PullReview
  28. DecodeJSON(t, resp, &reviews)
  29. if !assert.Len(t, reviews, 6) {
  30. return
  31. }
  32. for _, r := range reviews {
  33. assert.EqualValues(t, pullIssue.HTMLURL(), r.HTMLPullURL)
  34. }
  35. assert.EqualValues(t, 8, reviews[3].ID)
  36. assert.EqualValues(t, "APPROVED", reviews[3].State)
  37. assert.EqualValues(t, 0, reviews[3].CodeCommentsCount)
  38. assert.True(t, reviews[3].Stale)
  39. assert.False(t, reviews[3].Official)
  40. assert.EqualValues(t, 10, reviews[5].ID)
  41. assert.EqualValues(t, "REQUEST_CHANGES", reviews[5].State)
  42. assert.EqualValues(t, 1, reviews[5].CodeCommentsCount)
  43. assert.EqualValues(t, -1, reviews[5].Reviewer.ID) // ghost user
  44. assert.False(t, reviews[5].Stale)
  45. assert.True(t, reviews[5].Official)
  46. // test GetPullReview
  47. 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)
  48. resp = session.MakeRequest(t, req, http.StatusOK)
  49. var review api.PullReview
  50. DecodeJSON(t, resp, &review)
  51. assert.EqualValues(t, *reviews[3], review)
  52. 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)
  53. resp = session.MakeRequest(t, req, http.StatusOK)
  54. DecodeJSON(t, resp, &review)
  55. assert.EqualValues(t, *reviews[5], review)
  56. // test GetPullReviewComments
  57. comment := unittest.AssertExistsAndLoadBean(t, &issues_model.Comment{ID: 7}).(*issues_model.Comment)
  58. 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)
  59. resp = session.MakeRequest(t, req, http.StatusOK)
  60. var reviewComments []*api.PullReviewComment
  61. DecodeJSON(t, resp, &reviewComments)
  62. assert.Len(t, reviewComments, 1)
  63. assert.EqualValues(t, "Ghost", reviewComments[0].Poster.UserName)
  64. assert.EqualValues(t, "a review from a deleted user", reviewComments[0].Body)
  65. assert.EqualValues(t, comment.ID, reviewComments[0].ID)
  66. assert.EqualValues(t, comment.UpdatedUnix, reviewComments[0].Updated.Unix())
  67. assert.EqualValues(t, comment.HTMLURL(), reviewComments[0].HTMLURL)
  68. // test CreatePullReview
  69. 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{
  70. Body: "body1",
  71. // Event: "" # will result in PENDING
  72. Comments: []api.CreatePullReviewComment{
  73. {
  74. Path: "README.md",
  75. Body: "first new line",
  76. OldLineNum: 0,
  77. NewLineNum: 1,
  78. }, {
  79. Path: "README.md",
  80. Body: "first old line",
  81. OldLineNum: 1,
  82. NewLineNum: 0,
  83. }, {
  84. Path: "iso-8859-1.txt",
  85. Body: "this line contains a non-utf-8 character",
  86. OldLineNum: 0,
  87. NewLineNum: 1,
  88. },
  89. },
  90. })
  91. resp = session.MakeRequest(t, req, http.StatusOK)
  92. DecodeJSON(t, resp, &review)
  93. assert.EqualValues(t, 6, review.ID)
  94. assert.EqualValues(t, "PENDING", review.State)
  95. assert.EqualValues(t, 3, review.CodeCommentsCount)
  96. // test SubmitPullReview
  97. 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{
  98. Event: "APPROVED",
  99. Body: "just two nits",
  100. })
  101. resp = session.MakeRequest(t, req, http.StatusOK)
  102. DecodeJSON(t, resp, &review)
  103. assert.EqualValues(t, 6, review.ID)
  104. assert.EqualValues(t, "APPROVED", review.State)
  105. assert.EqualValues(t, 3, review.CodeCommentsCount)
  106. // test dismiss review
  107. 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{
  108. Message: "test",
  109. })
  110. resp = session.MakeRequest(t, req, http.StatusOK)
  111. DecodeJSON(t, resp, &review)
  112. assert.EqualValues(t, 6, review.ID)
  113. assert.True(t, review.Dismissed)
  114. // test dismiss review
  115. 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))
  116. resp = session.MakeRequest(t, req, http.StatusOK)
  117. DecodeJSON(t, resp, &review)
  118. assert.EqualValues(t, 6, review.ID)
  119. assert.False(t, review.Dismissed)
  120. // test DeletePullReview
  121. 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{
  122. Body: "just a comment",
  123. Event: "COMMENT",
  124. })
  125. resp = session.MakeRequest(t, req, http.StatusOK)
  126. DecodeJSON(t, resp, &review)
  127. assert.EqualValues(t, "COMMENT", review.State)
  128. assert.EqualValues(t, 0, review.CodeCommentsCount)
  129. 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)
  130. session.MakeRequest(t, req, http.StatusNoContent)
  131. // test CreatePullReview Comment without body but with comments
  132. 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{
  133. // Body: "",
  134. Event: "COMMENT",
  135. Comments: []api.CreatePullReviewComment{
  136. {
  137. Path: "README.md",
  138. Body: "first new line",
  139. OldLineNum: 0,
  140. NewLineNum: 1,
  141. }, {
  142. Path: "README.md",
  143. Body: "first old line",
  144. OldLineNum: 1,
  145. NewLineNum: 0,
  146. },
  147. },
  148. })
  149. var commentReview api.PullReview
  150. resp = session.MakeRequest(t, req, http.StatusOK)
  151. DecodeJSON(t, resp, &commentReview)
  152. assert.EqualValues(t, "COMMENT", commentReview.State)
  153. assert.EqualValues(t, 2, commentReview.CodeCommentsCount)
  154. assert.EqualValues(t, "", commentReview.Body)
  155. assert.EqualValues(t, false, commentReview.Dismissed)
  156. // test CreatePullReview Comment with body but without comments
  157. commentBody := "This is a body of the comment."
  158. 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{
  159. Body: commentBody,
  160. Event: "COMMENT",
  161. Comments: []api.CreatePullReviewComment{},
  162. })
  163. resp = session.MakeRequest(t, req, http.StatusOK)
  164. DecodeJSON(t, resp, &commentReview)
  165. assert.EqualValues(t, "COMMENT", commentReview.State)
  166. assert.EqualValues(t, 0, commentReview.CodeCommentsCount)
  167. assert.EqualValues(t, commentBody, commentReview.Body)
  168. assert.EqualValues(t, false, commentReview.Dismissed)
  169. // test CreatePullReview Comment without body and no comments
  170. 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{
  171. Body: "",
  172. Event: "COMMENT",
  173. Comments: []api.CreatePullReviewComment{},
  174. })
  175. resp = session.MakeRequest(t, req, http.StatusUnprocessableEntity)
  176. errMap := make(map[string]interface{})
  177. json.Unmarshal(resp.Body.Bytes(), &errMap)
  178. assert.EqualValues(t, "review event COMMENT requires a body or a comment", errMap["message"].(string))
  179. // test get review requests
  180. // to make it simple, use same api with get review
  181. pullIssue12 := unittest.AssertExistsAndLoadBean(t, &issues_model.Issue{ID: 12}).(*issues_model.Issue)
  182. assert.NoError(t, pullIssue12.LoadAttributes(db.DefaultContext))
  183. repo3 := unittest.AssertExistsAndLoadBean(t, &repo_model.Repository{ID: pullIssue12.RepoID}).(*repo_model.Repository)
  184. req = NewRequestf(t, http.MethodGet, "/api/v1/repos/%s/%s/pulls/%d/reviews?token=%s", repo3.OwnerName, repo3.Name, pullIssue12.Index, token)
  185. resp = session.MakeRequest(t, req, http.StatusOK)
  186. DecodeJSON(t, resp, &reviews)
  187. assert.EqualValues(t, 11, reviews[0].ID)
  188. assert.EqualValues(t, "REQUEST_REVIEW", reviews[0].State)
  189. assert.EqualValues(t, 0, reviews[0].CodeCommentsCount)
  190. assert.False(t, reviews[0].Stale)
  191. assert.True(t, reviews[0].Official)
  192. assert.EqualValues(t, "test_team", reviews[0].ReviewerTeam.Name)
  193. assert.EqualValues(t, 12, reviews[1].ID)
  194. assert.EqualValues(t, "REQUEST_REVIEW", reviews[1].State)
  195. assert.EqualValues(t, 0, reviews[0].CodeCommentsCount)
  196. assert.False(t, reviews[1].Stale)
  197. assert.True(t, reviews[1].Official)
  198. assert.EqualValues(t, 1, reviews[1].Reviewer.ID)
  199. }
  200. func TestAPIPullReviewRequest(t *testing.T) {
  201. defer prepareTestEnv(t)()
  202. pullIssue := unittest.AssertExistsAndLoadBean(t, &issues_model.Issue{ID: 3}).(*issues_model.Issue)
  203. assert.NoError(t, pullIssue.LoadAttributes(db.DefaultContext))
  204. repo := unittest.AssertExistsAndLoadBean(t, &repo_model.Repository{ID: pullIssue.RepoID}).(*repo_model.Repository)
  205. // Test add Review Request
  206. session := loginUser(t, "user2")
  207. token := getTokenForLoggedInUser(t, session)
  208. 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{
  209. Reviewers: []string{"user4@example.com", "user8"},
  210. })
  211. session.MakeRequest(t, req, http.StatusCreated)
  212. // poster of pr can't be reviewer
  213. 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{
  214. Reviewers: []string{"user1"},
  215. })
  216. session.MakeRequest(t, req, http.StatusUnprocessableEntity)
  217. // test user not exist
  218. 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{
  219. Reviewers: []string{"testOther"},
  220. })
  221. session.MakeRequest(t, req, http.StatusNotFound)
  222. // Test Remove Review Request
  223. session2 := loginUser(t, "user4")
  224. token2 := getTokenForLoggedInUser(t, session2)
  225. 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{
  226. Reviewers: []string{"user4"},
  227. })
  228. session.MakeRequest(t, req, http.StatusNoContent)
  229. // doer is not admin
  230. 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{
  231. Reviewers: []string{"user8"},
  232. })
  233. session.MakeRequest(t, req, http.StatusUnprocessableEntity)
  234. 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{
  235. Reviewers: []string{"user8"},
  236. })
  237. session.MakeRequest(t, req, http.StatusNoContent)
  238. // Test team review request
  239. pullIssue12 := unittest.AssertExistsAndLoadBean(t, &issues_model.Issue{ID: 12}).(*issues_model.Issue)
  240. assert.NoError(t, pullIssue12.LoadAttributes(db.DefaultContext))
  241. repo3 := unittest.AssertExistsAndLoadBean(t, &repo_model.Repository{ID: pullIssue12.RepoID}).(*repo_model.Repository)
  242. // Test add Team Review Request
  243. 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{
  244. TeamReviewers: []string{"team1", "owners"},
  245. })
  246. session.MakeRequest(t, req, http.StatusCreated)
  247. // Test add Team Review Request to not allowned
  248. 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{
  249. TeamReviewers: []string{"test_team"},
  250. })
  251. session.MakeRequest(t, req, http.StatusUnprocessableEntity)
  252. // Test add Team Review Request to not exist
  253. 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{
  254. TeamReviewers: []string{"not_exist_team"},
  255. })
  256. session.MakeRequest(t, req, http.StatusNotFound)
  257. // Test Remove team Review Request
  258. 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{
  259. TeamReviewers: []string{"team1"},
  260. })
  261. session.MakeRequest(t, req, http.StatusNoContent)
  262. // empty request test
  263. 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{})
  264. session.MakeRequest(t, req, http.StatusCreated)
  265. 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{})
  266. session.MakeRequest(t, req, http.StatusNoContent)
  267. }