All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.google.gerrit.server.restapi.change.CheckSubmitRequirement Maven / Gradle / Ivy

// Copyright (C) 2021 The Android Open Source Project
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.

package com.google.gerrit.server.restapi.change;

import com.google.common.collect.Iterables;
import com.google.gerrit.entities.RefNames;
import com.google.gerrit.entities.SubmitRequirement;
import com.google.gerrit.entities.SubmitRequirementExpression;
import com.google.gerrit.entities.SubmitRequirementResult;
import com.google.gerrit.extensions.common.SubmitRequirementInput;
import com.google.gerrit.extensions.common.SubmitRequirementResultInfo;
import com.google.gerrit.extensions.restapi.BadRequestException;
import com.google.gerrit.extensions.restapi.IdString;
import com.google.gerrit.extensions.restapi.ResourceConflictException;
import com.google.gerrit.extensions.restapi.ResourceNotFoundException;
import com.google.gerrit.extensions.restapi.Response;
import com.google.gerrit.extensions.restapi.RestApiException;
import com.google.gerrit.extensions.restapi.RestModifyView;
import com.google.gerrit.extensions.restapi.TopLevelResource;
import com.google.gerrit.server.change.ChangeResource;
import com.google.gerrit.server.change.SubmitRequirementsJson;
import com.google.gerrit.server.git.GitRepositoryManager;
import com.google.gerrit.server.notedb.ChangeNotes;
import com.google.gerrit.server.permissions.PermissionBackendException;
import com.google.gerrit.server.project.ProjectConfig;
import com.google.gerrit.server.project.SubmitRequirementsEvaluator;
import com.google.gerrit.server.query.change.ChangeData;
import com.google.inject.Inject;
import java.io.IOException;
import java.util.List;
import java.util.Map.Entry;
import java.util.Optional;
import java.util.stream.Collectors;
import org.eclipse.jgit.errors.ConfigInvalidException;
import org.eclipse.jgit.lib.ObjectId;
import org.eclipse.jgit.lib.Repository;
import org.kohsuke.args4j.Option;

/**
 * A rest view to evaluate (test) a {@link com.google.gerrit.entities.SubmitRequirement} on a given
 * change. The submit requirement can be supplied in one of two ways:
 *
 * 

1) Using the {@link SubmitRequirementInput}. * *

2) From a change to the {@link RefNames#REFS_CONFIG} branch and the name of the * submit-requirement. */ public class CheckSubmitRequirement implements RestModifyView { private final SubmitRequirementsEvaluator evaluator; @Option(name = "--sr-name") private String srName; @Option(name = "--refs-config-change-id") private String refsConfigChangeId; private final GitRepositoryManager repoManager; private final ProjectConfig.Factory projectConfigFactory; private final ChangeData.Factory changeDataFactory; private final ChangesCollection changesCollection; private final ChangeNotes.Factory changeNotesFactory; public void setSrName(String srName) { this.srName = srName; } public void setRefsConfigChangeId(String refsConfigChangeId) { this.refsConfigChangeId = refsConfigChangeId; } @Inject public CheckSubmitRequirement( SubmitRequirementsEvaluator evaluator, GitRepositoryManager repoManager, ProjectConfig.Factory projectConfigFactory, ChangeData.Factory changeDataFactory, ChangeNotes.Factory changeNotesFactory, ChangesCollection changesCollection) { this.evaluator = evaluator; this.repoManager = repoManager; this.projectConfigFactory = projectConfigFactory; this.changeDataFactory = changeDataFactory; this.changeNotesFactory = changeNotesFactory; this.changesCollection = changesCollection; } @Override public Response apply( ChangeResource resource, SubmitRequirementInput input) throws IOException, PermissionBackendException, RestApiException { if ((srName == null || refsConfigChangeId == null) && !(srName == null && refsConfigChangeId == null)) { throw new BadRequestException( "Both 'sr-name' and 'refs-config-change-id' parameters must be set"); } SubmitRequirement requirement = srName != null && refsConfigChangeId != null ? createSubmitRequirementFromRequestParams() : createSubmitRequirement(input); SubmitRequirementResult res = evaluator.evaluateRequirement(requirement, resource.getChangeData()); return Response.ok(SubmitRequirementsJson.toInfo(requirement, res)); } private SubmitRequirement createSubmitRequirement(SubmitRequirementInput input) throws BadRequestException { validateSubmitRequirementInput(input); return SubmitRequirement.builder() .setName(input.name) .setDescription(Optional.ofNullable(input.description)) .setApplicabilityExpression(SubmitRequirementExpression.of(input.applicabilityExpression)) .setSubmittabilityExpression( SubmitRequirementExpression.create(input.submittabilityExpression)) .setOverrideExpression(SubmitRequirementExpression.of(input.overrideExpression)) .setAllowOverrideInChildProjects( input.allowOverrideInChildProjects == null ? true : input.allowOverrideInChildProjects) .build(); } /** * Loads the submit-requirement identified by the name {@link #srName} from the latest patch-set * of the change with ID {@link #refsConfigChangeId}. * * @return a {@link SubmitRequirement} entity. * @throws BadRequestException If {@link #refsConfigChangeId} is a non-existent change or not in * the {@link RefNames#REFS_CONFIG} branch, if the submit-requirement with name {@link * #srName} does not exist or if the server failed to load the project due to other * exceptions. */ private SubmitRequirement createSubmitRequirementFromRequestParams() throws IOException, PermissionBackendException, RestApiException { ChangeResource refsConfigChange; try { refsConfigChange = changesCollection.parse( TopLevelResource.INSTANCE, IdString.fromDecoded(refsConfigChangeId)); } catch (ResourceNotFoundException e) { throw new BadRequestException( String.format("Change '%s' does not exist", refsConfigChangeId), e); } ChangeNotes notes = changeNotesFactory.createCheckedUsingIndexLookup(refsConfigChange.getId()); ChangeData changeData = changeDataFactory.create(notes); try (Repository git = repoManager.openRepository(changeData.project())) { if (!changeData.change().getDest().branch().equals(RefNames.REFS_CONFIG)) { throw new BadRequestException( String.format("Change '%s' is not in refs/meta/config branch.", refsConfigChangeId)); } ObjectId revisionId = changeData.currentPatchSet().commitId(); ProjectConfig cfg = projectConfigFactory.create(changeData.project()); try { cfg.load(git, revisionId); } catch (ConfigInvalidException e) { throw new ResourceConflictException( String.format( "Failed to load project config for change '%s' from revision '%s'", refsConfigChangeId, revisionId), e); } List> submitRequirements = cfg.getSubmitRequirementSections().entrySet().stream() .filter(entry -> entry.getKey().equals(srName)) .collect(Collectors.toList()); if (submitRequirements.isEmpty()) { throw new BadRequestException( String.format("No submit requirement matching name '%s'", srName)); } return Iterables.getOnlyElement(submitRequirements).getValue(); } } private void validateSubmitRequirementInput(SubmitRequirementInput input) throws BadRequestException { if (input.name == null) { throw new BadRequestException("Field 'name' is missing from input."); } if (input.submittabilityExpression == null) { throw new BadRequestException("Field 'submittability_expression' is missing from input."); } } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy