Você não pode selecionar mais de 25 tópicos Os tópicos devem começar com uma letra ou um número, podem incluir traços ('-') e podem ter até 35 caracteres.

api_pull_review_test.go 14KB

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