123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197 |
- /*
- * SonarQube
- * Copyright (C) 2009-2023 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.rule.ws;
-
- import com.google.common.io.Resources;
- import java.util.ArrayList;
- import java.util.Arrays;
- import java.util.Collections;
- import java.util.List;
- import java.util.Optional;
- import org.sonar.api.rule.RuleKey;
- import org.sonar.api.rule.RuleStatus;
- import org.sonar.api.rule.Severity;
- import org.sonar.api.rules.RuleType;
- import org.sonar.api.server.ws.Change;
- 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.KeyValueFormat;
- import org.sonar.db.DbClient;
- import org.sonar.db.DbSession;
- import org.sonar.db.rule.RuleDto;
- import org.sonar.db.rule.RuleParamDto;
- import org.sonar.server.rule.NewCustomRule;
- import org.sonar.server.rule.ReactivationException;
- import org.sonar.server.rule.RuleCreator;
- import org.sonarqube.ws.Rules;
-
- import static com.google.common.base.Strings.isNullOrEmpty;
- import static java.net.HttpURLConnection.HTTP_CONFLICT;
- import static java.util.Collections.singletonList;
- import static java.util.Optional.ofNullable;
- import static org.sonar.server.ws.WsUtils.writeProtobuf;
-
- public class CreateAction implements RulesWsAction {
-
- public static final String PARAM_CUSTOM_KEY = "customKey";
- public static final String PARAM_NAME = "name";
- public static final String PARAM_DESCRIPTION = "markdownDescription";
- public static final String PARAM_SEVERITY = "severity";
- public static final String PARAM_STATUS = "status";
- public static final String PARAM_TEMPLATE_KEY = "templateKey";
- public static final String PARAM_TYPE = "type";
- public static final String PARAMS = "params";
-
- public static final String PARAM_PREVENT_REACTIVATION = "preventReactivation";
- static final int KEY_MAXIMUM_LENGTH = 200;
- static final int NAME_MAXIMUM_LENGTH = 200;
-
- private final DbClient dbClient;
- private final RuleCreator ruleCreator;
- private final RuleMapper ruleMapper;
- private final RuleWsSupport ruleWsSupport;
-
- public CreateAction(DbClient dbClient, RuleCreator ruleCreator, RuleMapper ruleMapper, RuleWsSupport ruleWsSupport) {
- this.dbClient = dbClient;
- this.ruleCreator = ruleCreator;
- this.ruleMapper = ruleMapper;
- this.ruleWsSupport = ruleWsSupport;
- }
-
- @Override
- public void define(WebService.NewController controller) {
- WebService.NewAction action = controller
- .createAction("create")
- .setPost(true)
- .setDescription("Create a custom rule.<br>" +
- "Requires the 'Administer Quality Profiles' permission")
- .setResponseExample(Resources.getResource(getClass(), "create-example.json"))
- .setSince("4.4")
- .setChangelog(
- new Change("5.5", "Creating manual rule is not more possible"),
- new Change("10.0","Drop deprecated keys: 'custom_key', 'template_key', 'markdown_description', 'prevent_reactivation'"),
- new Change("10.2", "Add 'impacts', 'cleanCodeAttribute', 'cleanCodeAttributeCategory' fields to the response")
- )
- .setHandler(this);
-
- action
- .createParam(PARAM_CUSTOM_KEY)
- .setRequired(true)
- .setMaximumLength(KEY_MAXIMUM_LENGTH)
- .setDescription("Key of the custom rule")
- .setExampleValue("Todo_should_not_be_used");
-
- action
- .createParam(PARAM_TEMPLATE_KEY)
- .setRequired(true)
- .setDescription("Key of the template rule in order to create a custom rule (mandatory for custom rule)")
- .setExampleValue("java:XPath");
-
- action
- .createParam(PARAM_NAME)
- .setRequired(true)
- .setMaximumLength(NAME_MAXIMUM_LENGTH)
- .setDescription("Rule name")
- .setExampleValue("My custom rule");
-
- action
- .createParam(PARAM_DESCRIPTION)
- .setRequired(true)
- .setDescription("Rule description in <a href='/formatting/help'>markdown format</a>")
- .setExampleValue("Description of my custom rule");
-
- action
- .createParam(PARAM_SEVERITY)
- .setPossibleValues(Severity.ALL)
- .setDefaultValue(Severity.MAJOR)
- .setDescription("Rule severity");
-
- action
- .createParam(PARAM_STATUS)
- .setPossibleValues(
- Arrays.stream(RuleStatus.values())
- .filter(status -> !RuleStatus.REMOVED.equals(status))
- .toList())
- .setDefaultValue(RuleStatus.READY)
- .setDescription("Rule status");
-
- action.createParam(PARAMS)
- .setDescription("Parameters as semi-colon list of <key>=<value>, for example 'params=key1=v1;key2=v2' (Only for custom rule)");
-
- action
- .createParam(PARAM_PREVENT_REACTIVATION)
- .setBooleanPossibleValues()
- .setDefaultValue(false)
- .setDescription("If set to true and if the rule has been deactivated (status 'REMOVED'), a status 409 will be returned");
-
- action.createParam(PARAM_TYPE)
- .setPossibleValues(RuleType.names())
- .setDescription("Rule type")
- .setSince("6.7");
- }
-
- @Override
- public void handle(Request request, Response response) throws Exception {
- ruleWsSupport.checkQProfileAdminPermission();
- String customKey = request.mandatoryParam(PARAM_CUSTOM_KEY);
- try (DbSession dbSession = dbClient.openSession(false)) {
- try {
- NewCustomRule newRule = NewCustomRule.createForCustomRule(customKey, RuleKey.parse(request.mandatoryParam(PARAM_TEMPLATE_KEY)))
- .setName(request.mandatoryParam(PARAM_NAME))
- .setMarkdownDescription(request.mandatoryParam(PARAM_DESCRIPTION))
- .setSeverity(request.mandatoryParam(PARAM_SEVERITY))
- .setStatus(RuleStatus.valueOf(request.mandatoryParam(PARAM_STATUS)))
- .setPreventReactivation(request.mandatoryParamAsBoolean(PARAM_PREVENT_REACTIVATION));
- String params = request.param(PARAMS);
- if (!isNullOrEmpty(params)) {
- newRule.setParameters(KeyValueFormat.parse(params));
- }
- ofNullable(request.param(PARAM_TYPE)).ifPresent(t -> newRule.setType(RuleType.valueOf(t)));
- writeResponse(dbSession, request, response, ruleCreator.create(dbSession, newRule));
- } catch (ReactivationException e) {
- response.stream().setStatus(HTTP_CONFLICT);
- writeResponse(dbSession, request, response, e.ruleKey());
- }
- }
- }
-
- private void writeResponse(DbSession dbSession, Request request, Response response, RuleKey ruleKey) {
- writeProtobuf(createResponse(dbSession, ruleKey), request, response);
- }
-
- private Rules.CreateResponse createResponse(DbSession dbSession, RuleKey ruleKey) {
- RuleDto rule = dbClient.ruleDao().selectByKey(dbSession, ruleKey)
- .orElseThrow(() -> new IllegalStateException(String.format("Cannot load rule, that has just been created '%s'", ruleKey)));
- List<RuleDto> templateRules = new ArrayList<>();
- if (rule.isCustomRule()) {
- Optional<RuleDto> templateRule = dbClient.ruleDao().selectByUuid(rule.getTemplateUuid(), dbSession);
- templateRule.ifPresent(templateRules::add);
- }
- List<RuleParamDto> ruleParameters = dbClient.ruleDao().selectRuleParamsByRuleUuids(dbSession, singletonList(rule.getUuid()));
- SearchAction.SearchResult searchResult = new SearchAction.SearchResult()
- .setRuleParameters(ruleParameters)
- .setTemplateRules(templateRules)
- .setTotal(1L);
- return Rules.CreateResponse.newBuilder()
- .setRule(ruleMapper.toWsRule(rule, searchResult, Collections.emptySet()))
- .build();
- }
- }
|