diff options
-rw-r--r-- | server/sonar-server/src/main/java/org/sonar/server/computation/step/ValidateProjectStep.java | 111 |
1 files changed, 68 insertions, 43 deletions
diff --git a/server/sonar-server/src/main/java/org/sonar/server/computation/step/ValidateProjectStep.java b/server/sonar-server/src/main/java/org/sonar/server/computation/step/ValidateProjectStep.java index f3a7aaf6e11..58c1292fb63 100644 --- a/server/sonar-server/src/main/java/org/sonar/server/computation/step/ValidateProjectStep.java +++ b/server/sonar-server/src/main/java/org/sonar/server/computation/step/ValidateProjectStep.java @@ -22,12 +22,13 @@ package org.sonar.server.computation.step; import com.google.common.base.Function; import com.google.common.base.Joiner; -import com.google.common.collect.Maps; +import com.google.common.collect.FluentIterable; import java.util.ArrayList; import java.util.List; import java.util.Map; import javax.annotation.CheckForNull; import javax.annotation.Nonnull; +import javax.annotation.Nullable; import org.sonar.api.CoreProperties; import org.sonar.api.config.Settings; import org.sonar.api.utils.MessageException; @@ -50,6 +51,7 @@ import org.sonar.server.db.DbClient; * <li>project or module key is not valid</li> * <li>module key already exists in another project (same module key cannot exists in different projects)</li> * <li>module key is already used as a project key</li> + * <li>date of the analysis is before last analysis</li> * </ol> */ public class ValidateProjectStep implements ComputationStep { @@ -72,15 +74,10 @@ public class ValidateProjectStep implements ComputationStep { public void execute() { DbSession session = dbClient.openSession(false); try { - List<ComponentDto> modules = dbClient.componentDao().selectModulesFromProjectKey(session, treeRootHolder.getRoot().getKey()); - Map<String, ComponentDto> modulesByKey = Maps.uniqueIndex(modules, new Function<ComponentDto, String>() { - @Override - public String apply(@Nonnull ComponentDto input) { - return input.key(); - } - }); + List<ComponentDto> baseModules = dbClient.componentDao().selectModulesFromProjectKey(session, treeRootHolder.getRoot().getKey()); + Map<String, ComponentDto> baseModulesByKey = FluentIterable.from(baseModules).uniqueIndex(ComponentDtoToKey.INSTANCE); ValidateProjectsVisitor visitor = new ValidateProjectsVisitor(session, dbClient.componentDao(), - settings.getBoolean(CoreProperties.CORE_PREVENT_AUTOMATIC_PROJECT_CREATION), modulesByKey); + settings.getBoolean(CoreProperties.CORE_PREVENT_AUTOMATIC_PROJECT_CREATION), baseModulesByKey); visitor.visit(treeRootHolder.getRoot()); if (!visitor.validationMessages.isEmpty()) { @@ -100,64 +97,83 @@ public class ValidateProjectStep implements ComputationStep { private final DbSession session; private final ComponentDao componentDao; private final boolean preventAutomaticProjectCreation; - private final Map<String, ComponentDto> modulesByKey; + private final Map<String, ComponentDto> baseModulesByKey; private final List<String> validationMessages = new ArrayList<>(); - private Component root; + private Component rawProject; - public ValidateProjectsVisitor(DbSession session, ComponentDao componentDao, boolean preventAutomaticProjectCreation, Map<String, ComponentDto> modulesByKey) { + public ValidateProjectsVisitor(DbSession session, ComponentDao componentDao, boolean preventAutomaticProjectCreation, Map<String, ComponentDto> baseModulesByKey) { super(Component.Type.MODULE, Order.PRE_ORDER); this.session = session; this.componentDao = componentDao; this.preventAutomaticProjectCreation = preventAutomaticProjectCreation; - this.modulesByKey = modulesByKey; + this.baseModulesByKey = baseModulesByKey; } @Override - public void visitProject(Component project) { - this.root = project; + public void visitProject(Component rawProject) { + this.rawProject = rawProject; validateBranch(); - validateBatchKey(project); + validateBatchKey(rawProject); + + String rawProjectKey = rawProject.getKey(); + ComponentDto baseProject = loadBaseComponent(rawProjectKey); + validateWhenProvisioningEnforced(baseProject, rawProjectKey); + validateProjectKey(baseProject, rawProjectKey); + } - String projectKey = project.getKey(); - ComponentDto projectDto = loadComponent(projectKey); - if (projectDto == null) { + private void validateWhenProvisioningEnforced(@Nullable ComponentDto baseProject, String rawProjectKey){ + if (baseProject == null) { if (preventAutomaticProjectCreation) { - validationMessages.add(String.format("Unable to scan non-existing project '%s'", projectKey)); + validationMessages.add(String.format("Unable to scan non-existing project '%s'", rawProjectKey)); } - } else if (!projectDto.projectUuid().equals(projectDto.uuid())) { + } + } + + private void validateProjectKey(@Nullable ComponentDto baseProject, String rawProjectKey){ + if (baseProject != null && !baseProject.projectUuid().equals(baseProject.uuid())) { // Project key is already used as a module of another project - ComponentDto anotherProject = componentDao.selectByUuid(session, projectDto.projectUuid()); + ComponentDto anotherBaseProject = componentDao.selectByUuid(session, baseProject.projectUuid()); validationMessages.add(String.format("The project \"%s\" is already defined in SonarQube but as a module of project \"%s\". " - + "If you really want to stop directly analysing project \"%s\", please first delete it from SonarQube and then relaunch the analysis of project \"%s\".", - projectKey, anotherProject.key(), anotherProject.key(), projectKey)); + + "If you really want to stop directly analysing project \"%s\", please first delete it from SonarQube and then relaunch the analysis of project \"%s\".", + rawProjectKey, anotherBaseProject.key(), anotherBaseProject.key(), rawProjectKey)); } } @Override - public void visitModule(Component module) { - String projectKey = root.getKey(); - String moduleKey = module.getKey(); - validateBatchKey(module); + public void visitModule(Component rawModule) { + String rawProjectKey = rawProject.getKey(); + String rawModuleKey = rawModule.getKey(); + validateBatchKey(rawModule); - ComponentDto moduleDto = loadComponent(moduleKey); - if (moduleDto == null) { + ComponentDto baseModule = loadBaseComponent(rawModuleKey); + if (baseModule == null) { return; } - if (moduleDto.projectUuid().equals(moduleDto.uuid())) { + + validateModuleIsNotAlreadyUsedAsProject(baseModule, rawProjectKey, rawModuleKey); + validateModuleKeyIsNotAlreadyUsedInAnotherProject(baseModule, rawModuleKey); + } + + private void validateModuleIsNotAlreadyUsedAsProject(ComponentDto baseModule, String rawProjectKey, String rawModuleKey){ + if (baseModule.projectUuid().equals(baseModule.uuid())) { // module is actually a project validationMessages.add(String.format("The project \"%s\" is already defined in SonarQube but not as a module of project \"%s\". " - + "If you really want to stop directly analysing project \"%s\", please first delete it from SonarQube and then relaunch the analysis of project \"%s\".", - moduleKey, projectKey, moduleKey, projectKey)); - } else if (!moduleDto.projectUuid().equals(root.getUuid())) { - ComponentDto projectModule = componentDao.selectByUuid(session, moduleDto.projectUuid()); - validationMessages.add(String.format("Module \"%s\" is already part of project \"%s\"", moduleKey, projectModule.key())); + + "If you really want to stop directly analysing project \"%s\", please first delete it from SonarQube and then relaunch the analysis of project \"%s\".", + rawModuleKey, rawProjectKey, rawModuleKey, rawProjectKey)); } } - private void validateBatchKey(Component component) { - String batchKey = reportReader.readComponent(component.getRef()).getKey(); + private void validateModuleKeyIsNotAlreadyUsedInAnotherProject(ComponentDto baseModule, String rawModuleKey){ + if (!baseModule.projectUuid().equals(baseModule.uuid()) && !baseModule.projectUuid().equals(rawProject.getUuid())) { + ComponentDto projectModule = componentDao.selectByUuid(session, baseModule.projectUuid()); + validationMessages.add(String.format("Module \"%s\" is already part of project \"%s\"", rawModuleKey, projectModule.key())); + } + } + + private void validateBatchKey(Component rawComponent) { + String batchKey = reportReader.readComponent(rawComponent.getRef()).getKey(); if (!ComponentKeys.isValidModuleKey(batchKey)) { validationMessages.add(String.format("\"%s\" is not a valid project or module key. " + "Allowed characters are alphanumeric, '-', '_', '.' and ':', with at least one non-digit.", batchKey)); @@ -177,13 +193,22 @@ public class ValidateProjectStep implements ComputationStep { } } - private ComponentDto loadComponent(String componentKey) { - ComponentDto componentDto = modulesByKey.get(componentKey); - if (componentDto == null) { + private ComponentDto loadBaseComponent(String rawComponentKey) { + ComponentDto baseComponent = baseModulesByKey.get(rawComponentKey); + if (baseComponent == null) { // Load component from key to be able to detect issue (try to analyze a module, etc.) - return componentDao.selectNullableByKey(session, componentKey); + return componentDao.selectNullableByKey(session, rawComponentKey); } - return componentDto; + return baseComponent; + } + } + + private enum ComponentDtoToKey implements Function<ComponentDto, String> { + INSTANCE; + + @Override + public String apply(@Nonnull ComponentDto input) { + return input.key(); } } } |