Ви не можете вибрати більше 25 тем Теми мають розпочинатися з літери або цифри, можуть містити дефіси (-) і не повинні перевищувати 35 символів.

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259
  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.webhook.ws;
  21. import java.util.Optional;
  22. import org.junit.Rule;
  23. import org.junit.Test;
  24. import org.sonar.api.config.Configuration;
  25. import org.sonar.api.resources.ResourceTypes;
  26. import org.sonar.api.server.ws.WebService;
  27. import org.sonar.api.web.UserRole;
  28. import org.sonar.db.DbClient;
  29. import org.sonar.db.DbTester;
  30. import org.sonar.db.component.ComponentDbTester;
  31. import org.sonar.db.project.ProjectDto;
  32. import org.sonar.db.webhook.WebhookDbTester;
  33. import org.sonar.db.webhook.WebhookDto;
  34. import org.sonar.server.component.ComponentFinder;
  35. import org.sonar.server.exceptions.ForbiddenException;
  36. import org.sonar.server.exceptions.NotFoundException;
  37. import org.sonar.server.exceptions.UnauthorizedException;
  38. import org.sonar.server.tester.UserSessionRule;
  39. import org.sonar.server.ws.TestRequest;
  40. import org.sonar.server.ws.TestResponse;
  41. import org.sonar.server.ws.WsActionTester;
  42. import static java.net.HttpURLConnection.HTTP_NO_CONTENT;
  43. import static org.assertj.core.api.Assertions.assertThat;
  44. import static org.assertj.core.api.Assertions.assertThatThrownBy;
  45. import static org.assertj.core.api.AssertionsForClassTypes.tuple;
  46. import static org.mockito.Mockito.mock;
  47. import static org.sonar.api.web.UserRole.ADMIN;
  48. import static org.sonar.db.DbTester.create;
  49. import static org.sonar.db.permission.GlobalPermission.ADMINISTER;
  50. import static org.sonar.server.tester.UserSessionRule.standalone;
  51. import static org.sonar.server.ws.KeyExamples.NAME_WEBHOOK_EXAMPLE_001;
  52. import static org.sonar.server.ws.KeyExamples.URL_WEBHOOK_EXAMPLE_001;
  53. public class UpdateActionTest {
  54. @Rule
  55. public UserSessionRule userSession = standalone();
  56. @Rule
  57. public DbTester db = create();
  58. private final DbClient dbClient = db.getDbClient();
  59. private final WebhookDbTester webhookDbTester = db.webhooks();
  60. private final ComponentDbTester componentDbTester = db.components();
  61. private final Configuration configuration = mock(Configuration.class);
  62. private final NetworkInterfaceProvider networkInterfaceProvider = mock(NetworkInterfaceProvider.class);
  63. private final WebhookSupport webhookSupport = new WebhookSupport(userSession, configuration, networkInterfaceProvider);
  64. private final ResourceTypes resourceTypes = mock(ResourceTypes.class);
  65. private final ComponentFinder componentFinder = new ComponentFinder(dbClient, resourceTypes);
  66. private final UpdateAction underTest = new UpdateAction(dbClient, userSession, webhookSupport, componentFinder);
  67. private final WsActionTester wsActionTester = new WsActionTester(underTest);
  68. @Test
  69. public void test_ws_definition() {
  70. WebService.Action action = wsActionTester.getDef();
  71. assertThat(action).isNotNull();
  72. assertThat(action.isInternal()).isFalse();
  73. assertThat(action.isPost()).isTrue();
  74. assertThat(action.params())
  75. .extracting(WebService.Param::key, WebService.Param::isRequired)
  76. .containsExactlyInAnyOrder(
  77. tuple("webhook", true),
  78. tuple("name", true),
  79. tuple("url", true),
  80. tuple("secret", false));
  81. }
  82. @Test
  83. public void update_a_project_webhook_with_required_fields() {
  84. ProjectDto project = componentDbTester.insertPrivateProjectDto();
  85. WebhookDto dto = webhookDbTester.insertWebhook(project);
  86. userSession.logIn().addProjectPermission(ADMIN, project);
  87. TestResponse response = wsActionTester.newRequest()
  88. .setParam("webhook", dto.getUuid())
  89. .setParam("name", NAME_WEBHOOK_EXAMPLE_001)
  90. .setParam("url", URL_WEBHOOK_EXAMPLE_001)
  91. .execute();
  92. assertThat(response.getStatus()).isEqualTo(HTTP_NO_CONTENT);
  93. Optional<WebhookDto> reloaded = webhookDbTester.selectWebhook(dto.getUuid());
  94. assertThat(reloaded).isPresent();
  95. assertThat(reloaded.get().getName()).isEqualTo(NAME_WEBHOOK_EXAMPLE_001);
  96. assertThat(reloaded.get().getUrl()).isEqualTo(URL_WEBHOOK_EXAMPLE_001);
  97. assertThat(reloaded.get().getProjectUuid()).isEqualTo(dto.getProjectUuid());
  98. assertThat(reloaded.get().getSecret()).isEqualTo(dto.getSecret());
  99. }
  100. @Test
  101. public void update_with_empty_secrets_removes_the_secret() {
  102. ProjectDto project = componentDbTester.insertPrivateProjectDto();
  103. WebhookDto dto = webhookDbTester.insertWebhook(project);
  104. userSession.logIn().addProjectPermission(UserRole.ADMIN, project);
  105. TestResponse response = wsActionTester.newRequest()
  106. .setParam("webhook", dto.getUuid())
  107. .setParam("name", NAME_WEBHOOK_EXAMPLE_001)
  108. .setParam("url", URL_WEBHOOK_EXAMPLE_001)
  109. .setParam("secret", "")
  110. .execute();
  111. assertThat(response.getStatus()).isEqualTo(HTTP_NO_CONTENT);
  112. Optional<WebhookDto> reloaded = webhookDbTester.selectWebhook(dto.getUuid());
  113. assertThat(reloaded).isPresent();
  114. assertThat(reloaded.get().getName()).isEqualTo(NAME_WEBHOOK_EXAMPLE_001);
  115. assertThat(reloaded.get().getUrl()).isEqualTo(URL_WEBHOOK_EXAMPLE_001);
  116. assertThat(reloaded.get().getProjectUuid()).isEqualTo(dto.getProjectUuid());
  117. assertThat(reloaded.get().getSecret()).isEqualTo(null);
  118. }
  119. @Test
  120. public void update_a_project_webhook_with_all_fields() {
  121. ProjectDto project = componentDbTester.insertPrivateProjectDto();
  122. WebhookDto dto = webhookDbTester.insertWebhook(project);
  123. userSession.logIn().addProjectPermission(ADMIN, project);
  124. TestResponse response = wsActionTester.newRequest()
  125. .setParam("webhook", dto.getUuid())
  126. .setParam("name", NAME_WEBHOOK_EXAMPLE_001)
  127. .setParam("url", URL_WEBHOOK_EXAMPLE_001)
  128. .setParam("secret", "a_new_secret")
  129. .execute();
  130. assertThat(response.getStatus()).isEqualTo(HTTP_NO_CONTENT);
  131. Optional<WebhookDto> reloaded = webhookDbTester.selectWebhook(dto.getUuid());
  132. assertThat(reloaded).isPresent();
  133. assertThat(reloaded.get().getName()).isEqualTo(NAME_WEBHOOK_EXAMPLE_001);
  134. assertThat(reloaded.get().getUrl()).isEqualTo(URL_WEBHOOK_EXAMPLE_001);
  135. assertThat(reloaded.get().getProjectUuid()).isEqualTo(dto.getProjectUuid());
  136. assertThat(reloaded.get().getSecret()).isEqualTo("a_new_secret");
  137. }
  138. @Test
  139. public void update_a_global_webhook() {
  140. WebhookDto dto = webhookDbTester.insertGlobalWebhook();
  141. userSession.logIn().addPermission(ADMINISTER);
  142. TestResponse response = wsActionTester.newRequest()
  143. .setParam("webhook", dto.getUuid())
  144. .setParam("name", NAME_WEBHOOK_EXAMPLE_001)
  145. .setParam("url", URL_WEBHOOK_EXAMPLE_001)
  146. .setParam("secret", "a_new_secret")
  147. .execute();
  148. assertThat(response.getStatus()).isEqualTo(HTTP_NO_CONTENT);
  149. Optional<WebhookDto> reloaded = webhookDbTester.selectWebhook(dto.getUuid());
  150. assertThat(reloaded).isPresent();
  151. assertThat(reloaded.get().getName()).isEqualTo(NAME_WEBHOOK_EXAMPLE_001);
  152. assertThat(reloaded.get().getUrl()).isEqualTo(URL_WEBHOOK_EXAMPLE_001);
  153. assertThat(reloaded.get().getProjectUuid()).isNull();
  154. assertThat(reloaded.get().getSecret()).isEqualTo("a_new_secret");
  155. }
  156. @Test
  157. public void fail_if_webhook_does_not_exist() {
  158. userSession.logIn().addPermission(ADMINISTER);
  159. TestRequest request = wsActionTester.newRequest()
  160. .setParam("webhook", "inexistent-webhook-uuid")
  161. .setParam("name", NAME_WEBHOOK_EXAMPLE_001)
  162. .setParam("url", URL_WEBHOOK_EXAMPLE_001);
  163. assertThatThrownBy(request::execute)
  164. .isInstanceOf(NotFoundException.class)
  165. .hasMessage("No webhook with key 'inexistent-webhook-uuid'");
  166. }
  167. @Test
  168. public void fail_if_not_logged_in() {
  169. WebhookDto dto = webhookDbTester.insertGlobalWebhook();
  170. userSession.anonymous();
  171. TestRequest request = wsActionTester.newRequest()
  172. .setParam("webhook", dto.getUuid())
  173. .setParam("name", NAME_WEBHOOK_EXAMPLE_001)
  174. .setParam("url", URL_WEBHOOK_EXAMPLE_001);
  175. assertThatThrownBy(request::execute)
  176. .isInstanceOf(UnauthorizedException.class);
  177. }
  178. @Test
  179. public void fail_if_no_permission_on_webhook_scope_project() {
  180. ProjectDto project = componentDbTester.insertPrivateProjectDto();
  181. WebhookDto dto = webhookDbTester.insertWebhook(project);
  182. userSession.logIn();
  183. TestRequest request = wsActionTester.newRequest()
  184. .setParam("webhook", dto.getUuid())
  185. .setParam("name", NAME_WEBHOOK_EXAMPLE_001)
  186. .setParam("url", URL_WEBHOOK_EXAMPLE_001);
  187. assertThatThrownBy(request::execute)
  188. .isInstanceOf(ForbiddenException.class)
  189. .hasMessage("Insufficient privileges");
  190. }
  191. @Test
  192. public void fail_if_no_permission_on_webhook_scope_global() {
  193. WebhookDto dto = webhookDbTester.insertGlobalWebhook();
  194. userSession.logIn();
  195. TestRequest request = wsActionTester.newRequest()
  196. .setParam("webhook", dto.getUuid())
  197. .setParam("name", NAME_WEBHOOK_EXAMPLE_001)
  198. .setParam("url", URL_WEBHOOK_EXAMPLE_001);
  199. assertThatThrownBy(request::execute)
  200. .isInstanceOf(ForbiddenException.class)
  201. .hasMessage("Insufficient privileges");
  202. }
  203. @Test
  204. public void fail_if_url_is_not_valid() {
  205. ProjectDto project = componentDbTester.insertPrivateProjectDto();
  206. WebhookDto dto = webhookDbTester.insertWebhook(project);
  207. userSession.logIn().addProjectPermission(ADMIN, project);
  208. TestRequest request = wsActionTester.newRequest()
  209. .setParam("webhook", dto.getUuid())
  210. .setParam("name", NAME_WEBHOOK_EXAMPLE_001)
  211. .setParam("url", "htp://www.wrong-protocol.com/");
  212. assertThatThrownBy(request::execute)
  213. .isInstanceOf(IllegalArgumentException.class);
  214. }
  215. @Test
  216. public void fail_if_credential_in_url_is_have_a_wrong_format() {
  217. ProjectDto project = componentDbTester.insertPrivateProjectDto();
  218. WebhookDto dto = webhookDbTester.insertWebhook(project);
  219. userSession.logIn().addProjectPermission(ADMIN, project);
  220. TestRequest request = wsActionTester.newRequest()
  221. .setParam("webhook", dto.getUuid())
  222. .setParam("name", NAME_WEBHOOK_EXAMPLE_001)
  223. .setParam("url", "http://:www.wrong-protocol.com/");
  224. assertThatThrownBy(request::execute)
  225. .isInstanceOf(IllegalArgumentException.class);
  226. }
  227. }