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.

issue_test.go 8.9KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325
  1. // Copyright 2017 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 models
  5. import (
  6. "sort"
  7. "testing"
  8. "time"
  9. "github.com/stretchr/testify/assert"
  10. "xorm.io/builder"
  11. )
  12. func TestIssue_ReplaceLabels(t *testing.T) {
  13. assert.NoError(t, PrepareTestDatabase())
  14. testSuccess := func(issueID int64, labelIDs []int64) {
  15. issue := AssertExistsAndLoadBean(t, &Issue{ID: issueID}).(*Issue)
  16. repo := AssertExistsAndLoadBean(t, &Repository{ID: issue.RepoID}).(*Repository)
  17. doer := AssertExistsAndLoadBean(t, &User{ID: repo.OwnerID}).(*User)
  18. labels := make([]*Label, len(labelIDs))
  19. for i, labelID := range labelIDs {
  20. labels[i] = AssertExistsAndLoadBean(t, &Label{ID: labelID, RepoID: repo.ID}).(*Label)
  21. }
  22. assert.NoError(t, issue.ReplaceLabels(labels, doer))
  23. AssertCount(t, &IssueLabel{IssueID: issueID}, len(labelIDs))
  24. for _, labelID := range labelIDs {
  25. AssertExistsAndLoadBean(t, &IssueLabel{IssueID: issueID, LabelID: labelID})
  26. }
  27. }
  28. testSuccess(1, []int64{2})
  29. testSuccess(1, []int64{1, 2})
  30. testSuccess(1, []int64{})
  31. }
  32. func TestIssueAPIURL(t *testing.T) {
  33. assert.NoError(t, PrepareTestDatabase())
  34. issue := AssertExistsAndLoadBean(t, &Issue{ID: 1}).(*Issue)
  35. err := issue.LoadAttributes()
  36. assert.NoError(t, err)
  37. assert.Equal(t, "https://try.gitea.io/api/v1/repos/user2/repo1/issues/1", issue.APIURL())
  38. }
  39. func TestGetIssuesByIDs(t *testing.T) {
  40. assert.NoError(t, PrepareTestDatabase())
  41. testSuccess := func(expectedIssueIDs []int64, nonExistentIssueIDs []int64) {
  42. issues, err := GetIssuesByIDs(append(expectedIssueIDs, nonExistentIssueIDs...))
  43. assert.NoError(t, err)
  44. actualIssueIDs := make([]int64, len(issues))
  45. for i, issue := range issues {
  46. actualIssueIDs[i] = issue.ID
  47. }
  48. assert.Equal(t, expectedIssueIDs, actualIssueIDs)
  49. }
  50. testSuccess([]int64{1, 2, 3}, []int64{})
  51. testSuccess([]int64{1, 2, 3}, []int64{NonexistentID})
  52. }
  53. func TestGetParticipantsByIssueID(t *testing.T) {
  54. assert.NoError(t, PrepareTestDatabase())
  55. checkParticipants := func(issueID int64, userIDs []int) {
  56. participants, err := GetParticipantsByIssueID(issueID)
  57. if assert.NoError(t, err) {
  58. participantsIDs := make([]int, len(participants))
  59. for i, u := range participants {
  60. participantsIDs[i] = int(u.ID)
  61. }
  62. sort.Ints(participantsIDs)
  63. sort.Ints(userIDs)
  64. assert.Equal(t, userIDs, participantsIDs)
  65. }
  66. }
  67. // User 1 is issue1 poster (see fixtures/issue.yml)
  68. // User 2 only labeled issue1 (see fixtures/comment.yml)
  69. // Users 3 and 5 made actual comments (see fixtures/comment.yml)
  70. // User 3 is inactive, thus not active participant
  71. checkParticipants(1, []int{5})
  72. }
  73. func TestIssue_AddLabel(t *testing.T) {
  74. var tests = []struct {
  75. issueID int64
  76. labelID int64
  77. doerID int64
  78. }{
  79. {1, 2, 2}, // non-pull-request, not-already-added label
  80. {1, 1, 2}, // non-pull-request, already-added label
  81. {2, 2, 2}, // pull-request, not-already-added label
  82. {2, 1, 2}, // pull-request, already-added label
  83. }
  84. for _, test := range tests {
  85. assert.NoError(t, PrepareTestDatabase())
  86. issue := AssertExistsAndLoadBean(t, &Issue{ID: test.issueID}).(*Issue)
  87. label := AssertExistsAndLoadBean(t, &Label{ID: test.labelID}).(*Label)
  88. doer := AssertExistsAndLoadBean(t, &User{ID: test.doerID}).(*User)
  89. assert.NoError(t, issue.AddLabel(doer, label))
  90. AssertExistsAndLoadBean(t, &IssueLabel{IssueID: test.issueID, LabelID: test.labelID})
  91. }
  92. }
  93. func TestIssue_AddLabels(t *testing.T) {
  94. var tests = []struct {
  95. issueID int64
  96. labelIDs []int64
  97. doerID int64
  98. }{
  99. {1, []int64{1, 2}, 2}, // non-pull-request
  100. {1, []int64{}, 2}, // non-pull-request, empty
  101. {2, []int64{1, 2}, 2}, // pull-request
  102. {2, []int64{}, 1}, // pull-request, empty
  103. }
  104. for _, test := range tests {
  105. assert.NoError(t, PrepareTestDatabase())
  106. issue := AssertExistsAndLoadBean(t, &Issue{ID: test.issueID}).(*Issue)
  107. labels := make([]*Label, len(test.labelIDs))
  108. for i, labelID := range test.labelIDs {
  109. labels[i] = AssertExistsAndLoadBean(t, &Label{ID: labelID}).(*Label)
  110. }
  111. doer := AssertExistsAndLoadBean(t, &User{ID: test.doerID}).(*User)
  112. assert.NoError(t, issue.AddLabels(doer, labels))
  113. for _, labelID := range test.labelIDs {
  114. AssertExistsAndLoadBean(t, &IssueLabel{IssueID: test.issueID, LabelID: labelID})
  115. }
  116. }
  117. }
  118. func TestIssue_ClearLabels(t *testing.T) {
  119. var tests = []struct {
  120. issueID int64
  121. doerID int64
  122. }{
  123. {1, 2}, // non-pull-request, has labels
  124. {2, 2}, // pull-request, has labels
  125. {3, 2}, // pull-request, has no labels
  126. }
  127. for _, test := range tests {
  128. assert.NoError(t, PrepareTestDatabase())
  129. issue := AssertExistsAndLoadBean(t, &Issue{ID: test.issueID}).(*Issue)
  130. doer := AssertExistsAndLoadBean(t, &User{ID: test.doerID}).(*User)
  131. assert.NoError(t, issue.ClearLabels(doer))
  132. AssertNotExistsBean(t, &IssueLabel{IssueID: test.issueID})
  133. }
  134. }
  135. func TestUpdateIssueCols(t *testing.T) {
  136. assert.NoError(t, PrepareTestDatabase())
  137. issue := AssertExistsAndLoadBean(t, &Issue{}).(*Issue)
  138. const newTitle = "New Title for unit test"
  139. issue.Title = newTitle
  140. prevContent := issue.Content
  141. issue.Content = "This should have no effect"
  142. now := time.Now().Unix()
  143. assert.NoError(t, UpdateIssueCols(issue, "name"))
  144. then := time.Now().Unix()
  145. updatedIssue := AssertExistsAndLoadBean(t, &Issue{ID: issue.ID}).(*Issue)
  146. assert.EqualValues(t, newTitle, updatedIssue.Title)
  147. assert.EqualValues(t, prevContent, updatedIssue.Content)
  148. AssertInt64InRange(t, now, then, int64(updatedIssue.UpdatedUnix))
  149. }
  150. func TestIssues(t *testing.T) {
  151. assert.NoError(t, PrepareTestDatabase())
  152. for _, test := range []struct {
  153. Opts IssuesOptions
  154. ExpectedIssueIDs []int64
  155. }{
  156. {
  157. IssuesOptions{
  158. AssigneeID: 1,
  159. SortType: "oldest",
  160. },
  161. []int64{1, 6},
  162. },
  163. {
  164. IssuesOptions{
  165. RepoIDs: []int64{1, 3},
  166. SortType: "oldest",
  167. Page: 1,
  168. PageSize: 4,
  169. },
  170. []int64{1, 2, 3, 5},
  171. },
  172. {
  173. IssuesOptions{
  174. LabelIDs: []int64{1},
  175. Page: 1,
  176. PageSize: 4,
  177. },
  178. []int64{2, 1},
  179. },
  180. {
  181. IssuesOptions{
  182. LabelIDs: []int64{1, 2},
  183. Page: 1,
  184. PageSize: 4,
  185. },
  186. []int64{}, // issues with **both** label 1 and 2, none of these issues matches, TODO: add more tests
  187. },
  188. } {
  189. issues, err := Issues(&test.Opts)
  190. assert.NoError(t, err)
  191. if assert.Len(t, issues, len(test.ExpectedIssueIDs)) {
  192. for i, issue := range issues {
  193. assert.EqualValues(t, test.ExpectedIssueIDs[i], issue.ID)
  194. }
  195. }
  196. }
  197. }
  198. func TestGetUserIssueStats(t *testing.T) {
  199. assert.NoError(t, PrepareTestDatabase())
  200. for _, test := range []struct {
  201. Opts UserIssueStatsOptions
  202. ExpectedIssueStats IssueStats
  203. }{
  204. {
  205. UserIssueStatsOptions{
  206. UserID: 1,
  207. RepoID: 1,
  208. FilterMode: FilterModeAll,
  209. },
  210. IssueStats{
  211. YourRepositoriesCount: 0,
  212. AssignCount: 1,
  213. CreateCount: 1,
  214. OpenCount: 0,
  215. ClosedCount: 0,
  216. },
  217. },
  218. {
  219. UserIssueStatsOptions{
  220. UserID: 1,
  221. FilterMode: FilterModeAssign,
  222. },
  223. IssueStats{
  224. YourRepositoriesCount: 0,
  225. AssignCount: 2,
  226. CreateCount: 2,
  227. OpenCount: 2,
  228. ClosedCount: 0,
  229. },
  230. },
  231. {
  232. UserIssueStatsOptions{
  233. UserID: 1,
  234. FilterMode: FilterModeCreate,
  235. },
  236. IssueStats{
  237. YourRepositoriesCount: 0,
  238. AssignCount: 2,
  239. CreateCount: 2,
  240. OpenCount: 2,
  241. ClosedCount: 0,
  242. },
  243. },
  244. {
  245. UserIssueStatsOptions{
  246. UserID: 2,
  247. RepoSubQuery: builder.Select("repository.id").
  248. From("repository").
  249. Where(builder.In("repository.id", []int64{1, 2})),
  250. FilterMode: FilterModeAll,
  251. IsClosed: true,
  252. },
  253. IssueStats{
  254. YourRepositoriesCount: 2,
  255. AssignCount: 0,
  256. CreateCount: 2,
  257. OpenCount: 2,
  258. ClosedCount: 2,
  259. },
  260. },
  261. } {
  262. stats, err := GetUserIssueStats(test.Opts)
  263. if !assert.NoError(t, err) {
  264. continue
  265. }
  266. assert.Equal(t, test.ExpectedIssueStats, *stats)
  267. }
  268. }
  269. func TestIssue_loadTotalTimes(t *testing.T) {
  270. assert.NoError(t, PrepareTestDatabase())
  271. ms, err := GetIssueByID(2)
  272. assert.NoError(t, err)
  273. assert.NoError(t, ms.loadTotalTimes(x))
  274. assert.Equal(t, int64(3662), ms.TotalTrackedTime)
  275. }
  276. func TestIssue_SearchIssueIDsByKeyword(t *testing.T) {
  277. assert.NoError(t, PrepareTestDatabase())
  278. total, ids, err := SearchIssueIDsByKeyword("issue2", 1, 10, 0)
  279. assert.NoError(t, err)
  280. assert.EqualValues(t, 1, total)
  281. assert.EqualValues(t, []int64{2}, ids)
  282. total, ids, err = SearchIssueIDsByKeyword("first", 1, 10, 0)
  283. assert.NoError(t, err)
  284. assert.EqualValues(t, 1, total)
  285. assert.EqualValues(t, []int64{1}, ids)
  286. total, ids, err = SearchIssueIDsByKeyword("for", 1, 10, 0)
  287. assert.NoError(t, err)
  288. assert.EqualValues(t, 4, total)
  289. assert.EqualValues(t, []int64{1, 2, 3, 5}, ids)
  290. // issue1's comment id 2
  291. total, ids, err = SearchIssueIDsByKeyword("good", 1, 10, 0)
  292. assert.NoError(t, err)
  293. assert.EqualValues(t, 1, total)
  294. assert.EqualValues(t, []int64{1}, ids)
  295. }