123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306 |
- /*
- * 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.component;
-
- import java.util.Collection;
- import java.util.Optional;
- import java.util.Set;
- import java.util.stream.Collectors;
- import javax.annotation.Nullable;
- import org.sonar.api.resources.Qualifiers;
- import org.sonar.api.resources.ResourceType;
- import org.sonar.api.resources.ResourceTypes;
- import org.sonar.api.resources.Scopes;
- import org.sonar.db.DbClient;
- import org.sonar.db.DbSession;
- import org.sonar.db.component.BranchDto;
- import org.sonar.db.component.ComponentDto;
- import org.sonar.db.entity.EntityDto;
- import org.sonar.db.project.ProjectDto;
- import org.sonar.server.exceptions.NotFoundException;
-
- import static com.google.common.base.Preconditions.checkArgument;
- import static java.lang.String.format;
- import static org.sonar.server.exceptions.BadRequestException.checkRequest;
-
- public class ComponentFinder {
- private static final String MSG_COMPONENT_ID_OR_KEY_TEMPLATE = "Either '%s' or '%s' must be provided";
- private static final String MSG_PARAMETER_MUST_NOT_BE_EMPTY = "The '%s' parameter must not be empty";
- private static final String LABEL_PROJECT = "Project";
- private static final String LABEL_COMPONENT = "Component";
- private static final String LABEL_PROJECT_NOT_FOUND = "Project '%s' not found";
- private static final String LABEL_ENTITY_NOT_FOUND = "Component '%s' not found";
-
- private final DbClient dbClient;
- private final ResourceTypes resourceTypes;
-
- public ComponentFinder(DbClient dbClient, ResourceTypes resourceTypes) {
- this.dbClient = dbClient;
- this.resourceTypes = resourceTypes;
- }
-
- public ComponentDto getByUuidOrKey(DbSession dbSession, @Nullable String componentUuid, @Nullable String componentKey, ParamNames parameterNames) {
- checkByUuidOrKey(componentUuid, componentKey, parameterNames);
-
- if (componentUuid != null) {
- return getByUuidFromMainBranch(dbSession, checkParamNotEmpty(componentUuid, parameterNames.getUuidParam()));
- }
-
- return getByKey(dbSession, checkParamNotEmpty(componentKey, parameterNames.getKeyParam()));
- }
-
- public EntityDto getEntityByUuidOrKey(DbSession dbSession, @Nullable String entityUuid, @Nullable String entityKey, ParamNames parameterNames) {
- checkByUuidOrKey(entityUuid, entityKey, parameterNames);
-
- if (entityUuid != null) {
- return getEntityByUuid(dbSession, checkParamNotEmpty(entityUuid, parameterNames.getUuidParam()));
- }
- return getEntityByKey(dbSession, checkParamNotEmpty(entityKey, parameterNames.getKeyParam()));
- }
-
- public EntityDto getEntityByKey(DbSession dbSession, String entityKey) {
- return dbClient.entityDao().selectByKey(dbSession, entityKey)
- .orElseThrow(() -> new NotFoundException(String.format(LABEL_ENTITY_NOT_FOUND, entityKey)));
- }
-
- public EntityDto getEntityByUuid(DbSession dbSession, String entityUuid) {
- return dbClient.entityDao().selectByUuid(dbSession, entityUuid)
- .orElseThrow(() -> new NotFoundException(String.format(LABEL_ENTITY_NOT_FOUND, entityUuid)));
- }
-
- public ProjectDto getProjectByKey(DbSession dbSession, String projectKey) {
- return dbClient.projectDao().selectProjectByKey(dbSession, projectKey)
- .orElseThrow(() -> new NotFoundException(String.format(LABEL_PROJECT_NOT_FOUND, projectKey)));
- }
-
- public ProjectDto getApplicationByKey(DbSession dbSession, String applicationKey) {
- return dbClient.projectDao().selectApplicationByKey(dbSession, applicationKey)
- .orElseThrow(() -> new NotFoundException(String.format("Application '%s' not found", applicationKey)));
- }
-
- public ProjectDto getProjectOrApplicationByKey(DbSession dbSession, String projectKey) {
- return dbClient.projectDao().selectProjectOrAppByKey(dbSession, projectKey)
- .orElseThrow(() -> new NotFoundException(String.format(LABEL_PROJECT_NOT_FOUND, projectKey)));
- }
-
- public ProjectDto getProjectByUuid(DbSession dbSession, String projectUuid) {
- return dbClient.projectDao().selectByUuid(dbSession, projectUuid)
- .filter(p -> Qualifiers.PROJECT.equals(p.getQualifier()))
- .orElseThrow(() -> new NotFoundException(String.format(LABEL_PROJECT_NOT_FOUND, projectUuid)));
- }
-
- public ProjectDto getProjectByUuidOrKey(DbSession dbSession, @Nullable String projectUuid, @Nullable String projectKey, ParamNames parameterNames) {
- checkByUuidOrKey(projectUuid, projectKey, parameterNames);
-
- if (projectUuid != null) {
- return getProjectByUuid(dbSession, checkParamNotEmpty(projectUuid, parameterNames.getUuidParam()));
- }
-
- return getProjectByKey(dbSession, checkParamNotEmpty(projectKey, parameterNames.getKeyParam()));
- }
-
- private static String checkParamNotEmpty(String value, String param) {
- checkArgument(!value.isEmpty(), MSG_PARAMETER_MUST_NOT_BE_EMPTY, param);
- return value;
- }
-
- public BranchDto getBranchByUuid(DbSession dbSession, String branchUuid) {
- return dbClient.branchDao().selectByUuid(dbSession, branchUuid)
- .orElseThrow(() -> new NotFoundException(String.format("Branch uuid '%s' not found", branchUuid)));
- }
-
- public BranchDto getBranchOrPullRequest(DbSession dbSession, EntityDto entity, @Nullable String branchKey, @Nullable String pullRequestKey) {
- return getBranchOrPullRequest(dbSession, entity.getUuid(), entity.getKey(), branchKey, pullRequestKey);
- }
-
- public ProjectAndBranch getAppOrProjectAndBranch(DbSession dbSession, String projectKey, @Nullable String branchKey, @Nullable String pullRequestKey) {
- ProjectDto projectOrApp = getProjectOrApplicationByKey(dbSession, projectKey);
- BranchDto branch = getBranchOrPullRequest(dbSession, projectOrApp, branchKey, pullRequestKey);
- return new ProjectAndBranch(projectOrApp, branch);
- }
-
- public ProjectAndBranch getProjectAndBranch(DbSession dbSession, String projectKey, @Nullable String branchKey, @Nullable String pullRequestKey) {
- ProjectDto project = getProjectByKey(dbSession, projectKey);
- BranchDto branch = getBranchOrPullRequest(dbSession, project, branchKey, pullRequestKey);
- return new ProjectAndBranch(project, branch);
- }
-
- public static class ProjectAndBranch {
- private final ProjectDto project;
- private final BranchDto branch;
-
- public ProjectAndBranch(ProjectDto project, BranchDto branch) {
- this.project = project;
- this.branch = branch;
- }
-
- public ProjectDto getProject() {
- return project;
- }
-
- public BranchDto getBranch() {
- return branch;
- }
- }
-
- public BranchDto getBranchOrPullRequest(DbSession dbSession, String projectUuid, String projectKey, @Nullable String branchKey, @Nullable String pullRequestKey) {
- if (branchKey != null) {
- return dbClient.branchDao().selectByBranchKey(dbSession, projectUuid, branchKey)
- .orElseThrow(() -> new NotFoundException(String.format("Branch '%s' in project '%s' not found", branchKey, projectKey)));
- } else if (pullRequestKey != null) {
- return dbClient.branchDao().selectByPullRequestKey(dbSession, projectUuid, pullRequestKey)
- .orElseThrow(() -> new NotFoundException(String.format("Pull request '%s' in project '%s' not found", pullRequestKey, projectKey)));
- }
- return dbClient.branchDao().selectMainBranchByProjectUuid(dbSession, projectUuid)
- .orElseThrow(() -> new NotFoundException(String.format("Main branch in project '%s' not found", projectKey)));
- }
-
- public BranchDto getMainBranch(DbSession dbSession, ProjectDto projectDto) {
- return dbClient.branchDao().selectMainBranchByProjectUuid(dbSession, projectDto.getUuid())
- .orElseThrow(() -> new IllegalStateException(String.format("Can't find main branch for project '%s'", projectDto.getKey())));
- }
-
- private static void checkByUuidOrKey(@Nullable String componentUuid, @Nullable String componentKey, ParamNames parameterNames) {
- checkArgument(componentUuid != null ^ componentKey != null, MSG_COMPONENT_ID_OR_KEY_TEMPLATE, parameterNames.getUuidParam(), parameterNames.getKeyParam());
- }
-
- public ComponentDto getByKey(DbSession dbSession, String key) {
- return getByKey(dbSession, key, LABEL_COMPONENT);
- }
-
- private ComponentDto getByKey(DbSession dbSession, String key, String label) {
- return checkComponent(dbSession, dbClient.componentDao().selectByKey(dbSession, key), "%s key '%s' not found", label, key);
- }
-
- public ComponentDto getByUuidFromMainBranch(DbSession dbSession, String uuid) {
- return getByUuidFromMainBranch(dbSession, uuid, LABEL_COMPONENT);
- }
-
- private ComponentDto getByUuidFromMainBranch(DbSession dbSession, String uuid, String label) {
- return checkComponent(dbSession, dbClient.componentDao().selectByUuid(dbSession, uuid), "%s id '%s' not found", label, uuid);
- }
-
- private ComponentDto checkComponent(DbSession session, Optional<ComponentDto> componentDto, String message, Object... messageArguments) {
- if (componentDto.isPresent() && componentDto.get().isEnabled()) {
- if (dbClient.branchDao().selectByUuid(session, componentDto.get().branchUuid()).map(BranchDto::isMain).orElse(true)) {
- return componentDto.get();
- }
- }
- throw new NotFoundException(format(message, messageArguments));
- }
-
- public ComponentDto getRootComponentByUuidOrKey(DbSession dbSession, @Nullable String projectUuid, @Nullable String projectKey) {
- ComponentDto project;
- if (projectUuid != null) {
- project = getByUuidFromMainBranch(dbSession, projectUuid, LABEL_PROJECT);
- } else {
- project = getByKey(dbSession, projectKey, LABEL_PROJECT);
- }
- checkIsProject(project);
-
- return project;
- }
-
- private ComponentDto checkIsProject(ComponentDto component) {
- Set<String> rootQualifiers = getRootQualifiers(resourceTypes);
-
- checkRequest(component.scope().equals(Scopes.PROJECT) && rootQualifiers.contains(component.qualifier()),
- format(
- "Component '%s' (id: %s) must be a project%s.",
- component.getKey(), component.uuid(),
- rootQualifiers.contains(Qualifiers.VIEW) ? " or a view" : ""));
-
- return component;
- }
-
- private static Set<String> getRootQualifiers(ResourceTypes resourceTypes) {
- Collection<ResourceType> rootTypes = resourceTypes.getRoots();
- return rootTypes
- .stream()
- .map(ResourceType::getQualifier)
- .collect(Collectors.toSet());
- }
-
- public ComponentDto getByKeyAndBranch(DbSession dbSession, String key, String branch) {
- Optional<ComponentDto> componentDto = dbClient.componentDao().selectByKeyAndBranch(dbSession, key, branch);
- if (componentDto.isPresent() && componentDto.get().isEnabled()) {
- return componentDto.get();
- }
- throw new NotFoundException(format("Component '%s' on branch '%s' not found", key, branch));
- }
-
- public ComponentDto getByKeyAndPullRequest(DbSession dbSession, String key, String pullRequest) {
- Optional<ComponentDto> componentDto = dbClient.componentDao().selectByKeyAndPullRequest(dbSession, key, pullRequest);
- if (componentDto.isPresent() && componentDto.get().isEnabled()) {
- return componentDto.get();
- }
- throw new NotFoundException(format("Component '%s' of pull request '%s' not found", key, pullRequest));
- }
-
- public ComponentDto getByKeyAndOptionalBranchOrPullRequest(DbSession dbSession, String key, @Nullable String branch, @Nullable String pullRequest) {
- checkArgument(branch == null || pullRequest == null, "Either branch or pull request can be provided, not both");
- if (branch != null) {
- return getByKeyAndBranch(dbSession, key, branch);
- } else if (pullRequest != null) {
- return getByKeyAndPullRequest(dbSession, key, pullRequest);
- }
- return getByKey(dbSession, key);
- }
-
- public Optional<ComponentDto> getOptionalByKeyAndOptionalBranchOrPullRequest(DbSession dbSession, String key, @Nullable String branch, @Nullable String pullRequest) {
- checkArgument(branch == null || pullRequest == null, "Either branch or pull request can be provided, not both");
- if (branch != null) {
- return dbClient.componentDao().selectByKeyAndBranch(dbSession, key, branch);
- } else if (pullRequest != null) {
- return dbClient.componentDao().selectByKeyAndPullRequest(dbSession, key, pullRequest);
- }
- return dbClient.componentDao().selectByKey(dbSession, key);
- }
-
- public enum ParamNames {
- PROJECT_ID_AND_KEY("projectId", "projectKey"),
- PROJECT_UUID_AND_KEY("projectUuid", "projectKey"),
- PROJECT_UUID_AND_PROJECT("projectUuid", "project"),
- UUID_AND_KEY("uuid", "key"),
- ID_AND_KEY("id", "key"),
- COMPONENT_ID_AND_KEY("componentId", "componentKey"),
- BASE_COMPONENT_ID_AND_KEY("baseComponentId", "component"),
- DEVELOPER_ID_AND_KEY("developerId", "developerKey"),
- COMPONENT_ID_AND_COMPONENT("componentId", "component"),
- PROJECT_ID_AND_PROJECT("projectId", "project"),
- PROJECT_ID_AND_FROM("projectId", "from");
-
- private final String uuidParamName;
- private final String keyParamName;
-
- ParamNames(String uuidParamName, String keyParamName) {
- this.uuidParamName = uuidParamName;
- this.keyParamName = keyParamName;
- }
-
- public String getUuidParam() {
- return uuidParamName;
- }
-
- public String getKeyParam() {
- return keyParamName;
- }
- }
- }
|