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.");
    }
  }
}