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.

ChangelogActionTest.java 16KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377
  1. /*
  2. * SonarQube
  3. * Copyright (C) 2009-2020 SonarSource SA
  4. * mailto:info AT sonarsource DOT com
  5. *
  6. * This program is free software; you can redistribute it and/or
  7. * modify it under the terms of the GNU Lesser General Public
  8. * License as published by the Free Software Foundation; either
  9. * version 3 of the License, or (at your option) any later version.
  10. *
  11. * This program is distributed in the hope that it will be useful,
  12. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  14. * Lesser General Public License for more details.
  15. *
  16. * You should have received a copy of the GNU Lesser General Public License
  17. * along with this program; if not, write to the Free Software Foundation,
  18. * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
  19. */
  20. package org.sonar.server.issue.ws;
  21. import java.util.Date;
  22. import javax.annotation.Nullable;
  23. import org.junit.Before;
  24. import org.junit.Rule;
  25. import org.junit.Test;
  26. import org.junit.rules.ExpectedException;
  27. import org.sonar.api.server.ws.WebService;
  28. import org.sonar.api.utils.DateUtils;
  29. import org.sonar.api.utils.System2;
  30. import org.sonar.core.issue.FieldDiffs;
  31. import org.sonar.db.DbTester;
  32. import org.sonar.db.component.ComponentDto;
  33. import org.sonar.db.issue.IssueDto;
  34. import org.sonar.db.organization.OrganizationDto;
  35. import org.sonar.db.rule.RuleDefinitionDto;
  36. import org.sonar.db.rule.RuleDto;
  37. import org.sonar.db.user.UserDto;
  38. import org.sonar.db.user.UserTesting;
  39. import org.sonar.server.exceptions.ForbiddenException;
  40. import org.sonar.server.exceptions.NotFoundException;
  41. import org.sonar.server.issue.AvatarResolverImpl;
  42. import org.sonar.server.issue.IssueChangeWSSupport;
  43. import org.sonar.server.issue.IssueFinder;
  44. import org.sonar.server.tester.UserSessionRule;
  45. import org.sonar.server.ws.TestRequest;
  46. import org.sonar.server.ws.WsActionTester;
  47. import org.sonarqube.ws.Common.Changelog.Diff;
  48. import org.sonarqube.ws.Issues.ChangelogWsResponse;
  49. import static java.util.Optional.ofNullable;
  50. import static org.assertj.core.api.Assertions.assertThat;
  51. import static org.assertj.core.api.Assertions.assertThatThrownBy;
  52. import static org.assertj.core.groups.Tuple.tuple;
  53. import static org.sonar.api.web.UserRole.CODEVIEWER;
  54. import static org.sonar.api.web.UserRole.USER;
  55. import static org.sonar.db.component.ComponentTesting.newFileDto;
  56. import static org.sonar.db.issue.IssueTesting.newDto;
  57. import static org.sonar.db.rule.RuleTesting.newRuleDto;
  58. import static org.sonar.db.user.UserTesting.newUserDto;
  59. import static org.sonar.test.JsonAssert.assertJson;
  60. public class ChangelogActionTest {
  61. @Rule
  62. public ExpectedException expectedException = ExpectedException.none();
  63. @Rule
  64. public DbTester db = DbTester.create(System2.INSTANCE);
  65. @Rule
  66. public UserSessionRule userSession = UserSessionRule.standalone();
  67. private ComponentDto project;
  68. private ComponentDto file;
  69. private IssueFinder issueFinder = new IssueFinder(db.getDbClient(), userSession);
  70. private IssueChangeWSSupport issueChangeSupport = new IssueChangeWSSupport(db.getDbClient(), new AvatarResolverImpl(), userSession);
  71. private ChangelogAction underTest = new ChangelogAction(db.getDbClient(), issueFinder, userSession, issueChangeSupport);
  72. private WsActionTester tester = new WsActionTester(underTest);
  73. @Before
  74. public void setUp() {
  75. project = db.components().insertPrivateProject();
  76. file = db.components().insertComponent(newFileDto(project));
  77. }
  78. @Test
  79. public void return_changelog() {
  80. UserDto user = insertUser();
  81. IssueDto issueDto = insertNewIssue();
  82. userSession.logIn("john")
  83. .addMembership(db.getDefaultOrganization())
  84. .addProjectPermission(USER, project, file);
  85. db.issues().insertFieldDiffs(issueDto, new FieldDiffs().setUserUuid(user.getUuid()).setDiff("severity", "MAJOR", "BLOCKER").setCreationDate(new Date()));
  86. ChangelogWsResponse result = call(issueDto.getKey());
  87. assertThat(result.getChangelogList()).hasSize(1);
  88. assertThat(result.getChangelogList().get(0).getUser()).isNotNull().isEqualTo(user.getLogin());
  89. assertThat(result.getChangelogList().get(0).getUserName()).isNotNull().isEqualTo(user.getName());
  90. assertThat(result.getChangelogList().get(0).getIsUserActive()).isTrue();
  91. assertThat(result.getChangelogList().get(0).getAvatar()).isNotNull().isEqualTo("93942e96f5acd83e2e047ad8fe03114d");
  92. assertThat(result.getChangelogList().get(0).getCreationDate()).isNotEmpty();
  93. assertThat(result.getChangelogList().get(0).getDiffsList()).extracting(Diff::getKey, Diff::getOldValue, Diff::getNewValue).containsOnly(tuple("severity", "MAJOR", "BLOCKER"));
  94. }
  95. @Test
  96. public void return_empty_changelog_when_not_member() {
  97. UserDto user = insertUser();
  98. IssueDto issueDto = insertNewIssue();
  99. userSession.logIn("john")
  100. .addProjectPermission(USER, project, file);
  101. db.issues().insertFieldDiffs(issueDto, new FieldDiffs().setUserUuid(user.getUuid()).setDiff("severity", "MAJOR", "BLOCKER").setCreationDate(new Date()));
  102. ChangelogWsResponse result = call(issueDto.getKey());
  103. assertThat(result.getChangelogList()).hasSize(0);
  104. }
  105. @Test
  106. public void changelog_of_file_move_contains_file_names() {
  107. RuleDto rule = db.rules().insertRule(newRuleDto());
  108. OrganizationDto org = db.organizations().insert();
  109. ComponentDto project = db.components().insertPrivateProject(org);
  110. ComponentDto file1 = db.components().insertComponent(newFileDto(project));
  111. ComponentDto file2 = db.components().insertComponent(newFileDto(project));
  112. IssueDto issueDto = db.issues().insertIssue(newDto(rule, file2, project));
  113. userSession.logIn("john")
  114. .addMembership(org)
  115. .addProjectPermission(USER, project, file);
  116. db.issues().insertFieldDiffs(issueDto, new FieldDiffs().setDiff("file", file1.uuid(), file2.uuid()).setCreationDate(new Date()));
  117. ChangelogWsResponse result = call(issueDto.getKey());
  118. assertThat(result.getChangelogList()).hasSize(1);
  119. assertThat(result.getChangelogList().get(0).hasUser()).isFalse();
  120. assertThat(result.getChangelogList().get(0).getCreationDate()).isNotEmpty();
  121. assertThat(result.getChangelogList().get(0).getDiffsList()).extracting(Diff::getKey, Diff::getOldValue, Diff::getNewValue)
  122. .containsOnly(tuple("file", file1.longName(), file2.longName()));
  123. }
  124. @Test
  125. public void changelog_of_file_move_is_empty_when_files_does_not_exists() {
  126. IssueDto issueDto = insertNewIssue();
  127. userSession.logIn("john")
  128. .addMembership(db.getDefaultOrganization())
  129. .addProjectPermission(USER, project, file);
  130. db.issues().insertFieldDiffs(issueDto, new FieldDiffs().setDiff("file", "UNKNOWN_1", "UNKNOWN_2").setCreationDate(new Date()));
  131. ChangelogWsResponse result = call(issueDto.getKey());
  132. assertThat(result.getChangelogList()).hasSize(1);
  133. assertThat(result.getChangelogList().get(0).getDiffsList()).extracting(Diff::getKey, Diff::hasOldValue, Diff::hasNewValue)
  134. .containsOnly(tuple("file", false, false));
  135. }
  136. @Test
  137. public void return_changelog_on_user_without_email() {
  138. UserDto user = db.users().insertUser(UserTesting.newUserDto("john", "John", null));
  139. IssueDto issueDto = insertNewIssue();
  140. userSession.logIn("john")
  141. .addMembership(db.getDefaultOrganization())
  142. .addProjectPermission(USER, project, file);
  143. db.issues().insertFieldDiffs(issueDto, new FieldDiffs().setUserUuid(user.getUuid()).setDiff("severity", "MAJOR", "BLOCKER").setCreationDate(new Date()));
  144. ChangelogWsResponse result = call(issueDto.getKey());
  145. assertThat(result.getChangelogList()).hasSize(1);
  146. assertThat(result.getChangelogList().get(0).getUser()).isNotNull().isEqualTo(user.getLogin());
  147. assertThat(result.getChangelogList().get(0).getUserName()).isNotNull().isEqualTo(user.getName());
  148. assertThat(result.getChangelogList().get(0).hasAvatar()).isFalse();
  149. }
  150. @Test
  151. public void return_changelog_not_having_user() {
  152. IssueDto issueDto = insertNewIssue();
  153. userSession.logIn("john")
  154. .addMembership(db.getDefaultOrganization())
  155. .addProjectPermission(USER, project, file);
  156. db.issues().insertFieldDiffs(issueDto, new FieldDiffs().setUserUuid(null).setDiff("severity", "MAJOR", "BLOCKER").setCreationDate(new Date()));
  157. ChangelogWsResponse result = call(issueDto.getKey());
  158. assertThat(result.getChangelogList()).hasSize(1);
  159. assertThat(result.getChangelogList().get(0).hasUser()).isFalse();
  160. assertThat(result.getChangelogList().get(0).hasUserName()).isFalse();
  161. assertThat(result.getChangelogList().get(0).hasAvatar()).isFalse();
  162. assertThat(result.getChangelogList().get(0).getDiffsList()).isNotEmpty();
  163. }
  164. @Test
  165. public void return_changelog_on_none_existing_user() {
  166. IssueDto issueDto = insertNewIssue();
  167. userSession.logIn("john")
  168. .addMembership(db.getDefaultOrganization())
  169. .addProjectPermission(USER, project, file);
  170. db.issues().insertFieldDiffs(issueDto, new FieldDiffs().setUserUuid("UNKNOWN").setDiff("severity", "MAJOR", "BLOCKER").setCreationDate(new Date()));
  171. ChangelogWsResponse result = call(issueDto.getKey());
  172. assertThat(result.getChangelogList()).hasSize(1);
  173. assertThat(result.getChangelogList().get(0).hasUser()).isFalse();
  174. assertThat(result.getChangelogList().get(0).hasUserName()).isFalse();
  175. assertThat(result.getChangelogList().get(0).hasAvatar()).isFalse();
  176. assertThat(result.getChangelogList().get(0).getDiffsList()).isNotEmpty();
  177. }
  178. @Test
  179. public void return_changelog_on_deactivated_user() {
  180. UserDto user = db.users().insertDisabledUser();
  181. IssueDto issueDto = insertNewIssue();
  182. userSession.logIn("john")
  183. .addMembership(db.getDefaultOrganization())
  184. .addProjectPermission(USER, project, file);
  185. db.issues().insertFieldDiffs(issueDto, new FieldDiffs().setUserUuid(user.getUuid()).setDiff("severity", "MAJOR", "BLOCKER").setCreationDate(new Date()));
  186. ChangelogWsResponse result = call(issueDto.getKey());
  187. assertThat(result.getChangelogList()).hasSize(1);
  188. assertThat(result.getChangelogList().get(0).getUser()).isEqualTo(user.getLogin());
  189. assertThat(result.getChangelogList().get(0).getIsUserActive()).isFalse();
  190. assertThat(result.getChangelogList().get(0).getUserName()).isEqualTo(user.getName());
  191. assertThat(result.getChangelogList().get(0).hasAvatar()).isFalse();
  192. assertThat(result.getChangelogList().get(0).getDiffsList()).isNotEmpty();
  193. }
  194. @Test
  195. public void return_multiple_diffs() {
  196. UserDto user = insertUser();
  197. IssueDto issueDto = insertNewIssue();
  198. userSession.logIn("john")
  199. .addMembership(db.getDefaultOrganization())
  200. .addProjectPermission(USER, project, file);
  201. db.issues().insertFieldDiffs(issueDto, new FieldDiffs().setUserUuid(user.getUuid())
  202. .setDiff("severity", "MAJOR", "BLOCKER").setCreationDate(new Date())
  203. .setDiff("status", "RESOLVED", "CLOSED").setCreationDate(new Date()));
  204. ChangelogWsResponse result = call(issueDto.getKey());
  205. assertThat(result.getChangelogList()).hasSize(1);
  206. assertThat(result.getChangelogList().get(0).getDiffsList()).extracting(Diff::getKey, Diff::getOldValue, Diff::getNewValue)
  207. .containsOnly(tuple("severity", "MAJOR", "BLOCKER"), tuple("status", "RESOLVED", "CLOSED"));
  208. }
  209. @Test
  210. public void return_changelog_when_no_old_value() {
  211. UserDto user = insertUser();
  212. IssueDto issueDto = insertNewIssue();
  213. userSession.logIn("john")
  214. .addMembership(db.getDefaultOrganization())
  215. .addProjectPermission(USER, project, file);
  216. db.issues().insertFieldDiffs(issueDto, new FieldDiffs().setUserUuid(user.getUuid()).setDiff("severity", null, "BLOCKER").setCreationDate(new Date()));
  217. ChangelogWsResponse result = call(issueDto.getKey());
  218. assertThat(result.getChangelogList()).hasSize(1);
  219. assertThat(result.getChangelogList().get(0).getDiffsList().get(0).hasOldValue()).isFalse();
  220. }
  221. @Test
  222. public void return_changelog_when_no_new_value() {
  223. UserDto user = insertUser();
  224. IssueDto issueDto = insertNewIssue();
  225. userSession.logIn("john")
  226. .addMembership(db.getDefaultOrganization())
  227. .addProjectPermission(USER, project, file);
  228. db.issues().insertFieldDiffs(issueDto, new FieldDiffs().setUserUuid(user.getUuid()).setDiff("severity", "MAJOR", null).setCreationDate(new Date()));
  229. ChangelogWsResponse result = call(issueDto.getKey());
  230. assertThat(result.getChangelogList()).hasSize(1);
  231. assertThat(result.getChangelogList().get(0).getDiffsList().get(0).hasNewValue()).isFalse();
  232. }
  233. @Test
  234. public void return_many_changelog() {
  235. UserDto user = insertUser();
  236. IssueDto issueDto = insertNewIssue();
  237. userSession.logIn("john")
  238. .addMembership(db.getDefaultOrganization())
  239. .addProjectPermission(USER, project, file);
  240. db.issues().insertFieldDiffs(issueDto,
  241. new FieldDiffs().setUserUuid(user.getUuid()).setDiff("severity", "MAJOR", "BLOCKER").setCreationDate(new Date()),
  242. new FieldDiffs().setDiff("status", "RESOLVED", "CLOSED").setCreationDate(new Date()));
  243. ChangelogWsResponse result = call(issueDto.getKey());
  244. assertThat(result.getChangelogList()).hasSize(2);
  245. }
  246. @Test
  247. public void replace_technical_debt_key_by_effort() {
  248. UserDto user = insertUser();
  249. IssueDto issueDto = insertNewIssue();
  250. userSession.logIn("john")
  251. .addMembership(db.getDefaultOrganization())
  252. .addProjectPermission(USER, project, file);
  253. db.issues().insertFieldDiffs(issueDto, new FieldDiffs().setUserUuid(user.getUuid()).setDiff("technicalDebt", "10", "20").setCreationDate(new Date()));
  254. ChangelogWsResponse result = call(issueDto.getKey());
  255. assertThat(result.getChangelogList()).hasSize(1);
  256. assertThat(result.getChangelogList().get(0).getDiffsList()).extracting(Diff::getKey, Diff::getOldValue, Diff::getNewValue).containsOnly(tuple("effort", "10", "20"));
  257. }
  258. @Test
  259. public void return_empty_changelog_when_no_changes_on_issue() {
  260. IssueDto issueDto = insertNewIssue();
  261. userSession.logIn("john").addProjectPermission(USER, project, file);
  262. ChangelogWsResponse result = call(issueDto.getKey());
  263. assertThat(result.getChangelogList()).isEmpty();
  264. }
  265. @Test
  266. public void fail_when_not_enough_permission() {
  267. IssueDto issueDto = insertNewIssue();
  268. userSession.logIn("john").addProjectPermission(CODEVIEWER, project, file);
  269. expectedException.expect(ForbiddenException.class);
  270. call(issueDto.getKey());
  271. }
  272. @Test
  273. public void fail_when_trying_to_get_changelog_of_hotspot() {
  274. IssueDto issueDto = db.issues().insertHotspot();
  275. userSession.logIn("john").addProjectPermission(USER, project, file);
  276. assertThatThrownBy(() -> call(issueDto.getKey()))
  277. .isInstanceOf(NotFoundException.class)
  278. .hasMessage("Issue with key '%s' does not exist", issueDto.getKey());
  279. }
  280. @Test
  281. public void test_example() {
  282. UserDto user = db.users().insertUser(newUserDto("john.smith", "John Smith", "john@smith.com"));
  283. IssueDto issueDto = insertNewIssue();
  284. userSession.logIn("john")
  285. .addMembership(db.getDefaultOrganization())
  286. .addProjectPermission(USER, project, file);
  287. db.issues().insertFieldDiffs(issueDto, new FieldDiffs()
  288. .setUserUuid(user.getUuid())
  289. .setDiff("severity", "MAJOR", "BLOCKER").setCreationDate(new Date())
  290. .setCreationDate(DateUtils.parseDateTime("2014-03-04T23:03:44+0100")));
  291. String result = tester.newRequest().setParam("issue", issueDto.getKey()).execute().getInput();
  292. assertJson(result).isSimilarTo(getClass().getResource("changelog-example.json"));
  293. }
  294. @Test
  295. public void test_definition() {
  296. WebService.Action action = tester.getDef();
  297. assertThat(action.key()).isEqualTo("changelog");
  298. assertThat(action.isPost()).isFalse();
  299. assertThat(action.isInternal()).isFalse();
  300. assertThat(action.params()).hasSize(1);
  301. assertThat(action.responseExample()).isNotNull();
  302. }
  303. private ChangelogWsResponse call(@Nullable String issueKey) {
  304. TestRequest request = tester.newRequest();
  305. ofNullable(issueKey).ifPresent(e -> request.setParam("issue", e));
  306. return request.executeProtobuf(ChangelogWsResponse.class);
  307. }
  308. private IssueDto insertNewIssue() {
  309. RuleDefinitionDto rule = db.rules().insertIssueRule();
  310. return db.issues().insertIssue(rule, project, file);
  311. }
  312. private UserDto insertUser() {
  313. return db.users().insertUser(user -> user.setEmail("test@email.com"));
  314. }
  315. }