aboutsummaryrefslogtreecommitdiffstats
path: root/server/sonar-webserver-core/src/main/java
diff options
context:
space:
mode:
authorDejan Milisavljevic <dejan.milisavljevic@sonarsource.com>2024-10-10 16:21:04 +0200
committersonartech <sonartech@sonarsource.com>2024-10-16 20:03:01 +0000
commit5038a353ca5917a647e5613859fbf62e756f4a21 (patch)
tree06f18963373820c45f2f08e355271a6b03379a07 /server/sonar-webserver-core/src/main/java
parent32cec354bfc868d591e6356ee85728c5ecba51d1 (diff)
downloadsonarqube-5038a353ca5917a647e5613859fbf62e756f4a21.tar.gz
sonarqube-5038a353ca5917a647e5613859fbf62e756f4a21.zip
SONAR-23250 Initial population of active_rules with impacts
Diffstat (limited to 'server/sonar-webserver-core/src/main/java')
-rw-r--r--server/sonar-webserver-core/src/main/java/org/sonar/server/rule/registration/ActiveRulesImpactInitializer.java97
-rw-r--r--server/sonar-webserver-core/src/main/java/org/sonar/server/rule/registration/RulesRegistrant.java8
2 files changed, 104 insertions, 1 deletions
diff --git a/server/sonar-webserver-core/src/main/java/org/sonar/server/rule/registration/ActiveRulesImpactInitializer.java b/server/sonar-webserver-core/src/main/java/org/sonar/server/rule/registration/ActiveRulesImpactInitializer.java
new file mode 100644
index 00000000000..bcce3553966
--- /dev/null
+++ b/server/sonar-webserver-core/src/main/java/org/sonar/server/rule/registration/ActiveRulesImpactInitializer.java
@@ -0,0 +1,97 @@
+/*
+ * SonarQube
+ * Copyright (C) 2009-2024 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.registration;
+
+import java.util.Collection;
+import java.util.HashMap;
+import java.util.List;
+import java.util.Map;
+import org.sonar.api.issue.impact.Severity;
+import org.sonar.api.issue.impact.SoftwareQuality;
+import org.sonar.api.rules.RuleType;
+import org.sonar.api.server.rule.RulesDefinition;
+import org.sonar.db.DbClient;
+import org.sonar.db.DbSession;
+import org.sonar.db.issue.ImpactDto;
+import org.sonar.db.qualityprofile.ActiveRuleDto;
+import org.sonar.db.rule.RuleDto;
+import org.sonar.db.rule.SeverityUtil;
+import org.sonar.server.property.InternalProperties;
+import org.sonar.server.qualityprofile.QProfileImpactSeverityMapper;
+
+import static java.util.stream.Collectors.toUnmodifiableMap;
+
+/**
+ * Creates the initial impacts on active_rules, based on the rules. This class is necessary only up to 10 LTA version and can be removed
+ * later
+ */
+public class ActiveRulesImpactInitializer {
+ private static final String ACTIVE_RULE_IMPACT_INITIAL_POPULATION_DONE = "activeRules.impacts.populated";
+
+ private final InternalProperties internalProperties;
+ private final DbClient dbClient;
+
+ ActiveRulesImpactInitializer(InternalProperties internalProperties, DbClient dbClient) {
+ this.internalProperties = internalProperties;
+ this.dbClient = dbClient;
+ }
+
+ public void createImpactsOnActiveRules(RulesRegistrationContext context, RulesDefinition.Repository repoDef, DbSession dbSession) {
+
+ if (Boolean.parseBoolean(internalProperties.read(ACTIVE_RULE_IMPACT_INITIAL_POPULATION_DONE).orElse("false"))) {
+ return;
+ }
+
+ Map<String, RuleDto> rules = new HashMap<>(repoDef.rules().size());
+
+ for (RulesDefinition.Rule ruleDef : repoDef.rules()) {
+ context.getDbRuleFor(ruleDef).ifPresent(ruleDto -> rules.put(ruleDto.getUuid(), ruleDto));
+ }
+
+ context.getAllModified().forEach(r -> rules.put(r.getUuid(), r));
+
+ List<ActiveRuleDto> activeRules = dbClient.activeRuleDao().selectByRepository(dbSession, repoDef.key(), repoDef.language());
+
+ for (ActiveRuleDto activeRuleDto : activeRules) {
+ RuleDto rule = rules.get(activeRuleDto.getRuleUuid());
+ if (rule == null || rule.getEnumType() == RuleType.SECURITY_HOTSPOT) {
+ continue;
+ }
+
+ Map<SoftwareQuality, Severity> impacts = toImpactMap(rule.getDefaultImpacts());
+
+ if (!activeRuleDto.getSeverity().equals(rule.getSeverity())) {
+ impacts = QProfileImpactSeverityMapper.mapImpactSeverities(SeverityUtil.getSeverityFromOrdinal(activeRuleDto.getSeverity()), impacts, rule.getEnumType());
+ }
+ activeRuleDto.setImpacts(impacts);
+ dbClient.activeRuleDao().update(dbSession, activeRuleDto);
+ }
+
+ }
+
+ private static Map<SoftwareQuality, Severity> toImpactMap(Collection<ImpactDto> impacts) {
+ return impacts.stream()
+ .collect(toUnmodifiableMap(ImpactDto::getSoftwareQuality, ImpactDto::getSeverity));
+ }
+
+ public void markInitialPopulationDone() {
+ internalProperties.write(ACTIVE_RULE_IMPACT_INITIAL_POPULATION_DONE, "true");
+ }
+}
diff --git a/server/sonar-webserver-core/src/main/java/org/sonar/server/rule/registration/RulesRegistrant.java b/server/sonar-webserver-core/src/main/java/org/sonar/server/rule/registration/RulesRegistrant.java
index 9a022988bab..bafd97fda77 100644
--- a/server/sonar-webserver-core/src/main/java/org/sonar/server/rule/registration/RulesRegistrant.java
+++ b/server/sonar-webserver-core/src/main/java/org/sonar/server/rule/registration/RulesRegistrant.java
@@ -83,11 +83,13 @@ public class RulesRegistrant implements Startable {
private final QualityProfileChangesUpdater qualityProfileChangesUpdater;
private final SonarQubeVersion sonarQubeVersion;
private final DetectPluginChange detectPluginChange;
+ private final ActiveRulesImpactInitializer activeRulesImpactInitializer;
public RulesRegistrant(RuleDefinitionsLoader defLoader, QProfileRules qProfileRules, DbClient dbClient, RuleIndexer ruleIndexer,
ActiveRuleIndexer activeRuleIndexer, System2 system2, WebServerRuleFinder webServerRuleFinder,
MetadataIndex metadataIndex, RulesKeyVerifier rulesKeyVerifier, StartupRuleUpdater startupRuleUpdater,
- NewRuleCreator newRuleCreator, QualityProfileChangesUpdater qualityProfileChangesUpdater, SonarQubeVersion sonarQubeVersion, DetectPluginChange detectPluginChange) {
+ NewRuleCreator newRuleCreator, QualityProfileChangesUpdater qualityProfileChangesUpdater, SonarQubeVersion sonarQubeVersion,
+ DetectPluginChange detectPluginChange, ActiveRulesImpactInitializer activeRulesImpactInitializer) {
this.defLoader = defLoader;
this.qProfileRules = qProfileRules;
this.dbClient = dbClient;
@@ -102,6 +104,7 @@ public class RulesRegistrant implements Startable {
this.qualityProfileChangesUpdater = qualityProfileChangesUpdater;
this.sonarQubeVersion = sonarQubeVersion;
this.detectPluginChange = detectPluginChange;
+ this.activeRulesImpactInitializer = activeRulesImpactInitializer;
}
@Override
@@ -125,12 +128,15 @@ public class RulesRegistrant implements Startable {
throw new IllegalStateException("Language is mandatory for repository " + repoDef.key());
}
Set<PluginRuleUpdate> pluginRuleUpdates = registerRules(rulesRegistrationContext, repoDef.rules(), dbSession);
+
if (!repoDef.isExternal()) {
// External rules are not part of quality profiles
+ activeRulesImpactInitializer.createImpactsOnActiveRules(rulesRegistrationContext, repoDef, dbSession);
qualityProfileChangesUpdater.createQprofileChangesForRuleUpdates(dbSession, pluginRuleUpdates);
}
dbSession.commit();
}
+ activeRulesImpactInitializer.markInitialPopulationDone();
processRemainingDbRules(rulesRegistrationContext, dbSession);
List<ActiveRuleChange> changes = anyPluginChanged ? removeActiveRulesOnStillExistingRepositories(dbSession, rulesRegistrationContext, rulesRepositories) : List.of();
dbSession.commit();