mirror of
https://github.com/scm-manager/scm-manager.git
synced 2025-11-18 03:01:05 +01:00
Add revisions of merged branches before merge
This commit is contained in:
@@ -15,18 +15,22 @@ public class MergeCommandResult {
|
|||||||
|
|
||||||
private final Collection<String> filesWithConflict;
|
private final Collection<String> filesWithConflict;
|
||||||
private final String newHeadRevision;
|
private final String newHeadRevision;
|
||||||
|
private final String targetRevision;
|
||||||
|
private final String revisionToMerge;
|
||||||
|
|
||||||
private MergeCommandResult(Collection<String> filesWithConflict, String newHeadRevision) {
|
private MergeCommandResult(Collection<String> filesWithConflict, String targetRevision, String revisionToMerge, String newHeadRevision) {
|
||||||
this.filesWithConflict = filesWithConflict;
|
this.filesWithConflict = filesWithConflict;
|
||||||
|
this.targetRevision = targetRevision;
|
||||||
|
this.revisionToMerge = revisionToMerge;
|
||||||
this.newHeadRevision = newHeadRevision;
|
this.newHeadRevision = newHeadRevision;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static MergeCommandResult success(String newHeadRevision) {
|
public static MergeCommandResult success(String targetRevision, String revisionToMerge, String newHeadRevision) {
|
||||||
return new MergeCommandResult(emptyList(), newHeadRevision);
|
return new MergeCommandResult(emptyList(), targetRevision, revisionToMerge, newHeadRevision);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static MergeCommandResult failure(Collection<String> filesWithConflict) {
|
public static MergeCommandResult failure(String targetRevision, String revisionToMerge, Collection<String> filesWithConflict) {
|
||||||
return new MergeCommandResult(new HashSet<>(filesWithConflict), null);
|
return new MergeCommandResult(new HashSet<>(filesWithConflict), targetRevision, revisionToMerge, null);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -45,7 +49,25 @@ public class MergeCommandResult {
|
|||||||
return unmodifiableCollection(filesWithConflict);
|
return unmodifiableCollection(filesWithConflict);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the revision of the new head of the target branch, if the merge was successful ({@link #isSuccess()})
|
||||||
|
*/
|
||||||
public String getNewHeadRevision() {
|
public String getNewHeadRevision() {
|
||||||
return newHeadRevision;
|
return newHeadRevision;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the revision of the target branch prior to the merge.
|
||||||
|
*/
|
||||||
|
public String getTargetRevision() {
|
||||||
|
return targetRevision;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the revision of the branch that was merged into the target (or in case of a conflict of the revision that
|
||||||
|
* should have been merged).
|
||||||
|
*/
|
||||||
|
public String getRevisionToMerge() {
|
||||||
|
return revisionToMerge;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -23,7 +23,7 @@ class GitFastForwardIfPossible extends GitMergeStrategy {
|
|||||||
MergeResult fastForwardResult = mergeWithFastForwardOnlyMode();
|
MergeResult fastForwardResult = mergeWithFastForwardOnlyMode();
|
||||||
if (fastForwardResult.getMergeStatus().isSuccessful()) {
|
if (fastForwardResult.getMergeStatus().isSuccessful()) {
|
||||||
push();
|
push();
|
||||||
return MergeCommandResult.success(fastForwardResult.getNewHead().toString());
|
return createSuccessResult(fastForwardResult.getNewHead().name());
|
||||||
} else {
|
} else {
|
||||||
return fallbackMerge.run();
|
return fallbackMerge.run();
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -29,7 +29,7 @@ class GitMergeCommit extends GitMergeStrategy {
|
|||||||
if (result.getMergeStatus().isSuccessful()) {
|
if (result.getMergeStatus().isSuccessful()) {
|
||||||
RevCommit revCommit = doCommit().orElseThrow(() -> new NoChangesMadeException(getRepository()));
|
RevCommit revCommit = doCommit().orElseThrow(() -> new NoChangesMadeException(getRepository()));
|
||||||
push();
|
push();
|
||||||
return MergeCommandResult.success(extractRevisionFromRevCommit(revCommit));
|
return createSuccessResult(extractRevisionFromRevCommit(revCommit));
|
||||||
} else {
|
} else {
|
||||||
return analyseFailure(result);
|
return analyseFailure(result);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -26,37 +26,57 @@ abstract class GitMergeStrategy extends AbstractGitCommand.GitCloneWorker<MergeC
|
|||||||
"",
|
"",
|
||||||
"Automatic merge by SCM-Manager.");
|
"Automatic merge by SCM-Manager.");
|
||||||
|
|
||||||
private final String target;
|
private final String targetBranch;
|
||||||
private final String toMerge;
|
private final ObjectId targetRevision;
|
||||||
|
private final String branchToMerge;
|
||||||
|
private final ObjectId revisionToMerge;
|
||||||
private final Person author;
|
private final Person author;
|
||||||
private final String messageTemplate;
|
private final String messageTemplate;
|
||||||
|
|
||||||
GitMergeStrategy(Git clone, MergeCommandRequest request, GitContext context, sonia.scm.repository.Repository repository) {
|
GitMergeStrategy(Git clone, MergeCommandRequest request, GitContext context, sonia.scm.repository.Repository repository) {
|
||||||
super(clone, context, repository);
|
super(clone, context, repository);
|
||||||
this.target = request.getTargetBranch();
|
this.targetBranch = request.getTargetBranch();
|
||||||
this.toMerge = request.getBranchToMerge();
|
this.branchToMerge = request.getBranchToMerge();
|
||||||
this.author = request.getAuthor();
|
this.author = request.getAuthor();
|
||||||
this.messageTemplate = request.getMessageTemplate();
|
this.messageTemplate = request.getMessageTemplate();
|
||||||
|
try {
|
||||||
|
this.targetRevision = resolveRevision(request.getTargetBranch());
|
||||||
|
this.revisionToMerge = resolveRevision(request.getBranchToMerge());
|
||||||
|
} catch (IOException e) {
|
||||||
|
throw new InternalRepositoryException(repository, "Could not resolve revisions of target branch or branch to merge", e);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
MergeResult doMergeInClone(MergeCommand mergeCommand) throws IOException {
|
MergeResult doMergeInClone(MergeCommand mergeCommand) throws IOException {
|
||||||
MergeResult result;
|
MergeResult result;
|
||||||
try {
|
try {
|
||||||
ObjectId sourceRevision = resolveRevision(toMerge);
|
ObjectId sourceRevision = resolveRevision(branchToMerge);
|
||||||
mergeCommand
|
mergeCommand
|
||||||
.setCommit(false) // we want to set the author manually
|
.setCommit(false) // we want to set the author manually
|
||||||
.include(toMerge, sourceRevision);
|
.include(branchToMerge, sourceRevision);
|
||||||
|
|
||||||
result = mergeCommand.call();
|
result = mergeCommand.call();
|
||||||
} catch (GitAPIException e) {
|
} catch (GitAPIException e) {
|
||||||
throw new InternalRepositoryException(getContext().getRepository(), "could not merge branch " + toMerge + " into " + target, e);
|
throw new InternalRepositoryException(getContext().getRepository(), "could not merge branch " + branchToMerge + " into " + targetBranch, e);
|
||||||
}
|
}
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
Optional<RevCommit> doCommit() {
|
Optional<RevCommit> doCommit() {
|
||||||
logger.debug("merged branch {} into {}", toMerge, target);
|
logger.debug("merged branch {} into {}", branchToMerge, targetBranch);
|
||||||
return doCommit(MessageFormat.format(determineMessageTemplate(), toMerge, target), author);
|
return doCommit(MessageFormat.format(determineMessageTemplate(), branchToMerge, targetBranch), author);
|
||||||
|
}
|
||||||
|
|
||||||
|
MergeCommandResult createSuccessResult(String newRevision) {
|
||||||
|
return MergeCommandResult.success(targetRevision.name(), revisionToMerge.name(), newRevision);
|
||||||
|
}
|
||||||
|
|
||||||
|
ObjectId getTargetRevision() {
|
||||||
|
return targetRevision;
|
||||||
|
}
|
||||||
|
|
||||||
|
ObjectId getRevisionToMerge() {
|
||||||
|
return revisionToMerge;
|
||||||
}
|
}
|
||||||
|
|
||||||
private String determineMessageTemplate() {
|
private String determineMessageTemplate() {
|
||||||
@@ -68,7 +88,7 @@ abstract class GitMergeStrategy extends AbstractGitCommand.GitCloneWorker<MergeC
|
|||||||
}
|
}
|
||||||
|
|
||||||
MergeCommandResult analyseFailure(MergeResult result) {
|
MergeCommandResult analyseFailure(MergeResult result) {
|
||||||
logger.info("could not merge branch {} into {} due to conflict in paths {}", toMerge, target, result.getConflicts().keySet());
|
logger.info("could not merge branch {} into {} due to conflict in paths {}", branchToMerge, targetBranch, result.getConflicts().keySet());
|
||||||
return MergeCommandResult.failure(result.getConflicts().keySet());
|
return MergeCommandResult.failure(targetRevision.name(), revisionToMerge.name(), result.getConflicts().keySet());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -9,8 +9,6 @@ import sonia.scm.repository.Repository;
|
|||||||
import sonia.scm.repository.api.MergeCommandResult;
|
import sonia.scm.repository.api.MergeCommandResult;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.util.Collections;
|
|
||||||
import java.util.Optional;
|
|
||||||
|
|
||||||
import static sonia.scm.repository.spi.GitRevisionExtractor.extractRevisionFromRevCommit;
|
import static sonia.scm.repository.spi.GitRevisionExtractor.extractRevisionFromRevCommit;
|
||||||
|
|
||||||
@@ -29,7 +27,7 @@ class GitMergeWithSquash extends GitMergeStrategy {
|
|||||||
if (result.getMergeStatus().isSuccessful()) {
|
if (result.getMergeStatus().isSuccessful()) {
|
||||||
RevCommit revCommit = doCommit().orElseThrow(() -> new NoChangesMadeException(getRepository()));
|
RevCommit revCommit = doCommit().orElseThrow(() -> new NoChangesMadeException(getRepository()));
|
||||||
push();
|
push();
|
||||||
return MergeCommandResult.success(extractRevisionFromRevCommit(revCommit));
|
return MergeCommandResult.success(getTargetRevision().name(), revCommit.name(), extractRevisionFromRevCommit(revCommit));
|
||||||
} else {
|
} else {
|
||||||
return analyseFailure(result);
|
return analyseFailure(result);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -71,6 +71,8 @@ public class GitMergeCommandTest extends AbstractGitCommandTestBase {
|
|||||||
MergeCommandResult mergeCommandResult = command.merge(request);
|
MergeCommandResult mergeCommandResult = command.merge(request);
|
||||||
|
|
||||||
assertThat(mergeCommandResult.isSuccess()).isTrue();
|
assertThat(mergeCommandResult.isSuccess()).isTrue();
|
||||||
|
assertThat(mergeCommandResult.getRevisionToMerge()).isEqualTo("91b99de908fcd04772798a31c308a64aea1a5523");
|
||||||
|
assertThat(mergeCommandResult.getTargetRevision()).isEqualTo("fcd0ef1831e4002ac43ea539f4094334c79ea9ec");
|
||||||
|
|
||||||
Repository repository = createContext().open();
|
Repository repository = createContext().open();
|
||||||
Iterable<RevCommit> commits = new Git(repository).log().add(repository.resolve("master")).setMaxCount(1).call();
|
Iterable<RevCommit> commits = new Git(repository).log().add(repository.resolve("master")).setMaxCount(1).call();
|
||||||
@@ -229,6 +231,8 @@ public class GitMergeCommandTest extends AbstractGitCommandTestBase {
|
|||||||
|
|
||||||
Repository repository = createContext().open();
|
Repository repository = createContext().open();
|
||||||
assertThat(mergeCommandResult.isSuccess()).isTrue();
|
assertThat(mergeCommandResult.isSuccess()).isTrue();
|
||||||
|
assertThat(mergeCommandResult.getRevisionToMerge()).isEqualTo(mergeCommandResult.getNewHeadRevision());
|
||||||
|
assertThat(mergeCommandResult.getTargetRevision()).isEqualTo("fcd0ef1831e4002ac43ea539f4094334c79ea9ec");
|
||||||
|
|
||||||
Iterable<RevCommit> commits = new Git(repository).log().add(repository.resolve("master")).setMaxCount(1).call();
|
Iterable<RevCommit> commits = new Git(repository).log().add(repository.resolve("master")).setMaxCount(1).call();
|
||||||
RevCommit mergeCommit = commits.iterator().next();
|
RevCommit mergeCommit = commits.iterator().next();
|
||||||
@@ -279,6 +283,9 @@ public class GitMergeCommandTest extends AbstractGitCommandTestBase {
|
|||||||
request.setAuthor(new Person("Dirk Gently", "dirk@holistic.det"));
|
request.setAuthor(new Person("Dirk Gently", "dirk@holistic.det"));
|
||||||
|
|
||||||
MergeCommandResult mergeCommandResult = command.merge(request);
|
MergeCommandResult mergeCommandResult = command.merge(request);
|
||||||
|
assertThat(mergeCommandResult.getNewHeadRevision()).isEqualTo("35597e9e98fe53167266583848bfef985c2adb27");
|
||||||
|
assertThat(mergeCommandResult.getRevisionToMerge()).isEqualTo("35597e9e98fe53167266583848bfef985c2adb27");
|
||||||
|
assertThat(mergeCommandResult.getTargetRevision()).isEqualTo("fcd0ef1831e4002ac43ea539f4094334c79ea9ec");
|
||||||
|
|
||||||
assertThat(mergeCommandResult.isSuccess()).isTrue();
|
assertThat(mergeCommandResult.isSuccess()).isTrue();
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user