Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance. Project price only 1 $
You can buy this project and download/modify it how often you want.
/*
* SonarLint Core - Server Connection
* Copyright (C) 2016-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.sonarsource.sonarlint.core.serverconnection;
import java.util.Map;
import java.util.Set;
import java.util.function.Function;
import java.util.stream.Collectors;
import org.sonarsource.sonarlint.core.commons.Language;
import org.sonarsource.sonarlint.core.commons.log.SonarLintLogger;
import org.sonarsource.sonarlint.core.commons.progress.ProgressMonitor;
import org.sonarsource.sonarlint.core.serverapi.ServerApi;
import org.sonarsource.sonarlint.core.serverapi.branches.ServerBranch;
import org.sonarsource.sonarlint.core.serverapi.qualityprofile.QualityProfile;
import org.sonarsource.sonarlint.core.serverconnection.storage.StorageException;
import static java.util.stream.Collectors.toSet;
public class LocalStorageSynchronizer {
private static final SonarLintLogger LOG = SonarLintLogger.get();
private final Set enabledLanguageKeys;
private final ConnectionStorage storage;
private final ServerInfoSynchronizer serverInfoSynchronizer;
private final PluginsSynchronizer pluginsSynchronizer;
public LocalStorageSynchronizer(Set enabledLanguages, Set embeddedPluginKeys, ServerInfoSynchronizer serverInfoSynchronizer, ConnectionStorage storage) {
this.enabledLanguageKeys = enabledLanguages.stream().map(Language::getLanguageKey).collect(toSet());
this.storage = storage;
this.pluginsSynchronizer = new PluginsSynchronizer(enabledLanguages, storage, embeddedPluginKeys);
this.serverInfoSynchronizer = serverInfoSynchronizer;
}
public SynchronizationResult synchronize(ServerApi serverApi, Set projectKeys, ProgressMonitor progressMonitor) {
serverInfoSynchronizer.synchronize(serverApi);
var anyPluginUpdated = pluginsSynchronizer.synchronize(serverApi, progressMonitor);
projectKeys.stream()
.collect(Collectors.toMap(Function.identity(), projectKey -> synchronizeAnalyzerConfig(serverApi, projectKey, progressMonitor)))
.forEach((projectKey, analyzerConfig) -> storage.project(projectKey).analyzerConfiguration().store(analyzerConfig));
var branchByProjectKey = projectKeys.stream()
.collect(Collectors.toMap(Function.identity(), projectKey -> synchronizeProjectBranches(serverApi, projectKey)));
branchByProjectKey
.forEach((projectKey, branches) -> storage.project(projectKey).branches().store(branches));
return new SynchronizationResult(anyPluginUpdated);
}
private AnalyzerConfiguration synchronizeAnalyzerConfig(ServerApi serverApi, String projectKey, ProgressMonitor progressMonitor) {
LOG.info("[SYNC] Synchronizing analyzer configuration for project '{}'", projectKey);
Map currentRuleSets;
int currentSchemaVersion;
try {
var analyzerConfiguration = storage.project(projectKey).analyzerConfiguration().read();
currentRuleSets = analyzerConfiguration.getRuleSetByLanguageKey();
currentSchemaVersion = analyzerConfiguration.getSchemaVersion();
} catch (StorageException e) {
currentRuleSets = Map.of();
currentSchemaVersion = 0;
}
var shouldForceRuleSetUpdate = outdatedSchema(currentSchemaVersion);
var currentRuleSetsFinal = currentRuleSets;
var settings = new Settings(serverApi.settings().getProjectSettings(projectKey));
var ruleSetsByLanguageKey = serverApi.qualityProfile().getQualityProfiles(projectKey).stream()
.filter(qualityProfile -> enabledLanguageKeys.contains(qualityProfile.getLanguage()))
.collect(Collectors.toMap(QualityProfile::getLanguage, profile -> toRuleSet(serverApi, currentRuleSetsFinal, profile, shouldForceRuleSetUpdate, progressMonitor)));
return new AnalyzerConfiguration(settings, ruleSetsByLanguageKey, AnalyzerConfiguration.CURRENT_SCHEMA_VERSION);
}
private static RuleSet toRuleSet(ServerApi serverApi, Map currentRuleSets, QualityProfile profile, boolean forceUpdate,
ProgressMonitor progressMonitor) {
var language = profile.getLanguage();
if (forceUpdate ||
newlySupportedLanguage(currentRuleSets, language) ||
profileModifiedSinceLastSync(currentRuleSets, profile, language)) {
var profileKey = profile.getKey();
LOG.info("[SYNC] Fetching rule set for language '{}' from profile '{}'", language, profileKey);
var profileActiveRules = serverApi.rules().getAllActiveRules(profileKey, progressMonitor);
return new RuleSet(profileActiveRules, profile.getRulesUpdatedAt());
} else {
LOG.info("[SYNC] Active rules for '{}' are up-to-date", language);
return currentRuleSets.get(language);
}
}
private static boolean profileModifiedSinceLastSync(Map currentRuleSets, QualityProfile profile, String language) {
return !currentRuleSets.get(language).getLastModified().equals(profile.getRulesUpdatedAt());
}
private static boolean newlySupportedLanguage(Map currentRuleSets, String language) {
return !currentRuleSets.containsKey(language);
}
private static boolean outdatedSchema(int currentSchemaVersion) {
return currentSchemaVersion < AnalyzerConfiguration.CURRENT_SCHEMA_VERSION;
}
private static ProjectBranches synchronizeProjectBranches(ServerApi serverApi, String projectKey) {
LOG.info("[SYNC] Synchronizing project branches for project '{}'", projectKey);
var allBranches = serverApi.branches().getAllBranches(projectKey);
var mainBranch = allBranches.stream().filter(ServerBranch::isMain).findFirst().map(ServerBranch::getName)
.orElseThrow(() -> new IllegalStateException("No main branch for project '" + projectKey + "'"));
return new ProjectBranches(allBranches.stream().map(ServerBranch::getName).collect(toSet()), mainBranch);
}
}