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

com.google.gerrit.server.change.RebaseChangeOp Maven / Gradle / Ivy

There is a newer version: 3.11.1
Show newest version
// Copyright (C) 2015 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.change;

import static com.google.common.base.Preconditions.checkState;
import static com.google.common.collect.ImmutableSet.toImmutableSet;
import static com.google.gerrit.server.project.ProjectCache.illegalState;

import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableSet;
import com.google.gerrit.entities.PatchSet;
import com.google.gerrit.extensions.restapi.BadRequestException;
import com.google.gerrit.extensions.restapi.MergeConflictException;
import com.google.gerrit.extensions.restapi.ResourceConflictException;
import com.google.gerrit.extensions.restapi.RestApiException;
import com.google.gerrit.server.ChangeUtil;
import com.google.gerrit.server.CurrentUser;
import com.google.gerrit.server.IdentifiedUser;
import com.google.gerrit.server.change.RebaseUtil.Base;
import com.google.gerrit.server.git.CodeReviewCommit;
import com.google.gerrit.server.git.CodeReviewCommit.CodeReviewRevWalk;
import com.google.gerrit.server.git.GroupCollector;
import com.google.gerrit.server.git.MergeUtil;
import com.google.gerrit.server.notedb.ChangeNotes;
import com.google.gerrit.server.permissions.PermissionBackendException;
import com.google.gerrit.server.project.InvalidChangeOperationException;
import com.google.gerrit.server.project.NoSuchChangeException;
import com.google.gerrit.server.project.ProjectCache;
import com.google.gerrit.server.project.ProjectState;
import com.google.gerrit.server.update.BatchUpdateOp;
import com.google.gerrit.server.update.ChangeContext;
import com.google.gerrit.server.update.PostUpdateContext;
import com.google.gerrit.server.update.RepoContext;
import com.google.inject.Inject;
import com.google.inject.assistedinject.Assisted;
import java.io.IOException;
import java.util.List;
import java.util.Map;
import org.eclipse.jgit.diff.Sequence;
import org.eclipse.jgit.dircache.DirCache;
import org.eclipse.jgit.lib.CommitBuilder;
import org.eclipse.jgit.lib.ObjectId;
import org.eclipse.jgit.lib.PersonIdent;
import org.eclipse.jgit.merge.MergeResult;
import org.eclipse.jgit.merge.ResolveMerger;
import org.eclipse.jgit.merge.ThreeWayMerger;
import org.eclipse.jgit.revwalk.RevCommit;
import org.eclipse.jgit.revwalk.RevWalk;

/**
 * BatchUpdate operation that rebases a change.
 *
 * 

Can only be executed in a {@link com.google.gerrit.server.update.BatchUpdate} set has a {@link * CodeReviewRevWalk} set as {@link RevWalk} (set via {@link * com.google.gerrit.server.update.BatchUpdate#setRepository(org.eclipse.jgit.lib.Repository, * RevWalk, org.eclipse.jgit.lib.ObjectInserter)}). */ public class RebaseChangeOp implements BatchUpdateOp { public interface Factory { RebaseChangeOp create(ChangeNotes notes, PatchSet originalPatchSet, ObjectId baseCommitId); } private final PatchSetInserter.Factory patchSetInserterFactory; private final MergeUtil.Factory mergeUtilFactory; private final RebaseUtil rebaseUtil; private final ChangeResource.Factory changeResourceFactory; private final ChangeNotes notes; private final PatchSet originalPatchSet; private final IdentifiedUser.GenericFactory identifiedUserFactory; private final ProjectCache projectCache; private ObjectId baseCommitId; private PersonIdent committerIdent; private boolean fireRevisionCreated = true; private boolean validate = true; private boolean checkAddPatchSetPermission = true; private boolean forceContentMerge; private boolean allowConflicts; private boolean detailedCommitMessage; private boolean postMessage = true; private boolean sendEmail = true; private boolean storeCopiedVotes = true; private boolean matchAuthorToCommitterDate = false; private CodeReviewCommit rebasedCommit; private PatchSet.Id rebasedPatchSetId; private PatchSetInserter patchSetInserter; private PatchSet rebasedPatchSet; @Inject RebaseChangeOp( PatchSetInserter.Factory patchSetInserterFactory, MergeUtil.Factory mergeUtilFactory, RebaseUtil rebaseUtil, ChangeResource.Factory changeResourceFactory, IdentifiedUser.GenericFactory identifiedUserFactory, ProjectCache projectCache, @Assisted ChangeNotes notes, @Assisted PatchSet originalPatchSet, @Assisted ObjectId baseCommitId) { this.patchSetInserterFactory = patchSetInserterFactory; this.mergeUtilFactory = mergeUtilFactory; this.rebaseUtil = rebaseUtil; this.changeResourceFactory = changeResourceFactory; this.identifiedUserFactory = identifiedUserFactory; this.projectCache = projectCache; this.notes = notes; this.originalPatchSet = originalPatchSet; this.baseCommitId = baseCommitId; } public RebaseChangeOp setCommitterIdent(PersonIdent committerIdent) { this.committerIdent = committerIdent; return this; } public RebaseChangeOp setValidate(boolean validate) { this.validate = validate; return this; } public RebaseChangeOp setCheckAddPatchSetPermission(boolean checkAddPatchSetPermission) { this.checkAddPatchSetPermission = checkAddPatchSetPermission; return this; } public RebaseChangeOp setFireRevisionCreated(boolean fireRevisionCreated) { this.fireRevisionCreated = fireRevisionCreated; return this; } public RebaseChangeOp setForceContentMerge(boolean forceContentMerge) { this.forceContentMerge = forceContentMerge; return this; } /** * Allows the rebase to succeed if there are conflicts. * *

This setting requires that {@link #forceContentMerge} is set {@code true}. If {@link * #forceContentMerge} is {@code false} this setting has no effect. * * @see #setForceContentMerge(boolean) */ public RebaseChangeOp setAllowConflicts(boolean allowConflicts) { this.allowConflicts = allowConflicts; return this; } public RebaseChangeOp setDetailedCommitMessage(boolean detailedCommitMessage) { this.detailedCommitMessage = detailedCommitMessage; return this; } public RebaseChangeOp setPostMessage(boolean postMessage) { this.postMessage = postMessage; return this; } /** * We always want to store copied votes except when the change is getting submitted and a new * patch-set is created on submit (using submit strategies such as "REBASE_ALWAYS"). In such * cases, we already store the votes of the new patch-sets in SubmitStrategyOp#saveApprovals. We * should not also store the copied votes. */ public RebaseChangeOp setStoreCopiedVotes(boolean storeCopiedVotes) { this.storeCopiedVotes = storeCopiedVotes; return this; } public RebaseChangeOp setSendEmail(boolean sendEmail) { this.sendEmail = sendEmail; return this; } public RebaseChangeOp setMatchAuthorToCommitterDate(boolean matchAuthorToCommitterDate) { this.matchAuthorToCommitterDate = matchAuthorToCommitterDate; return this; } @Override public void updateRepo(RepoContext ctx) throws MergeConflictException, InvalidChangeOperationException, RestApiException, IOException, NoSuchChangeException, PermissionBackendException { // Ok that originalPatchSet was not read in a transaction, since we just // need its revision. RevWalk rw = ctx.getRevWalk(); RevCommit original = rw.parseCommit(originalPatchSet.commitId()); rw.parseBody(original); RevCommit baseCommit = rw.parseCommit(baseCommitId); CurrentUser changeOwner = identifiedUserFactory.create(notes.getChange().getOwner()); String newCommitMessage; if (detailedCommitMessage) { rw.parseBody(baseCommit); newCommitMessage = newMergeUtil() .createCommitMessageOnSubmit(original, baseCommit, notes, originalPatchSet.id()); } else { newCommitMessage = original.getFullMessage(); } rebasedCommit = rebaseCommit(ctx, original, baseCommit, newCommitMessage); Base base = rebaseUtil.parseBase( new RevisionResource( changeResourceFactory.create(notes, changeOwner), originalPatchSet), baseCommitId.name()); rebasedPatchSetId = ChangeUtil.nextPatchSetIdFromChangeRefs( ctx.getRepoView().getRefs(originalPatchSet.id().changeId().toRefPrefix()).keySet(), notes.getChange().currentPatchSetId()); patchSetInserter = patchSetInserterFactory .create(notes, rebasedPatchSetId, rebasedCommit) .setDescription("Rebase") .setFireRevisionCreated(fireRevisionCreated) .setCheckAddPatchSetPermission(checkAddPatchSetPermission) .setValidate(validate) .setSendEmail(sendEmail) // The votes are automatically copied and they don't count as copied votes. See // method's javadoc. .setStoreCopiedVotes(storeCopiedVotes); if (!rebasedCommit.getFilesWithGitConflicts().isEmpty() && !notes.getChange().isWorkInProgress()) { patchSetInserter.setWorkInProgress(true); } if (postMessage) { patchSetInserter.setMessage( messageForRebasedChange(rebasedPatchSetId, originalPatchSet.id(), rebasedCommit)); } if (base != null && !base.notes().getChange().isMerged()) { if (!base.notes().getChange().isMerged()) { // Add to end of relation chain for open base change. patchSetInserter.setGroups(base.patchSet().groups()); } else { // If the base is merged, start a new relation chain. patchSetInserter.setGroups(GroupCollector.getDefaultGroups(rebasedCommit)); } } ctx.getRevWalk().getObjectReader().getCreatedFromInserter().flush(); patchSetInserter.updateRepo(ctx); } private static String messageForRebasedChange( PatchSet.Id rebasePatchSetId, PatchSet.Id originalPatchSetId, CodeReviewCommit commit) { StringBuilder stringBuilder = new StringBuilder( String.format( "Patch Set %d: Patch Set %d was rebased", rebasePatchSetId.get(), originalPatchSetId.get())); if (!commit.getFilesWithGitConflicts().isEmpty()) { stringBuilder.append("\n\nThe following files contain Git conflicts:\n"); commit.getFilesWithGitConflicts().stream() .sorted() .forEach(filePath -> stringBuilder.append("* ").append(filePath).append("\n")); } return stringBuilder.toString(); } @Override public boolean updateChange(ChangeContext ctx) throws ResourceConflictException, IOException, BadRequestException { boolean ret = patchSetInserter.updateChange(ctx); rebasedPatchSet = patchSetInserter.getPatchSet(); return ret; } @Override public void postUpdate(PostUpdateContext ctx) { patchSetInserter.postUpdate(ctx); } public CodeReviewCommit getRebasedCommit() { checkState(rebasedCommit != null, "getRebasedCommit() only valid after updateRepo"); return rebasedCommit; } public PatchSet.Id getPatchSetId() { checkState(rebasedPatchSetId != null, "getPatchSetId() only valid after updateRepo"); return rebasedPatchSetId; } public PatchSet getPatchSet() { checkState(rebasedPatchSet != null, "getPatchSet() only valid after executing update"); return rebasedPatchSet; } private MergeUtil newMergeUtil() { ProjectState project = projectCache.get(notes.getProjectName()).orElseThrow(illegalState(notes.getProjectName())); return forceContentMerge ? mergeUtilFactory.create(project, true) : mergeUtilFactory.create(project); } /** * Rebase a commit. * * @param ctx repo context. * @param original the commit to rebase. * @param base base to rebase against. * @return the rebased commit. * @throws MergeConflictException the rebase failed due to a merge conflict. * @throws IOException the merge failed for another reason. */ private CodeReviewCommit rebaseCommit( RepoContext ctx, RevCommit original, ObjectId base, String commitMessage) throws ResourceConflictException, IOException { RevCommit parentCommit = original.getParent(0); if (base.equals(parentCommit)) { throw new ResourceConflictException("Change is already up to date."); } ThreeWayMerger merger = newMergeUtil().newThreeWayMerger(ctx.getInserter(), ctx.getRepoView().getConfig()); merger.setBase(parentCommit); DirCache dc = DirCache.newInCore(); if (allowConflicts && merger instanceof ResolveMerger) { // The DirCache must be set on ResolveMerger before calling // ResolveMerger#merge(AnyObjectId...) otherwise the entries in DirCache don't get populated. ((ResolveMerger) merger).setDirCache(dc); } boolean success = merger.merge(original, base); ObjectId tree; ImmutableSet filesWithGitConflicts; if (success) { filesWithGitConflicts = null; tree = merger.getResultTreeId(); } else { List conflicts = ImmutableList.of(); if (merger instanceof ResolveMerger) { conflicts = ((ResolveMerger) merger).getUnmergedPaths(); } if (!allowConflicts || !(merger instanceof ResolveMerger)) { throw new MergeConflictException( "The change could not be rebased due to a conflict during merge.\n\n" + MergeUtil.createConflictMessage(conflicts)); } Map> mergeResults = ((ResolveMerger) merger).getMergeResults(); filesWithGitConflicts = mergeResults.entrySet().stream() .filter(e -> e.getValue().containsConflicts()) .map(Map.Entry::getKey) .collect(toImmutableSet()); tree = MergeUtil.mergeWithConflicts( ctx.getRevWalk(), ctx.getInserter(), dc, "PATCH SET", original, "BASE", ctx.getRevWalk().parseCommit(base), mergeResults); } CommitBuilder cb = new CommitBuilder(); cb.setTreeId(tree); cb.setParentId(base); cb.setAuthor(original.getAuthorIdent()); cb.setMessage(commitMessage); if (committerIdent != null) { cb.setCommitter(committerIdent); } else { cb.setCommitter(ctx.getIdentifiedUser().newCommitterIdent(ctx.getWhen(), ctx.getTimeZone())); } if (matchAuthorToCommitterDate) { cb.setAuthor( new PersonIdent( cb.getAuthor(), cb.getCommitter().getWhen(), cb.getCommitter().getTimeZone())); } ObjectId objectId = ctx.getInserter().insert(cb); ctx.getInserter().flush(); CodeReviewCommit commit = ((CodeReviewRevWalk) ctx.getRevWalk()).parseCommit(objectId); commit.setFilesWithGitConflicts(filesWithGitConflicts); return commit; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy