aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--server/sonar-alm-client/build.gradle1
-rw-r--r--server/sonar-alm-client/src/main/java/org/sonar/alm/client/azure/AzureDevOpsHttpClient.java158
-rw-r--r--server/sonar-alm-client/src/main/java/org/sonar/alm/client/azure/GsonAzureError.java41
-rw-r--r--server/sonar-alm-client/src/main/java/org/sonar/alm/client/azure/GsonAzureProject.java48
-rw-r--r--server/sonar-alm-client/src/main/java/org/sonar/alm/client/azure/GsonAzureProjectList.java56
-rw-r--r--server/sonar-alm-client/src/main/java/org/sonar/alm/client/azure/GsonAzureRepo.java63
-rw-r--r--server/sonar-alm-client/src/main/java/org/sonar/alm/client/azure/GsonAzureRepoList.java49
-rw-r--r--server/sonar-alm-client/src/test/java/org/sonar/alm/client/azure/AzureDevOpsHttpClientTest.java273
-rw-r--r--server/sonar-webserver-webapi/src/main/java/org/sonar/server/almintegration/ws/AlmIntegrationsWSModule.java6
-rw-r--r--server/sonar-webserver-webapi/src/main/java/org/sonar/server/almintegration/ws/azure/ImportAzureProjectAction.java163
-rw-r--r--server/sonar-webserver-webapi/src/main/java/org/sonar/server/almintegration/ws/azure/ListAzureProjectsAction.java115
-rw-r--r--server/sonar-webserver-webapi/src/main/java/org/sonar/server/almintegration/ws/azure/SearchAzureReposAction.java235
-rw-r--r--server/sonar-webserver-webapi/src/main/java/org/sonar/server/almintegration/ws/azure/package-info.java23
-rw-r--r--server/sonar-webserver-webapi/src/test/java/org/sonar/server/almintegration/ws/azure/ImportAzureProjectActionTest.java238
-rw-r--r--server/sonar-webserver-webapi/src/test/java/org/sonar/server/almintegration/ws/azure/ListAzureProjectsActionTest.java187
-rw-r--r--server/sonar-webserver-webapi/src/test/java/org/sonar/server/almintegration/ws/azure/SearchAzureReposActionTest.java367
-rw-r--r--server/sonar-webserver/src/main/java/org/sonar/server/platform/platformlevel/PlatformLevel4.java2
17 files changed, 2025 insertions, 0 deletions
diff --git a/server/sonar-alm-client/build.gradle b/server/sonar-alm-client/build.gradle
index afe80fc3278..f546848d630 100644
--- a/server/sonar-alm-client/build.gradle
+++ b/server/sonar-alm-client/build.gradle
@@ -6,6 +6,7 @@ dependencies {
compile 'com.google.code.gson:gson'
compile 'com.google.guava:guava'
compile 'com.squareup.okhttp3:okhttp'
+ compile 'commons-codec:commons-codec'
testCompile project(':sonar-plugin-api-impl')
diff --git a/server/sonar-alm-client/src/main/java/org/sonar/alm/client/azure/AzureDevOpsHttpClient.java b/server/sonar-alm-client/src/main/java/org/sonar/alm/client/azure/AzureDevOpsHttpClient.java
new file mode 100644
index 00000000000..c4f2f4f0c57
--- /dev/null
+++ b/server/sonar-alm-client/src/main/java/org/sonar/alm/client/azure/AzureDevOpsHttpClient.java
@@ -0,0 +1,158 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2021 SonarSource SA
+ * mailto:info AT sonarsource DOT com
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 3 of the License, or (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with this program; if not, write to the Free Software Foundation,
+ * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ */
+package org.sonar.alm.client.azure;
+
+import com.google.common.base.Strings;
+import com.google.gson.Gson;
+import com.google.gson.GsonBuilder;
+import com.google.gson.JsonSyntaxException;
+
+import java.io.IOException;
+import java.net.HttpURLConnection;
+import java.nio.charset.StandardCharsets;
+import java.util.function.Function;
+import javax.annotation.Nullable;
+
+import okhttp3.OkHttpClient;
+import okhttp3.Request;
+import okhttp3.RequestBody;
+import okhttp3.Response;
+import okhttp3.ResponseBody;
+import org.apache.commons.codec.binary.Base64;
+import org.sonar.alm.client.TimeoutConfiguration;
+import org.sonar.api.server.ServerSide;
+import org.sonar.api.utils.log.Logger;
+import org.sonar.api.utils.log.Loggers;
+import org.sonarqube.ws.client.OkHttpClientBuilder;
+
+import static org.sonar.api.internal.apachecommons.lang.StringUtils.isBlank;
+import static org.sonar.api.internal.apachecommons.lang.StringUtils.substringBeforeLast;
+
+@ServerSide
+public class AzureDevOpsHttpClient {
+ private static final Logger LOG = Loggers.get(AzureDevOpsHttpClient.class);
+
+ protected static final String GET = "GET";
+ protected static final String UNABLE_TO_CONTACT_AZURE_SERVER = "Unable to contact Azure DevOps server";
+
+ protected final OkHttpClient client;
+
+ public AzureDevOpsHttpClient(TimeoutConfiguration timeoutConfiguration) {
+ client = new OkHttpClientBuilder()
+ .setConnectTimeoutMs(timeoutConfiguration.getConnectTimeout())
+ .setReadTimeoutMs(timeoutConfiguration.getReadTimeout())
+ .build();
+ }
+
+ public GsonAzureProjectList getProjects(String serverUrl, String token) {
+ String url = String.format("%s/_apis/projects", getTrimmedUrl(serverUrl));
+ LOG.debug(String.format("get projects : [%s]", url));
+ return doGet(token, url, r -> buildGson().fromJson(r.body().charStream(), GsonAzureProjectList.class));
+ }
+
+
+ public GsonAzureRepoList getRepos(String serverUrl, String token, @Nullable String projectName) {
+ String url;
+ if (projectName != null && !projectName.isEmpty()) {
+ url = String.format("%s/%s/_apis/git/repositories", getTrimmedUrl(serverUrl), projectName);
+ } else {
+ url = String.format("%s/_apis/git/repositories", getTrimmedUrl(serverUrl));
+ }
+ LOG.debug(String.format("get repos : [%s]", url));
+ return doGet(token, url, r -> buildGson().fromJson(r.body().charStream(), GsonAzureRepoList.class));
+ }
+
+ public GsonAzureRepo getRepo(String serverUrl, String token, String projectName, String repositoryName) {
+ String url = String.format("%s/%s/_apis/git/repositories/%s", getTrimmedUrl(serverUrl), projectName, repositoryName);
+ LOG.debug(String.format("get repo : [%s]", url));
+ return doGet(token, url, r -> buildGson().fromJson(r.body().charStream(), GsonAzureRepo.class));
+ }
+
+ protected <G> G doGet(String token, String url, Function<Response, G> handler) {
+ Request request = prepareRequestWithToken(token, GET, url, null);
+ return doCall(request, handler);
+ }
+
+ protected <G> G doCall(Request request, Function<Response, G> handler) {
+ try (Response response = client.newCall(request).execute()) {
+ checkResponseIsSuccessful(response);
+ return handler.apply(response);
+ } catch (JsonSyntaxException e) {
+ throw new IllegalArgumentException(UNABLE_TO_CONTACT_AZURE_SERVER + ", got an unexpected response", e);
+ } catch (IOException e) {
+ throw new IllegalArgumentException(UNABLE_TO_CONTACT_AZURE_SERVER, e);
+ }
+ }
+
+ protected static Request prepareRequestWithToken(String token, String method, String url, @Nullable RequestBody body) {
+ return new Request.Builder()
+ .method(method, body)
+ .url(url)
+ .addHeader("Authorization", encodeToken("accessToken:" + token))
+ .build();
+ }
+
+ protected static void checkResponseIsSuccessful(Response response) throws IOException {
+ if (!response.isSuccessful()) {
+ LOG.debug(UNABLE_TO_CONTACT_AZURE_SERVER + ": {} {}", response.request().url().toString(), response.code());
+ if (response.code() == HttpURLConnection.HTTP_UNAUTHORIZED) {
+ throw new IllegalArgumentException("Invalid personal access token");
+ }
+ ResponseBody responseBody = response.body();
+ String body = responseBody == null ? "" : responseBody.string();
+ String errorMessage = generateErrorMessage(body, UNABLE_TO_CONTACT_AZURE_SERVER);
+ LOG.info(String.format("Azure API call to [%s] failed with %s http code. Azure response content : [%s]", response.request().url().toString(), response.code(), body));
+ throw new IllegalArgumentException(errorMessage);
+ }
+ }
+
+ protected static String generateErrorMessage(String body, String defaultMessage) {
+ GsonAzureError gsonAzureError = null;
+ try {
+ gsonAzureError = buildGson().fromJson(body, GsonAzureError.class);
+ } catch (JsonSyntaxException e) {
+ // not a json payload, ignore the error
+ }
+ if (gsonAzureError != null && !Strings.isNullOrEmpty(gsonAzureError.message())) {
+ return defaultMessage + " : " + gsonAzureError.message();
+ } else {
+ return defaultMessage;
+ }
+ }
+
+ protected static String getTrimmedUrl(String rawUrl) {
+ if (isBlank(rawUrl)) {
+ return rawUrl;
+ }
+ if (rawUrl.endsWith("/")) {
+ return substringBeforeLast(rawUrl, "/");
+ }
+ return rawUrl;
+ }
+
+ protected static String encodeToken(String token) {
+ return String.format("BASIC %s", Base64.encodeBase64String(token.getBytes(StandardCharsets.UTF_8)));
+ }
+
+ protected static Gson buildGson() {
+ return new GsonBuilder()
+ .create();
+ }
+}
diff --git a/server/sonar-alm-client/src/main/java/org/sonar/alm/client/azure/GsonAzureError.java b/server/sonar-alm-client/src/main/java/org/sonar/alm/client/azure/GsonAzureError.java
new file mode 100644
index 00000000000..e4e91799e24
--- /dev/null
+++ b/server/sonar-alm-client/src/main/java/org/sonar/alm/client/azure/GsonAzureError.java
@@ -0,0 +1,41 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2021 SonarSource SA
+ * mailto:info AT sonarsource DOT com
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 3 of the License, or (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with this program; if not, write to the Free Software Foundation,
+ * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ */
+package org.sonar.alm.client.azure;
+
+import com.google.gson.annotations.SerializedName;
+import javax.annotation.Nullable;
+
+public class GsonAzureError {
+ @SerializedName("message")
+ private final String message;
+
+ public GsonAzureError(@Nullable String message) {
+ this.message = message;
+ }
+
+ public GsonAzureError() {
+ // http://stackoverflow.com/a/18645370/229031
+ this(null);
+ }
+
+ public String message() {
+ return message;
+ }
+}
diff --git a/server/sonar-alm-client/src/main/java/org/sonar/alm/client/azure/GsonAzureProject.java b/server/sonar-alm-client/src/main/java/org/sonar/alm/client/azure/GsonAzureProject.java
new file mode 100644
index 00000000000..c206f02d61d
--- /dev/null
+++ b/server/sonar-alm-client/src/main/java/org/sonar/alm/client/azure/GsonAzureProject.java
@@ -0,0 +1,48 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2021 SonarSource SA
+ * mailto:info AT sonarsource DOT com
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 3 of the License, or (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with this program; if not, write to the Free Software Foundation,
+ * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ */
+package org.sonar.alm.client.azure;
+
+import com.google.gson.annotations.SerializedName;
+
+public class GsonAzureProject {
+
+ @SerializedName("name")
+ private String name;
+
+ @SerializedName("description")
+ private String description;
+
+ public GsonAzureProject() {
+ // http://stackoverflow.com/a/18645370/229031
+ }
+
+ public GsonAzureProject(String name, String description) {
+ this.name = name;
+ this.description = description;
+ }
+
+ public String getName() {
+ return name;
+ }
+
+ public String getDescription() {
+ return description;
+ }
+}
diff --git a/server/sonar-alm-client/src/main/java/org/sonar/alm/client/azure/GsonAzureProjectList.java b/server/sonar-alm-client/src/main/java/org/sonar/alm/client/azure/GsonAzureProjectList.java
new file mode 100644
index 00000000000..e594bbbf90a
--- /dev/null
+++ b/server/sonar-alm-client/src/main/java/org/sonar/alm/client/azure/GsonAzureProjectList.java
@@ -0,0 +1,56 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2021 SonarSource SA
+ * mailto:info AT sonarsource DOT com
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 3 of the License, or (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with this program; if not, write to the Free Software Foundation,
+ * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ */
+package org.sonar.alm.client.azure;
+
+import com.google.gson.annotations.SerializedName;
+
+import java.util.ArrayList;
+import java.util.List;
+
+public class GsonAzureProjectList {
+
+ @SerializedName("value")
+ private List<GsonAzureProject> values;
+
+ public GsonAzureProjectList() {
+ // http://stackoverflow.com/a/18645370/229031
+ this(new ArrayList<>());
+ }
+
+ public GsonAzureProjectList(List<GsonAzureProject> values) {
+ this.values = values;
+ }
+
+ public List<GsonAzureProject> getValues() {
+ return values;
+ }
+
+ public GsonAzureProjectList setValues(List<GsonAzureProject> values) {
+ this.values = values;
+ return this;
+ }
+
+ @Override
+ public String toString() {
+ return "{" +
+ "values=" + values +
+ '}';
+ }
+}
diff --git a/server/sonar-alm-client/src/main/java/org/sonar/alm/client/azure/GsonAzureRepo.java b/server/sonar-alm-client/src/main/java/org/sonar/alm/client/azure/GsonAzureRepo.java
new file mode 100644
index 00000000000..60327fc5c63
--- /dev/null
+++ b/server/sonar-alm-client/src/main/java/org/sonar/alm/client/azure/GsonAzureRepo.java
@@ -0,0 +1,63 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2021 SonarSource SA
+ * mailto:info AT sonarsource DOT com
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 3 of the License, or (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with this program; if not, write to the Free Software Foundation,
+ * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ */
+package org.sonar.alm.client.azure;
+
+import com.google.gson.annotations.SerializedName;
+
+public class GsonAzureRepo {
+ @SerializedName("id")
+ private String id;
+
+ @SerializedName("name")
+ private String name;
+
+ @SerializedName("url")
+ private String url;
+
+ @SerializedName("project")
+ private GsonAzureProject project;
+
+ public GsonAzureRepo() {
+ // http://stackoverflow.com/a/18645370/229031
+ }
+
+ public GsonAzureRepo(String id, String name, String url, GsonAzureProject project) {
+ this.id = id;
+ this.name = name;
+ this.url = url;
+ this.project = project;
+ }
+
+ public String getId() {
+ return id;
+ }
+
+ public String getName() {
+ return name;
+ }
+
+ public String getUrl() {
+ return url;
+ }
+
+ public GsonAzureProject getProject() {
+ return project;
+ }
+}
diff --git a/server/sonar-alm-client/src/main/java/org/sonar/alm/client/azure/GsonAzureRepoList.java b/server/sonar-alm-client/src/main/java/org/sonar/alm/client/azure/GsonAzureRepoList.java
new file mode 100644
index 00000000000..eaeffc8a5c6
--- /dev/null
+++ b/server/sonar-alm-client/src/main/java/org/sonar/alm/client/azure/GsonAzureRepoList.java
@@ -0,0 +1,49 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2021 SonarSource SA
+ * mailto:info AT sonarsource DOT com
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 3 of the License, or (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with this program; if not, write to the Free Software Foundation,
+ * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ */
+package org.sonar.alm.client.azure;
+
+import com.google.gson.Gson;
+import com.google.gson.annotations.SerializedName;
+import java.util.ArrayList;
+import java.util.List;
+
+public class GsonAzureRepoList {
+
+ @SerializedName("value")
+ private List<GsonAzureRepo> values;
+
+ public GsonAzureRepoList() {
+ // http://stackoverflow.com/a/18645370/229031
+ this(new ArrayList<>());
+ }
+
+ public GsonAzureRepoList(List<GsonAzureRepo> values) {
+ this.values = values;
+ }
+
+ static GsonAzureRepoList parse(String json) {
+ return new Gson().fromJson(json, GsonAzureRepoList.class);
+ }
+
+ public List<GsonAzureRepo> getValues() {
+ return values;
+ }
+
+}
diff --git a/server/sonar-alm-client/src/test/java/org/sonar/alm/client/azure/AzureDevOpsHttpClientTest.java b/server/sonar-alm-client/src/test/java/org/sonar/alm/client/azure/AzureDevOpsHttpClientTest.java
new file mode 100644
index 00000000000..c296f79b5af
--- /dev/null
+++ b/server/sonar-alm-client/src/test/java/org/sonar/alm/client/azure/AzureDevOpsHttpClientTest.java
@@ -0,0 +1,273 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2021 SonarSource SA
+ * mailto:info AT sonarsource DOT com
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 3 of the License, or (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with this program; if not, write to the Free Software Foundation,
+ * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ */
+package org.sonar.alm.client.azure;
+
+import okhttp3.mockwebserver.MockResponse;
+import okhttp3.mockwebserver.MockWebServer;
+import okhttp3.mockwebserver.RecordedRequest;
+import org.junit.After;
+import org.junit.Before;
+import org.junit.Rule;
+import org.junit.Test;
+import org.sonar.alm.client.ConstantTimeoutConfiguration;
+import org.sonar.alm.client.TimeoutConfiguration;
+import org.sonar.api.utils.log.LogTester;
+import org.sonar.api.utils.log.LoggerLevel;
+
+import static org.assertj.core.api.Assertions.assertThat;
+import static org.assertj.core.api.Assertions.assertThatThrownBy;
+import static org.assertj.core.api.Assertions.tuple;
+import javax.annotation.Nullable;
+import java.io.IOException;
+import java.util.concurrent.TimeUnit;
+
+public class AzureDevOpsHttpClientTest {
+ public static final String UNABLE_TO_CONTACT_AZURE = "Unable to contact Azure DevOps server, got an unexpected response";
+ @Rule
+ public LogTester logTester = new LogTester();
+
+ private static final String NON_JSON_PAYLOAD = "non json payload";
+ private final MockWebServer server = new MockWebServer();
+ private AzureDevOpsHttpClient underTest;
+
+ @Before
+ public void prepare() throws IOException {
+ server.start();
+
+ TimeoutConfiguration timeoutConfiguration = new ConstantTimeoutConfiguration(10_000);
+ underTest = new AzureDevOpsHttpClient(timeoutConfiguration);
+ }
+
+ @After
+ public void stopServer() throws IOException {
+ server.shutdown();
+ }
+
+ @Test
+ public void get_projects() throws InterruptedException {
+ enqueueResponse(200, " { \"count\": 2,\n" +
+ " \"value\": [\n" +
+ " {\n" +
+ " \"id\": \"3311cd05-3f00-4a5e-b47f-df94a9982b6e\",\n" +
+ " \"name\": \"Project 1\",\n" +
+ " \"description\": \"Project Description\",\n" +
+ " \"url\": \"https://ado.sonarqube.com/DefaultCollection/_apis/projects/3311cd05-3f00-4a5e-b47f-df94a9982b6e\",\n" +
+ " \"state\": \"wellFormed\",\n" +
+ " \"revision\": 63,\n" +
+ " \"visibility\": \"private\"\n" +
+ " }," +
+ "{\n" +
+ " \"id\": \"3be0f34d-c931-4ff8-8d37-18a83663bd3c\",\n" +
+ " \"name\": \"Project 2\",\n" +
+ " \"url\": \"https://ado.sonarqube.com/DefaultCollection/_apis/projects/3be0f34d-c931-4ff8-8d37-18a83663bd3c\",\n" +
+ " \"state\": \"wellFormed\",\n" +
+ " \"revision\": 52,\n" +
+ " \"visibility\": \"private\"\n" +
+ " }]}");
+
+ GsonAzureProjectList projects = underTest.getProjects(server.url("").toString(), "token");
+
+ RecordedRequest request = server.takeRequest(10, TimeUnit.SECONDS);
+ String azureDevOpsUrlCall = request.getRequestUrl().toString();
+ assertThat(azureDevOpsUrlCall).isEqualTo(server.url("") + "_apis/projects");
+ assertThat(request.getMethod()).isEqualTo("GET");
+
+ assertThat(logTester.logs(LoggerLevel.DEBUG)).hasSize(1);
+ assertThat(logTester.logs(LoggerLevel.DEBUG))
+ .contains("get projects : [" + server.url("").toString() + "_apis/projects]");
+ assertThat(projects.getValues()).hasSize(2);
+ assertThat(projects.getValues())
+ .extracting(GsonAzureProject::getName, GsonAzureProject::getDescription)
+ .containsExactly(tuple("Project 1", "Project Description"), tuple("Project 2", null));
+ }
+
+ @Test
+ public void get_projects_non_json_payload() {
+ enqueueResponse(200, NON_JSON_PAYLOAD);
+
+ assertThatThrownBy(() -> underTest.getProjects(server.url("").toString(), "token"))
+ .isInstanceOf(IllegalArgumentException.class)
+ .hasMessage(UNABLE_TO_CONTACT_AZURE);
+ }
+
+ @Test
+ public void get_projects_with_invalid_pat() {
+ enqueueResponse(401);
+
+ assertThatThrownBy(() -> underTest.getProjects(server.url("").toString(), "invalid-token"))
+ .isInstanceOf(IllegalArgumentException.class)
+ .hasMessage("Invalid personal access token");
+ }
+
+ @Test
+ public void get_projects_with_server_error() {
+ enqueueResponse(500);
+
+ assertThatThrownBy(() -> underTest.getProjects(server.url("").toString(), "token"))
+ .isInstanceOf(IllegalArgumentException.class)
+ .hasMessage("Unable to contact Azure DevOps server");
+ }
+
+ @Test
+ public void get_repos_with_project_name() throws InterruptedException {
+ enqueueResponse(200, "{\n" +
+ " \"value\": [\n" +
+ " {\n" +
+ " \"id\": \"741248a4-285e-4a6d-af52-1a49d8070638\",\n" +
+ " \"name\": \"Repository 1\",\n" +
+ " \"url\": \"https://ado.sonarqube.com/repositories/\",\n" +
+ " \"project\": {\n" +
+ " \"id\": \"c88ddb32-ced8-420d-ab34-764133038b34\",\n" +
+ " \"name\": \"projectName\",\n" +
+ " \"url\": \"https://ado.sonarqube.com/DefaultCollection/_apis/projects/c88ddb32-ced8-420d-ab34-764133038b34\",\n" +
+ " \"state\": \"wellFormed\",\n" +
+ " \"revision\": 29,\n" +
+ " \"visibility\": \"private\",\n" +
+ " \"lastUpdateTime\": \"2020-11-11T09:38:03.3Z\"\n" +
+ " },\n" +
+ " \"size\": 0\n" +
+ " }\n" +
+ " ],\n" +
+ " \"count\": 1\n" +
+ "}");
+
+ GsonAzureRepoList repos = underTest.getRepos(server.url("").toString(), "token", "projectName");
+
+ RecordedRequest request = server.takeRequest(10, TimeUnit.SECONDS);
+ String azureDevOpsUrlCall = request.getRequestUrl().toString();
+ assertThat(azureDevOpsUrlCall).isEqualTo(server.url("") + "projectName/_apis/git/repositories");
+ assertThat(request.getMethod()).isEqualTo("GET");
+
+ assertThat(logTester.logs(LoggerLevel.DEBUG)).hasSize(1);
+ assertThat(logTester.logs(LoggerLevel.DEBUG))
+ .contains("get repos : [" + server.url("").toString() + "projectName/_apis/git/repositories]");
+ assertThat(repos.getValues()).hasSize(1);
+ assertThat(repos.getValues())
+ .extracting(GsonAzureRepo::getName, GsonAzureRepo::getUrl, r -> r.getProject().getName())
+ .containsExactly(tuple("Repository 1", "https://ado.sonarqube.com/repositories/", "projectName"));
+ }
+
+ @Test
+ public void get_repos_without_project_name() throws InterruptedException {
+ enqueueResponse(200, "{ \"value\": [], \"count\": 0 }");
+
+ GsonAzureRepoList repos = underTest.getRepos(server.url("").toString(), "token", null);
+
+ RecordedRequest request = server.takeRequest(10, TimeUnit.SECONDS);
+ String azureDevOpsUrlCall = request.getRequestUrl().toString();
+ assertThat(azureDevOpsUrlCall).isEqualTo(server.url("") + "_apis/git/repositories");
+ assertThat(request.getMethod()).isEqualTo("GET");
+ assertThat(repos.getValues()).isEmpty();
+ }
+
+ @Test
+ public void get_repos_non_json_payload() {
+ enqueueResponse(200, NON_JSON_PAYLOAD);
+
+ assertThatThrownBy(() -> underTest.getRepos(server.url("").toString(), "token", null))
+ .isInstanceOf(IllegalArgumentException.class)
+ .hasMessage(UNABLE_TO_CONTACT_AZURE);
+ }
+
+ @Test
+ public void get_repo() throws InterruptedException {
+ enqueueResponse(200, "{ " +
+ " \"id\": \"Repo-Id-1\",\n" +
+ " \"name\": \"Repo-Name-1\",\n" +
+ " \"url\": \"https://ado.sonarqube.com/DefaultCollection/Repo-Id-1\",\n" +
+ " \"project\": {\n" +
+ " \"id\": \"84ea9d51-0c8a-44ad-be92-b2af7fe2c299\",\n" +
+ " \"name\": \"Project-Name\",\n" +
+ " \"description\": \"Project's description\" \n" +
+ " },\n" +
+ " \"size\": 0" +
+ "}");
+
+ GsonAzureRepo repo = underTest.getRepo(server.url("").toString(), "token", "Project-Name", "Repo-Name-1");
+
+ RecordedRequest request = server.takeRequest(10, TimeUnit.SECONDS);
+ String azureDevOpsUrlCall = request.getRequestUrl().toString();
+ assertThat(azureDevOpsUrlCall).isEqualTo(server.url("") + "Project-Name/_apis/git/repositories/Repo-Name-1");
+ assertThat(request.getMethod()).isEqualTo("GET");
+
+ assertThat(logTester.logs(LoggerLevel.DEBUG)).hasSize(1);
+ assertThat(logTester.logs(LoggerLevel.DEBUG))
+ .contains("get repo : [" + server.url("").toString() + "Project-Name/_apis/git/repositories/Repo-Name-1]");
+ assertThat(repo.getId()).isEqualTo("Repo-Id-1");
+ assertThat(repo.getName()).isEqualTo("Repo-Name-1");
+ assertThat(repo.getUrl()).isEqualTo("https://ado.sonarqube.com/DefaultCollection/Repo-Id-1");
+ assertThat(repo.getProject().getName()).isEqualTo("Project-Name");
+ }
+
+ @Test
+ public void get_repo_non_json_payload() {
+ enqueueResponse(200, NON_JSON_PAYLOAD);
+
+ assertThatThrownBy(() -> underTest.getRepo(server.url("").toString(), "token", "projectName", "repoName"))
+ .isInstanceOf(IllegalArgumentException.class)
+ .hasMessage(UNABLE_TO_CONTACT_AZURE);
+ }
+
+ @Test
+ public void get_repo_json_error_payload() {
+ enqueueResponse(400,
+ "{'message':'TF200016: The following project does not exist: projectName. Verify that the name of the project is correct and that the project exists on the specified Azure DevOps Server.'}");
+
+ assertThatThrownBy(() -> underTest.getRepo(server.url("").toString(), "token", "projectName", "repoName"))
+ .isInstanceOf(IllegalArgumentException.class)
+ .hasMessage(
+ "Unable to contact Azure DevOps server : TF200016: The following project does not exist: projectName. Verify that the name of the project is correct and that the project exists on the specified Azure DevOps Server.");
+ }
+
+ private void enqueueResponse(int responseCode) {
+ enqueueResponse(responseCode, "");
+ }
+
+ private void enqueueResponse(int responseCode, @Nullable String body) {
+ server.enqueue(new MockResponse()
+ .setHeader("Content-Type", "application/json;charset=UTF-8")
+ .setResponseCode(responseCode)
+ .setBody(body));
+ }
+
+ @Test
+ public void trim_url() {
+ assertThat(AzureDevOpsHttpClient.getTrimmedUrl("http://localhost:4564/"))
+ .isEqualTo("http://localhost:4564");
+ }
+
+ @Test
+ public void trim_url_without_ending_slash() {
+ assertThat(AzureDevOpsHttpClient.getTrimmedUrl("http://localhost:4564"))
+ .isEqualTo("http://localhost:4564");
+ }
+
+ @Test
+ public void trim_null_url() {
+ assertThat(AzureDevOpsHttpClient.getTrimmedUrl(null))
+ .isNull();
+ }
+
+ @Test
+ public void trim_empty_url() {
+ assertThat(AzureDevOpsHttpClient.getTrimmedUrl(""))
+ .isEmpty();
+ }
+}
diff --git a/server/sonar-webserver-webapi/src/main/java/org/sonar/server/almintegration/ws/AlmIntegrationsWSModule.java b/server/sonar-webserver-webapi/src/main/java/org/sonar/server/almintegration/ws/AlmIntegrationsWSModule.java
index 65b65cac982..0496dae204c 100644
--- a/server/sonar-webserver-webapi/src/main/java/org/sonar/server/almintegration/ws/AlmIntegrationsWSModule.java
+++ b/server/sonar-webserver-webapi/src/main/java/org/sonar/server/almintegration/ws/AlmIntegrationsWSModule.java
@@ -20,6 +20,9 @@
package org.sonar.server.almintegration.ws;
import org.sonar.core.platform.Module;
+import org.sonar.server.almintegration.ws.azure.ImportAzureProjectAction;
+import org.sonar.server.almintegration.ws.azure.ListAzureProjectsAction;
+import org.sonar.server.almintegration.ws.azure.SearchAzureReposAction;
import org.sonar.server.almintegration.ws.bitbucketserver.ImportBitbucketServerProjectAction;
import org.sonar.server.almintegration.ws.bitbucketserver.ListBitbucketServerProjectsAction;
import org.sonar.server.almintegration.ws.bitbucketserver.SearchBitbucketServerReposAction;
@@ -35,6 +38,9 @@ public class AlmIntegrationsWSModule extends Module {
SearchBitbucketServerReposAction.class,
ImportGitLabProjectAction.class,
SearchGitlabReposAction.class,
+ ImportAzureProjectAction.class,
+ ListAzureProjectsAction.class,
+ SearchAzureReposAction.class,
AlmIntegrationsWs.class);
}
}
diff --git a/server/sonar-webserver-webapi/src/main/java/org/sonar/server/almintegration/ws/azure/ImportAzureProjectAction.java b/server/sonar-webserver-webapi/src/main/java/org/sonar/server/almintegration/ws/azure/ImportAzureProjectAction.java
new file mode 100644
index 00000000000..de7a2a6fc8b
--- /dev/null
+++ b/server/sonar-webserver-webapi/src/main/java/org/sonar/server/almintegration/ws/azure/ImportAzureProjectAction.java
@@ -0,0 +1,163 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2021 SonarSource SA
+ * mailto:info AT sonarsource DOT com
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 3 of the License, or (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with this program; if not, write to the Free Software Foundation,
+ * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ */
+package org.sonar.server.almintegration.ws.azure;
+
+import com.google.common.annotations.VisibleForTesting;
+
+import java.util.Optional;
+
+import org.sonar.alm.client.azure.AzureDevOpsHttpClient;
+import org.sonar.alm.client.azure.GsonAzureRepo;
+import org.sonar.api.server.ws.Request;
+import org.sonar.api.server.ws.Response;
+import org.sonar.api.server.ws.WebService;
+import org.sonar.db.DbClient;
+import org.sonar.db.DbSession;
+import org.sonar.db.alm.pat.AlmPatDto;
+import org.sonar.db.alm.setting.AlmSettingDto;
+import org.sonar.db.alm.setting.ProjectAlmSettingDto;
+import org.sonar.db.component.ComponentDto;
+import org.sonar.server.almintegration.ws.AlmIntegrationsWsAction;
+import org.sonar.server.almintegration.ws.ImportHelper;
+import org.sonar.server.component.ComponentUpdater;
+import org.sonar.server.project.ProjectDefaultVisibility;
+import org.sonar.server.user.UserSession;
+import org.sonarqube.ws.Projects.CreateWsResponse;
+
+import static java.util.Objects.requireNonNull;
+import static org.sonar.api.resources.Qualifiers.PROJECT;
+import static org.sonar.server.almintegration.ws.ImportHelper.PARAM_ALM_SETTING;
+import static org.sonar.server.almintegration.ws.ImportHelper.toCreateResponse;
+import static org.sonar.server.component.NewComponent.newComponentBuilder;
+import static org.sonar.server.ws.WsUtils.writeProtobuf;
+
+public class ImportAzureProjectAction implements AlmIntegrationsWsAction {
+
+ private static final String PARAM_REPOSITORY_NAME = "repositoryName";
+ private static final String PARAM_PROJECT_NAME = "projectName";
+
+ private final DbClient dbClient;
+ private final UserSession userSession;
+ private final AzureDevOpsHttpClient azureDevOpsHttpClient;
+ private final ProjectDefaultVisibility projectDefaultVisibility;
+ private final ComponentUpdater componentUpdater;
+ private final ImportHelper importHelper;
+
+ public ImportAzureProjectAction(DbClient dbClient, UserSession userSession, AzureDevOpsHttpClient azureDevOpsHttpClient,
+ ProjectDefaultVisibility projectDefaultVisibility, ComponentUpdater componentUpdater,
+ ImportHelper importHelper) {
+ this.dbClient = dbClient;
+ this.userSession = userSession;
+ this.azureDevOpsHttpClient = azureDevOpsHttpClient;
+ this.projectDefaultVisibility = projectDefaultVisibility;
+ this.componentUpdater = componentUpdater;
+ this.importHelper = importHelper;
+ }
+
+ @Override
+ public void define(WebService.NewController context) {
+ WebService.NewAction action = context.createAction("import_azure_project")
+ .setDescription("Create a SonarQube project with the information from the provided Azure DevOps project.<br/>" +
+ "Autoconfigure pull request decoration mechanism.<br/>" +
+ "Requires the 'Create Projects' permission")
+ .setPost(true)
+ .setInternal(true)
+ .setSince("8.6")
+ .setHandler(this);
+
+ action.createParam(PARAM_ALM_SETTING)
+ .setRequired(true)
+ .setMaximumLength(200)
+ .setDescription("ALM setting key");
+
+ action.createParam(PARAM_PROJECT_NAME)
+ .setRequired(true)
+ .setMaximumLength(200)
+ .setDescription("Azure project name");
+
+ action.createParam(PARAM_REPOSITORY_NAME)
+ .setRequired(true)
+ .setMaximumLength(200)
+ .setDescription("Azure repository name");
+ }
+
+ @Override
+ public void handle(Request request, Response response) {
+ CreateWsResponse createResponse = doHandle(request);
+ writeProtobuf(createResponse, request, response);
+ }
+
+ private CreateWsResponse doHandle(Request request) {
+ importHelper.checkProvisionProjectPermission();
+ AlmSettingDto almSettingDto = importHelper.getAlmSetting(request);
+ String userUuid = importHelper.getUserUuid();
+ try (DbSession dbSession = dbClient.openSession(false)) {
+
+ Optional<AlmPatDto> almPatDto = dbClient.almPatDao().selectByUserAndAlmSetting(dbSession, userUuid, almSettingDto);
+ String pat = almPatDto.map(AlmPatDto::getPersonalAccessToken)
+ .orElseThrow(() -> new IllegalArgumentException(String.format("personal access token for '%s' is missing", almSettingDto.getKey())));
+
+ String projectName = request.mandatoryParam(PARAM_PROJECT_NAME);
+ String repositoryName = request.mandatoryParam(PARAM_REPOSITORY_NAME);
+
+ String url = requireNonNull(almSettingDto.getUrl(), "ALM url cannot be null");
+ GsonAzureRepo repo = azureDevOpsHttpClient.getRepo(url, pat, projectName, repositoryName);
+
+ ComponentDto componentDto = createProject(dbSession, repo);
+ populatePRSetting(dbSession, repo, componentDto, almSettingDto);
+
+ return toCreateResponse(componentDto);
+ }
+ }
+
+ private ComponentDto createProject(DbSession dbSession, GsonAzureRepo repo) {
+ boolean visibility = projectDefaultVisibility.get(dbSession).isPrivate();
+ return componentUpdater.create(dbSession, newComponentBuilder()
+ .setKey(generateProjectKey(repo.getProject().getName(), repo.getName()))
+ .setName(repo.getName())
+ .setPrivate(visibility)
+ .setQualifier(PROJECT)
+ .build(),
+ userSession.isLoggedIn() ? userSession.getUuid() : null);
+ }
+
+ private void populatePRSetting(DbSession dbSession, GsonAzureRepo repo, ComponentDto componentDto, AlmSettingDto almSettingDto) {
+ ProjectAlmSettingDto projectAlmSettingDto = new ProjectAlmSettingDto()
+ .setAlmSettingUuid(almSettingDto.getUuid())
+ .setAlmRepo(repo.getName())
+ .setAlmSlug(repo.getProject().getName())
+ .setProjectUuid(componentDto.uuid())
+ .setMonorepo(false);
+ dbClient.projectAlmSettingDao().insertOrUpdate(dbSession, projectAlmSettingDto);
+ dbSession.commit();
+ }
+
+ @VisibleForTesting
+ String generateProjectKey(String projectName, String repoName) {
+ String sqProjectKey = projectName + "_" + repoName;
+
+ if (sqProjectKey.length() > 250) {
+ sqProjectKey = sqProjectKey.substring(sqProjectKey.length() - 250);
+ }
+
+ return sqProjectKey.replace(" ", "_");
+ }
+
+}
diff --git a/server/sonar-webserver-webapi/src/main/java/org/sonar/server/almintegration/ws/azure/ListAzureProjectsAction.java b/server/sonar-webserver-webapi/src/main/java/org/sonar/server/almintegration/ws/azure/ListAzureProjectsAction.java
new file mode 100644
index 00000000000..7bd8a3f9ac9
--- /dev/null
+++ b/server/sonar-webserver-webapi/src/main/java/org/sonar/server/almintegration/ws/azure/ListAzureProjectsAction.java
@@ -0,0 +1,115 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2021 SonarSource SA
+ * mailto:info AT sonarsource DOT com
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 3 of the License, or (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with this program; if not, write to the Free Software Foundation,
+ * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ */
+package org.sonar.server.almintegration.ws.azure;
+
+import java.util.List;
+import java.util.Optional;
+import java.util.stream.Collectors;
+
+import org.sonar.alm.client.azure.AzureDevOpsHttpClient;
+import org.sonar.alm.client.azure.GsonAzureProject;
+import org.sonar.alm.client.azure.GsonAzureProjectList;
+import org.sonar.api.server.ws.Request;
+import org.sonar.api.server.ws.Response;
+import org.sonar.api.server.ws.WebService;
+import org.sonar.db.DbClient;
+import org.sonar.db.DbSession;
+import org.sonar.db.alm.pat.AlmPatDto;
+import org.sonar.db.alm.setting.AlmSettingDto;
+import org.sonar.server.almintegration.ws.AlmIntegrationsWsAction;
+import org.sonar.server.exceptions.NotFoundException;
+import org.sonar.server.user.UserSession;
+import org.sonarqube.ws.AlmIntegrations.AzureProject;
+import org.sonarqube.ws.AlmIntegrations.ListAzureProjectsWsResponse;
+
+import static java.util.Comparator.comparing;
+import static java.util.Objects.requireNonNull;
+import static org.sonar.db.permission.GlobalPermission.PROVISION_PROJECTS;
+import static org.sonar.server.ws.WsUtils.writeProtobuf;
+
+public class ListAzureProjectsAction implements AlmIntegrationsWsAction {
+
+ private static final String PARAM_ALM_SETTING = "almSetting";
+
+ private final DbClient dbClient;
+ private final UserSession userSession;
+ private final AzureDevOpsHttpClient azureDevOpsHttpClient;
+
+ public ListAzureProjectsAction(DbClient dbClient, UserSession userSession, AzureDevOpsHttpClient azureDevOpsHttpClient) {
+ this.dbClient = dbClient;
+ this.userSession = userSession;
+ this.azureDevOpsHttpClient = azureDevOpsHttpClient;
+ }
+
+ @Override
+ public void define(WebService.NewController context) {
+ WebService.NewAction action = context.createAction("list_azure_projects")
+ .setDescription("List Azure projects<br/>" +
+ "Requires the 'Create Projects' permission")
+ .setPost(false)
+ .setSince("8.6")
+ .setHandler(this);
+
+ action.createParam(PARAM_ALM_SETTING)
+ .setRequired(true)
+ .setMaximumLength(200)
+ .setDescription("ALM setting key");
+ }
+
+ @Override
+ public void handle(Request request, Response response) {
+
+ ListAzureProjectsWsResponse wsResponse = doHandle(request);
+ writeProtobuf(wsResponse, request, response);
+ }
+
+ private ListAzureProjectsWsResponse doHandle(Request request) {
+
+ try (DbSession dbSession = dbClient.openSession(false)) {
+ userSession.checkLoggedIn().checkPermission(PROVISION_PROJECTS);
+
+ String almSettingKey = request.mandatoryParam(PARAM_ALM_SETTING);
+ String userUuid = requireNonNull(userSession.getUuid(), "User UUID is not null");
+ AlmSettingDto almSettingDto = dbClient.almSettingDao().selectByKey(dbSession, almSettingKey)
+ .orElseThrow(() -> new NotFoundException(String.format("ALM Setting '%s' not found", almSettingKey)));
+ Optional<AlmPatDto> almPatDto = dbClient.almPatDao().selectByUserAndAlmSetting(dbSession, userUuid, almSettingDto);
+ String pat = almPatDto.map(AlmPatDto::getPersonalAccessToken).orElseThrow(() -> new IllegalArgumentException("No personal access token found"));
+
+ String url = requireNonNull(almSettingDto.getUrl(), "URL cannot be null");
+ GsonAzureProjectList projectList = azureDevOpsHttpClient.getProjects(url, pat);
+
+ List<AzureProject> values = projectList.getValues().stream()
+ .map(ListAzureProjectsAction::toAzureProject)
+ .sorted(comparing(AzureProject::getName, String::compareToIgnoreCase))
+ .collect(Collectors.toList());
+ ListAzureProjectsWsResponse.Builder builder = ListAzureProjectsWsResponse.newBuilder()
+ .addAllProjects(values);
+ return builder.build();
+ }
+ }
+
+ private static AzureProject toAzureProject(GsonAzureProject project) {
+ return AzureProject.newBuilder()
+ .setName(project.getName())
+ .setDescription(Optional.ofNullable(project.getDescription()).orElse(""))
+ .build();
+ }
+
+}
diff --git a/server/sonar-webserver-webapi/src/main/java/org/sonar/server/almintegration/ws/azure/SearchAzureReposAction.java b/server/sonar-webserver-webapi/src/main/java/org/sonar/server/almintegration/ws/azure/SearchAzureReposAction.java
new file mode 100644
index 00000000000..33a3825c528
--- /dev/null
+++ b/server/sonar-webserver-webapi/src/main/java/org/sonar/server/almintegration/ws/azure/SearchAzureReposAction.java
@@ -0,0 +1,235 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2021 SonarSource SA
+ * mailto:info AT sonarsource DOT com
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 3 of the License, or (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with this program; if not, write to the Free Software Foundation,
+ * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ */
+package org.sonar.server.almintegration.ws.azure;
+
+import java.util.List;
+import java.util.Map;
+import java.util.Objects;
+import java.util.Optional;
+import java.util.Set;
+import java.util.function.BinaryOperator;
+import java.util.function.Function;
+import java.util.stream.Collectors;
+import javax.annotation.Nullable;
+
+import org.sonar.alm.client.azure.AzureDevOpsHttpClient;
+import org.sonar.alm.client.azure.GsonAzureRepo;
+import org.sonar.alm.client.azure.GsonAzureRepoList;
+import org.sonar.api.server.ws.Request;
+import org.sonar.api.server.ws.Response;
+import org.sonar.api.server.ws.WebService;
+import org.sonar.api.utils.log.Logger;
+import org.sonar.api.utils.log.Loggers;
+import org.sonar.db.DbClient;
+import org.sonar.db.DbSession;
+import org.sonar.db.alm.pat.AlmPatDto;
+import org.sonar.db.alm.setting.AlmSettingDto;
+import org.sonar.db.alm.setting.ProjectAlmSettingDto;
+import org.sonar.db.project.ProjectDto;
+import org.sonar.server.almintegration.ws.AlmIntegrationsWsAction;
+import org.sonar.server.exceptions.NotFoundException;
+import org.sonar.server.user.UserSession;
+import org.sonarqube.ws.AlmIntegrations.AzureRepo;
+import org.sonarqube.ws.AlmIntegrations.SearchAzureReposWsResponse;
+
+import static java.util.Comparator.comparing;
+import static java.util.Objects.requireNonNull;
+import static java.util.stream.Collectors.toList;
+import static java.util.stream.Collectors.toMap;
+import static java.util.stream.Collectors.toSet;
+import static org.apache.commons.lang.StringUtils.containsIgnoreCase;
+import static org.sonar.db.permission.GlobalPermission.PROVISION_PROJECTS;
+import static org.sonar.server.ws.WsUtils.writeProtobuf;
+
+public class SearchAzureReposAction implements AlmIntegrationsWsAction {
+
+ private static final Logger LOG = Loggers.get(SearchAzureReposAction.class);
+
+ private static final String PARAM_ALM_SETTING = "almSetting";
+ private static final String PARAM_PROJECT_NAME = "projectName";
+ private static final String PARAM_SEARCH_QUERY = "searchQuery";
+
+ private final DbClient dbClient;
+ private final UserSession userSession;
+ private final AzureDevOpsHttpClient azureDevOpsHttpClient;
+
+ public SearchAzureReposAction(DbClient dbClient, UserSession userSession,
+ AzureDevOpsHttpClient azureDevOpsHttpClient) {
+ this.dbClient = dbClient;
+ this.userSession = userSession;
+ this.azureDevOpsHttpClient = azureDevOpsHttpClient;
+ }
+
+ @Override
+ public void define(WebService.NewController context) {
+ WebService.NewAction action = context.createAction("search_azure_repos")
+ .setDescription("Search the Azure repositories<br/>" +
+ "Requires the 'Create Projects' permission")
+ .setPost(false)
+ .setSince("8.6")
+ .setHandler(this);
+
+ action.createParam(PARAM_ALM_SETTING)
+ .setRequired(true)
+ .setMaximumLength(200)
+ .setDescription("ALM setting key");
+ action.createParam(PARAM_PROJECT_NAME)
+ .setRequired(false)
+ .setMaximumLength(200)
+ .setDescription("Project name filter");
+ action.createParam(PARAM_SEARCH_QUERY)
+ .setRequired(false)
+ .setMaximumLength(200)
+ .setDescription("Search query filter");
+ }
+
+ @Override
+ public void handle(Request request, Response response) {
+
+ SearchAzureReposWsResponse wsResponse = doHandle(request);
+ writeProtobuf(wsResponse, request, response);
+
+ }
+
+ private SearchAzureReposWsResponse doHandle(Request request) {
+
+ try (DbSession dbSession = dbClient.openSession(false)) {
+ userSession.checkLoggedIn().checkPermission(PROVISION_PROJECTS);
+
+ String almSettingKey = request.mandatoryParam(PARAM_ALM_SETTING);
+ String userUuid = requireNonNull(userSession.getUuid(), "User UUID cannot be null");
+ AlmSettingDto almSettingDto = dbClient.almSettingDao().selectByKey(dbSession, almSettingKey)
+ .orElseThrow(() -> new NotFoundException(String.format("ALM Setting '%s' not found", almSettingKey)));
+ Optional<AlmPatDto> almPatDto = dbClient.almPatDao().selectByUserAndAlmSetting(dbSession, userUuid, almSettingDto);
+
+ String projectKey = request.param(PARAM_PROJECT_NAME);
+ String searchQuery = request.param(PARAM_SEARCH_QUERY);
+ String pat = almPatDto.map(AlmPatDto::getPersonalAccessToken).orElseThrow(() -> new IllegalArgumentException("No personal access token found"));
+ String url = requireNonNull(almSettingDto.getUrl(), "ALM url cannot be null");
+
+ GsonAzureRepoList gsonAzureRepoList = azureDevOpsHttpClient.getRepos(url, pat, projectKey);
+
+ Map<ProjectKeyName, ProjectDto> sqProjectsKeyByAzureKey = getSqProjectsKeyByCustomKey(dbSession, almSettingDto, gsonAzureRepoList);
+
+ List<AzureRepo> repositories = gsonAzureRepoList.getValues()
+ .stream()
+ .filter(r -> isSearchOnlyByProjectName(searchQuery) || doesSearchCriteriaMatchProjectOrRepo(r, searchQuery))
+ .map(repo -> toAzureRepo(repo, sqProjectsKeyByAzureKey))
+ .sorted(comparing(AzureRepo::getName, String::compareToIgnoreCase))
+ .collect(toList());
+
+ LOG.debug(repositories.toString());
+
+ return SearchAzureReposWsResponse.newBuilder()
+ .addAllRepositories(repositories)
+ .build();
+ }
+ }
+
+ private Map<ProjectKeyName, ProjectDto> getSqProjectsKeyByCustomKey(DbSession dbSession, AlmSettingDto almSettingDto,
+ GsonAzureRepoList azureProjectList) {
+ Set<String> projectNames = azureProjectList.getValues().stream().map(r -> r.getProject().getName()).collect(toSet());
+ Set<ProjectKeyName> azureProjectsAndRepos = azureProjectList.getValues().stream().map(ProjectKeyName::from).collect(toSet());
+
+ List<ProjectAlmSettingDto> projectAlmSettingDtos = dbClient.projectAlmSettingDao()
+ .selectByAlmSettingAndSlugs(dbSession, almSettingDto, projectNames);
+
+ Map<String, ProjectAlmSettingDto> filteredProjectsByUuid = projectAlmSettingDtos
+ .stream()
+ .filter(p -> azureProjectsAndRepos.contains(ProjectKeyName.from(p)))
+ .collect(toMap(ProjectAlmSettingDto::getProjectUuid, Function.identity()));
+
+ Set<String> projectUuids = filteredProjectsByUuid.values().stream().map(ProjectAlmSettingDto::getProjectUuid).collect(toSet());
+
+ return dbClient.projectDao().selectByUuids(dbSession, projectUuids)
+ .stream()
+ .collect(Collectors.toMap(
+ projectDto -> ProjectKeyName.from(filteredProjectsByUuid.get(projectDto.getUuid())),
+ p -> p,
+ resolveNameCollisionOperatorByNaturalOrder()));
+ }
+
+ private static boolean isSearchOnlyByProjectName(@Nullable String criteria) {
+ return criteria == null || criteria.isEmpty();
+ }
+
+ private static boolean doesSearchCriteriaMatchProjectOrRepo(GsonAzureRepo repo, String criteria) {
+ boolean matchProject = containsIgnoreCase(repo.getProject().getName(), criteria);
+ boolean matchRepo = containsIgnoreCase(repo.getName(), criteria);
+ return matchProject || matchRepo;
+ }
+
+ private static AzureRepo toAzureRepo(GsonAzureRepo azureRepo, Map<ProjectKeyName, ProjectDto> sqProjectsKeyByAzureKey) {
+ AzureRepo.Builder builder = AzureRepo.newBuilder()
+ .setName(azureRepo.getName())
+ .setProjectName(azureRepo.getProject().getName());
+
+ ProjectDto projectDto = sqProjectsKeyByAzureKey.get(ProjectKeyName.from(azureRepo));
+ if (projectDto != null) {
+ builder.setSqProjectName(projectDto.getName());
+ builder.setSqProjectKey(projectDto.getKey());
+ }
+
+ return builder.build();
+ }
+
+ private static BinaryOperator<ProjectDto> resolveNameCollisionOperatorByNaturalOrder() {
+ return (a, b) -> b.getKey().compareTo(a.getKey()) > 0 ? a : b;
+ }
+
+ static class ProjectKeyName {
+ final String projectName;
+ final String repoName;
+
+ ProjectKeyName(String projectName, String repoName) {
+ this.projectName = projectName;
+ this.repoName = repoName;
+ }
+
+ public static ProjectKeyName from(ProjectAlmSettingDto project) {
+ return new ProjectKeyName(project.getAlmSlug(), project.getAlmRepo());
+ }
+
+ public static ProjectKeyName from(GsonAzureRepo gsonAzureRepo) {
+ return new ProjectKeyName(gsonAzureRepo.getProject().getName(), gsonAzureRepo.getName());
+ }
+
+ @Override
+ public boolean equals(Object o) {
+ if (this == o) {
+ return true;
+ }
+
+ if (o == null || getClass() != o.getClass()) {
+ return false;
+ }
+
+ ProjectKeyName that = (ProjectKeyName) o;
+ return Objects.equals(projectName, that.projectName) &&
+ Objects.equals(repoName, that.repoName);
+ }
+
+ @Override
+ public int hashCode() {
+ return Objects.hash(projectName, repoName);
+ }
+ }
+
+}
diff --git a/server/sonar-webserver-webapi/src/main/java/org/sonar/server/almintegration/ws/azure/package-info.java b/server/sonar-webserver-webapi/src/main/java/org/sonar/server/almintegration/ws/azure/package-info.java
new file mode 100644
index 00000000000..23ca978e3db
--- /dev/null
+++ b/server/sonar-webserver-webapi/src/main/java/org/sonar/server/almintegration/ws/azure/package-info.java
@@ -0,0 +1,23 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2021 SonarSource SA
+ * mailto:info AT sonarsource DOT com
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 3 of the License, or (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with this program; if not, write to the Free Software Foundation,
+ * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ */
+@ParametersAreNonnullByDefault
+package org.sonar.server.almintegration.ws.azure;
+
+import javax.annotation.ParametersAreNonnullByDefault;
diff --git a/server/sonar-webserver-webapi/src/test/java/org/sonar/server/almintegration/ws/azure/ImportAzureProjectActionTest.java b/server/sonar-webserver-webapi/src/test/java/org/sonar/server/almintegration/ws/azure/ImportAzureProjectActionTest.java
new file mode 100644
index 00000000000..22b2439e197
--- /dev/null
+++ b/server/sonar-webserver-webapi/src/test/java/org/sonar/server/almintegration/ws/azure/ImportAzureProjectActionTest.java
@@ -0,0 +1,238 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2021 SonarSource SA
+ * mailto:info AT sonarsource DOT com
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 3 of the License, or (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with this program; if not, write to the Free Software Foundation,
+ * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ */
+package org.sonar.server.almintegration.ws.azure;
+
+import java.util.Optional;
+import java.util.stream.IntStream;
+
+import org.junit.Before;
+import org.junit.Rule;
+import org.junit.Test;
+import org.sonar.alm.client.azure.AzureDevOpsHttpClient;
+import org.sonar.alm.client.azure.GsonAzureProject;
+import org.sonar.alm.client.azure.GsonAzureRepo;
+import org.sonar.api.server.ws.WebService;
+import org.sonar.api.utils.System2;
+import org.sonar.core.i18n.I18n;
+import org.sonar.core.util.SequenceUuidFactory;
+import org.sonar.db.DbTester;
+import org.sonar.db.alm.pat.AlmPatDto;
+import org.sonar.db.alm.setting.AlmSettingDto;
+import org.sonar.db.alm.setting.ProjectAlmSettingDto;
+import org.sonar.db.project.ProjectDto;
+import org.sonar.db.user.UserDto;
+import org.sonar.server.almintegration.ws.ImportHelper;
+import org.sonar.server.component.ComponentUpdater;
+import org.sonar.server.es.TestProjectIndexers;
+import org.sonar.server.exceptions.BadRequestException;
+import org.sonar.server.exceptions.ForbiddenException;
+import org.sonar.server.exceptions.NotFoundException;
+import org.sonar.server.exceptions.UnauthorizedException;
+import org.sonar.server.favorite.FavoriteUpdater;
+import org.sonar.server.permission.PermissionTemplateService;
+import org.sonar.server.project.ProjectDefaultVisibility;
+import org.sonar.server.project.Visibility;
+import org.sonar.server.tester.UserSessionRule;
+import org.sonar.server.ws.TestRequest;
+import org.sonar.server.ws.WsActionTester;
+import org.sonarqube.ws.Projects;
+
+import static java.util.stream.Collectors.joining;
+import static org.assertj.core.api.Assertions.assertThat;
+import static org.assertj.core.api.Assertions.assertThatThrownBy;
+import static org.assertj.core.api.Assertions.tuple;
+import static org.mockito.ArgumentMatchers.any;
+import static org.mockito.Mockito.mock;
+import static org.mockito.Mockito.when;
+import static org.sonar.db.alm.integration.pat.AlmPatsTesting.newAlmPatDto;
+import static org.sonar.db.permission.GlobalPermission.PROVISION_PROJECTS;
+import static org.sonar.db.permission.GlobalPermission.SCAN;
+
+public class ImportAzureProjectActionTest {
+
+ @Rule
+ public UserSessionRule userSession = UserSessionRule.standalone();
+ @Rule
+ public DbTester db = DbTester.create();
+
+ private final AzureDevOpsHttpClient azureDevOpsHttpClient = mock(AzureDevOpsHttpClient.class);
+
+ private final ComponentUpdater componentUpdater = new ComponentUpdater(db.getDbClient(), mock(I18n.class), System2.INSTANCE,
+ mock(PermissionTemplateService.class), new FavoriteUpdater(db.getDbClient()), new TestProjectIndexers(), new SequenceUuidFactory());
+
+ private final ImportHelper importHelper = new ImportHelper(db.getDbClient(), userSession);
+ private final ProjectDefaultVisibility projectDefaultVisibility = mock(ProjectDefaultVisibility.class);
+ private final ImportAzureProjectAction importAzureProjectAction = new ImportAzureProjectAction(db.getDbClient(), userSession,
+ azureDevOpsHttpClient, projectDefaultVisibility, componentUpdater, importHelper);
+ private final WsActionTester ws = new WsActionTester(importAzureProjectAction);
+
+ @Before
+ public void before() {
+ when(projectDefaultVisibility.get(any())).thenReturn(Visibility.PRIVATE);
+ }
+
+ @Test
+ public void import_project() {
+ UserDto user = db.users().insertUser();
+ userSession.logIn(user).addPermission(PROVISION_PROJECTS);
+ AlmSettingDto almSetting = db.almSettings().insertAzureAlmSetting();
+ db.almPats().insert(dto -> {
+ dto.setAlmSettingUuid(almSetting.getUuid());
+ dto.setPersonalAccessToken(almSetting.getPersonalAccessToken());
+ dto.setUserUuid(user.getUuid());
+ });
+ GsonAzureRepo repo = getGsonAzureRepo();
+ when(azureDevOpsHttpClient.getRepo(almSetting.getUrl(), almSetting.getPersonalAccessToken(), "project-name", "repo-name"))
+ .thenReturn(repo);
+
+ Projects.CreateWsResponse response = ws.newRequest()
+ .setParam("almSetting", almSetting.getKey())
+ .setParam("projectName", "project-name")
+ .setParam("repositoryName", "repo-name")
+ .executeProtobuf(Projects.CreateWsResponse.class);
+
+ Projects.CreateWsResponse.Project result = response.getProject();
+ assertThat(result.getKey()).isEqualTo(repo.getProject().getName() + "_" + repo.getName());
+ assertThat(result.getName()).isEqualTo(repo.getName());
+
+ Optional<ProjectDto> projectDto = db.getDbClient().projectDao().selectProjectByKey(db.getSession(), result.getKey());
+ assertThat(projectDto).isPresent();
+ Optional<ProjectAlmSettingDto> projectAlmSettingDto = db.getDbClient().projectAlmSettingDao().selectByProject(db.getSession(), projectDto.get());
+ assertThat(projectAlmSettingDto.get().getAlmRepo()).isEqualTo("repo-name");
+ assertThat(projectAlmSettingDto.get().getAlmSettingUuid()).isEqualTo(almSetting.getUuid());
+ assertThat(projectAlmSettingDto.get().getAlmSlug()).isEqualTo("project-name");
+ }
+
+ @Test
+ public void fail_when_not_logged_in() {
+ TestRequest request = ws.newRequest()
+ .setParam("almSetting", "azure")
+ .setParam("projectName", "project-name")
+ .setParam("repositoryName", "repo-name");
+
+ assertThatThrownBy(() -> request.execute())
+ .isInstanceOf(UnauthorizedException.class);
+ }
+
+ @Test
+ public void fail_when_missing_project_creator_permission() {
+ UserDto user = db.users().insertUser();
+ userSession.logIn(user).addPermission(SCAN);
+
+ TestRequest request = ws.newRequest()
+ .setParam("almSetting", "azure")
+ .setParam("projectName", "project-name")
+ .setParam("repositoryName", "repo-name");
+
+ assertThatThrownBy(() -> request.execute())
+ .isInstanceOf(ForbiddenException.class)
+ .hasMessage("Insufficient privileges");
+ }
+
+ @Test
+ public void check_pat_is_missing() {
+ UserDto user = db.users().insertUser();
+ userSession.logIn(user).addPermission(PROVISION_PROJECTS);
+ AlmSettingDto almSetting = db.almSettings().insertAzureAlmSetting();
+
+ TestRequest request = ws.newRequest()
+ .setParam("almSetting", almSetting.getKey())
+ .setParam("projectName", "project-name")
+ .setParam("repositoryName", "repo-name");
+
+ assertThatThrownBy(() -> request.execute())
+ .isInstanceOf(IllegalArgumentException.class)
+ .hasMessage("personal access token for '" + almSetting.getKey() + "' is missing");
+ }
+
+ @Test
+ public void fail_check_alm_setting_not_found() {
+ UserDto user = db.users().insertUser();
+ userSession.logIn(user).addPermission(PROVISION_PROJECTS);
+ AlmPatDto almPatDto = newAlmPatDto();
+ db.getDbClient().almPatDao().insert(db.getSession(), almPatDto);
+
+ TestRequest request = ws.newRequest()
+ .setParam("almSetting", "testKey");
+
+ assertThatThrownBy(() -> request.execute())
+ .isInstanceOf(NotFoundException.class)
+ .hasMessage("ALM Setting 'testKey' not found");
+ }
+
+ @Test
+ public void fail_project_already_exists() {
+ UserDto user = db.users().insertUser();
+ userSession.logIn(user).addPermission(PROVISION_PROJECTS);
+ AlmSettingDto almSetting = db.almSettings().insertAzureAlmSetting();
+ db.almPats().insert(dto -> {
+ dto.setAlmSettingUuid(almSetting.getUuid());
+ dto.setPersonalAccessToken(almSetting.getPersonalAccessToken());
+ dto.setUserUuid(user.getUuid());
+ });
+ GsonAzureRepo repo = getGsonAzureRepo();
+ String projectKey = repo.getProject().getName() + "_" + repo.getName();
+ db.components().insertPublicProject(p -> p.setDbKey(projectKey));
+
+ when(azureDevOpsHttpClient.getRepo(almSetting.getUrl(), almSetting.getPersonalAccessToken(), "project-name", "repo-name")).thenReturn(repo);
+ TestRequest request = ws.newRequest()
+ .setParam("almSetting", almSetting.getKey())
+ .setParam("projectName", "project-name")
+ .setParam("repositoryName", "repo-name");
+
+ assertThatThrownBy(() -> request.execute())
+ .isInstanceOf(BadRequestException.class)
+ .hasMessage("Could not create null, key already exists: " + projectKey);
+ }
+
+ @Test
+ public void sanitize_project_and_repo_names_with_invalid_characters() {
+ assertThat(importAzureProjectAction.generateProjectKey("project name", "repo name"))
+ .isEqualTo("project_name_repo_name");
+ }
+
+ @Test
+ public void sanitize_long_project_and_repo_names() {
+ String projectName = IntStream.range(0, 260).mapToObj(i -> "a").collect(joining());
+
+ assertThat(importAzureProjectAction.generateProjectKey(projectName, "repo name"))
+ .hasSize(250);
+ }
+
+ @Test
+ public void define() {
+ WebService.Action def = ws.getDef();
+
+ assertThat(def.since()).isEqualTo("8.6");
+ assertThat(def.isPost()).isTrue();
+ assertThat(def.params())
+ .extracting(WebService.Param::key, WebService.Param::isRequired)
+ .containsExactlyInAnyOrder(
+ tuple("almSetting", true),
+ tuple("projectName", true),
+ tuple("repositoryName", true));
+ }
+
+ private GsonAzureRepo getGsonAzureRepo() {
+ return new GsonAzureRepo("repo-id", "repo-name", "repo-url",
+ new GsonAzureProject("project-name", "project-description"));
+ }
+
+}
diff --git a/server/sonar-webserver-webapi/src/test/java/org/sonar/server/almintegration/ws/azure/ListAzureProjectsActionTest.java b/server/sonar-webserver-webapi/src/test/java/org/sonar/server/almintegration/ws/azure/ListAzureProjectsActionTest.java
new file mode 100644
index 00000000000..aca6f3e978c
--- /dev/null
+++ b/server/sonar-webserver-webapi/src/test/java/org/sonar/server/almintegration/ws/azure/ListAzureProjectsActionTest.java
@@ -0,0 +1,187 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2021 SonarSource SA
+ * mailto:info AT sonarsource DOT com
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 3 of the License, or (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with this program; if not, write to the Free Software Foundation,
+ * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ */
+package org.sonar.server.almintegration.ws.azure;
+
+import com.google.common.collect.ImmutableList;
+
+import java.util.List;
+
+import org.junit.Before;
+import org.junit.Rule;
+import org.junit.Test;
+import org.junit.rules.ExpectedException;
+import org.sonar.alm.client.azure.AzureDevOpsHttpClient;
+import org.sonar.alm.client.azure.GsonAzureProject;
+import org.sonar.alm.client.azure.GsonAzureProjectList;
+import org.sonar.api.server.ws.WebService;
+import org.sonar.db.DbTester;
+import org.sonar.db.alm.pat.AlmPatDto;
+import org.sonar.db.alm.setting.AlmSettingDto;
+import org.sonar.db.user.UserDto;
+import org.sonar.server.exceptions.ForbiddenException;
+import org.sonar.server.exceptions.NotFoundException;
+import org.sonar.server.exceptions.UnauthorizedException;
+import org.sonar.server.tester.UserSessionRule;
+import org.sonar.server.ws.TestRequest;
+import org.sonar.server.ws.WsActionTester;
+import org.sonarqube.ws.AlmIntegrations.AzureProject;
+import org.sonarqube.ws.AlmIntegrations.ListAzureProjectsWsResponse;
+
+import static org.assertj.core.api.Assertions.assertThat;
+import static org.assertj.core.api.Assertions.assertThatThrownBy;
+import static org.assertj.core.api.Assertions.tuple;
+import static org.mockito.ArgumentMatchers.anyString;
+import static org.mockito.Mockito.mock;
+import static org.mockito.Mockito.when;
+import static org.sonar.db.alm.integration.pat.AlmPatsTesting.newAlmPatDto;
+import static org.sonar.db.permission.GlobalPermission.PROVISION_PROJECTS;
+
+public class ListAzureProjectsActionTest {
+ @Rule
+ public ExpectedException expectedException = ExpectedException.none();
+ @Rule
+ public UserSessionRule userSession = UserSessionRule.standalone();
+ @Rule
+ public DbTester db = DbTester.create();
+
+ private final AzureDevOpsHttpClient azureDevOpsHttpClient = mock(AzureDevOpsHttpClient.class);
+ private final WsActionTester ws = new WsActionTester(new ListAzureProjectsAction(db.getDbClient(), userSession, azureDevOpsHttpClient));
+
+ @Before
+ public void before() {
+ mockClient(ImmutableList.of(new GsonAzureProject("name", "description"),
+ new GsonAzureProject("name", null)));
+ }
+
+ private void mockClient(List<GsonAzureProject> projects) {
+ GsonAzureProjectList projectList = new GsonAzureProjectList();
+ projectList.setValues(projects);
+ when(azureDevOpsHttpClient.getProjects(anyString(), anyString())).thenReturn(projectList);
+ }
+
+ @Test
+ public void list_projects() {
+ AlmSettingDto almSetting = insertAlmSetting();
+
+ ListAzureProjectsWsResponse response = ws.newRequest()
+ .setParam("almSetting", almSetting.getKey())
+ .executeProtobuf(ListAzureProjectsWsResponse.class);
+
+ assertThat(response.getProjectsCount()).isEqualTo(2);
+ assertThat(response.getProjectsList())
+ .extracting(AzureProject::getName, AzureProject::getDescription)
+ .containsExactly(tuple("name", "description"), tuple("name", ""));
+ }
+
+ @Test
+ public void list_projects_alphabetically_sorted() {
+ mockClient(ImmutableList.of(new GsonAzureProject("BBB project", "BBB project description"),
+ new GsonAzureProject("AAA project 1", "AAA project description"),
+ new GsonAzureProject("zzz project", "zzz project description"),
+ new GsonAzureProject("aaa project", "aaa project description")));
+ AlmSettingDto almSetting = insertAlmSetting();
+
+ ListAzureProjectsWsResponse response = ws.newRequest()
+ .setParam("almSetting", almSetting.getKey())
+ .executeProtobuf(ListAzureProjectsWsResponse.class);
+
+ assertThat(response.getProjectsCount()).isEqualTo(4);
+ assertThat(response.getProjectsList())
+ .extracting(AzureProject::getName, AzureProject::getDescription)
+ .containsExactly(tuple("aaa project", "aaa project description"), tuple("AAA project 1", "AAA project description"),
+ tuple("BBB project", "BBB project description"), tuple("zzz project", "zzz project description"));
+ }
+
+ @Test
+ public void check_pat_is_missing() {
+ insertUser();
+ AlmSettingDto almSetting = db.almSettings().insertAzureAlmSetting();
+
+ TestRequest request = ws.newRequest()
+ .setParam("almSetting", almSetting.getKey());
+
+ assertThatThrownBy(request::execute)
+ .isInstanceOf(IllegalArgumentException.class)
+ .hasMessage("No personal access token found");
+ }
+
+ @Test
+ public void fail_check_alm_setting_not_found() {
+ insertUser();
+ AlmPatDto almPatDto = newAlmPatDto();
+ db.getDbClient().almPatDao().insert(db.getSession(), almPatDto);
+
+ TestRequest request = ws.newRequest()
+ .setParam("almSetting", "testKey");
+
+ assertThatThrownBy(request::execute)
+ .isInstanceOf(NotFoundException.class)
+ .hasMessage("ALM Setting 'testKey' not found");
+ }
+
+ @Test
+ public void fail_when_not_logged_in() {
+ TestRequest request = ws.newRequest()
+ .setParam("almSetting", "anyvalue");
+
+ assertThatThrownBy(request::execute)
+ .isInstanceOf(UnauthorizedException.class);
+ }
+
+ @Test
+ public void fail_when_no_creation_project_permission() {
+ UserDto user = db.users().insertUser();
+ userSession.logIn(user);
+
+ TestRequest request = ws.newRequest()
+ .setParam("almSetting", "anyvalue");
+
+ assertThatThrownBy(request::execute)
+ .isInstanceOf(ForbiddenException.class)
+ .hasMessage("Insufficient privileges");
+ }
+
+ @Test
+ public void definition() {
+ WebService.Action def = ws.getDef();
+
+ assertThat(def.since()).isEqualTo("8.6");
+ assertThat(def.isPost()).isFalse();
+ assertThat(def.params())
+ .extracting(WebService.Param::key, WebService.Param::isRequired)
+ .containsExactlyInAnyOrder(tuple("almSetting", true));
+ }
+
+ private UserDto insertUser() {
+ UserDto user = db.users().insertUser();
+ userSession.logIn(user).addPermission(PROVISION_PROJECTS);
+ return user;
+ }
+
+ private AlmSettingDto insertAlmSetting() {
+ UserDto user = insertUser();
+ AlmSettingDto almSetting = db.almSettings().insertAzureAlmSetting();
+ db.almPats().insert(dto -> {
+ dto.setAlmSettingUuid(almSetting.getUuid());
+ dto.setUserUuid(user.getUuid());
+ });
+ return almSetting;
+ }
+}
diff --git a/server/sonar-webserver-webapi/src/test/java/org/sonar/server/almintegration/ws/azure/SearchAzureReposActionTest.java b/server/sonar-webserver-webapi/src/test/java/org/sonar/server/almintegration/ws/azure/SearchAzureReposActionTest.java
new file mode 100644
index 00000000000..ed736553046
--- /dev/null
+++ b/server/sonar-webserver-webapi/src/test/java/org/sonar/server/almintegration/ws/azure/SearchAzureReposActionTest.java
@@ -0,0 +1,367 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2021 SonarSource SA
+ * mailto:info AT sonarsource DOT com
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 3 of the License, or (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with this program; if not, write to the Free Software Foundation,
+ * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ */
+package org.sonar.server.almintegration.ws.azure;
+
+import com.google.common.collect.ImmutableList;
+import org.jetbrains.annotations.NotNull;
+import org.junit.Before;
+import org.junit.Rule;
+import org.junit.Test;
+import org.sonar.alm.client.azure.AzureDevOpsHttpClient;
+import org.sonar.alm.client.azure.GsonAzureProject;
+import org.sonar.alm.client.azure.GsonAzureRepo;
+import org.sonar.alm.client.azure.GsonAzureRepoList;
+import org.sonar.api.server.ws.WebService;
+import org.sonar.db.DbTester;
+import org.sonar.db.alm.pat.AlmPatDto;
+import org.sonar.db.alm.setting.AlmSettingDto;
+import org.sonar.db.project.ProjectDto;
+import org.sonar.db.user.UserDto;
+import org.sonar.server.exceptions.ForbiddenException;
+import org.sonar.server.exceptions.NotFoundException;
+import org.sonar.server.exceptions.UnauthorizedException;
+import org.sonar.server.tester.UserSessionRule;
+import org.sonar.server.ws.TestRequest;
+import org.sonar.server.ws.WsActionTester;
+
+import static java.util.Collections.emptyList;
+import static org.assertj.core.api.Assertions.assertThat;
+import static org.assertj.core.api.Assertions.assertThatThrownBy;
+import static org.assertj.core.api.Assertions.tuple;
+import static org.mockito.ArgumentMatchers.any;
+import static org.mockito.Mockito.mock;
+import static org.mockito.Mockito.when;
+import static org.sonar.db.alm.integration.pat.AlmPatsTesting.newAlmPatDto;
+import static org.sonar.db.permission.GlobalPermission.PROVISION_PROJECTS;
+import static org.sonarqube.ws.AlmIntegrations.AzureRepo;
+import static org.sonarqube.ws.AlmIntegrations.SearchAzureReposWsResponse;
+
+public class SearchAzureReposActionTest {
+
+ @Rule
+ public UserSessionRule userSession = UserSessionRule.standalone();
+ @Rule
+ public DbTester db = DbTester.create();
+
+ private AzureDevOpsHttpClient azureDevOpsHttpClient = mock(AzureDevOpsHttpClient.class);
+ private WsActionTester ws = new WsActionTester(new SearchAzureReposAction(db.getDbClient(), userSession, azureDevOpsHttpClient));
+
+ @Before
+ public void before() {
+ mockClient(new GsonAzureRepoList(ImmutableList.of(getGsonAzureRepo("project-1", "repoName-1"),
+ getGsonAzureRepo("project-2", "repoName-2"))));
+ }
+
+ @Test
+ public void define() {
+ WebService.Action def = ws.getDef();
+
+ assertThat(def.since()).isEqualTo("8.6");
+ assertThat(def.isPost()).isFalse();
+ assertThat(def.params())
+ .extracting(WebService.Param::key, WebService.Param::isRequired)
+ .containsExactlyInAnyOrder(
+ tuple("almSetting", true),
+ tuple("projectName", false),
+ tuple("searchQuery", false));
+ }
+
+ @Test
+ public void search_repos() {
+ AlmSettingDto almSetting = insertAlmSetting();
+
+ SearchAzureReposWsResponse response = ws.newRequest()
+ .setParam("almSetting", almSetting.getKey())
+ .executeProtobuf(SearchAzureReposWsResponse.class);
+
+ assertThat(response.getRepositoriesList())
+ .extracting(AzureRepo::getName, AzureRepo::getProjectName)
+ .containsExactlyInAnyOrder(
+ tuple("repoName-1", "project-1"), tuple("repoName-2", "project-2"));
+ }
+
+ @Test
+ public void search_repos_alphabetically_sorted() {
+ mockClient(new GsonAzureRepoList(ImmutableList.of(getGsonAzureRepo("project-1", "Z-repo"),
+ getGsonAzureRepo("project-1", "A-repo-1"), getGsonAzureRepo("project-1", "a-repo"),
+ getGsonAzureRepo("project-1", "b-repo"))));
+
+ AlmSettingDto almSetting = insertAlmSetting();
+
+ SearchAzureReposWsResponse response = ws.newRequest()
+ .setParam("almSetting", almSetting.getKey())
+ .executeProtobuf(SearchAzureReposWsResponse.class);
+
+ assertThat(response.getRepositoriesList())
+ .extracting(AzureRepo::getName, AzureRepo::getProjectName)
+ .containsExactly(
+ tuple("a-repo", "project-1"), tuple("A-repo-1", "project-1"),
+ tuple("b-repo", "project-1"), tuple("Z-repo", "project-1"));
+ }
+
+ @Test
+ public void search_repos_with_project_already_set_up() {
+ AlmSettingDto almSetting = insertAlmSetting();
+
+ ProjectDto projectDto2 = insertProject(almSetting, "repoName-2", "project-2");
+
+ SearchAzureReposWsResponse response = ws.newRequest()
+ .setParam("almSetting", almSetting.getKey())
+ .executeProtobuf(SearchAzureReposWsResponse.class);
+
+ assertThat(response.getRepositoriesCount()).isEqualTo(2);
+
+ assertThat(response.getRepositoriesList())
+ .extracting(AzureRepo::getName, AzureRepo::getProjectName,
+ AzureRepo::getSqProjectKey, AzureRepo::getSqProjectName)
+ .containsExactlyInAnyOrder(
+ tuple("repoName-1", "project-1", "", ""),
+ tuple("repoName-2", "project-2", projectDto2.getKey(), projectDto2.getName()));
+ }
+
+ @Test
+ public void search_repos_with_project_already_set_u_and_collision_is_handled() {
+ AlmSettingDto almSetting = insertAlmSetting();
+
+ ProjectDto projectDto2 = insertProject(almSetting, "repoName-2", "project-2");
+ insertProject(almSetting, "repoName-2", "project-2");
+
+ SearchAzureReposWsResponse response = ws.newRequest()
+ .setParam("almSetting", almSetting.getKey())
+ .executeProtobuf(SearchAzureReposWsResponse.class);
+
+ assertThat(response.getRepositoriesCount()).isEqualTo(2);
+
+ assertThat(response.getRepositoriesList())
+ .extracting(AzureRepo::getName, AzureRepo::getProjectName,
+ AzureRepo::getSqProjectKey, AzureRepo::getSqProjectName)
+ .containsExactlyInAnyOrder(
+ tuple("repoName-1", "project-1", "", ""),
+ tuple("repoName-2", "project-2", projectDto2.getKey(), projectDto2.getName()));
+ }
+
+ @Test
+ public void search_repos_with_projects_already_set_up_and_no_collision() {
+ mockClient(new GsonAzureRepoList(ImmutableList.of(getGsonAzureRepo("project-1", "repoName-1"),
+ getGsonAzureRepo("project", "1-repoName-1"))));
+ AlmSettingDto almSetting = insertAlmSetting();
+
+ ProjectDto projectDto1 = insertProject(almSetting, "repoName-1", "project-1");
+ ProjectDto projectDto2 = insertProject(almSetting, "1-repoName-1", "project");
+
+ SearchAzureReposWsResponse response = ws.newRequest()
+ .setParam("almSetting", almSetting.getKey())
+ .executeProtobuf(SearchAzureReposWsResponse.class);
+
+ assertThat(response.getRepositoriesCount()).isEqualTo(2);
+
+ assertThat(response.getRepositoriesList())
+ .extracting(AzureRepo::getName, AzureRepo::getProjectName,
+ AzureRepo::getSqProjectKey, AzureRepo::getSqProjectName)
+ .containsExactlyInAnyOrder(
+ tuple("repoName-1", "project-1", projectDto1.getKey(), projectDto1.getName()),
+ tuple("1-repoName-1", "project", projectDto2.getKey(), projectDto2.getName()));
+ }
+
+ @Test
+ public void search_repos_with_same_name_and_different_project() {
+ mockClient(new GsonAzureRepoList(ImmutableList.of(getGsonAzureRepo("project-1", "repoName-1"),
+ getGsonAzureRepo("project-2", "repoName-1"))));
+ AlmSettingDto almSetting = insertAlmSetting();
+
+ ProjectDto projectDto1 = insertProject(almSetting, "repoName-1", "project-1");
+ ProjectDto projectDto2 = insertProject(almSetting, "repoName-1", "project-2");
+
+ SearchAzureReposWsResponse response = ws.newRequest()
+ .setParam("almSetting", almSetting.getKey())
+ .executeProtobuf(SearchAzureReposWsResponse.class);
+
+ assertThat(response.getRepositoriesCount()).isEqualTo(2);
+
+ assertThat(response.getRepositoriesList())
+ .extracting(AzureRepo::getName, AzureRepo::getProjectName,
+ AzureRepo::getSqProjectKey, AzureRepo::getSqProjectName)
+ .containsExactlyInAnyOrder(
+ tuple("repoName-1", "project-1", projectDto1.getKey(), projectDto1.getName()),
+ tuple("repoName-1", "project-2", projectDto2.getKey(), projectDto2.getName()));
+ }
+
+ @Test
+ public void search_repos_with_project_name() {
+ AlmSettingDto almSetting = insertAlmSetting();
+
+ SearchAzureReposWsResponse response = ws.newRequest()
+ .setParam("almSetting", almSetting.getKey())
+ .setParam("projectName", "project-1")
+ .executeProtobuf(SearchAzureReposWsResponse.class);
+
+ assertThat(response.getRepositoriesList())
+ .extracting(AzureRepo::getName, AzureRepo::getProjectName)
+ .containsExactlyInAnyOrder(
+ tuple("repoName-1", "project-1"), tuple("repoName-2", "project-2"));
+ }
+
+ @Test
+ public void search_repos_with_project_name_and_empty_criteria() {
+ AlmSettingDto almSetting = insertAlmSetting();
+
+ SearchAzureReposWsResponse response = ws.newRequest()
+ .setParam("almSetting", almSetting.getKey())
+ .setParam("projectName", "project-1")
+ .setParam("searchQuery", "")
+ .executeProtobuf(SearchAzureReposWsResponse.class);
+
+ assertThat(response.getRepositoriesList())
+ .extracting(AzureRepo::getName, AzureRepo::getProjectName)
+ .containsExactlyInAnyOrder(
+ tuple("repoName-1", "project-1"), tuple("repoName-2", "project-2"));
+ }
+
+ @Test
+ public void search_and_filter_repos_with_repo_name() {
+ AlmSettingDto almSetting = insertAlmSetting();
+
+ SearchAzureReposWsResponse response = ws.newRequest()
+ .setParam("almSetting", almSetting.getKey())
+ .setParam("searchQuery", "repoName-2")
+ .executeProtobuf(SearchAzureReposWsResponse.class);
+
+ assertThat(response.getRepositoriesList())
+ .extracting(AzureRepo::getName, AzureRepo::getProjectName)
+ .containsExactlyInAnyOrder(tuple("repoName-2", "project-2"));
+ }
+
+ @Test
+ public void search_and_filter_repos_with_matching_repo_and_project_name() {
+ mockClient(new GsonAzureRepoList(ImmutableList.of(getGsonAzureRepo("big-project", "repo-1"),
+ getGsonAzureRepo("big-project", "repo-2"),
+ getGsonAzureRepo("big-project", "big-repo"),
+ getGsonAzureRepo("project", "big-repo"),
+ getGsonAzureRepo("project", "small-repo"))));
+ AlmSettingDto almSetting = insertAlmSetting();
+
+ SearchAzureReposWsResponse response = ws.newRequest()
+ .setParam("almSetting", almSetting.getKey())
+ .setParam("searchQuery", "big")
+ .executeProtobuf(SearchAzureReposWsResponse.class);
+
+ assertThat(response.getRepositoriesList())
+ .extracting(AzureRepo::getName, AzureRepo::getProjectName)
+ .containsExactlyInAnyOrder(tuple("repo-1", "big-project"), tuple("repo-2", "big-project"),
+ tuple("big-repo", "big-project"), tuple("big-repo", "project"));
+ }
+
+ @Test
+ public void return_empty_list_when_there_are_no_azure_repos() {
+ when(azureDevOpsHttpClient.getRepos(any(), any(), any())).thenReturn(new GsonAzureRepoList(emptyList()));
+
+ AlmSettingDto almSetting = insertAlmSetting();
+
+ SearchAzureReposWsResponse response = ws.newRequest()
+ .setParam("almSetting", almSetting.getKey())
+ .executeProtobuf(SearchAzureReposWsResponse.class);
+
+ assertThat(response.getRepositoriesList()).isEmpty();
+ }
+
+ @Test
+ public void check_pat_is_missing() {
+ insertUser();
+ AlmSettingDto almSetting = db.almSettings().insertAzureAlmSetting();
+
+ TestRequest request = ws.newRequest()
+ .setParam("almSetting", almSetting.getKey());
+
+ assertThatThrownBy(request::execute)
+ .isInstanceOf(IllegalArgumentException.class)
+ .hasMessage("No personal access token found");
+ }
+
+ @Test
+ public void fail_check_pat_alm_setting_not_found() {
+ insertUser();
+ AlmPatDto almPatDto = newAlmPatDto();
+ db.getDbClient().almPatDao().insert(db.getSession(), almPatDto);
+
+ TestRequest request = ws.newRequest()
+ .setParam("almSetting", "testKey");
+
+ assertThatThrownBy(request::execute)
+ .isInstanceOf(NotFoundException.class)
+ .hasMessage("ALM Setting 'testKey' not found");
+ }
+
+ @Test
+ public void fail_when_not_logged_in() {
+ TestRequest request = ws.newRequest()
+ .setParam("almSetting", "anyvalue");
+
+ assertThatThrownBy(request::execute)
+ .isInstanceOf(UnauthorizedException.class);
+ }
+
+ @Test
+ public void fail_when_no_creation_project_permission() {
+ UserDto user = db.users().insertUser();
+ userSession.logIn(user);
+
+ TestRequest request = ws.newRequest()
+ .setParam("almSetting", "anyvalue");
+
+ assertThatThrownBy(request::execute)
+ .isInstanceOf(ForbiddenException.class)
+ .hasMessage("Insufficient privileges");
+ }
+
+ private ProjectDto insertProject(AlmSettingDto almSetting, String repoName, String projectName) {
+ ProjectDto projectDto1 = db.components().insertPrivateProjectDto();
+ db.almSettings().insertAzureProjectAlmSetting(almSetting, projectDto1, projectAlmSettingDto -> projectAlmSettingDto.setAlmRepo(repoName),
+ projectAlmSettingDto -> projectAlmSettingDto.setAlmSlug(projectName));
+ return projectDto1;
+ }
+
+ private void mockClient(GsonAzureRepoList repoList) {
+ when(azureDevOpsHttpClient.getRepos(any(), any(), any())).thenReturn(repoList);
+ }
+
+ private AlmSettingDto insertAlmSetting() {
+ UserDto user = insertUser();
+ AlmSettingDto almSetting = db.almSettings().insertAzureAlmSetting();
+ db.almPats().insert(dto -> {
+ dto.setAlmSettingUuid(almSetting.getUuid());
+ dto.setUserUuid(user.getUuid());
+ dto.setPersonalAccessToken(almSetting.getPersonalAccessToken());
+ });
+ return almSetting;
+ }
+
+ @NotNull
+ private UserDto insertUser() {
+ UserDto user = db.users().insertUser();
+ userSession.logIn(user).addPermission(PROVISION_PROJECTS);
+ return user;
+ }
+
+ private GsonAzureRepo getGsonAzureRepo(String projectName, String repoName) {
+ GsonAzureProject project = new GsonAzureProject(projectName, "the best project ever");
+ GsonAzureRepo gsonAzureRepo = new GsonAzureRepo("repo-id", repoName, "url", project);
+ return gsonAzureRepo;
+ }
+}
diff --git a/server/sonar-webserver/src/main/java/org/sonar/server/platform/platformlevel/PlatformLevel4.java b/server/sonar-webserver/src/main/java/org/sonar/server/platform/platformlevel/PlatformLevel4.java
index d6761042cc3..b7deaf622b6 100644
--- a/server/sonar-webserver/src/main/java/org/sonar/server/platform/platformlevel/PlatformLevel4.java
+++ b/server/sonar-webserver/src/main/java/org/sonar/server/platform/platformlevel/PlatformLevel4.java
@@ -21,6 +21,7 @@ package org.sonar.server.platform.platformlevel;
import java.util.List;
import org.sonar.alm.client.TimeoutConfigurationImpl;
+import org.sonar.alm.client.azure.AzureDevOpsHttpClient;
import org.sonar.alm.client.bitbucketserver.BitbucketServerRestClient;
import org.sonar.alm.client.gitlab.GitlabHttpClient;
import org.sonar.api.profiles.AnnotationProfileParser;
@@ -494,6 +495,7 @@ public class PlatformLevel4 extends PlatformLevel {
ImportHelper.class,
BitbucketServerRestClient.class,
GitlabHttpClient.class,
+ AzureDevOpsHttpClient.class,
AlmIntegrationsWSModule.class,
// Branch