Merge "Recover from merges that failed after updateRepo"
This commit is contained in:
		| @@ -219,22 +219,22 @@ public abstract class AbstractSubmit extends AbstractDaemonTest { | ||||
|   } | ||||
|  | ||||
|   protected void submit(String changeId) throws Exception { | ||||
|     submit(changeId, new SubmitInput(), null, null); | ||||
|     submit(changeId, new SubmitInput(), null, null, true); | ||||
|   } | ||||
|  | ||||
|   protected void submit(String changeId, SubmitInput input) throws Exception { | ||||
|     submit(changeId, input, null, null); | ||||
|     submit(changeId, input, null, null, true); | ||||
|   } | ||||
|  | ||||
|   protected void submitWithConflict(String changeId, | ||||
|       String expectedError) throws Exception { | ||||
|     submit(changeId, new SubmitInput(), ResourceConflictException.class, | ||||
|         expectedError); | ||||
|         expectedError, true); | ||||
|   } | ||||
|  | ||||
|   private void submit(String changeId, SubmitInput input, | ||||
|   protected void submit(String changeId, SubmitInput input, | ||||
|       Class<? extends RestApiException> expectedExceptionType, | ||||
|       String expectedExceptionMsg) throws Exception { | ||||
|       String expectedExceptionMsg, boolean checkMergeResult) throws Exception { | ||||
|     approve(changeId); | ||||
|     try { | ||||
|       gApi.changes().id(changeId).current().submit(input); | ||||
| @@ -258,7 +258,9 @@ public abstract class AbstractSubmit extends AbstractDaemonTest { | ||||
|     } | ||||
|     ChangeInfo change = gApi.changes().id(changeId).info(); | ||||
|     assertThat(change.status).isEqualTo(ChangeStatus.MERGED); | ||||
|     checkMergeResult(change); | ||||
|     if (checkMergeResult) { | ||||
|       checkMergeResult(change); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   private void checkMergeResult(ChangeInfo change) throws Exception { | ||||
|   | ||||
| @@ -17,11 +17,23 @@ package com.google.gerrit.acceptance.rest.change; | ||||
| import static com.google.common.truth.Truth.assertThat; | ||||
| import static com.google.common.truth.TruthJUnit.assume; | ||||
|  | ||||
| import com.google.common.collect.Iterables; | ||||
| import com.google.gerrit.acceptance.PushOneCommit; | ||||
| import com.google.gerrit.acceptance.TestProjectInput; | ||||
| import com.google.gerrit.extensions.api.changes.SubmitInput; | ||||
| import com.google.gerrit.extensions.client.ChangeStatus; | ||||
| import com.google.gerrit.extensions.client.InheritableBoolean; | ||||
| import com.google.gerrit.extensions.common.ChangeInfo; | ||||
| import com.google.gerrit.extensions.common.ChangeMessageInfo; | ||||
| import com.google.gerrit.extensions.restapi.ResourceConflictException; | ||||
| import com.google.gerrit.reviewdb.client.Change; | ||||
| import com.google.gerrit.reviewdb.client.PatchSet; | ||||
| import com.google.gerrit.server.change.Submit.TestSubmitInput; | ||||
|  | ||||
| import org.eclipse.jgit.lib.ObjectId; | ||||
| import org.eclipse.jgit.lib.Repository; | ||||
| import org.eclipse.jgit.revwalk.RevCommit; | ||||
| import org.eclipse.jgit.revwalk.RevWalk; | ||||
| import org.junit.Test; | ||||
|  | ||||
| public abstract class AbstractSubmitByMerge extends AbstractSubmit { | ||||
| @@ -118,4 +130,58 @@ public abstract class AbstractSubmitByMerge extends AbstractSubmit { | ||||
|     assertThat(head.getParent(0)).isEqualTo(change1.getCommit()); | ||||
|     assertThat(head.getParent(1)).isEqualTo(change2.getCommit()); | ||||
|   } | ||||
|  | ||||
|   @Test | ||||
|   public void repairChangeStateAfterFailure() throws Exception { | ||||
|     RevCommit initialHead = getRemoteHead(); | ||||
|     PushOneCommit.Result change = | ||||
|         createChange("Change 1", "a.txt", "content"); | ||||
|     submit(change.getChangeId()); | ||||
|     RevCommit afterChange1Head = getRemoteHead(); | ||||
|  | ||||
|     testRepo.reset(initialHead); | ||||
|     PushOneCommit.Result change2 = | ||||
|         createChange("Change 2", "b.txt", "other content"); | ||||
|     Change.Id id2 = change2.getChange().getId(); | ||||
|     SubmitInput failAfterRefUpdates = | ||||
|         new TestSubmitInput(new SubmitInput(), true); | ||||
|     submit(change2.getChangeId(), failAfterRefUpdates, | ||||
|         ResourceConflictException.class, "Failing after ref updates", true); | ||||
|  | ||||
|     // Bad: ref advanced but change wasn't updated. | ||||
|     PatchSet.Id psId1 = new PatchSet.Id(id2, 1); | ||||
|     ChangeInfo info = gApi.changes().id(id2.get()).get(); | ||||
|     assertThat(info.status).isEqualTo(ChangeStatus.NEW); | ||||
|     assertThat(info.revisions.get(info.currentRevision)._number).isEqualTo(1); | ||||
|     ChangeMessageInfo lastMessage = Iterables.getLast(info.messages); | ||||
|  | ||||
|     RevCommit tip; | ||||
|     try (Repository repo = repoManager.openRepository(project); | ||||
|         RevWalk rw = new RevWalk(repo)) { | ||||
|       ObjectId rev1 = repo.exactRef(psId1.toRefName()).getObjectId(); | ||||
|       assertThat(rev1).isNotNull(); | ||||
|  | ||||
|       tip = rw.parseCommit(repo.exactRef("refs/heads/master").getObjectId()); | ||||
|       assertThat(tip.getParentCount()).isEqualTo(2); | ||||
|       assertThat(tip.getParent(0)).isEqualTo(afterChange1Head); | ||||
|       assertThat(tip.getParent(1)).isEqualTo(change2.getCommit()); | ||||
|     } | ||||
|  | ||||
|     // Skip checking the merge result; in the fixup case, the newRev in | ||||
|     // ChangeMergedEvent won't match the current branch tip. | ||||
|     submit(change2.getChangeId(), new SubmitInput(), null, null, false); | ||||
|  | ||||
|     // Change status and patch set entities were updated, and branch tip stayed | ||||
|     // the same. | ||||
|     info = gApi.changes().id(id2.get()).get(); | ||||
|     assertThat(info.status).isEqualTo(ChangeStatus.MERGED); | ||||
|     assertThat(info.revisions.get(info.currentRevision)._number).isEqualTo(1); | ||||
|     assertThat(Iterables.getLast(info.messages).message) | ||||
|         .isEqualTo(lastMessage.message); | ||||
|  | ||||
|     try (Repository repo = repoManager.openRepository(project)) { | ||||
|       assertThat(repo.exactRef("refs/heads/master").getObjectId()) | ||||
|           .isEqualTo(tip); | ||||
|     } | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -16,15 +16,25 @@ package com.google.gerrit.acceptance.rest.change; | ||||
|  | ||||
| import static com.google.common.truth.Truth.assertThat; | ||||
|  | ||||
| import com.google.common.collect.Iterables; | ||||
| import com.google.gerrit.acceptance.PushOneCommit; | ||||
| import com.google.gerrit.acceptance.TestProjectInput; | ||||
| import com.google.gerrit.extensions.api.changes.SubmitInput; | ||||
| import com.google.gerrit.extensions.client.ChangeStatus; | ||||
| import com.google.gerrit.extensions.client.InheritableBoolean; | ||||
| import com.google.gerrit.extensions.client.ListChangesOption; | ||||
| import com.google.gerrit.extensions.client.SubmitType; | ||||
| import com.google.gerrit.extensions.common.ChangeInfo; | ||||
| import com.google.gerrit.extensions.common.ChangeMessageInfo; | ||||
| import com.google.gerrit.extensions.restapi.ResourceConflictException; | ||||
| import com.google.gerrit.reviewdb.client.Change; | ||||
| import com.google.gerrit.reviewdb.client.PatchSet; | ||||
| import com.google.gerrit.server.change.Submit.TestSubmitInput; | ||||
|  | ||||
| import org.eclipse.jgit.lib.ObjectId; | ||||
| import org.eclipse.jgit.lib.Repository; | ||||
| import org.eclipse.jgit.revwalk.RevCommit; | ||||
| import org.eclipse.jgit.revwalk.RevWalk; | ||||
| import org.junit.Test; | ||||
|  | ||||
| import java.util.List; | ||||
| @@ -257,4 +267,64 @@ public class SubmitByCherryPickIT extends AbstractSubmit { | ||||
|     assertNew(change2.getChangeId()); | ||||
|     assertNew(change3.getChangeId()); | ||||
|   } | ||||
|  | ||||
|   @Test | ||||
|   public void repairChangeStateAfterFailure() throws Exception { | ||||
|     RevCommit initialHead = getRemoteHead(); | ||||
|     PushOneCommit.Result change = | ||||
|         createChange("Change 1", "a.txt", "content"); | ||||
|     submit(change.getChangeId()); | ||||
|  | ||||
|     RevCommit oldHead = getRemoteHead(); | ||||
|     testRepo.reset(initialHead); | ||||
|     PushOneCommit.Result change2 = | ||||
|         createChange("Change 2", "b.txt", "other content"); | ||||
|     Change.Id id2 = change2.getChange().getId(); | ||||
|     SubmitInput failAfterRefUpdates = | ||||
|         new TestSubmitInput(new SubmitInput(), true); | ||||
|     submit(change2.getChangeId(), failAfterRefUpdates, | ||||
|         ResourceConflictException.class, "Failing after ref updates", true); | ||||
|  | ||||
|     // Bad: ref advanced but change wasn't updated. | ||||
|     PatchSet.Id psId1 = new PatchSet.Id(id2, 1); | ||||
|     PatchSet.Id psId2 = new PatchSet.Id(id2, 2); | ||||
|     ChangeInfo info = gApi.changes().id(id2.get()).get(); | ||||
|     assertThat(info.status).isEqualTo(ChangeStatus.NEW); | ||||
|     assertThat(info.revisions.get(info.currentRevision)._number).isEqualTo(1); | ||||
|     assertThat(getPatchSet(psId2)).isNull(); | ||||
|     ChangeMessageInfo lastMessage = Iterables.getLast(info.messages); | ||||
|  | ||||
|     ObjectId rev2; | ||||
|     try (Repository repo = repoManager.openRepository(project); | ||||
|         RevWalk rw = new RevWalk(repo)) { | ||||
|       ObjectId rev1 = repo.exactRef(psId1.toRefName()).getObjectId(); | ||||
|       assertThat(rev1).isNotNull(); | ||||
|  | ||||
|       rev2 = repo.exactRef(psId2.toRefName()).getObjectId(); | ||||
|       assertThat(rev2).isNotNull(); | ||||
|       assertThat(rev2).isNotEqualTo(rev1); | ||||
|       assertThat(rw.parseCommit(rev2).getParent(0)).isEqualTo(oldHead); | ||||
|  | ||||
|       assertThat(repo.exactRef("refs/heads/master").getObjectId()) | ||||
|           .isEqualTo(rev2); | ||||
|     } | ||||
|  | ||||
|     submit(change2.getChangeId()); | ||||
|  | ||||
|     // Change status and patch set entities were updated, and branch tip stayed | ||||
|     // the same. | ||||
|     info = gApi.changes().id(id2.get()).get(); | ||||
|     assertThat(info.status).isEqualTo(ChangeStatus.MERGED); | ||||
|     assertThat(info.revisions.get(info.currentRevision)._number).isEqualTo(2); | ||||
|     PatchSet ps2 = getPatchSet(psId2); | ||||
|     assertThat(ps2).isNotNull(); | ||||
|     assertThat(ps2.getRevision().get()).isEqualTo(rev2.name()); | ||||
|     assertThat(Iterables.getLast(info.messages).message) | ||||
|         .isEqualTo(lastMessage.message); | ||||
|  | ||||
|     try (Repository repo = repoManager.openRepository(project)) { | ||||
|       assertThat(repo.exactRef("refs/heads/master").getObjectId()) | ||||
|           .isEqualTo(rev2); | ||||
|     } | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -16,12 +16,23 @@ package com.google.gerrit.acceptance.rest.change; | ||||
|  | ||||
| import static com.google.common.truth.Truth.assertThat; | ||||
|  | ||||
| import com.google.common.collect.Iterables; | ||||
| import com.google.gerrit.acceptance.PushOneCommit; | ||||
| import com.google.gerrit.extensions.api.changes.SubmitInput; | ||||
| import com.google.gerrit.extensions.client.ChangeStatus; | ||||
| import com.google.gerrit.extensions.client.SubmitType; | ||||
| import com.google.gerrit.extensions.common.ActionInfo; | ||||
| import com.google.gerrit.extensions.common.ChangeInfo; | ||||
| import com.google.gerrit.extensions.common.ChangeMessageInfo; | ||||
| import com.google.gerrit.extensions.restapi.ResourceConflictException; | ||||
| import com.google.gerrit.reviewdb.client.Change; | ||||
| import com.google.gerrit.reviewdb.client.PatchSet; | ||||
| import com.google.gerrit.server.change.Submit.TestSubmitInput; | ||||
|  | ||||
| import org.eclipse.jgit.lib.ObjectId; | ||||
| import org.eclipse.jgit.lib.Repository; | ||||
| import org.eclipse.jgit.revwalk.RevCommit; | ||||
| import org.eclipse.jgit.revwalk.RevWalk; | ||||
| import org.junit.Test; | ||||
|  | ||||
| import java.util.Map; | ||||
| @@ -107,4 +118,44 @@ public class SubmitByFastForwardIT extends AbstractSubmit { | ||||
|     assertThat(getRemoteHead()).isEqualTo(oldHead); | ||||
|     assertSubmitter(change.getChangeId(), 1); | ||||
|   } | ||||
|  | ||||
|   @Test | ||||
|   public void repairChangeStateAfterFailure() throws Exception { | ||||
|     PushOneCommit.Result change = createChange("Change 1", "a.txt", "content"); | ||||
|     Change.Id id = change.getChange().getId(); | ||||
|     SubmitInput failAfterRefUpdates = | ||||
|         new TestSubmitInput(new SubmitInput(), true); | ||||
|     submit(change.getChangeId(), failAfterRefUpdates, | ||||
|         ResourceConflictException.class, "Failing after ref updates", true); | ||||
|  | ||||
|     // Bad: ref advanced but change wasn't updated. | ||||
|     PatchSet.Id psId = new PatchSet.Id(id, 1); | ||||
|     ChangeInfo info = gApi.changes().id(id.get()).get(); | ||||
|     assertThat(info.status).isEqualTo(ChangeStatus.NEW); | ||||
|     assertThat(info.revisions.get(info.currentRevision)._number).isEqualTo(1); | ||||
|     ChangeMessageInfo lastMessage = Iterables.getLast(info.messages); | ||||
|  | ||||
|     ObjectId rev; | ||||
|     try (Repository repo = repoManager.openRepository(project); | ||||
|         RevWalk rw = new RevWalk(repo)) { | ||||
|       rev = repo.exactRef(psId.toRefName()).getObjectId(); | ||||
|       assertThat(rev).isNotNull(); | ||||
|       assertThat(repo.exactRef("refs/heads/master").getObjectId()) | ||||
|           .isEqualTo(rev); | ||||
|     } | ||||
|  | ||||
|     submit(change.getChangeId()); | ||||
|  | ||||
|     // Change status was updated, and branch tip stayed the same. | ||||
|     info = gApi.changes().id(id.get()).get(); | ||||
|     assertThat(info.status).isEqualTo(ChangeStatus.MERGED); | ||||
|     assertThat(info.revisions.get(info.currentRevision)._number).isEqualTo(1); | ||||
|     assertThat(Iterables.getLast(info.messages).message) | ||||
|         .isEqualTo(lastMessage.message); | ||||
|  | ||||
|     try (Repository repo = repoManager.openRepository(project)) { | ||||
|       assertThat(repo.exactRef("refs/heads/master").getObjectId()) | ||||
|           .isEqualTo(rev); | ||||
|     } | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -16,10 +16,19 @@ package com.google.gerrit.acceptance.rest.change; | ||||
|  | ||||
| import static com.google.common.truth.Truth.assertThat; | ||||
|  | ||||
| import com.google.common.collect.Iterables; | ||||
| import com.google.gerrit.acceptance.PushOneCommit; | ||||
| import com.google.gerrit.acceptance.TestProjectInput; | ||||
| import com.google.gerrit.extensions.api.changes.SubmitInput; | ||||
| import com.google.gerrit.extensions.client.ChangeStatus; | ||||
| import com.google.gerrit.extensions.client.InheritableBoolean; | ||||
| import com.google.gerrit.extensions.client.SubmitType; | ||||
| import com.google.gerrit.extensions.common.ChangeInfo; | ||||
| import com.google.gerrit.extensions.common.ChangeMessageInfo; | ||||
| import com.google.gerrit.extensions.restapi.ResourceConflictException; | ||||
| import com.google.gerrit.reviewdb.client.Change; | ||||
| import com.google.gerrit.reviewdb.client.PatchSet; | ||||
| import com.google.gerrit.server.change.Submit.TestSubmitInput; | ||||
|  | ||||
| import org.eclipse.jgit.lib.ObjectId; | ||||
| import org.eclipse.jgit.lib.Repository; | ||||
| @@ -154,6 +163,66 @@ public class SubmitByRebaseIfNecessaryIT extends AbstractSubmit { | ||||
|     assertNoSubmitter(change2.getChangeId(), 1); | ||||
|   } | ||||
|  | ||||
|   @Test | ||||
|   public void repairChangeStateAfterFailure() throws Exception { | ||||
|     RevCommit initialHead = getRemoteHead(); | ||||
|     PushOneCommit.Result change = | ||||
|         createChange("Change 1", "a.txt", "content"); | ||||
|     submit(change.getChangeId()); | ||||
|  | ||||
|     RevCommit oldHead = getRemoteHead(); | ||||
|     testRepo.reset(initialHead); | ||||
|     PushOneCommit.Result change2 = | ||||
|         createChange("Change 2", "b.txt", "other content"); | ||||
|     Change.Id id2 = change2.getChange().getId(); | ||||
|     SubmitInput failAfterRefUpdates = | ||||
|         new TestSubmitInput(new SubmitInput(), true); | ||||
|     submit(change2.getChangeId(), failAfterRefUpdates, | ||||
|         ResourceConflictException.class, "Failing after ref updates", true); | ||||
|  | ||||
|     // Bad: ref advanced but change wasn't updated. | ||||
|     PatchSet.Id psId1 = new PatchSet.Id(id2, 1); | ||||
|     PatchSet.Id psId2 = new PatchSet.Id(id2, 2); | ||||
|     ChangeInfo info = gApi.changes().id(id2.get()).get(); | ||||
|     assertThat(info.status).isEqualTo(ChangeStatus.NEW); | ||||
|     assertThat(info.revisions.get(info.currentRevision)._number).isEqualTo(1); | ||||
|     assertThat(getPatchSet(psId2)).isNull(); | ||||
|     ChangeMessageInfo lastMessage = Iterables.getLast(info.messages); | ||||
|  | ||||
|     ObjectId rev2; | ||||
|     try (Repository repo = repoManager.openRepository(project); | ||||
|         RevWalk rw = new RevWalk(repo)) { | ||||
|       ObjectId rev1 = repo.exactRef(psId1.toRefName()).getObjectId(); | ||||
|       assertThat(rev1).isNotNull(); | ||||
|  | ||||
|       rev2 = repo.exactRef(psId2.toRefName()).getObjectId(); | ||||
|       assertThat(rev2).isNotNull(); | ||||
|       assertThat(rev2).isNotEqualTo(rev1); | ||||
|       assertThat(rw.parseCommit(rev2).getParent(0)).isEqualTo(oldHead); | ||||
|  | ||||
|       assertThat(repo.exactRef("refs/heads/master").getObjectId()) | ||||
|           .isEqualTo(rev2); | ||||
|     } | ||||
|  | ||||
|     submit(change2.getChangeId()); | ||||
|  | ||||
|     // Change status and patch set entities were updated, and branch tip stayed | ||||
|     // the same. | ||||
|     info = gApi.changes().id(id2.get()).get(); | ||||
|     assertThat(info.status).isEqualTo(ChangeStatus.MERGED); | ||||
|     assertThat(info.revisions.get(info.currentRevision)._number).isEqualTo(2); | ||||
|     PatchSet ps2 = getPatchSet(psId2); | ||||
|     assertThat(ps2).isNotNull(); | ||||
|     assertThat(ps2.getRevision().get()).isEqualTo(rev2.name()); | ||||
|     assertThat(Iterables.getLast(info.messages).message) | ||||
|         .isEqualTo(lastMessage.message); | ||||
|  | ||||
|     try (Repository repo = repoManager.openRepository(project)) { | ||||
|       assertThat(repo.exactRef("refs/heads/master").getObjectId()) | ||||
|           .isEqualTo(rev2); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   private RevCommit parse(ObjectId id) throws Exception { | ||||
|     try (Repository repo = repoManager.openRepository(project); | ||||
|         RevWalk rw = new RevWalk(repo)) { | ||||
|   | ||||
| @@ -14,6 +14,7 @@ | ||||
|  | ||||
| package com.google.gerrit.server.change; | ||||
|  | ||||
| import com.google.common.annotations.VisibleForTesting; | ||||
| import com.google.common.base.MoreObjects; | ||||
| import com.google.common.base.Predicate; | ||||
| import com.google.common.base.Strings; | ||||
| @@ -94,6 +95,22 @@ public class Submit implements RestModifyView<RevisionResource, SubmitInput>, | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   /** | ||||
|    * Subclass of {@link SubmitInput} with special bits that may be flipped for | ||||
|    * testing purposes only. | ||||
|    */ | ||||
|   @VisibleForTesting | ||||
|   public static class TestSubmitInput extends SubmitInput { | ||||
|     public final boolean failAfterRefUpdates; | ||||
|  | ||||
|     @SuppressWarnings("deprecation") | ||||
|     public TestSubmitInput(SubmitInput base, boolean failAfterRefUpdates) { | ||||
|       this.onBehalfOf = base.onBehalfOf; | ||||
|       this.waitForMerge = base.waitForMerge; | ||||
|       this.failAfterRefUpdates = failAfterRefUpdates; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   private final Provider<ReviewDb> dbProvider; | ||||
|   private final GitRepositoryManager repoManager; | ||||
|   private final ChangeData.Factory changeDataFactory; | ||||
| @@ -185,7 +202,7 @@ public class Submit implements RestModifyView<RevisionResource, SubmitInput>, | ||||
|  | ||||
|     try (MergeOp op = mergeOpProvider.get()) { | ||||
|       ReviewDb db = dbProvider.get(); | ||||
|       op.merge(db, change, caller, true); | ||||
|       op.merge(db, change, caller, true, input); | ||||
|       change = db.changes().get(change.getId()); | ||||
|     } | ||||
|  | ||||
|   | ||||
| @@ -38,6 +38,7 @@ import com.google.common.hash.Hashing; | ||||
| import com.google.gerrit.common.TimeUtil; | ||||
| import com.google.gerrit.common.data.SubmitRecord; | ||||
| import com.google.gerrit.common.data.SubmitTypeRecord; | ||||
| import com.google.gerrit.extensions.api.changes.SubmitInput; | ||||
| import com.google.gerrit.extensions.client.SubmitType; | ||||
| import com.google.gerrit.extensions.restapi.ResourceConflictException; | ||||
| import com.google.gerrit.extensions.restapi.RestApiException; | ||||
| @@ -335,6 +336,7 @@ public class MergeOp implements AutoCloseable { | ||||
|  | ||||
|   private CommitStatus commits; | ||||
|   private ReviewDb db; | ||||
|   private SubmitInput submitInput; | ||||
|  | ||||
|   @Inject | ||||
|   MergeOp(ChangeControl.GenericFactory changeControlFactory, | ||||
| @@ -544,7 +546,9 @@ public class MergeOp implements AutoCloseable { | ||||
|   } | ||||
|  | ||||
|   public void merge(ReviewDb db, Change change, IdentifiedUser caller, | ||||
|       boolean checkSubmitRules) throws OrmException, RestApiException { | ||||
|       boolean checkSubmitRules, SubmitInput submitInput) | ||||
|       throws OrmException, RestApiException { | ||||
|     this.submitInput = submitInput; | ||||
|     this.caller = caller; | ||||
|     updateSubmissionId(change); | ||||
|     this.db = db; | ||||
| @@ -627,7 +631,7 @@ public class MergeOp implements AutoCloseable { | ||||
|  | ||||
|       BatchUpdate.execute( | ||||
|           batchUpdates(projects), | ||||
|           new SubmitStrategyListener(strategies, commits)); | ||||
|           new SubmitStrategyListener(submitInput, strategies, commits)); | ||||
|  | ||||
|       SubmoduleOp subOp = subOpProvider.get(); | ||||
|       for (Branch.NameKey branch : branches) { | ||||
|   | ||||
| @@ -14,6 +14,8 @@ | ||||
|  | ||||
| package com.google.gerrit.server.git; | ||||
|  | ||||
| import static com.google.common.base.Preconditions.checkState; | ||||
|  | ||||
| import com.google.common.base.Strings; | ||||
| import com.google.common.collect.Multimap; | ||||
| import com.google.gerrit.common.data.SubmitTypeRecord; | ||||
| @@ -84,11 +86,15 @@ public class MergeSuperSet { | ||||
|       throws MissingObjectException, IncorrectObjectTypeException, IOException, | ||||
|       OrmException { | ||||
|     ChangeData cd = changeDataFactory.create(db, change.getId()); | ||||
|     ChangeSet result; | ||||
|     if (Submit.wholeTopicEnabled(cfg)) { | ||||
|       return completeChangeSetIncludingTopics(db, new ChangeSet(cd)); | ||||
|       result = completeChangeSetIncludingTopics(db, new ChangeSet(cd)); | ||||
|     } else { | ||||
|       return completeChangeSetWithoutTopic(db, new ChangeSet(cd)); | ||||
|       result = completeChangeSetWithoutTopic(db, new ChangeSet(cd)); | ||||
|     } | ||||
|     checkState(result.ids().contains(change.getId()), | ||||
|         "change %s missing from result %s", change.getId(), result); | ||||
|     return result; | ||||
|   } | ||||
|  | ||||
|   private ChangeSet completeChangeSetWithoutTopic(ReviewDb db, ChangeSet changes) | ||||
| @@ -132,8 +138,13 @@ public class MergeSuperSet { | ||||
|           } | ||||
|  | ||||
|           List<String> hashes = new ArrayList<>(); | ||||
|           // Always include the input, even if merged. This allows | ||||
|           // SubmitStrategyOp to correct the situation later. | ||||
|           hashes.add(objIdStr); | ||||
|           for (RevCommit c : rw) { | ||||
|             hashes.add(c.name()); | ||||
|             if (!c.equals(commit)) { | ||||
|               hashes.add(c.name()); | ||||
|             } | ||||
|           } | ||||
|  | ||||
|           if (!hashes.isEmpty()) { | ||||
|   | ||||
| @@ -672,13 +672,15 @@ public class MergeUtil { | ||||
|  | ||||
|   public Set<Change.Id> findUnmergedChanges(Set<Change.Id> expected, | ||||
|       CodeReviewRevWalk rw, RevFlag canMergeFlag, CodeReviewCommit oldTip, | ||||
|       CodeReviewCommit mergeTip) throws IntegrationException { | ||||
|       CodeReviewCommit mergeTip, Iterable<Change.Id> alreadyMerged) | ||||
|       throws IntegrationException { | ||||
|     if (mergeTip == null) { | ||||
|       return expected; | ||||
|     } | ||||
|  | ||||
|     try { | ||||
|       Set<Change.Id> found = Sets.newHashSetWithExpectedSize(expected.size()); | ||||
|       Iterables.addAll(found, alreadyMerged); | ||||
|       rw.resetRetain(canMergeFlag); | ||||
|       rw.sort(RevSort.TOPO); | ||||
|       rw.markStart(mergeTip); | ||||
| @@ -705,4 +707,16 @@ public class MergeUtil { | ||||
|       throw new IntegrationException("Cannot check if changes were merged", e); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   public static CodeReviewCommit findAnyMergedInto(CodeReviewRevWalk rw, | ||||
|       Iterable<CodeReviewCommit> commits, CodeReviewCommit tip) throws IOException { | ||||
|     for (CodeReviewCommit c : commits) { | ||||
|       // TODO(dborowitz): Seems like this could get expensive for many patch | ||||
|       // sets. Is there a more efficient implementation? | ||||
|       if (rw.isMergedInto(c, tip)) { | ||||
|         return c; | ||||
|       } | ||||
|     } | ||||
|     return null; | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -1840,7 +1840,7 @@ public class ReceiveCommits { | ||||
|     RevisionResource rsrc = new RevisionResource(changes.parse(changeCtl), ps); | ||||
|     try (MergeOp op = mergeOpProvider.get()) { | ||||
|       op.merge(db, rsrc.getChange(), | ||||
|           changeCtl.getUser().asIdentifiedUser(), false); | ||||
|           changeCtl.getUser().asIdentifiedUser(), false, null); | ||||
|     } | ||||
|     addMessage(""); | ||||
|     Change c = db.changes().get(rsrc.getChange().getId()); | ||||
|   | ||||
| @@ -15,38 +15,55 @@ | ||||
| package com.google.gerrit.server.git.strategy; | ||||
|  | ||||
| import com.google.common.base.CharMatcher; | ||||
| import com.google.gerrit.extensions.api.changes.SubmitInput; | ||||
| import com.google.gerrit.extensions.restapi.ResourceConflictException; | ||||
| import com.google.gerrit.reviewdb.client.Change; | ||||
| import com.google.gerrit.server.change.Submit.TestSubmitInput; | ||||
| import com.google.gerrit.server.git.BatchUpdate; | ||||
| import com.google.gerrit.server.git.CodeReviewCommit; | ||||
| import com.google.gerrit.server.git.IntegrationException; | ||||
| import com.google.gerrit.server.git.MergeOp.CommitStatus; | ||||
|  | ||||
| import java.util.ArrayList; | ||||
| import java.util.Collection; | ||||
| import java.util.List; | ||||
| import java.util.Set; | ||||
|  | ||||
| public class SubmitStrategyListener extends BatchUpdate.Listener { | ||||
|   private final Collection<SubmitStrategy> strategies; | ||||
|   private final CommitStatus commits; | ||||
|   private final boolean failAfterRefUpdates; | ||||
|  | ||||
|   public SubmitStrategyListener(Collection<SubmitStrategy> strategies, | ||||
|       CommitStatus commits) { | ||||
|   public SubmitStrategyListener(SubmitInput input, | ||||
|       Collection<SubmitStrategy> strategies, CommitStatus commits) { | ||||
|     this.strategies = strategies; | ||||
|     this.commits = commits; | ||||
|     if (input instanceof TestSubmitInput) { | ||||
|       failAfterRefUpdates = ((TestSubmitInput) input).failAfterRefUpdates; | ||||
|     } else { | ||||
|       failAfterRefUpdates = false; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   @Override | ||||
|   public void afterUpdateRepos() throws ResourceConflictException { | ||||
|     try { | ||||
|       markCleanMerges(); | ||||
|       checkCommitStatus(); | ||||
|       findUnmergedChanges(); | ||||
|       List<Change.Id> alreadyMerged = checkCommitStatus(); | ||||
|       findUnmergedChanges(alreadyMerged); | ||||
|     } catch (IntegrationException e) { | ||||
|       throw new ResourceConflictException(e.getMessage(), e); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   private void findUnmergedChanges() | ||||
|   @Override | ||||
|   public void afterRefUpdates() throws ResourceConflictException { | ||||
|     if (failAfterRefUpdates) { | ||||
|       throw new ResourceConflictException("Failing after ref updates"); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   private void findUnmergedChanges(List<Change.Id> alreadyMerged) | ||||
|       throws ResourceConflictException, IntegrationException { | ||||
|     for (SubmitStrategy strategy : strategies) { | ||||
|       if (strategy instanceof CherryPick) { | ||||
| @@ -57,7 +74,7 @@ public class SubmitStrategyListener extends BatchUpdate.Listener { | ||||
|       Set<Change.Id> unmerged = args.mergeUtil.findUnmergedChanges( | ||||
|           args.commits.getChangeIds(args.destBranch), args.rw, | ||||
|           args.canMergeFlag, args.mergeTip.getInitialTip(), | ||||
|           args.mergeTip.getCurrentTip()); | ||||
|           args.mergeTip.getCurrentTip(), alreadyMerged); | ||||
|       for (Change.Id id : unmerged) { | ||||
|         commits.problem(id, | ||||
|             "internal error: change not reachable from new branch tip"); | ||||
| @@ -74,22 +91,28 @@ public class SubmitStrategyListener extends BatchUpdate.Listener { | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   private void checkCommitStatus() throws ResourceConflictException { | ||||
|   private List<Change.Id> checkCommitStatus() throws ResourceConflictException { | ||||
|     List<Change.Id> alreadyMerged = | ||||
|         new ArrayList<>(commits.getChangeIds().size()); | ||||
|     for (Change.Id id : commits.getChangeIds()) { | ||||
|       CodeReviewCommit commit = commits.get(id); | ||||
|       CommitMergeStatus s = commit != null ? commit.getStatusCode() : null; | ||||
|       if (s == null) { | ||||
|         commits.problem(id, | ||||
|             "internal error: change not processed by merge strategy"); | ||||
|         return; | ||||
|         continue; | ||||
|       } | ||||
|       switch (s) { | ||||
|         case CLEAN_MERGE: | ||||
|         case CLEAN_REBASE: | ||||
|         case CLEAN_PICK: | ||||
|         case ALREADY_MERGED: | ||||
|           break; // Merge strategy accepted this change. | ||||
|  | ||||
|         case ALREADY_MERGED: | ||||
|           // Already an ancestor of tip. | ||||
|           alreadyMerged.add(commit.getPatchsetId().getParentKey()); | ||||
|           break; | ||||
|  | ||||
|         case PATH_CONFLICT: | ||||
|         case REBASE_MERGE_CONFLICT: | ||||
|         case MANUAL_RECURSIVE_MERGE: | ||||
| @@ -112,6 +135,7 @@ public class SubmitStrategyListener extends BatchUpdate.Listener { | ||||
|       } | ||||
|     } | ||||
|     commits.maybeFailVerbose(); | ||||
|     return alreadyMerged; | ||||
|   } | ||||
|  | ||||
|   @Override | ||||
|   | ||||
| @@ -33,6 +33,7 @@ import com.google.gerrit.reviewdb.client.PatchSetApproval; | ||||
| import com.google.gerrit.reviewdb.client.Project; | ||||
| import com.google.gerrit.reviewdb.client.RefNames; | ||||
| import com.google.gerrit.reviewdb.server.ReviewDb; | ||||
| import com.google.gerrit.reviewdb.server.ReviewDbUtil; | ||||
| import com.google.gerrit.server.ChangeUtil; | ||||
| import com.google.gerrit.server.IdentifiedUser; | ||||
| import com.google.gerrit.server.git.BatchUpdate; | ||||
| @@ -40,18 +41,28 @@ import com.google.gerrit.server.git.BatchUpdate.ChangeContext; | ||||
| import com.google.gerrit.server.git.BatchUpdate.Context; | ||||
| import com.google.gerrit.server.git.BatchUpdate.RepoContext; | ||||
| 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.IntegrationException; | ||||
| import com.google.gerrit.server.git.LabelNormalizer; | ||||
| import com.google.gerrit.server.git.MergeUtil; | ||||
| import com.google.gerrit.server.git.ProjectConfig; | ||||
| import com.google.gerrit.server.notedb.ChangeUpdate; | ||||
| import com.google.gerrit.server.project.ProjectState; | ||||
| import com.google.gwtorm.server.OrmException; | ||||
|  | ||||
| import org.eclipse.jgit.errors.IncorrectObjectTypeException; | ||||
| import org.eclipse.jgit.errors.MissingObjectException; | ||||
| import org.eclipse.jgit.lib.ObjectId; | ||||
| import org.eclipse.jgit.lib.Ref; | ||||
| import org.eclipse.jgit.transport.ReceiveCommand; | ||||
| import org.slf4j.Logger; | ||||
| import org.slf4j.LoggerFactory; | ||||
|  | ||||
| import java.io.IOException; | ||||
| import java.util.ArrayList; | ||||
| import java.util.Collection; | ||||
| import java.util.Collections; | ||||
| import java.util.List; | ||||
| import java.util.Map; | ||||
| import java.util.Objects; | ||||
| @@ -68,6 +79,7 @@ abstract class SubmitStrategyOp extends BatchUpdate.Op { | ||||
|   private ObjectId mergeResultRev; | ||||
|   private PatchSet mergedPatchSet; | ||||
|   private Change updatedChange; | ||||
|   private CodeReviewCommit alreadyMerged; | ||||
|  | ||||
|   protected SubmitStrategyOp(SubmitStrategy.Arguments args, | ||||
|       CodeReviewCommit toMerge) { | ||||
| @@ -93,18 +105,27 @@ abstract class SubmitStrategyOp extends BatchUpdate.Op { | ||||
|  | ||||
|   @Override | ||||
|   public final void updateRepo(RepoContext ctx) throws Exception { | ||||
|     logDebug("{}#updateRepo for change {}", getClass().getSimpleName(), | ||||
|         toMerge.change().getId()); | ||||
|     // Run the submit strategy implementation and record the merge tip state so | ||||
|     // we can create the ref update. | ||||
|     CodeReviewCommit tipBefore = args.mergeTip.getCurrentTip(); | ||||
|     updateRepoImpl(ctx); | ||||
|     alreadyMerged = getAlreadyMergedCommit(ctx); | ||||
|     if (alreadyMerged == null) { | ||||
|       updateRepoImpl(ctx); | ||||
|     } else { | ||||
|       logDebug("Already merged as {}", alreadyMerged.name()); | ||||
|     } | ||||
|     CodeReviewCommit tipAfter = args.mergeTip.getCurrentTip(); | ||||
|  | ||||
|     if (Objects.equals(tipBefore, tipAfter)) { | ||||
|       logDebug("Did not move tip", getClass().getSimpleName()); | ||||
|       return; | ||||
|     } else if (tipAfter == null) { | ||||
|       logDebug("No merge tip, no update to perform"); | ||||
|       return; | ||||
|     } | ||||
|     logDebug("Moved tip from {} to {}", tipBefore, tipAfter); | ||||
|  | ||||
|     checkProjectConfig(ctx, tipAfter); | ||||
|  | ||||
| @@ -133,27 +154,97 @@ abstract class SubmitStrategyOp extends BatchUpdate.Op { | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   private CodeReviewCommit getAlreadyMergedCommit(RepoContext ctx) | ||||
|       throws IOException { | ||||
|     CodeReviewCommit tip = args.mergeTip.getInitialTip(); | ||||
|     if (tip == null) { | ||||
|       return null; | ||||
|     } | ||||
|     CodeReviewRevWalk rw = (CodeReviewRevWalk) ctx.getRevWalk(); | ||||
|     Change.Id id = getId(); | ||||
|  | ||||
|     Collection<Ref> refs = ctx.getRepository().getRefDatabase() | ||||
|         .getRefs(id.toRefPrefix()).values(); | ||||
|     List<CodeReviewCommit> commits = new ArrayList<>(refs.size()); | ||||
|     for (Ref ref : refs) { | ||||
|       PatchSet.Id psId = PatchSet.Id.fromRef(ref.getName()); | ||||
|       if (psId == null) { | ||||
|         continue; | ||||
|       } | ||||
|       try { | ||||
|         CodeReviewCommit c = rw.parseCommit(ref.getObjectId()); | ||||
|         c.setPatchsetId(psId); | ||||
|         commits.add(c); | ||||
|       } catch (MissingObjectException | IncorrectObjectTypeException e) { | ||||
|         continue; // Bogus ref, can't be merged into tip so we don't care. | ||||
|       } | ||||
|     } | ||||
|     Collections.sort(commits, ReviewDbUtil.intKeyOrdering().reverse() | ||||
|         .onResultOf( | ||||
|           new Function<CodeReviewCommit, PatchSet.Id>() { | ||||
|             @Override | ||||
|             public PatchSet.Id apply(CodeReviewCommit in) { | ||||
|               return in.getPatchsetId(); | ||||
|             } | ||||
|           })); | ||||
|     CodeReviewCommit result = MergeUtil.findAnyMergedInto(rw, commits, tip); | ||||
|     if (result == null) { | ||||
|       return null; | ||||
|     } | ||||
|  | ||||
|     // Some patch set of this change is actually merged into the target | ||||
|     // branch, most likely because a previous run of MergeOp failed after | ||||
|     // updateRepo, during updateChange. | ||||
|     // | ||||
|     // Do the best we can to clean this up: mark the change as merged and set | ||||
|     // the current patch set. Don't touch the dest branch at all. This can | ||||
|     // lead to some odd situations like another change in the set merging in | ||||
|     // a different patch set of this change, but that's unavoidable at this | ||||
|     // point.  At least the change will end up in the right state. | ||||
|     // | ||||
|     // TODO(dborowitz): Consider deleting later junk patch set refs. They | ||||
|     // presumably don't have PatchSets pointing to them. | ||||
|     rw.parseBody(result); | ||||
|     result.add(args.canMergeFlag); | ||||
|     PatchSet.Id psId = result.getPatchsetId(); | ||||
|     result.copyFrom(toMerge); | ||||
|     result.setPatchsetId(psId); // Got overwriten by copyFrom. | ||||
|     result.setStatusCode(CommitMergeStatus.ALREADY_MERGED); | ||||
|     args.commits.put(result); | ||||
|     return result; | ||||
|   } | ||||
|  | ||||
|   @Override | ||||
|   public final boolean updateChange(ChangeContext ctx) throws Exception { | ||||
|     logDebug("{}#updateChange for change {}", getClass().getSimpleName(), | ||||
|         toMerge.change().getId()); | ||||
|     toMerge.setControl(ctx.getControl()); // Update change and notes from ctx. | ||||
|     PatchSet newPatchSet = updateChangeImpl(ctx); | ||||
|     PatchSet.Id oldPsId = checkNotNull(toMerge.getPatchsetId()); | ||||
|     PatchSet.Id newPsId = checkNotNull(ctx.getChange().currentPatchSetId()); | ||||
|     if (newPatchSet == null) { | ||||
|       checkState(oldPsId.equals(newPsId), | ||||
|           "patch set advanced from %s to %s but updateChangeImpl did not return" | ||||
|           + " new patch set instance", oldPsId, newPsId); | ||||
|       // Ok to use stale notes to get the old patch set, which didn't change | ||||
|       // during the submit strategy. | ||||
|       mergedPatchSet = checkNotNull( | ||||
|           args.psUtil.get(ctx.getDb(), ctx.getNotes(), oldPsId), | ||||
|           "missing old patch set %s", oldPsId); | ||||
|     PatchSet.Id newPsId; | ||||
|  | ||||
|     if (alreadyMerged != null) { | ||||
|       alreadyMerged.setControl(ctx.getControl()); | ||||
|       mergedPatchSet = getOrCreateAlreadyMergedPatchSet(ctx); | ||||
|       newPsId = mergedPatchSet.getId(); | ||||
|     } else { | ||||
|       PatchSet.Id n = newPatchSet.getId(); | ||||
|       checkState(!n.equals(oldPsId) && n.equals(newPsId), | ||||
|           "current patch was %s and is now %s, but updateChangeImpl returned" | ||||
|           + " new patch set instance at %s", oldPsId, newPsId, n); | ||||
|       mergedPatchSet = newPatchSet; | ||||
|       PatchSet newPatchSet = updateChangeImpl(ctx); | ||||
|       newPsId = checkNotNull(ctx.getChange().currentPatchSetId()); | ||||
|       if (newPatchSet == null) { | ||||
|         checkState(oldPsId.equals(newPsId), | ||||
|             "patch set advanced from %s to %s but updateChangeImpl did not" | ||||
|             + " return new patch set instance", oldPsId, newPsId); | ||||
|         // Ok to use stale notes to get the old patch set, which didn't change | ||||
|         // during the submit strategy. | ||||
|         mergedPatchSet = checkNotNull( | ||||
|             args.psUtil.get(ctx.getDb(), ctx.getNotes(), oldPsId), | ||||
|             "missing old patch set %s", oldPsId); | ||||
|       } else { | ||||
|         PatchSet.Id n = newPatchSet.getId(); | ||||
|         checkState(!n.equals(oldPsId) && n.equals(newPsId), | ||||
|             "current patch was %s and is now %s, but updateChangeImpl returned" | ||||
|             + " new patch set instance at %s", oldPsId, newPsId, n); | ||||
|         mergedPatchSet = newPatchSet; | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     Change c = ctx.getChange(); | ||||
| @@ -168,8 +259,12 @@ abstract class SubmitStrategyOp extends BatchUpdate.Op { | ||||
|           id); | ||||
|       setApproval(ctx, args.caller); | ||||
|  | ||||
|       mergeResultRev = args.mergeTip != null | ||||
|           ? args.mergeTip.getMergeResults().get(commit) : null; | ||||
|       mergeResultRev = alreadyMerged == null | ||||
|           ? args.mergeTip.getMergeResults().get(commit) | ||||
|           // Our fixup code is not smart enough to find a merge commit | ||||
|           // corresponding to the merge result. This results in a different | ||||
|           // ChangeMergedEvent in the fixup case, but we'll just live with that. | ||||
|           : alreadyMerged; | ||||
|       String txt = s.getMessage(); | ||||
|  | ||||
|       ChangeMessage msg; | ||||
| @@ -197,6 +292,30 @@ abstract class SubmitStrategyOp extends BatchUpdate.Op { | ||||
|     return true; | ||||
|   } | ||||
|  | ||||
|   private PatchSet getOrCreateAlreadyMergedPatchSet(ChangeContext ctx) | ||||
|       throws IOException, OrmException { | ||||
|     PatchSet.Id psId = alreadyMerged.getPatchsetId(); | ||||
|     logDebug("Fixing up already-merged patch set {}", psId); | ||||
|     PatchSet prevPs = args.psUtil.current(ctx.getDb(), ctx.getNotes()); | ||||
|     ctx.getRevWalk().parseBody(alreadyMerged); | ||||
|     ctx.getChange().setCurrentPatchSet(psId, | ||||
|         alreadyMerged.getShortMessage(), | ||||
|         ctx.getChange().getOriginalSubject()); | ||||
|     PatchSet existing = args.psUtil.get(ctx.getDb(), ctx.getNotes(), psId); | ||||
|     if (existing != null) { | ||||
|       logDebug("Patch set row exists, only updating change"); | ||||
|       return existing; | ||||
|     } | ||||
|     // No patch set for the already merged commit, although we know it came form | ||||
|     // a patch set ref. Fix up the database. Note that this uses the current | ||||
|     // user as the uploader, which is as good a guess as any. | ||||
|     List<String> groups = prevPs != null | ||||
|         ? prevPs.getGroups() | ||||
|         : GroupCollector.getDefaultGroups(alreadyMerged); | ||||
|     return args.psUtil.insert(ctx.getDb(), ctx.getRevWalk(), | ||||
|         ctx.getUpdate(psId), psId, alreadyMerged, false, groups, null); | ||||
|   } | ||||
|  | ||||
|   private void setApproval(ChangeContext ctx, IdentifiedUser user) | ||||
|       throws OrmException { | ||||
|     Change.Id id = ctx.getChange().getId(); | ||||
|   | ||||
		Reference in New Issue
	
	Block a user
	 Dave Borowitz
					Dave Borowitz