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

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

The newest version!
// Copyright (C) 2022 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 static com.google.gerrit.server.project.ProjectCache.illegalState;

import com.google.gerrit.common.RawInputUtil;
import com.google.gerrit.entities.Comment.Range;
import com.google.gerrit.entities.FixReplacement;
import com.google.gerrit.entities.PatchSet;
import com.google.gerrit.entities.Project;
import com.google.gerrit.extensions.api.changes.ApplyPatchInput;
import com.google.gerrit.extensions.common.ApplyProvidedFixInput;
import com.google.gerrit.extensions.common.EditInfo;
import com.google.gerrit.extensions.restapi.AuthException;
import com.google.gerrit.extensions.restapi.BadRequestException;
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.server.change.RevisionResource;
import com.google.gerrit.server.edit.ChangeEdit;
import com.google.gerrit.server.edit.ChangeEditJson;
import com.google.gerrit.server.edit.ChangeEditModifier;
import com.google.gerrit.server.edit.CommitModification;
import com.google.gerrit.server.edit.tree.ChangeFileContentModification;
import com.google.gerrit.server.fixes.FixReplacementInterpreter;
import com.google.gerrit.server.git.GitRepositoryManager;
import com.google.gerrit.server.notedb.ChangeNotes;
import com.google.gerrit.server.patch.ApplyPatchUtil;
import com.google.gerrit.server.patch.MagicFile;
import com.google.gerrit.server.permissions.PermissionBackendException;
import com.google.gerrit.server.project.InvalidChangeOperationException;
import com.google.gerrit.server.project.ProjectCache;
import com.google.gerrit.server.project.ProjectState;
import com.google.inject.Inject;
import com.google.inject.Singleton;
import java.io.IOException;
import java.util.List;
import java.util.stream.Collectors;
import org.eclipse.jgit.lib.ObjectId;
import org.eclipse.jgit.lib.ObjectInserter;
import org.eclipse.jgit.lib.ObjectLoader;
import org.eclipse.jgit.lib.Repository;
import org.eclipse.jgit.patch.PatchApplier;
import org.eclipse.jgit.patch.PatchApplier.Result.Error;
import org.eclipse.jgit.revwalk.RevCommit;
import org.eclipse.jgit.revwalk.RevWalk;
import org.eclipse.jgit.treewalk.TreeWalk;

/** Applies a fix that is provided as part of the request body. */
@Singleton
public class ApplyProvidedFix implements RestModifyView {
  private final GitRepositoryManager gitRepositoryManager;
  private final FixReplacementInterpreter fixReplacementInterpreter;
  private final ChangeEditModifier changeEditModifier;
  private final ChangeEditJson changeEditJson;
  private final ProjectCache projectCache;

  @Inject
  public ApplyProvidedFix(
      GitRepositoryManager gitRepositoryManager,
      FixReplacementInterpreter fixReplacementInterpreter,
      ChangeEditModifier changeEditModifier,
      ChangeEditJson changeEditJson,
      ProjectCache projectCache) {
    this.gitRepositoryManager = gitRepositoryManager;
    this.fixReplacementInterpreter = fixReplacementInterpreter;
    this.changeEditModifier = changeEditModifier;
    this.changeEditJson = changeEditJson;
    this.projectCache = projectCache;
  }

  @Override
  public Response apply(
      RevisionResource revisionResource, ApplyProvidedFixInput applyProvidedFixInput)
      throws AuthException,
          BadRequestException,
          ResourceConflictException,
          IOException,
          ResourceNotFoundException,
          PermissionBackendException,
          RestApiException {
    if (applyProvidedFixInput == null) {
      throw new BadRequestException("applyProvidedFixInput is required");
    }
    if (applyProvidedFixInput.fixReplacementInfos == null) {
      throw new BadRequestException("applyProvidedFixInput.fixReplacementInfos is required");
    }
    Project.NameKey project = revisionResource.getProject();
    ProjectState projectState = projectCache.get(project).orElseThrow(illegalState(project));
    PatchSet targetPatchSet = revisionResource.getPatchSet();

    ChangeNotes changeNotes = revisionResource.getNotes();
    PatchSet originPatchSetForFix =
        applyProvidedFixInput.originalPatchsetForFix != null
                && applyProvidedFixInput.originalPatchsetForFix > 0
            ? changeNotes
                .getPatchSets()
                .get(
                    PatchSet.id(
                        revisionResource.getChange().getId(),
                        applyProvidedFixInput.originalPatchsetForFix))
            : targetPatchSet;

    List fixReplacements =
        applyProvidedFixInput.fixReplacementInfos.stream()
            .map(fix -> new FixReplacement(fix.path, new Range(fix.range), fix.replacement))
            .collect(Collectors.toList());

    try (Repository repository = gitRepositoryManager.openRepository(project)) {
      CommitModification commitModification =
          getCommitModification(
              repository, projectState, originPatchSetForFix, targetPatchSet, fixReplacements);
      ChangeEdit changeEdit =
          changeEditModifier.combineWithModifiedPatchSetTree(
              repository, changeNotes, targetPatchSet, commitModification);

      return Response.ok(changeEditJson.toEditInfo(changeEdit, false));
    } catch (InvalidChangeOperationException e) {
      throw new ResourceConflictException(e.getMessage());
    }
  }

  /**
   * Returns CommitModification for fixes and rebase it if the fix is for an older patchset.
   *
   * 

The method creates CommitModification by applying {@code fixReplacements} to the {@code * basePatchSetForFix}. If the {@code targetPatchSetForFix} is different from the {@code * basePatchSetForFix}, CommitModification is created from the {@link PatchApplier.Result}, after * applying the patch generated from {@code basePatchSetForFix} to the {@code * targetPatchSetForFix}. * *

Note: if there is a fix for a commit message and commit messages are different in {@code * basePatchSetForFix} and {@code targetPatchSetForFix}, the method can't move the fix to the * {@code targetPatchSetForFix} and throws {@link ResourceConflictException}. This limitations * exists because the method uses ApplyPatchUtil which operates only on files. */ private CommitModification getCommitModification( Repository repository, ProjectState projectState, PatchSet basePatchSetForFix, PatchSet targetPatchSetForFix, List fixReplacements) throws IOException, InvalidChangeOperationException, RestApiException { CommitModification originCommitModification = fixReplacementInterpreter.toCommitModification( repository, projectState, basePatchSetForFix.commitId(), fixReplacements); if (basePatchSetForFix.id().equals(targetPatchSetForFix.id())) { return originCommitModification; } RevCommit originCommit = repository.parseCommit(basePatchSetForFix.commitId()); ObjectId newTreeId = ChangeEditModifier.createNewTree( repository, originCommit, originCommitModification.treeModifications()); CommitModification.Builder resultBuilder = CommitModification.builder(); String patch; try (RevWalk rw = new RevWalk(repository)) { ObjectId targetCommit = targetPatchSetForFix.commitId(); if (originCommitModification.newCommitMessage().isPresent()) { MagicFile originCommitMessageFile = MagicFile.forCommitMessage(rw.getObjectReader(), originCommit); String originCommitMessage = originCommitMessageFile.modifiableContent(); MagicFile targetCommitMessageFile = MagicFile.forCommitMessage(rw.getObjectReader(), targetCommit); String targetCommitMessage = targetCommitMessageFile.modifiableContent(); if (!originCommitMessage.equals(targetCommitMessage)) { throw new ResourceConflictException( "The fix attempts to modify commit message of an older patchset, but commit message" + " has been updated in a newer patchset. The fix can't be applied."); } resultBuilder.newCommitMessage(originCommitModification.newCommitMessage().get()); } patch = ApplyPatchUtil.getResultPatch( repository, repository.newObjectReader(), originCommit, rw.lookupTree(newTreeId)); PatchApplier.Result result; try (ObjectInserter oi = repository.newObjectInserter()) { ApplyPatchInput inp = new ApplyPatchInput(); inp.patch = patch; // Allow conflicts for showing more precise error message. inp.allowConflicts = true; result = ApplyPatchUtil.applyPatch(repository, oi, inp, repository.parseCommit(targetCommit)); if (!result.getErrors().isEmpty()) { String errorMessage = (result.getErrors().stream().anyMatch(Error::isGitConflict) ? "Merge conflict while applying a fix:\n" : "Error while applying a fix:\n") + result.getErrors().stream() .map(Error::toString) .collect(Collectors.joining("\n")); throw new ResourceConflictException(errorMessage); } oi.flush(); for (String path : result.getPaths()) { try (TreeWalk tw = TreeWalk.forPath(rw.getObjectReader(), path, result.getTreeId())) { ObjectLoader loader = rw.getObjectReader().open(tw.getObjectId(0)); resultBuilder.addTreeModification( new ChangeFileContentModification(path, RawInputUtil.create(loader.getBytes()))); } } } } return resultBuilder.build(); } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy