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.

ImportAzureProjectActionIT.java 22KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513
  1. /*
  2. * SonarQube
  3. * Copyright (C) 2009-2024 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.almintegration.ws.azure;
  21. import java.util.Optional;
  22. import org.junit.Before;
  23. import org.junit.Rule;
  24. import org.junit.Test;
  25. import org.sonar.alm.client.azure.AzureDevOpsHttpClient;
  26. import org.sonar.alm.client.azure.GsonAzureProject;
  27. import org.sonar.alm.client.azure.GsonAzureRepo;
  28. import org.sonar.api.config.internal.Encryption;
  29. import org.sonar.api.server.ws.WebService;
  30. import org.sonar.api.utils.System2;
  31. import org.sonar.core.platform.EditionProvider;
  32. import org.sonar.core.platform.PlatformEditionProvider;
  33. import org.sonar.core.util.SequenceUuidFactory;
  34. import org.sonar.db.DbTester;
  35. import org.sonar.db.alm.setting.AlmSettingDto;
  36. import org.sonar.db.alm.setting.ProjectAlmSettingDto;
  37. import org.sonar.db.component.BranchDto;
  38. import org.sonar.db.newcodeperiod.NewCodePeriodDto;
  39. import org.sonar.db.project.CreationMethod;
  40. import org.sonar.db.project.ProjectDto;
  41. import org.sonar.db.user.UserDto;
  42. import org.sonar.server.almintegration.ws.ImportHelper;
  43. import org.sonar.server.common.almintegration.ProjectKeyGenerator;
  44. import org.sonar.server.common.almsettings.DevOpsProjectCreatorFactory;
  45. import org.sonar.server.common.almsettings.azuredevops.AzureDevOpsProjectCreatorFactory;
  46. import org.sonar.server.common.component.ComponentUpdater;
  47. import org.sonar.server.common.newcodeperiod.NewCodeDefinitionResolver;
  48. import org.sonar.server.common.permission.PermissionTemplateService;
  49. import org.sonar.server.common.permission.PermissionUpdater;
  50. import org.sonar.server.common.project.ImportProjectService;
  51. import org.sonar.server.common.project.ProjectCreator;
  52. import org.sonar.server.es.TestIndexers;
  53. import org.sonar.server.exceptions.BadRequestException;
  54. import org.sonar.server.exceptions.ForbiddenException;
  55. import org.sonar.server.exceptions.NotFoundException;
  56. import org.sonar.server.exceptions.UnauthorizedException;
  57. import org.sonar.server.favorite.FavoriteUpdater;
  58. import org.sonar.server.l18n.I18nRule;
  59. import org.sonar.server.permission.PermissionService;
  60. import org.sonar.server.project.DefaultBranchNameResolver;
  61. import org.sonar.server.project.ProjectDefaultVisibility;
  62. import org.sonar.server.project.Visibility;
  63. import org.sonar.server.tester.UserSessionRule;
  64. import org.sonar.server.ws.TestRequest;
  65. import org.sonar.server.ws.WsActionTester;
  66. import org.sonarqube.ws.Projects;
  67. import static java.util.Objects.requireNonNull;
  68. import static org.assertj.core.api.Assertions.assertThat;
  69. import static org.assertj.core.api.Assertions.assertThatNoException;
  70. import static org.assertj.core.api.Assertions.assertThatThrownBy;
  71. import static org.assertj.core.api.Assertions.tuple;
  72. import static org.mockito.ArgumentMatchers.any;
  73. import static org.mockito.Mockito.mock;
  74. import static org.mockito.Mockito.verify;
  75. import static org.mockito.Mockito.when;
  76. import static org.sonar.db.component.BranchDto.DEFAULT_MAIN_BRANCH_NAME;
  77. import static org.sonar.db.newcodeperiod.NewCodePeriodType.NUMBER_OF_DAYS;
  78. import static org.sonar.db.newcodeperiod.NewCodePeriodType.REFERENCE_BRANCH;
  79. import static org.sonar.db.permission.GlobalPermission.PROVISION_PROJECTS;
  80. import static org.sonar.db.permission.GlobalPermission.SCAN;
  81. import static org.sonarqube.ws.client.project.ProjectsWsParameters.PARAM_NEW_CODE_DEFINITION_TYPE;
  82. import static org.sonarqube.ws.client.project.ProjectsWsParameters.PARAM_NEW_CODE_DEFINITION_VALUE;
  83. public class ImportAzureProjectActionIT {
  84. private static final String GENERATED_PROJECT_KEY = "TEST_PROJECT_KEY";
  85. @Rule
  86. public UserSessionRule userSession = UserSessionRule.standalone();
  87. @Rule
  88. public DbTester db = DbTester.create();
  89. @Rule
  90. public final I18nRule i18n = new I18nRule();
  91. private final AzureDevOpsHttpClient azureDevOpsHttpClient = mock(AzureDevOpsHttpClient.class);
  92. private final DefaultBranchNameResolver defaultBranchNameResolver = mock(DefaultBranchNameResolver.class);
  93. private final ComponentUpdater componentUpdater = new ComponentUpdater(db.getDbClient(), i18n, System2.INSTANCE,
  94. mock(PermissionTemplateService.class), new FavoriteUpdater(db.getDbClient()), new TestIndexers(), new SequenceUuidFactory(),
  95. defaultBranchNameResolver, mock(PermissionUpdater.class), mock(PermissionService.class));
  96. private final Encryption encryption = mock(Encryption.class);
  97. private final ImportHelper importHelper = new ImportHelper(db.getDbClient(), userSession);
  98. private final ProjectDefaultVisibility projectDefaultVisibility = mock(ProjectDefaultVisibility.class);
  99. private final ProjectKeyGenerator projectKeyGenerator = mock(ProjectKeyGenerator.class);
  100. private final PlatformEditionProvider editionProvider = mock(PlatformEditionProvider.class);
  101. private final NewCodeDefinitionResolver newCodeDefinitionResolver = new NewCodeDefinitionResolver(db.getDbClient(), editionProvider);
  102. private final ProjectCreator projectCreator = new ProjectCreator(userSession, projectDefaultVisibility, componentUpdater);
  103. private final DevOpsProjectCreatorFactory devOpsProjectCreatorFactory = new AzureDevOpsProjectCreatorFactory(db.getDbClient(), userSession, azureDevOpsHttpClient, projectCreator,
  104. projectKeyGenerator);
  105. private final ImportProjectService importProjectService = new ImportProjectService(db.getDbClient(), devOpsProjectCreatorFactory, userSession, componentUpdater,
  106. newCodeDefinitionResolver);
  107. private final ImportAzureProjectAction importAzureProjectAction = new ImportAzureProjectAction(
  108. importHelper, importProjectService);
  109. private final WsActionTester ws = new WsActionTester(importAzureProjectAction);
  110. @Before
  111. public void before() {
  112. when(projectDefaultVisibility.get(any())).thenReturn(Visibility.PRIVATE);
  113. when(projectKeyGenerator.generateUniqueProjectKey(any(), any())).thenReturn(GENERATED_PROJECT_KEY);
  114. when(defaultBranchNameResolver.getEffectiveMainBranchName()).thenReturn(DEFAULT_MAIN_BRANCH_NAME);
  115. }
  116. @Test
  117. public void import_project() {
  118. AlmSettingDto almSetting = configureUserAndAlmSettings();
  119. GsonAzureRepo repo = mockAzureInteractions(almSetting);
  120. Projects.CreateWsResponse.Project result = callWebserviceAndEnsureProjectIsCreated(almSetting, repo);
  121. ProjectDto projectDto = getProjectDto(result);
  122. Optional<ProjectAlmSettingDto> projectAlmSettingDto = db.getDbClient().projectAlmSettingDao().selectByProject(db.getSession(), projectDto);
  123. assertThat(projectAlmSettingDto.get().getAlmRepo()).isEqualTo("repo-name");
  124. assertThat(projectAlmSettingDto.get().getAlmSettingUuid()).isEqualTo(almSetting.getUuid());
  125. assertThat(projectAlmSettingDto.get().getAlmSlug()).isEqualTo("project-name");
  126. Optional<BranchDto> mainBranch = db.getDbClient()
  127. .branchDao()
  128. .selectByProject(db.getSession(), projectDto)
  129. .stream()
  130. .filter(BranchDto::isMain)
  131. .findFirst();
  132. assertThat(mainBranch).isPresent();
  133. assertThat(mainBranch.get().getKey()).hasToString("repo-default-branch");
  134. verify(projectKeyGenerator).generateUniqueProjectKey(repo.getProject().getName(), repo.getName());
  135. }
  136. @Test
  137. public void importProject_whenCallIsNotFromBrowser_shouldFlagTheProjectAsCreatedFromApi() {
  138. AlmSettingDto almSetting = configureUserAndAlmSettings();
  139. GsonAzureRepo repo = mockAzureInteractions(almSetting);
  140. Projects.CreateWsResponse.Project result = callWebserviceAndEnsureProjectIsCreated(almSetting, repo);
  141. assertThat(getProjectDto(result).getCreationMethod()).isEqualTo(CreationMethod.ALM_IMPORT_API);
  142. }
  143. @Test
  144. public void importProject_whenCallIsFromBrowser_shouldFlagTheProjectAsCreatedFromBrowser() {
  145. AlmSettingDto almSetting = configureUserAndAlmSettings();
  146. userSession.flagSessionAsGui();
  147. GsonAzureRepo repo = mockAzureInteractions(almSetting);
  148. Projects.CreateWsResponse.Project result = callWebserviceAndEnsureProjectIsCreated(almSetting, repo);
  149. assertThat(getProjectDto(result).getCreationMethod()).isEqualTo(CreationMethod.ALM_IMPORT_BROWSER);
  150. }
  151. private Projects.CreateWsResponse.Project callWebserviceAndEnsureProjectIsCreated(AlmSettingDto almSetting, GsonAzureRepo repo) {
  152. Projects.CreateWsResponse response = ws.newRequest()
  153. .setParam("almSetting", almSetting.getKey())
  154. .setParam("projectName", "project-name")
  155. .setParam("repositoryName", "repo-name")
  156. .executeProtobuf(Projects.CreateWsResponse.class);
  157. Projects.CreateWsResponse.Project result = response.getProject();
  158. assertThat(result.getKey()).isEqualTo(GENERATED_PROJECT_KEY);
  159. assertThat(result.getName()).isEqualTo(repo.getName());
  160. return result;
  161. }
  162. @Test
  163. public void import_project_with_NCD_developer_edition() {
  164. when(editionProvider.get()).thenReturn(Optional.of(EditionProvider.Edition.DEVELOPER));
  165. AlmSettingDto almSetting = configureUserAndAlmSettings();
  166. mockAzureInteractions(almSetting);
  167. Projects.CreateWsResponse response = ws.newRequest()
  168. .setParam("almSetting", almSetting.getKey())
  169. .setParam("projectName", "project-name")
  170. .setParam("repositoryName", "repo-name")
  171. .setParam(PARAM_NEW_CODE_DEFINITION_TYPE, "NUMBER_OF_DAYS")
  172. .setParam(PARAM_NEW_CODE_DEFINITION_VALUE, "30")
  173. .executeProtobuf(Projects.CreateWsResponse.class);
  174. Projects.CreateWsResponse.Project result = response.getProject();
  175. ProjectDto projectDto = getProjectDto(result);
  176. assertThat(db.getDbClient().newCodePeriodDao().selectByProject(db.getSession(), projectDto.getUuid()))
  177. .isPresent()
  178. .get()
  179. .extracting(NewCodePeriodDto::getType, NewCodePeriodDto::getValue, NewCodePeriodDto::getBranchUuid)
  180. .containsExactly(NUMBER_OF_DAYS, "30", null);
  181. }
  182. @Test
  183. public void import_project_with_NCD_community_edition() {
  184. when(editionProvider.get()).thenReturn(Optional.of(EditionProvider.Edition.COMMUNITY));
  185. AlmSettingDto almSetting = configureUserAndAlmSettings();
  186. mockAzureInteractions(almSetting);
  187. Projects.CreateWsResponse response = ws.newRequest()
  188. .setParam("almSetting", almSetting.getKey())
  189. .setParam("projectName", "project-name")
  190. .setParam("repositoryName", "repo-name")
  191. .setParam(PARAM_NEW_CODE_DEFINITION_TYPE, "NUMBER_OF_DAYS")
  192. .setParam(PARAM_NEW_CODE_DEFINITION_VALUE, "30")
  193. .executeProtobuf(Projects.CreateWsResponse.class);
  194. Projects.CreateWsResponse.Project result = response.getProject();
  195. Optional<ProjectDto> projectDto = db.getDbClient().projectDao().selectProjectByKey(db.getSession(), result.getKey());
  196. BranchDto branchDto = db.getDbClient().branchDao().selectMainBranchByProjectUuid(db.getSession(), projectDto.get().getUuid()).orElseThrow();
  197. assertThat(projectDto).isPresent();
  198. String projectUuid = projectDto.get().getUuid();
  199. assertThat(db.getDbClient().newCodePeriodDao().selectByBranch(db.getSession(), projectUuid, branchDto.getUuid()))
  200. .isPresent()
  201. .get()
  202. .extracting(NewCodePeriodDto::getType, NewCodePeriodDto::getValue, NewCodePeriodDto::getBranchUuid)
  203. .containsExactly(NUMBER_OF_DAYS, "30", branchDto.getUuid());
  204. }
  205. @Test
  206. public void import_project_throw_IAE_when_newCodeDefinitionValue_provided_and_no_newCodeDefinitionType() {
  207. when(editionProvider.get()).thenReturn(Optional.of(EditionProvider.Edition.DEVELOPER));
  208. AlmSettingDto almSetting = configureUserAndAlmSettings();
  209. mockAzureInteractions(almSetting);
  210. Projects.CreateWsResponse response = ws.newRequest()
  211. .setParam("almSetting", almSetting.getKey())
  212. .setParam("projectName", "project-name")
  213. .setParam("repositoryName", "repo-name")
  214. .setParam(PARAM_NEW_CODE_DEFINITION_TYPE, "NUMBER_OF_DAYS")
  215. .setParam(PARAM_NEW_CODE_DEFINITION_VALUE, "30")
  216. .executeProtobuf(Projects.CreateWsResponse.class);
  217. Projects.CreateWsResponse.Project result = response.getProject();
  218. ProjectDto projectDto = getProjectDto(result);
  219. assertThat(db.getDbClient().newCodePeriodDao().selectByProject(db.getSession(), projectDto.getUuid()))
  220. .isPresent()
  221. .get()
  222. .extracting(NewCodePeriodDto::getType, NewCodePeriodDto::getValue, NewCodePeriodDto::getBranchUuid)
  223. .containsExactly(NUMBER_OF_DAYS, "30", null);
  224. }
  225. @Test
  226. public void import_project_reference_branch_ncd_no_default_branch_name() {
  227. when(editionProvider.get()).thenReturn(Optional.of(EditionProvider.Edition.DEVELOPER));
  228. AlmSettingDto almSetting = configureUserAndAlmSettings();
  229. GsonAzureRepo repo = getEmptyGsonAzureRepo();
  230. when(azureDevOpsHttpClient.getRepo(almSetting.getUrl(), almSetting.getDecryptedPersonalAccessToken(encryption),
  231. "project-name", "repo-name"))
  232. .thenReturn(repo);
  233. Projects.CreateWsResponse response = ws.newRequest()
  234. .setParam("almSetting", almSetting.getKey())
  235. .setParam("projectName", "project-name")
  236. .setParam("repositoryName", "repo-name")
  237. .setParam(PARAM_NEW_CODE_DEFINITION_TYPE, "REFERENCE_BRANCH")
  238. .executeProtobuf(Projects.CreateWsResponse.class);
  239. Projects.CreateWsResponse.Project result = response.getProject();
  240. ProjectDto projectDto = getProjectDto(result);
  241. assertThat(db.getDbClient().newCodePeriodDao().selectByProject(db.getSession(), projectDto.getUuid()))
  242. .isPresent()
  243. .get()
  244. .extracting(NewCodePeriodDto::getType, NewCodePeriodDto::getValue)
  245. .containsExactly(REFERENCE_BRANCH, DEFAULT_MAIN_BRANCH_NAME);
  246. }
  247. @Test
  248. public void import_project_reference_branch_ncd() {
  249. when(editionProvider.get()).thenReturn(Optional.of(EditionProvider.Edition.DEVELOPER));
  250. AlmSettingDto almSetting = configureUserAndAlmSettings();
  251. mockAzureInteractions(almSetting);
  252. Projects.CreateWsResponse response = ws.newRequest()
  253. .setParam("almSetting", almSetting.getKey())
  254. .setParam("projectName", "project-name")
  255. .setParam("repositoryName", "repo-name")
  256. .setParam(PARAM_NEW_CODE_DEFINITION_TYPE, "REFERENCE_BRANCH")
  257. .executeProtobuf(Projects.CreateWsResponse.class);
  258. Projects.CreateWsResponse.Project result = response.getProject();
  259. ProjectDto projectDto = getProjectDto(result);
  260. assertThat(db.getDbClient().newCodePeriodDao().selectByProject(db.getSession(), projectDto.getUuid()))
  261. .isPresent()
  262. .get()
  263. .extracting(NewCodePeriodDto::getType, NewCodePeriodDto::getValue)
  264. .containsExactly(REFERENCE_BRANCH, "repo-default-branch");
  265. }
  266. @Test
  267. public void import_project_from_empty_repo() {
  268. AlmSettingDto almSetting = configureUserAndAlmSettings();
  269. GsonAzureRepo repo = getEmptyGsonAzureRepo();
  270. when(azureDevOpsHttpClient.getRepo(almSetting.getUrl(), almSetting.getDecryptedPersonalAccessToken(encryption),
  271. "project-name", "repo-name"))
  272. .thenReturn(repo);
  273. TestRequest request = ws.newRequest()
  274. .setParam("almSetting", almSetting.getKey())
  275. .setParam("projectName", "project-name")
  276. .setParam("repositoryName", "repo-name")
  277. .setParam(PARAM_NEW_CODE_DEFINITION_VALUE, "30");
  278. assertThatThrownBy(() -> request.executeProtobuf(Projects.CreateWsResponse.class))
  279. .isInstanceOf(IllegalArgumentException.class)
  280. .hasMessage("New code definition type is required when new code definition value is provided");
  281. }
  282. @Test
  283. public void fail_when_not_logged_in() {
  284. TestRequest request = ws.newRequest()
  285. .setParam("almSetting", "azure")
  286. .setParam("projectName", "project-name")
  287. .setParam("repositoryName", "repo-name");
  288. assertThatThrownBy(request::execute)
  289. .isInstanceOf(UnauthorizedException.class);
  290. }
  291. @Test
  292. public void fail_when_missing_project_creator_permission() {
  293. UserDto user = db.users().insertUser();
  294. userSession.logIn(user).addPermission(SCAN);
  295. TestRequest request = ws.newRequest()
  296. .setParam("almSetting", "azure")
  297. .setParam("projectName", "project-name")
  298. .setParam("repositoryName", "repo-name");
  299. assertThatThrownBy(request::execute)
  300. .isInstanceOf(ForbiddenException.class)
  301. .hasMessage("Insufficient privileges");
  302. }
  303. @Test
  304. public void check_pat_is_missing() {
  305. UserDto user = db.users().insertUser();
  306. userSession.logIn(user).addPermission(PROVISION_PROJECTS);
  307. AlmSettingDto almSetting = db.almSettings().insertAzureAlmSetting();
  308. TestRequest request = ws.newRequest()
  309. .setParam("almSetting", almSetting.getKey())
  310. .setParam("projectName", "project-name")
  311. .setParam("repositoryName", "repo-name");
  312. assertThatThrownBy(request::execute)
  313. .isInstanceOf(IllegalArgumentException.class)
  314. .hasMessage("personal access token for '" + almSetting.getKey() + "' is missing");
  315. }
  316. @Test
  317. public void fail_project_already_exists() {
  318. AlmSettingDto almSetting = configureUserAndAlmSettings();
  319. GsonAzureRepo repo = getGsonAzureRepo();
  320. db.components().insertPublicProject(p -> p.setKey(GENERATED_PROJECT_KEY)).getMainBranchComponent();
  321. when(azureDevOpsHttpClient.getRepo(almSetting.getUrl(), almSetting.getDecryptedPersonalAccessToken(encryption),
  322. "project-name", "repo-name")).thenReturn(repo);
  323. TestRequest request = ws.newRequest()
  324. .setParam("almSetting", almSetting.getKey())
  325. .setParam("projectName", "project-name")
  326. .setParam("repositoryName", "repo-name");
  327. assertThatThrownBy(request::execute)
  328. .isInstanceOf(BadRequestException.class)
  329. .hasMessage("Could not create Project with key: \"%s\". A similar key already exists: \"%s\"", GENERATED_PROJECT_KEY,
  330. GENERATED_PROJECT_KEY);
  331. }
  332. @Test
  333. public void importProject_whenAlmSettingKeyDoesNotExist_shouldThrow() {
  334. UserDto user = db.users().insertUser();
  335. userSession.logIn(user).addPermission(PROVISION_PROJECTS);
  336. TestRequest request = ws.newRequest()
  337. .setParam("almSetting", "unknown")
  338. .setParam("projectName", "project-name")
  339. .setParam("repositoryName", "repo-name");
  340. assertThatThrownBy(request::execute)
  341. .isInstanceOf(NotFoundException.class)
  342. .hasMessage("DevOps Platform configuration 'unknown' not found.");
  343. }
  344. @Test
  345. public void importProject_whenNoAlmSettingKeyAndNoConfig_shouldThrow() {
  346. UserDto user = db.users().insertUser();
  347. userSession.logIn(user).addPermission(PROVISION_PROJECTS);
  348. TestRequest request = ws.newRequest()
  349. .setParam("projectName", "project-name")
  350. .setParam("repositoryName", "repo-name");
  351. assertThatThrownBy(request::execute)
  352. .isInstanceOf(NotFoundException.class)
  353. .hasMessage("There is no AZURE_DEVOPS configuration for DevOps Platform. Please add one.");
  354. }
  355. @Test
  356. public void importProject_whenNoAlmSettingKeyAndMultipleConfigs_shouldThrow() {
  357. UserDto user = db.users().insertUser();
  358. userSession.logIn(user).addPermission(PROVISION_PROJECTS);
  359. db.almSettings().insertAzureAlmSetting();
  360. db.almSettings().insertAzureAlmSetting();
  361. TestRequest request = ws.newRequest()
  362. .setParam("projectName", "project-name")
  363. .setParam("repositoryName", "repo-name");
  364. assertThatThrownBy(request::execute)
  365. .isInstanceOf(IllegalArgumentException.class)
  366. .hasMessage("Parameter almSetting is required as there are multiple DevOps Platform configurations.");
  367. }
  368. @Test
  369. public void importProject_whenNoAlmSettingKeyAndOnlyOneConfig_shouldImport() {
  370. AlmSettingDto almSetting = configureUserAndAlmSettings();
  371. mockAzureInteractions(almSetting);
  372. TestRequest request = ws.newRequest()
  373. .setParam("projectName", "project-name")
  374. .setParam("repositoryName", "repo-name");
  375. assertThatNoException().isThrownBy(request::execute);
  376. }
  377. @Test
  378. public void define() {
  379. WebService.Action def = ws.getDef();
  380. assertThat(def.since()).isEqualTo("8.6");
  381. assertThat(def.isPost()).isTrue();
  382. assertThat(def.params())
  383. .extracting(WebService.Param::key, WebService.Param::isRequired)
  384. .containsExactlyInAnyOrder(
  385. tuple("almSetting", false),
  386. tuple("projectName", true),
  387. tuple("repositoryName", true),
  388. tuple(PARAM_NEW_CODE_DEFINITION_TYPE, false),
  389. tuple(PARAM_NEW_CODE_DEFINITION_VALUE, false));
  390. assertThat(def.deprecatedSince()).isEqualTo("10.5");
  391. }
  392. private AlmSettingDto configureUserAndAlmSettings() {
  393. UserDto user = db.users().insertUser();
  394. userSession.logIn(user).addPermission(PROVISION_PROJECTS);
  395. AlmSettingDto almSetting = db.almSettings().insertAzureAlmSetting();
  396. db.almPats().insert(dto -> {
  397. dto.setAlmSettingUuid(almSetting.getUuid());
  398. dto.setPersonalAccessToken(requireNonNull(almSetting.getDecryptedPersonalAccessToken(encryption)));
  399. dto.setUserUuid(user.getUuid());
  400. });
  401. return almSetting;
  402. }
  403. private GsonAzureRepo mockAzureInteractions(AlmSettingDto almSetting) {
  404. GsonAzureRepo repo = getGsonAzureRepo();
  405. when(azureDevOpsHttpClient.getRepo(almSetting.getUrl(), almSetting.getDecryptedPersonalAccessToken(encryption),
  406. "project-name", "repo-name"))
  407. .thenReturn(repo);
  408. return repo;
  409. }
  410. private ProjectDto getProjectDto(Projects.CreateWsResponse.Project result) {
  411. Optional<ProjectDto> projectDto = db.getDbClient().projectDao().selectProjectByKey(db.getSession(), result.getKey());
  412. assertThat(projectDto).isPresent();
  413. return projectDto.get();
  414. }
  415. private GsonAzureRepo getGsonAzureRepo() {
  416. return new GsonAzureRepo("repo-id", "repo-name", "repo-url",
  417. new GsonAzureProject("project-name", "project-description"),
  418. "refs/heads/repo-default-branch");
  419. }
  420. private GsonAzureRepo getEmptyGsonAzureRepo() {
  421. return new GsonAzureRepo("repo-id", "repo-name", "repo-url",
  422. new GsonAzureProject("project-name", "project-description"), null);
  423. }
  424. }