aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/license/ws/LicensesWs.java43
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/license/ws/LicensesWsModule.java32
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/license/ws/ListAction.java189
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/license/ws/package-info.java23
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/platform/platformlevel/PlatformLevel4.java4
-rw-r--r--server/sonar-server/src/main/resources/org/sonar/server/license/ws/list-example.json28
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/license/ws/LicensesWsModuleTest.java35
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/license/ws/ListActionTest.java317
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/setting/ws/ValuesActionTest.java58
-rw-r--r--sonar-ws/src/main/java/org/sonarqube/ws/client/license/LicensesService.java41
-rw-r--r--sonar-ws/src/main/java/org/sonarqube/ws/client/license/LicensesWsParameters.java32
-rw-r--r--sonar-ws/src/main/protobuf/ws-licenses.proto51
-rw-r--r--sonar-ws/src/test/java/org/sonarqube/ws/client/license/LicensesServiceTest.java49
13 files changed, 873 insertions, 29 deletions
diff --git a/server/sonar-server/src/main/java/org/sonar/server/license/ws/LicensesWs.java b/server/sonar-server/src/main/java/org/sonar/server/license/ws/LicensesWs.java
new file mode 100644
index 00000000000..06dc877ddcb
--- /dev/null
+++ b/server/sonar-server/src/main/java/org/sonar/server/license/ws/LicensesWs.java
@@ -0,0 +1,43 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2016 SonarSource SA
+ * mailto:contact 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.license.ws;
+
+import org.sonar.api.server.ws.WebService;
+
+import static org.sonarqube.ws.client.license.LicensesWsParameters.CONTROLLER_SETTINGS;
+
+public class LicensesWs implements WebService {
+
+ private final ListAction listAction;
+
+ public LicensesWs(ListAction listAction) {
+ this.listAction = listAction;
+ }
+
+ @Override
+ public void define(Context context) {
+ NewController controller = context.createController(CONTROLLER_SETTINGS)
+ .setDescription("Manage licenses")
+ .setSince("6.1");
+ listAction.define(controller);
+ controller.done();
+ }
+}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/license/ws/LicensesWsModule.java b/server/sonar-server/src/main/java/org/sonar/server/license/ws/LicensesWsModule.java
new file mode 100644
index 00000000000..656dcc175a4
--- /dev/null
+++ b/server/sonar-server/src/main/java/org/sonar/server/license/ws/LicensesWsModule.java
@@ -0,0 +1,32 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2016 SonarSource SA
+ * mailto:contact 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.license.ws;
+
+import org.sonar.core.platform.Module;
+
+public class LicensesWsModule extends Module {
+ @Override
+ protected void configureModule() {
+ add(
+ LicensesWs.class,
+ ListAction.class);
+ }
+}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/license/ws/ListAction.java b/server/sonar-server/src/main/java/org/sonar/server/license/ws/ListAction.java
new file mode 100644
index 00000000000..6ff20b00d23
--- /dev/null
+++ b/server/sonar-server/src/main/java/org/sonar/server/license/ws/ListAction.java
@@ -0,0 +1,189 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2016 SonarSource SA
+ * mailto:contact 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.license.ws;
+
+import java.util.List;
+import java.util.Map;
+import java.util.Objects;
+import java.util.Optional;
+import java.util.Set;
+import java.util.function.Function;
+import javax.annotation.Nullable;
+import org.sonar.api.config.License;
+import org.sonar.api.config.PropertyDefinition;
+import org.sonar.api.config.PropertyDefinitions;
+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.property.PropertyDto;
+import org.sonar.server.user.UserSession;
+import org.sonar.server.ws.WsAction;
+import org.sonarqube.ws.Licenses;
+import org.sonarqube.ws.Licenses.ListWsResponse;
+
+import static com.google.common.collect.Sets.newHashSet;
+import static org.sonar.api.CoreProperties.PERMANENT_SERVER_ID;
+import static org.sonar.api.PropertyType.LICENSE;
+import static org.sonar.core.permission.GlobalPermissions.SYSTEM_ADMIN;
+import static org.sonar.core.util.stream.Collectors.toSet;
+import static org.sonar.core.util.stream.Collectors.uniqueIndex;
+import static org.sonar.server.ws.WsUtils.writeProtobuf;
+import static org.sonarqube.ws.client.license.LicensesWsParameters.ACTION_LIST;
+
+public class ListAction implements WsAction {
+
+ private static final String ALL_SERVERS_VALUE = "*";
+
+ private final UserSession userSession;
+ private final PropertyDefinitions definitions;
+ private final DbClient dbClient;
+
+ public ListAction(UserSession userSession, PropertyDefinitions definitions, DbClient dbClient) {
+ this.userSession = userSession;
+ this.definitions = definitions;
+ this.dbClient = dbClient;
+ }
+
+ @Override
+ public void define(WebService.NewController context) {
+ context.createAction(ACTION_LIST)
+ .setDescription("List licenses settings.<br>" +
+ "Requires 'Administer System' permission")
+ .setResponseExample(getClass().getResource("list-example.json"))
+ .setSince("6.1")
+ .setInternal(true)
+ .setHandler(this);
+ }
+
+ @Override
+ public void handle(Request request, Response response) throws Exception {
+ userSession.checkPermission(SYSTEM_ADMIN);
+
+ DbSession dbSession = dbClient.openSession(true);
+ try {
+ writeProtobuf(doHandle(dbSession), request, response);
+ } finally {
+ dbClient.closeSession(dbSession);
+ }
+ }
+
+ private ListWsResponse doHandle(DbSession dbSession) {
+ Set<String> licenseSettingsKeys = definitions.getAll().stream()
+ .filter(definition -> LICENSE.equals(definition.type()))
+ .map(PropertyDefinition::key)
+ .collect(toSet());
+ Set<String> settingsKeys = newHashSet(licenseSettingsKeys);
+ settingsKeys.add(PERMANENT_SERVER_ID);
+ List<PropertyDto> properties = dbClient.propertiesDao().selectGlobalPropertiesByKeys(dbSession, settingsKeys);
+ return new ListResponseBuilder(licenseSettingsKeys, properties).build();
+ }
+
+ private static class ListResponseBuilder {
+ private final Optional<String> serverId;
+ private final Map<String, PropertyDto> licenseSettingsByKey;
+ private final Set<String> licenseSettingsKeys;
+
+ ListResponseBuilder(Set<String> licenseSettingsKeys, List<PropertyDto> properties) {
+ this.serverId = getServerId(properties);
+ this.licenseSettingsKeys = licenseSettingsKeys;
+ this.licenseSettingsByKey = properties.stream().collect(uniqueIndex(PropertyDto::getKey, Function.identity()));
+ }
+
+ ListWsResponse build() {
+ ListWsResponse.Builder wsResponse = ListWsResponse.newBuilder();
+ licenseSettingsKeys.forEach(key -> wsResponse.addLicenses(buildLicense(key, licenseSettingsByKey.get(key))));
+ return wsResponse.build();
+ }
+
+ private Licenses.License buildLicense(String key, @Nullable PropertyDto setting) {
+ Licenses.License.Builder licenseBuilder = Licenses.License.newBuilder().setKey(key);
+ if (setting != null) {
+ License license = License.readBase64(setting.getValue());
+ licenseBuilder.setValue(setting.getValue());
+ setProduct(licenseBuilder, license, setting);
+ setOrganization(licenseBuilder, license);
+ setExpiration(licenseBuilder, license);
+ setServerId(licenseBuilder, license);
+ setType(licenseBuilder, license);
+ setAdditionalProperties(licenseBuilder, license);
+ }
+ return licenseBuilder.build();
+ }
+
+ private static void setProduct(Licenses.License.Builder licenseBuilder, License license, PropertyDto setting) {
+ String product = license.getProduct();
+ if (product != null) {
+ licenseBuilder.setProduct(product);
+ }
+ if (product == null || !setting.getKey().contains(product)) {
+ licenseBuilder.setInvalidProduct(true);
+ }
+ }
+
+ private static void setOrganization(Licenses.License.Builder licenseBuilder, License license) {
+ String licenseOrganization = license.getOrganization();
+ if (licenseOrganization != null) {
+ licenseBuilder.setOrganization(licenseOrganization);
+ }
+ }
+
+ private void setServerId(Licenses.License.Builder licenseBuilder, License license) {
+ String licenseServerId = license.getServer();
+ if (licenseServerId != null) {
+ licenseBuilder.setServerId(licenseServerId);
+ }
+ if (!Objects.equals(ALL_SERVERS_VALUE, licenseServerId) && serverId.isPresent() && !serverId.get().equals(licenseServerId)) {
+ licenseBuilder.setInvalidServerId(true);
+ }
+ }
+
+ private static void setExpiration(Licenses.License.Builder licenseBuilder, License license) {
+ String expiration = license.getExpirationDateAsString();
+ if (expiration != null) {
+ licenseBuilder.setExpiration(expiration);
+ }
+ if (license.isExpired()) {
+ licenseBuilder.setInvalidExpiration(true);
+ }
+ }
+
+ private static void setType(Licenses.License.Builder licenseBuilder, License license) {
+ String type = license.getType();
+ if (type != null) {
+ licenseBuilder.setType(type);
+ }
+ }
+
+ private static void setAdditionalProperties(Licenses.License.Builder licenseBuilder, License license) {
+ Map<String, String> additionalProperties = license.additionalProperties();
+ if (!additionalProperties.isEmpty()) {
+ licenseBuilder.getAdditionalPropertiesBuilder().putAllAdditionalProperties(additionalProperties).build();
+ }
+ }
+
+ private static Optional<String> getServerId(List<PropertyDto> propertyDtos) {
+ Optional<PropertyDto> propertyDto = propertyDtos.stream().filter(setting -> setting.getKey().equals(PERMANENT_SERVER_ID)).findFirst();
+ return propertyDto.isPresent() ? Optional.of(propertyDto.get().getValue()) : Optional.empty();
+ }
+ }
+}
diff --git a/server/sonar-server/src/main/java/org/sonar/server/license/ws/package-info.java b/server/sonar-server/src/main/java/org/sonar/server/license/ws/package-info.java
new file mode 100644
index 00000000000..c31656ea2fe
--- /dev/null
+++ b/server/sonar-server/src/main/java/org/sonar/server/license/ws/package-info.java
@@ -0,0 +1,23 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2016 SonarSource SA
+ * mailto:contact 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.license.ws;
+
+import javax.annotation.ParametersAreNonnullByDefault;
diff --git a/server/sonar-server/src/main/java/org/sonar/server/platform/platformlevel/PlatformLevel4.java b/server/sonar-server/src/main/java/org/sonar/server/platform/platformlevel/PlatformLevel4.java
index fa0fcc45238..b2a65af5969 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/platform/platformlevel/PlatformLevel4.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/platform/platformlevel/PlatformLevel4.java
@@ -119,6 +119,7 @@ import org.sonar.server.issue.workflow.FunctionExecutor;
import org.sonar.server.issue.workflow.IssueWorkflow;
import org.sonar.server.issue.ws.IssueWsModule;
import org.sonar.server.language.ws.LanguageWs;
+import org.sonar.server.license.ws.LicensesWsModule;
import org.sonar.server.measure.MeasureFilterEngine;
import org.sonar.server.measure.MeasureFilterExecutor;
import org.sonar.server.measure.MeasureFilterFactory;
@@ -584,6 +585,9 @@ public class PlatformLevel4 extends PlatformLevel {
PropertiesWs.class,
SettingsWsModule.class,
+ // Licences
+ LicensesWsModule.class,
+
TypeValidationModule.class,
// Project Links
diff --git a/server/sonar-server/src/main/resources/org/sonar/server/license/ws/list-example.json b/server/sonar-server/src/main/resources/org/sonar/server/license/ws/list-example.json
new file mode 100644
index 00000000000..3ec21fa9f71
--- /dev/null
+++ b/server/sonar-server/src/main/resources/org/sonar/server/license/ws/list-example.json
@@ -0,0 +1,28 @@
+{
+ "licenses": [
+ {
+ "key": "sonar.devcockpit.license.secured",
+ "value": "T3JnYW5pc2F0aW9uOiBVbmtub3duIApTZXJ2ZXI6IDU0MzIxIApQcm9kdWN0OiBvdGhlciAKRXhwaXJhdGlvbjogMjAxMC0wMS0wMSAKVHlwZTogRVZBTFVBVElPTiAK",
+ "product": "other",
+ "organization": "Unknown",
+ "expiration": "2010-01-01",
+ "serverId": "54321",
+ "type": "EVALUATION",
+ "invalidProduct": true,
+ "invalidExpiration": true,
+ "invalidServerId": true
+ },
+ {
+ "key": "sonar.governance.license.secured",
+ "value": "T3JnYW5pc2F0aW9uOiBTb25hclNvdXJjZSAKU2VydmVyOiAxMjM0NSAKUHJvZHVjdDogZ292ZXJuYW5jZSAKRXhwaXJhdGlvbjogMjA5OS0wMS0wMSAKVHlwZTogUFJPRFVDVElPTiAKb3RoZXI6IHZhbHVlIAo\u003d",
+ "product": "governance",
+ "organization": "SonarSource",
+ "expiration": "2099-01-01",
+ "serverId": "12345",
+ "type": "PRODUCTION",
+ "additionalProperties": {
+ "other": "value"
+ }
+ }
+ ]
+}
diff --git a/server/sonar-server/src/test/java/org/sonar/server/license/ws/LicensesWsModuleTest.java b/server/sonar-server/src/test/java/org/sonar/server/license/ws/LicensesWsModuleTest.java
new file mode 100644
index 00000000000..dc1c9ff24c9
--- /dev/null
+++ b/server/sonar-server/src/test/java/org/sonar/server/license/ws/LicensesWsModuleTest.java
@@ -0,0 +1,35 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2016 SonarSource SA
+ * mailto:contact 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.license.ws;
+
+import org.junit.Test;
+import org.sonar.core.platform.ComponentContainer;
+
+import static org.assertj.core.api.Assertions.assertThat;
+
+public class LicensesWsModuleTest {
+ @Test
+ public void verify_count_of_added_components() {
+ ComponentContainer container = new ComponentContainer();
+ new LicensesWsModule().configure(container);
+ assertThat(container.size()).isEqualTo(2 + 2);
+ }
+}
diff --git a/server/sonar-server/src/test/java/org/sonar/server/license/ws/ListActionTest.java b/server/sonar-server/src/test/java/org/sonar/server/license/ws/ListActionTest.java
new file mode 100644
index 00000000000..15edb494b36
--- /dev/null
+++ b/server/sonar-server/src/test/java/org/sonar/server/license/ws/ListActionTest.java
@@ -0,0 +1,317 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2016 SonarSource SA
+ * mailto:contact 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.license.ws;
+
+import com.google.common.collect.ImmutableMap;
+import java.io.IOException;
+import java.nio.charset.StandardCharsets;
+import java.util.Collections;
+import java.util.Map;
+import javax.annotation.Nullable;
+import org.apache.commons.codec.binary.Base64;
+import org.junit.Rule;
+import org.junit.Test;
+import org.junit.rules.ExpectedException;
+import org.sonar.api.config.PropertyDefinition;
+import org.sonar.api.config.PropertyDefinitions;
+import org.sonar.api.server.ws.WebService;
+import org.sonar.api.utils.System2;
+import org.sonar.db.DbClient;
+import org.sonar.db.DbTester;
+import org.sonar.db.property.PropertyDbTester;
+import org.sonar.server.exceptions.ForbiddenException;
+import org.sonar.server.tester.UserSessionRule;
+import org.sonar.server.ws.TestRequest;
+import org.sonar.server.ws.WsActionTester;
+import org.sonar.test.JsonAssert;
+import org.sonarqube.ws.Licenses;
+import org.sonarqube.ws.Licenses.ListWsResponse;
+import org.sonarqube.ws.MediaTypes;
+
+import static org.assertj.core.api.Assertions.assertThat;
+import static org.assertj.core.api.Java6Assertions.entry;
+import static org.sonar.api.CoreProperties.PERMANENT_SERVER_ID;
+import static org.sonar.api.PropertyType.LICENSE;
+import static org.sonar.core.permission.GlobalPermissions.DASHBOARD_SHARING;
+import static org.sonar.core.permission.GlobalPermissions.SYSTEM_ADMIN;
+import static org.sonar.db.property.PropertyTesting.newGlobalPropertyDto;
+import static org.sonarqube.ws.MediaTypes.JSON;
+
+public class ListActionTest {
+
+ private static final String LICENSE_KEY_SAMPLE = "sonar.governance.license.secured";
+ private static final String ORGANISATION_SAMPLE = "SonarSource";
+ private static final String SERVER_ID_SAMPLE = "12345";
+ private static final String PRODUCT_SAMPLE = "governance";
+ private static final String TYPE_SAMPLE = "PRODUCTION";
+ private static final String EXPIRATION_SAMPLE = "2099-01-01";
+ @Rule
+ public ExpectedException expectedException = ExpectedException.none();
+
+ @Rule
+ public UserSessionRule userSession = UserSessionRule.standalone();
+
+ @Rule
+ public DbTester db = DbTester.create(System2.INSTANCE);
+
+ DbClient dbClient = db.getDbClient();
+ PropertyDbTester propertyDb = new PropertyDbTester(db);
+ PropertyDefinitions definitions = new PropertyDefinitions();
+
+ WsActionTester ws = new WsActionTester(new ListAction(userSession, definitions, dbClient));
+
+ @Test
+ public void return_licenses() throws Exception {
+ setUserAsSystemAdmin();
+ addServerIdSettings("12345");
+ String data = createBase64License("SonarSource", "governance", "12345", "2099-01-01", "PRODUCTION", ImmutableMap.of("other", "value"));
+ addLicenseSetting("sonar.governance.license.secured", data);
+
+ ListWsResponse result = executeRequest();
+
+ assertThat(result.getLicensesList()).hasSize(1);
+ Licenses.License license = result.getLicenses(0);
+ assertThat(license.getKey()).isEqualTo("sonar.governance.license.secured");
+ assertThat(license.getValue()).isEqualTo(data);
+ assertThat(license.getProduct()).isEqualTo("governance");
+ assertThat(license.getOrganization()).isEqualTo("SonarSource");
+ assertThat(license.getExpiration()).isEqualTo("2099-01-01");
+ assertThat(license.getType()).isEqualTo("PRODUCTION");
+ assertThat(license.getServerId()).isEqualTo("12345");
+ assertThat(license.getAdditionalProperties().getAdditionalProperties()).containsOnly(entry("other", "value"));
+
+ assertThat(license.hasInvalidProduct()).isFalse();
+ assertThat(license.hasInvalidExpiration()).isFalse();
+ assertThat(license.hasInvalidServerId()).isFalse();
+ }
+
+ @Test
+ public void return_licenses_even_if_no_value_set_in_database() throws Exception {
+ setUserAsSystemAdmin();
+ addServerIdSettings("12345");
+ definitions.addComponent(PropertyDefinition.builder("sonar.governance.license.secured").type(LICENSE).build());
+
+ ListWsResponse result = executeRequest();
+
+ assertThat(result.getLicensesList()).hasSize(1);
+ Licenses.License license = result.getLicenses(0);
+ assertThat(license.getKey()).isEqualTo("sonar.governance.license.secured");
+ assertThat(license.hasValue()).isFalse();
+ assertThat(license.hasProduct()).isFalse();
+ assertThat(license.hasOrganization()).isFalse();
+ assertThat(license.hasExpiration()).isFalse();
+ assertThat(license.hasType()).isFalse();
+ assertThat(license.hasServerId()).isFalse();
+ assertThat(license.hasAdditionalProperties()).isFalse();
+
+ assertThat(license.hasInvalidProduct()).isFalse();
+ assertThat(license.hasInvalidExpiration()).isFalse();
+ assertThat(license.hasInvalidServerId()).isFalse();
+ }
+
+ @Test
+ public void return_license_with_minimal_info() throws Exception {
+ setUserAsSystemAdmin();
+ addServerIdSettings(SERVER_ID_SAMPLE);
+ addLicenseSetting(LICENSE_KEY_SAMPLE, toBase64(""));
+
+ ListWsResponse result = executeRequest();
+
+ assertThat(result.getLicensesList()).hasSize(1);
+ Licenses.License license = result.getLicenses(0);
+ assertThat(license.getKey()).isEqualTo(LICENSE_KEY_SAMPLE);
+ assertThat(license.getValue()).isEmpty();
+ assertThat(license.hasProduct()).isFalse();
+ assertThat(license.hasOrganization()).isFalse();
+ assertThat(license.hasExpiration()).isFalse();
+ assertThat(license.hasType()).isFalse();
+ assertThat(license.hasServerId()).isFalse();
+ assertThat(license.hasAdditionalProperties()).isFalse();
+
+ assertThat(license.hasInvalidProduct()).isTrue();
+ assertThat(license.hasInvalidExpiration()).isFalse();
+ assertThat(license.hasInvalidServerId()).isTrue();
+ }
+
+ @Test
+ public void return_license_with_bad_product() throws Exception {
+ setUserAsSystemAdmin();
+ addServerIdSettings(SERVER_ID_SAMPLE);
+ addLicenseSetting(LICENSE_KEY_SAMPLE, createBase64License(ORGANISATION_SAMPLE, "Other", SERVER_ID_SAMPLE, EXPIRATION_SAMPLE, TYPE_SAMPLE, Collections.emptyMap()));
+
+ ListWsResponse result = executeRequest();
+
+ assertThat(result.getLicensesList()).hasSize(1);
+ Licenses.License license = result.getLicenses(0);
+ assertThat(license.getProduct()).isEqualTo("Other");
+ assertThat(license.getInvalidProduct()).isTrue();
+ assertThat(license.hasInvalidExpiration()).isFalse();
+ assertThat(license.hasInvalidServerId()).isFalse();
+ }
+
+ @Test
+ public void return_license_with_bad_server_id() throws Exception {
+ setUserAsSystemAdmin();
+ addServerIdSettings(SERVER_ID_SAMPLE);
+ addLicenseSetting(LICENSE_KEY_SAMPLE, createBase64License(ORGANISATION_SAMPLE, PRODUCT_SAMPLE, "Other", EXPIRATION_SAMPLE, TYPE_SAMPLE, Collections.emptyMap()));
+
+ ListWsResponse result = executeRequest();
+
+ assertThat(result.getLicensesList()).hasSize(1);
+ Licenses.License license = result.getLicenses(0);
+ assertThat(license.getServerId()).isEqualTo("Other");
+ assertThat(license.getInvalidServerId()).isTrue();
+ assertThat(license.hasInvalidProduct()).isFalse();
+ assertThat(license.hasInvalidExpiration()).isFalse();
+ }
+
+ @Test
+ public void does_not_return_invalid_server_id_when_all_servers_accepted_and_no_server_id_setting() throws Exception {
+ setUserAsSystemAdmin();
+ addLicenseSetting(LICENSE_KEY_SAMPLE, createBase64License(ORGANISATION_SAMPLE, PRODUCT_SAMPLE, "*", EXPIRATION_SAMPLE, TYPE_SAMPLE, Collections.emptyMap()));
+
+ ListWsResponse result = executeRequest();
+
+ assertThat(result.getLicensesList()).hasSize(1);
+ Licenses.License license = result.getLicenses(0);
+ assertThat(license.getServerId()).isEqualTo("*");
+ assertThat(license.hasInvalidServerId()).isFalse();
+ }
+
+ @Test
+ public void return_license_when_all_servers_are_accepted() throws Exception {
+ setUserAsSystemAdmin();
+ addServerIdSettings(SERVER_ID_SAMPLE);
+ addLicenseSetting(LICENSE_KEY_SAMPLE, createBase64License(ORGANISATION_SAMPLE, PRODUCT_SAMPLE, "*", EXPIRATION_SAMPLE, TYPE_SAMPLE, Collections.emptyMap()));
+
+ ListWsResponse result = executeRequest();
+
+ assertThat(result.getLicensesList()).hasSize(1);
+ Licenses.License license = result.getLicenses(0);
+ assertThat(license.getServerId()).isEqualTo("*");
+ assertThat(license.hasInvalidServerId()).isFalse();
+ }
+
+ @Test
+ public void return_license_when_expired() throws Exception {
+ setUserAsSystemAdmin();
+ addServerIdSettings(SERVER_ID_SAMPLE);
+ addLicenseSetting(LICENSE_KEY_SAMPLE,
+ createBase64License(ORGANISATION_SAMPLE, PRODUCT_SAMPLE, SERVER_ID_SAMPLE, "2010-01-01", TYPE_SAMPLE, Collections.emptyMap()));
+
+ ListWsResponse result = executeRequest();
+
+ assertThat(result.getLicensesList()).hasSize(1);
+ Licenses.License license = result.getLicenses(0);
+ assertThat(license.getExpiration()).isEqualTo("2010-01-01");
+ assertThat(license.getInvalidExpiration()).isTrue();
+ assertThat(license.hasInvalidProduct()).isFalse();
+ assertThat(license.hasInvalidServerId()).isFalse();
+ }
+
+ @Test
+ public void none_license_type_settings_are_not_returned() throws Exception {
+ setUserAsSystemAdmin();
+ definitions.addComponent(PropertyDefinition.builder("foo").build());
+ propertyDb.insertProperties(newGlobalPropertyDto().setKey("foo").setValue("value"));
+
+ ListWsResponse result = executeRequest();
+
+ assertThat(result.getLicensesList()).isEmpty();
+ }
+
+ @Test
+ public void fail_when_not_system_admin() throws Exception {
+ userSession.login("not-admin").setGlobalPermissions(DASHBOARD_SHARING);
+ definitions.addComponent(PropertyDefinition.builder("foo").build());
+
+ expectedException.expect(ForbiddenException.class);
+
+ executeRequest();
+ }
+
+ @Test
+ public void test_example_json_response() {
+ setUserAsSystemAdmin();
+ addServerIdSettings("12345");
+ addLicenseSetting("sonar.governance.license.secured", createBase64License("SonarSource", "governance", "12345", "2099-01-01", "PRODUCTION", ImmutableMap.of("other", "value")));
+ addLicenseSetting("sonar.devcockpit.license.secured", createBase64License("Unknown", "other", "54321", "2010-01-01", "EVALUATION", Collections.emptyMap()));
+
+ String result = ws.newRequest()
+ .setMediaType(JSON)
+ .execute()
+ .getInput();
+
+ JsonAssert.assertJson(ws.getDef().responseExampleAsString()).isSimilarTo(result);
+ }
+
+ @Test
+ public void test_ws_definition() {
+ WebService.Action action = ws.getDef();
+ assertThat(action).isNotNull();
+ assertThat(action.isInternal()).isTrue();
+ assertThat(action.isPost()).isFalse();
+ assertThat(action.responseExampleAsString()).isNotEmpty();
+ assertThat(action.params()).isEmpty();
+ }
+
+ private ListWsResponse executeRequest() {
+ TestRequest request = ws.newRequest()
+ .setMediaType(MediaTypes.PROTOBUF);
+ try {
+ return ListWsResponse.parseFrom(request.execute().getInputStream());
+ } catch (IOException e) {
+ throw new IllegalStateException(e);
+ }
+ }
+
+ private void setUserAsSystemAdmin() {
+ userSession.login("admin").setGlobalPermissions(SYSTEM_ADMIN);
+ }
+
+ private void addLicenseSetting(String key, String value) {
+ definitions.addComponent(PropertyDefinition.builder(key).type(LICENSE).build());
+ propertyDb.insertProperties(newGlobalPropertyDto().setKey(key).setValue(value));
+ }
+
+ private void addServerIdSettings(String serverId) {
+ propertyDb.insertProperties(newGlobalPropertyDto().setKey(PERMANENT_SERVER_ID).setValue(serverId));
+ }
+
+ private static String toBase64(String data) {
+ return Base64.encodeBase64String((data.getBytes(StandardCharsets.UTF_8)));
+ }
+
+ private static String createBase64License(@Nullable String organisation, @Nullable String product, @Nullable String serverId, @Nullable String expirationDate,
+ @Nullable String type, Map<String, String> additionalProperties) {
+ StringBuilder data = new StringBuilder();
+ data.append("Organisation: ").append(organisation).append(" \n");
+ data.append("Server: ").append(serverId).append(" \n");
+ data.append("Product: ").append(product).append(" \n");
+ data.append("Expiration: ").append(expirationDate).append(" \n");
+ data.append("Type: ").append(type).append(" \n");
+ for (Map.Entry<String, String> entry : additionalProperties.entrySet()) {
+ data.append(entry.getKey()).append(": ").append(entry.getValue()).append(" \n");
+ }
+ return toBase64(data.toString());
+ }
+
+}
diff --git a/server/sonar-server/src/test/java/org/sonar/server/setting/ws/ValuesActionTest.java b/server/sonar-server/src/test/java/org/sonar/server/setting/ws/ValuesActionTest.java
index 59ec14cf1d0..f32885cca81 100644
--- a/server/sonar-server/src/test/java/org/sonar/server/setting/ws/ValuesActionTest.java
+++ b/server/sonar-server/src/test/java/org/sonar/server/setting/ws/ValuesActionTest.java
@@ -78,12 +78,12 @@ public class ValuesActionTest {
DbClient dbClient = db.getDbClient();
PropertyDbTester propertyDb = new PropertyDbTester(db);
ComponentDbTester componentDb = new ComponentDbTester(db);
- PropertyDefinitions propertyDefinitions = new PropertyDefinitions();
- SettingsFinder settingsFinder = new SettingsFinder(dbClient, propertyDefinitions);
+ PropertyDefinitions definitions = new PropertyDefinitions();
+ SettingsFinder settingsFinder = new SettingsFinder(dbClient, definitions);
ComponentDto project;
- WsActionTester ws = new WsActionTester(new ValuesAction(dbClient, new ComponentFinder(dbClient), userSession, propertyDefinitions, settingsFinder));
+ WsActionTester ws = new WsActionTester(new ValuesAction(dbClient, new ComponentFinder(dbClient), userSession, definitions, settingsFinder));
@Before
public void setUp() throws Exception {
@@ -93,7 +93,7 @@ public class ValuesActionTest {
@Test
public void return_simple_value() throws Exception {
setUserAsSystemAdmin();
- propertyDefinitions.addComponent(PropertyDefinition
+ definitions.addComponent(PropertyDefinition
.builder("foo")
.build());
propertyDb.insertProperties(newGlobalPropertyDto().setKey("foo").setValue("one"));
@@ -114,13 +114,13 @@ public class ValuesActionTest {
setUserAsSystemAdmin();
// Property never defined, default value is returned
- propertyDefinitions.addComponent(PropertyDefinition.builder("default")
+ definitions.addComponent(PropertyDefinition.builder("default")
.multiValues(true)
.defaultValue("one,two")
.build());
// Property defined at global level
- propertyDefinitions.addComponent(PropertyDefinition.builder("global")
+ definitions.addComponent(PropertyDefinition.builder("global")
.multiValues(true)
.build());
propertyDb.insertProperties(newGlobalPropertyDto().setKey("global").setValue("three,four"));
@@ -144,7 +144,7 @@ public class ValuesActionTest {
@Test
public void return_multi_value_with_coma() throws Exception {
setUserAsSystemAdmin();
- propertyDefinitions.addComponent(PropertyDefinition.builder("global").multiValues(true).build());
+ definitions.addComponent(PropertyDefinition.builder("global").multiValues(true).build());
propertyDb.insertProperties(newGlobalPropertyDto().setKey("global").setValue("three,four%2Cfive"));
ValuesWsResponse result = executeRequestForGlobalProperties("global");
@@ -158,7 +158,7 @@ public class ValuesActionTest {
@Test
public void return_property_set() throws Exception {
setUserAsSystemAdmin();
- propertyDefinitions.addComponent(PropertyDefinition
+ definitions.addComponent(PropertyDefinition
.builder("foo")
.type(PropertyType.PROPERTY_SET)
.fields(asList(
@@ -180,7 +180,7 @@ public class ValuesActionTest {
@Test
public void return_property_set_for_component() throws Exception {
setUserAsSystemAdmin();
- propertyDefinitions.addComponent(PropertyDefinition
+ definitions.addComponent(PropertyDefinition
.builder("foo")
.type(PropertyType.PROPERTY_SET)
.fields(asList(
@@ -202,7 +202,7 @@ public class ValuesActionTest {
@Test
public void return_default_values() throws Exception {
setUserAsSystemAdmin();
- propertyDefinitions.addComponent(PropertyDefinition
+ definitions.addComponent(PropertyDefinition
.builder("foo")
.defaultValue("default")
.build());
@@ -216,7 +216,7 @@ public class ValuesActionTest {
@Test
public void return_global_values() throws Exception {
setUserAsSystemAdmin();
- propertyDefinitions.addComponent(PropertyDefinition.builder("property").defaultValue("default").build());
+ definitions.addComponent(PropertyDefinition.builder("property").defaultValue("default").build());
propertyDb.insertProperties(
// The property is overriding default value
newGlobalPropertyDto().setKey("property").setValue("one"));
@@ -230,7 +230,7 @@ public class ValuesActionTest {
@Test
public void return_project_values() throws Exception {
setUserAsSystemAdmin();
- propertyDefinitions.addComponent(PropertyDefinition.builder("property").defaultValue("default").build());
+ definitions.addComponent(PropertyDefinition.builder("property").defaultValue("default").build());
propertyDb.insertProperties(
newGlobalPropertyDto().setKey("property").setValue("one"),
// The property is overriding global value
@@ -245,7 +245,7 @@ public class ValuesActionTest {
@Test
public void return_is_inherited_to_true_when_property_is_defined_only_at_global_level() throws Exception {
setUserAsSystemAdmin();
- propertyDefinitions.addComponent(PropertyDefinition.builder("property").defaultValue("default").build());
+ definitions.addComponent(PropertyDefinition.builder("property").defaultValue("default").build());
// The property is not defined on project
propertyDb.insertProperties(newGlobalPropertyDto().setKey("property").setValue("one"));
@@ -271,7 +271,7 @@ public class ValuesActionTest {
@Test
public void return_empty_when_property_def_exists_but_no_value() throws Exception {
setUserAsSystemAdmin();
- propertyDefinitions.addComponent(PropertyDefinition
+ definitions.addComponent(PropertyDefinition
.builder("foo")
.build());
@@ -283,7 +283,7 @@ public class ValuesActionTest {
@Test
public void return_nothing_when_unknown_keys() throws Exception {
setUserAsSystemAdmin();
- propertyDefinitions.addComponent(PropertyDefinition
+ definitions.addComponent(PropertyDefinition
.builder("foo")
.defaultValue("default")
.build());
@@ -298,7 +298,7 @@ public class ValuesActionTest {
public void return_module_values() throws Exception {
setUserAsSystemAdmin();
ComponentDto module = componentDb.insertComponent(newModuleDto(project));
- propertyDefinitions.addComponent(PropertyDefinition.builder("property").defaultValue("default").build());
+ definitions.addComponent(PropertyDefinition.builder("property").defaultValue("default").build());
propertyDb.insertProperties(
newGlobalPropertyDto().setKey("property").setValue("one"),
// The property is overriding global value
@@ -314,7 +314,7 @@ public class ValuesActionTest {
public void return_inherited_values_on_module() throws Exception {
setUserAsSystemAdmin();
ComponentDto module = componentDb.insertComponent(newModuleDto(project));
- propertyDefinitions.addComponents(asList(
+ definitions.addComponents(asList(
PropertyDefinition.builder("defaultProperty").defaultValue("default").build(),
PropertyDefinition.builder("globalProperty").build(),
PropertyDefinition.builder("projectProperty").build(),
@@ -336,7 +336,7 @@ public class ValuesActionTest {
@Test
public void return_inherited_values_on_global_setting() throws Exception {
setUserAsSystemAdmin();
- propertyDefinitions.addComponents(asList(
+ definitions.addComponents(asList(
PropertyDefinition.builder("defaultProperty").defaultValue("default").build(),
PropertyDefinition.builder("globalProperty").build()));
propertyDb.insertProperties(
@@ -354,7 +354,7 @@ public class ValuesActionTest {
setUserAsSystemAdmin();
ComponentDto module = componentDb.insertComponent(newModuleDto(project));
ComponentDto subModule = componentDb.insertComponent(newModuleDto(module));
- propertyDefinitions.addComponents(asList(
+ definitions.addComponents(asList(
PropertyDefinition.builder("foo").defaultValue("default").build()));
propertyDb.insertProperties(
newGlobalPropertyDto().setKey("foo").setValue("global"),
@@ -372,7 +372,7 @@ public class ValuesActionTest {
setUserAsSystemAdmin();
ComponentDto module = componentDb.insertComponent(newModuleDto(project));
ComponentDto subModule = componentDb.insertComponent(newModuleDto(module));
- propertyDefinitions.addComponents(asList(
+ definitions.addComponents(asList(
PropertyDefinition.builder("foo").defaultValue("default1,default2").multiValues(true).build()));
propertyDb.insertProperties(
newGlobalPropertyDto().setKey("foo").setValue("global1,global2"),
@@ -390,7 +390,7 @@ public class ValuesActionTest {
setUserAsSystemAdmin();
ComponentDto module = componentDb.insertComponent(newModuleDto(project));
ComponentDto subModule = componentDb.insertComponent(newModuleDto(module));
- propertyDefinitions.addComponent(PropertyDefinition
+ definitions.addComponent(PropertyDefinition
.builder("foo")
.type(PropertyType.PROPERTY_SET)
.fields(asList(
@@ -412,7 +412,7 @@ public class ValuesActionTest {
setUserAsSystemAdmin();
ComponentDto module = componentDb.insertComponent(newModuleDto(project));
ComponentDto subModule = componentDb.insertComponent(newModuleDto(module));
- propertyDefinitions.addComponents(asList(
+ definitions.addComponents(asList(
PropertyDefinition.builder("simple").build(),
PropertyDefinition.builder("multi").multiValues(true).build(),
PropertyDefinition.builder("set")
@@ -447,7 +447,7 @@ public class ValuesActionTest {
@Test
public void return_value_of_deprecated_key() throws Exception {
setUserAsSystemAdmin();
- propertyDefinitions.addComponent(PropertyDefinition
+ definitions.addComponent(PropertyDefinition
.builder("foo")
.deprecatedKey("deprecated")
.build());
@@ -464,16 +464,16 @@ public class ValuesActionTest {
@Test
public void test_example_json_response() {
setUserAsSystemAdmin();
- propertyDefinitions.addComponent(PropertyDefinition
+ definitions.addComponent(PropertyDefinition
.builder("sonar.test.jira")
.defaultValue("abc")
.build());
- propertyDefinitions.addComponent(PropertyDefinition
+ definitions.addComponent(PropertyDefinition
.builder("sonar.autogenerated")
.multiValues(true)
.build());
propertyDb.insertProperties(newGlobalPropertyDto().setKey("sonar.autogenerated").setValue("val1,val2,val3"));
- propertyDefinitions.addComponent(PropertyDefinition
+ definitions.addComponent(PropertyDefinition
.builder("sonar.demo")
.type(PropertyType.PROPERTY_SET)
.fields(PropertyFieldDefinition.build("text").name("Text").build(),
@@ -502,7 +502,7 @@ public class ValuesActionTest {
@Test
public void fail_when_not_system_admin() throws Exception {
userSession.login("not-admin").setGlobalPermissions(DASHBOARD_SHARING);
- propertyDefinitions.addComponent(PropertyDefinition.builder("foo").build());
+ definitions.addComponent(PropertyDefinition.builder("foo").build());
expectedException.expect(ForbiddenException.class);
@@ -512,7 +512,7 @@ public class ValuesActionTest {
@Test
public void fail_when_not_project_admin() throws Exception {
userSession.login("project-admin").addProjectUuidPermissions(USER, project.uuid());
- propertyDefinitions.addComponent(PropertyDefinition.builder("foo").build());
+ definitions.addComponent(PropertyDefinition.builder("foo").build());
expectedException.expect(ForbiddenException.class);
@@ -522,7 +522,7 @@ public class ValuesActionTest {
@Test
public void fail_when_deprecated_key_and_new_key_are_used() throws Exception {
setUserAsSystemAdmin();
- propertyDefinitions.addComponent(PropertyDefinition
+ definitions.addComponent(PropertyDefinition
.builder("foo")
.deprecatedKey("deprecated")
.build());
diff --git a/sonar-ws/src/main/java/org/sonarqube/ws/client/license/LicensesService.java b/sonar-ws/src/main/java/org/sonarqube/ws/client/license/LicensesService.java
new file mode 100644
index 00000000000..93fb452ac0d
--- /dev/null
+++ b/sonar-ws/src/main/java/org/sonarqube/ws/client/license/LicensesService.java
@@ -0,0 +1,41 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2016 SonarSource SA
+ * mailto:contact 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.sonarqube.ws.client.license;
+
+import org.sonarqube.ws.Licenses.ListWsResponse;
+import org.sonarqube.ws.client.BaseService;
+import org.sonarqube.ws.client.GetRequest;
+import org.sonarqube.ws.client.WsConnector;
+
+import static org.sonarqube.ws.client.license.LicensesWsParameters.ACTION_LIST;
+import static org.sonarqube.ws.client.license.LicensesWsParameters.CONTROLLER_SETTINGS;
+
+public class LicensesService extends BaseService {
+
+ public LicensesService(WsConnector wsConnector) {
+ super(wsConnector, CONTROLLER_SETTINGS);
+ }
+
+ public ListWsResponse list() {
+ GetRequest getRequest = new GetRequest(path(ACTION_LIST));
+ return call(getRequest, ListWsResponse.parser());
+ }
+
+}
diff --git a/sonar-ws/src/main/java/org/sonarqube/ws/client/license/LicensesWsParameters.java b/sonar-ws/src/main/java/org/sonarqube/ws/client/license/LicensesWsParameters.java
new file mode 100644
index 00000000000..a035f684519
--- /dev/null
+++ b/sonar-ws/src/main/java/org/sonarqube/ws/client/license/LicensesWsParameters.java
@@ -0,0 +1,32 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2016 SonarSource SA
+ * mailto:contact 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.sonarqube.ws.client.license;
+
+public class LicensesWsParameters {
+ public static final String CONTROLLER_SETTINGS = "api/licenses";
+
+ public static final String ACTION_LIST = "list";
+
+ private LicensesWsParameters() {
+ // Only static stuff
+ }
+
+}
diff --git a/sonar-ws/src/main/protobuf/ws-licenses.proto b/sonar-ws/src/main/protobuf/ws-licenses.proto
new file mode 100644
index 00000000000..77f1002ffba
--- /dev/null
+++ b/sonar-ws/src/main/protobuf/ws-licenses.proto
@@ -0,0 +1,51 @@
+// SonarQube, open source software quality management tool.
+// Copyright (C) 2008-2015 SonarSource
+// mailto:contact AT sonarsource DOT com
+//
+// SonarQube 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.
+//
+// SonarQube 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.
+
+syntax = "proto2";
+
+package sonarqube.ws.licenses;
+
+option java_package = "org.sonarqube.ws";
+option java_outer_classname = "Licenses";
+option optimize_for = SPEED;
+
+// Response of GET api/licenses/list
+message ListWsResponse {
+ repeated License licenses = 1;
+}
+
+message License {
+ optional string key = 1;
+ optional string value = 2;
+ optional string product = 3;
+ optional string organization = 4;
+ optional string expiration = 5;
+ optional string serverId = 6;
+ optional string type = 7;
+ optional AdditionalProperties additionalProperties = 8;
+ optional bool invalidProduct = 9;
+ optional bool invalidExpiration = 10;
+ optional bool invalidServerId = 11;
+}
+
+message AdditionalProperties {
+ map<string, string> additionalProperties = 1;
+}
+
+
+
diff --git a/sonar-ws/src/test/java/org/sonarqube/ws/client/license/LicensesServiceTest.java b/sonar-ws/src/test/java/org/sonarqube/ws/client/license/LicensesServiceTest.java
new file mode 100644
index 00000000000..8b305df0514
--- /dev/null
+++ b/sonar-ws/src/test/java/org/sonarqube/ws/client/license/LicensesServiceTest.java
@@ -0,0 +1,49 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2016 SonarSource SA
+ * mailto:contact 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.sonarqube.ws.client.license;
+
+import org.junit.Rule;
+import org.junit.Test;
+import org.sonarqube.ws.Licenses.ListWsResponse;
+import org.sonarqube.ws.client.GetRequest;
+import org.sonarqube.ws.client.ServiceTester;
+import org.sonarqube.ws.client.WsConnector;
+
+import static org.assertj.core.api.Assertions.assertThat;
+import static org.mockito.Mockito.mock;
+
+public class LicensesServiceTest {
+
+ @Rule
+ public ServiceTester<LicensesService> serviceTester = new ServiceTester<>(new LicensesService(mock(WsConnector.class)));
+
+ private LicensesService underTest = serviceTester.getInstanceUnderTest();
+
+ @Test
+ public void list_definitions() {
+ underTest.list();
+ GetRequest getRequest = serviceTester.getGetRequest();
+
+ assertThat(serviceTester.getGetParser()).isSameAs(ListWsResponse.parser());
+ serviceTester.assertThat(getRequest).andNoOtherParam();
+ }
+
+}