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.

ChangelogActionIT.java 13KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326
  1. /*
  2. * SonarQube
  3. * Copyright (C) 2009-2023 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.sonar.api.server.ws.WebService;
  27. import org.sonar.api.utils.DateUtils;
  28. import org.sonar.api.utils.System2;
  29. import org.sonar.core.issue.FieldDiffs;
  30. import org.sonar.db.DbTester;
  31. import org.sonar.db.component.ComponentDto;
  32. import org.sonar.db.issue.IssueDto;
  33. import org.sonar.db.rule.RuleDto;
  34. import org.sonar.db.user.UserDto;
  35. import org.sonar.db.user.UserTesting;
  36. import org.sonar.server.exceptions.ForbiddenException;
  37. import org.sonar.server.exceptions.NotFoundException;
  38. import org.sonar.server.common.avatar.AvatarResolverImpl;
  39. import org.sonar.server.issue.IssueChangeWSSupport;
  40. import org.sonar.server.issue.IssueFinder;
  41. import org.sonar.server.tester.UserSessionRule;
  42. import org.sonar.server.ws.TestRequest;
  43. import org.sonar.server.ws.WsActionTester;
  44. import org.sonarqube.ws.Common;
  45. import org.sonarqube.ws.Common.Changelog.Diff;
  46. import org.sonarqube.ws.Issues.ChangelogWsResponse;
  47. import static java.util.Optional.ofNullable;
  48. import static org.assertj.core.api.Assertions.assertThat;
  49. import static org.assertj.core.api.Assertions.assertThatThrownBy;
  50. import static org.assertj.core.groups.Tuple.tuple;
  51. import static org.sonar.api.web.UserRole.CODEVIEWER;
  52. import static org.sonar.api.web.UserRole.USER;
  53. import static org.sonar.db.component.ComponentTesting.newFileDto;
  54. import static org.sonar.db.user.UserTesting.newUserDto;
  55. import static org.sonar.test.JsonAssert.assertJson;
  56. public class ChangelogActionIT {
  57. @Rule
  58. public DbTester db = DbTester.create(System2.INSTANCE);
  59. @Rule
  60. public UserSessionRule userSession = UserSessionRule.standalone();
  61. private ComponentDto project;
  62. private ComponentDto file;
  63. private final IssueFinder issueFinder = new IssueFinder(db.getDbClient(), userSession);
  64. private final IssueChangeWSSupport issueChangeSupport = new IssueChangeWSSupport(db.getDbClient(), new AvatarResolverImpl(), userSession);
  65. private final ChangelogAction underTest = new ChangelogAction(db.getDbClient(), issueFinder, issueChangeSupport);
  66. private final WsActionTester tester = new WsActionTester(underTest);
  67. @Before
  68. public void setUp() {
  69. project = db.components().insertPrivateProject().getMainBranchComponent();
  70. file = db.components().insertComponent(newFileDto(project));
  71. }
  72. @Test
  73. public void return_changelog() {
  74. UserDto user = insertUser();
  75. IssueDto issueDto = insertNewIssue();
  76. userSession.logIn("john")
  77. .addProjectPermission(USER, project, file);
  78. FieldDiffs fieldDiffs = createFieldDiffs(user);
  79. db.issues().insertFieldDiffs(issueDto, fieldDiffs);
  80. ChangelogWsResponse result = call(issueDto.getKey());
  81. assertThat(result.getChangelogList()).hasSize(1);
  82. Common.Changelog changelog = result.getChangelogList().get(0);
  83. assertThat(changelog.getUser()).isNotNull().isEqualTo(user.getLogin());
  84. assertThat(changelog.getUserName()).isNotNull().isEqualTo(user.getName());
  85. assertThat(changelog.getIsUserActive()).isTrue();
  86. assertThat(changelog.getAvatar()).isNotNull().isEqualTo("93942e96f5acd83e2e047ad8fe03114d");
  87. assertThat(changelog.getCreationDate()).isNotEmpty();
  88. assertThat(changelog.getExternalUser()).isEqualTo(fieldDiffs.externalUser().orElse(null));
  89. assertThat(changelog.getWebhookSource()).isEqualTo(fieldDiffs.webhookSource().orElse(null));
  90. assertThat(changelog.getDiffsList()).extracting(Diff::getKey, Diff::getOldValue, Diff::getNewValue).containsOnly(tuple("severity", "MAJOR", "BLOCKER"));
  91. }
  92. private static FieldDiffs createFieldDiffs(UserDto user) {
  93. return new FieldDiffs()
  94. .setUserUuid(user.getUuid())
  95. .setDiff("severity", "MAJOR", "BLOCKER")
  96. .setCreationDate(new Date())
  97. .setExternalUser("toto")
  98. .setWebhookSource("github");
  99. }
  100. @Test
  101. public void return_changelog_on_user_without_email() {
  102. UserDto user = db.users().insertUser(UserTesting.newUserDto("john", "John", null));
  103. IssueDto issueDto = insertNewIssue();
  104. userSession.logIn("john")
  105. .addProjectPermission(USER, project, file);
  106. db.issues().insertFieldDiffs(issueDto, new FieldDiffs().setUserUuid(user.getUuid()).setDiff("severity", "MAJOR", "BLOCKER").setCreationDate(new Date()));
  107. ChangelogWsResponse result = call(issueDto.getKey());
  108. assertThat(result.getChangelogList()).hasSize(1);
  109. assertThat(result.getChangelogList().get(0).getUser()).isNotNull().isEqualTo(user.getLogin());
  110. assertThat(result.getChangelogList().get(0).getUserName()).isNotNull().isEqualTo(user.getName());
  111. assertThat(result.getChangelogList().get(0).hasAvatar()).isFalse();
  112. }
  113. @Test
  114. public void return_changelog_not_having_user() {
  115. IssueDto issueDto = insertNewIssue();
  116. userSession.logIn("john")
  117. .addProjectPermission(USER, project, file);
  118. db.issues().insertFieldDiffs(issueDto, new FieldDiffs().setUserUuid(null).setDiff("severity", "MAJOR", "BLOCKER").setCreationDate(new Date()));
  119. ChangelogWsResponse result = call(issueDto.getKey());
  120. assertThat(result.getChangelogList()).hasSize(1);
  121. assertThat(result.getChangelogList().get(0).hasUser()).isFalse();
  122. assertThat(result.getChangelogList().get(0).hasUserName()).isFalse();
  123. assertThat(result.getChangelogList().get(0).hasAvatar()).isFalse();
  124. assertThat(result.getChangelogList().get(0).getDiffsList()).isNotEmpty();
  125. }
  126. @Test
  127. public void return_changelog_on_none_existing_user() {
  128. IssueDto issueDto = insertNewIssue();
  129. userSession.logIn("john")
  130. .addProjectPermission(USER, project, file);
  131. db.issues().insertFieldDiffs(issueDto, new FieldDiffs().setUserUuid("UNKNOWN").setDiff("severity", "MAJOR", "BLOCKER").setCreationDate(new Date()));
  132. ChangelogWsResponse result = call(issueDto.getKey());
  133. assertThat(result.getChangelogList()).hasSize(1);
  134. assertThat(result.getChangelogList().get(0).hasUser()).isFalse();
  135. assertThat(result.getChangelogList().get(0).hasUserName()).isFalse();
  136. assertThat(result.getChangelogList().get(0).hasAvatar()).isFalse();
  137. assertThat(result.getChangelogList().get(0).getDiffsList()).isNotEmpty();
  138. }
  139. @Test
  140. public void return_changelog_on_deactivated_user() {
  141. UserDto user = db.users().insertDisabledUser();
  142. IssueDto issueDto = insertNewIssue();
  143. userSession.logIn("john")
  144. .addProjectPermission(USER, project, file);
  145. db.issues().insertFieldDiffs(issueDto, new FieldDiffs().setUserUuid(user.getUuid()).setDiff("severity", "MAJOR", "BLOCKER").setCreationDate(new Date()));
  146. ChangelogWsResponse result = call(issueDto.getKey());
  147. assertThat(result.getChangelogList()).hasSize(1);
  148. assertThat(result.getChangelogList().get(0).getUser()).isEqualTo(user.getLogin());
  149. assertThat(result.getChangelogList().get(0).getIsUserActive()).isFalse();
  150. assertThat(result.getChangelogList().get(0).getUserName()).isEqualTo(user.getName());
  151. assertThat(result.getChangelogList().get(0).hasAvatar()).isFalse();
  152. assertThat(result.getChangelogList().get(0).getDiffsList()).isNotEmpty();
  153. }
  154. @Test
  155. public void return_multiple_diffs() {
  156. UserDto user = insertUser();
  157. IssueDto issueDto = insertNewIssue();
  158. userSession.logIn("john")
  159. .addProjectPermission(USER, project, file);
  160. db.issues().insertFieldDiffs(issueDto, new FieldDiffs().setUserUuid(user.getUuid())
  161. .setDiff("severity", "MAJOR", "BLOCKER").setCreationDate(new Date())
  162. .setDiff("status", "RESOLVED", "CLOSED").setCreationDate(new Date()));
  163. ChangelogWsResponse result = call(issueDto.getKey());
  164. assertThat(result.getChangelogList()).hasSize(1);
  165. assertThat(result.getChangelogList().get(0).getDiffsList()).extracting(Diff::getKey, Diff::getOldValue, Diff::getNewValue)
  166. .containsOnly(tuple("severity", "MAJOR", "BLOCKER"), tuple("status", "RESOLVED", "CLOSED"));
  167. }
  168. @Test
  169. public void return_changelog_when_no_old_value() {
  170. UserDto user = insertUser();
  171. IssueDto issueDto = insertNewIssue();
  172. userSession.logIn("john")
  173. .addProjectPermission(USER, project, file);
  174. db.issues().insertFieldDiffs(issueDto, new FieldDiffs().setUserUuid(user.getUuid()).setDiff("severity", null, "BLOCKER").setCreationDate(new Date()));
  175. ChangelogWsResponse result = call(issueDto.getKey());
  176. assertThat(result.getChangelogList()).hasSize(1);
  177. assertThat(result.getChangelogList().get(0).getDiffsList().get(0).hasOldValue()).isFalse();
  178. }
  179. @Test
  180. public void return_changelog_when_no_new_value() {
  181. UserDto user = insertUser();
  182. IssueDto issueDto = insertNewIssue();
  183. userSession.logIn("john")
  184. .addProjectPermission(USER, project, file);
  185. db.issues().insertFieldDiffs(issueDto, new FieldDiffs().setUserUuid(user.getUuid()).setDiff("severity", "MAJOR", null).setCreationDate(new Date()));
  186. ChangelogWsResponse result = call(issueDto.getKey());
  187. assertThat(result.getChangelogList()).hasSize(1);
  188. assertThat(result.getChangelogList().get(0).getDiffsList().get(0).hasNewValue()).isFalse();
  189. }
  190. @Test
  191. public void return_many_changelog() {
  192. UserDto user = insertUser();
  193. IssueDto issueDto = insertNewIssue();
  194. userSession.logIn("john")
  195. .addProjectPermission(USER, project, file);
  196. db.issues().insertFieldDiffs(issueDto,
  197. new FieldDiffs().setUserUuid(user.getUuid()).setDiff("severity", "MAJOR", "BLOCKER").setCreationDate(new Date()),
  198. new FieldDiffs().setDiff("status", "RESOLVED", "CLOSED").setCreationDate(new Date()));
  199. ChangelogWsResponse result = call(issueDto.getKey());
  200. assertThat(result.getChangelogList()).hasSize(2);
  201. }
  202. @Test
  203. public void replace_technical_debt_key_by_effort() {
  204. UserDto user = insertUser();
  205. IssueDto issueDto = insertNewIssue();
  206. userSession.logIn("john")
  207. .addProjectPermission(USER, project, file);
  208. db.issues().insertFieldDiffs(issueDto, new FieldDiffs().setUserUuid(user.getUuid()).setDiff("technicalDebt", "10", "20").setCreationDate(new Date()));
  209. ChangelogWsResponse result = call(issueDto.getKey());
  210. assertThat(result.getChangelogList()).hasSize(1);
  211. assertThat(result.getChangelogList().get(0).getDiffsList()).extracting(Diff::getKey, Diff::getOldValue, Diff::getNewValue).containsOnly(tuple("effort", "10", "20"));
  212. }
  213. @Test
  214. public void return_empty_changelog_when_no_changes_on_issue() {
  215. IssueDto issueDto = insertNewIssue();
  216. userSession.logIn("john").addProjectPermission(USER, project, file);
  217. ChangelogWsResponse result = call(issueDto.getKey());
  218. assertThat(result.getChangelogList()).isEmpty();
  219. }
  220. @Test
  221. public void fail_when_not_enough_permission() {
  222. IssueDto issueDto = insertNewIssue();
  223. userSession.logIn("john").addProjectPermission(CODEVIEWER, project, file);
  224. assertThatThrownBy(() -> call(issueDto.getKey()))
  225. .isInstanceOf(ForbiddenException.class);
  226. }
  227. @Test
  228. public void fail_when_trying_to_get_changelog_of_hotspot() {
  229. IssueDto issueDto = db.issues().insertHotspot();
  230. userSession.logIn("john").addProjectPermission(USER, project, file);
  231. String issueDtoKey = issueDto.getKey();
  232. assertThatThrownBy(() -> call(issueDtoKey))
  233. .isInstanceOf(NotFoundException.class)
  234. .hasMessage("Issue with key '%s' does not exist", issueDtoKey);
  235. }
  236. @Test
  237. public void test_example() {
  238. UserDto user = db.users().insertUser(newUserDto("john.smith", "John Smith", "john@smith.com"));
  239. IssueDto issueDto = insertNewIssue();
  240. userSession.logIn("john")
  241. .addProjectPermission(USER, project, file);
  242. db.issues().insertFieldDiffs(issueDto, new FieldDiffs()
  243. .setUserUuid(user.getUuid())
  244. .setDiff("severity", "MAJOR", "BLOCKER")
  245. .setWebhookSource("github")
  246. .setExternalUser("toto")
  247. .setCreationDate(new Date())
  248. .setCreationDate(DateUtils.parseDateTime("2014-03-04T23:03:44+0100")));
  249. String result = tester.newRequest().setParam("issue", issueDto.getKey()).execute().getInput();
  250. assertJson(result).isSimilarTo(getClass().getResource("changelog-example.json"));
  251. }
  252. @Test
  253. public void test_definition() {
  254. WebService.Action action = tester.getDef();
  255. assertThat(action.key()).isEqualTo("changelog");
  256. assertThat(action.isPost()).isFalse();
  257. assertThat(action.isInternal()).isFalse();
  258. assertThat(action.params()).hasSize(1);
  259. assertThat(action.responseExample()).isNotNull();
  260. }
  261. private ChangelogWsResponse call(@Nullable String issueKey) {
  262. TestRequest request = tester.newRequest();
  263. ofNullable(issueKey).ifPresent(e -> request.setParam("issue", e));
  264. return request.executeProtobuf(ChangelogWsResponse.class);
  265. }
  266. private IssueDto insertNewIssue() {
  267. RuleDto rule = db.rules().insertIssueRule();
  268. return db.issues().insertIssue(rule, project, file);
  269. }
  270. private UserDto insertUser() {
  271. return db.users().insertUser(user -> user.setEmail("test@email.com"));
  272. }
  273. }