mirror of
https://github.com/scm-manager/scm-manager.git
synced 2025-11-01 11:05:56 +01:00
return revision on merge
This commit is contained in:
@@ -13,17 +13,19 @@ import static java.util.Collections.unmodifiableCollection;
|
|||||||
*/
|
*/
|
||||||
public class MergeCommandResult {
|
public class MergeCommandResult {
|
||||||
private final Collection<String> filesWithConflict;
|
private final Collection<String> filesWithConflict;
|
||||||
|
private static String mergeCommitRevision;
|
||||||
|
|
||||||
private MergeCommandResult(Collection<String> filesWithConflict) {
|
public MergeCommandResult(Collection<String> filesWithConflict, String mergeCommitRevision) {
|
||||||
this.filesWithConflict = filesWithConflict;
|
this.filesWithConflict = filesWithConflict;
|
||||||
|
this.mergeCommitRevision = mergeCommitRevision;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static MergeCommandResult success() {
|
public static MergeCommandResult success() {
|
||||||
return new MergeCommandResult(emptyList());
|
return new MergeCommandResult(emptyList(), mergeCommitRevision);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static MergeCommandResult failure(Collection<String> filesWithConflict) {
|
public static MergeCommandResult failure(Collection<String> filesWithConflict) {
|
||||||
return new MergeCommandResult(new HashSet<>(filesWithConflict));
|
return new MergeCommandResult(new HashSet<>(filesWithConflict), "");
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -41,4 +43,8 @@ public class MergeCommandResult {
|
|||||||
public Collection<String> getFilesWithConflict() {
|
public Collection<String> getFilesWithConflict() {
|
||||||
return unmodifiableCollection(filesWithConflict);
|
return unmodifiableCollection(filesWithConflict);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public String getMergeCommitRevision() {
|
||||||
|
return mergeCommitRevision;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,6 +2,7 @@ package sonia.scm.repository.spi;
|
|||||||
|
|
||||||
import com.google.common.base.Strings;
|
import com.google.common.base.Strings;
|
||||||
import org.eclipse.jgit.diff.DiffEntry;
|
import org.eclipse.jgit.diff.DiffEntry;
|
||||||
|
import org.eclipse.jgit.errors.MissingObjectException;
|
||||||
import org.eclipse.jgit.lib.ObjectId;
|
import org.eclipse.jgit.lib.ObjectId;
|
||||||
import org.eclipse.jgit.lib.Repository;
|
import org.eclipse.jgit.lib.Repository;
|
||||||
import org.eclipse.jgit.revwalk.RevCommit;
|
import org.eclipse.jgit.revwalk.RevCommit;
|
||||||
@@ -10,6 +11,8 @@ import org.eclipse.jgit.revwalk.RevWalk;
|
|||||||
import org.eclipse.jgit.treewalk.EmptyTreeIterator;
|
import org.eclipse.jgit.treewalk.EmptyTreeIterator;
|
||||||
import org.eclipse.jgit.treewalk.TreeWalk;
|
import org.eclipse.jgit.treewalk.TreeWalk;
|
||||||
import org.eclipse.jgit.treewalk.filter.PathFilter;
|
import org.eclipse.jgit.treewalk.filter.PathFilter;
|
||||||
|
import sonia.scm.ContextEntry;
|
||||||
|
import sonia.scm.NotFoundException;
|
||||||
import sonia.scm.repository.GitUtil;
|
import sonia.scm.repository.GitUtil;
|
||||||
import sonia.scm.util.Util;
|
import sonia.scm.util.Util;
|
||||||
|
|
||||||
@@ -38,7 +41,15 @@ final class Differ implements AutoCloseable {
|
|||||||
RevWalk walk = new RevWalk(repository);
|
RevWalk walk = new RevWalk(repository);
|
||||||
|
|
||||||
ObjectId revision = repository.resolve(request.getRevision());
|
ObjectId revision = repository.resolve(request.getRevision());
|
||||||
RevCommit commit = walk.parseCommit(revision);
|
if (revision == null) {
|
||||||
|
throw NotFoundException.notFound(ContextEntry.ContextBuilder.entity("revision not found", request.getRevision()));
|
||||||
|
}
|
||||||
|
RevCommit commit;
|
||||||
|
try {
|
||||||
|
commit = walk.parseCommit(revision);
|
||||||
|
} catch (MissingObjectException ex) {
|
||||||
|
throw NotFoundException.notFound(ContextEntry.ContextBuilder.entity("revision not found", request.getRevision()));
|
||||||
|
}
|
||||||
|
|
||||||
walk.markStart(commit);
|
walk.markStart(commit);
|
||||||
commit = walk.next();
|
commit = walk.next();
|
||||||
@@ -46,34 +57,25 @@ final class Differ implements AutoCloseable {
|
|||||||
treeWalk.reset();
|
treeWalk.reset();
|
||||||
treeWalk.setRecursive(true);
|
treeWalk.setRecursive(true);
|
||||||
|
|
||||||
if (Util.isNotEmpty(request.getPath()))
|
if (Util.isNotEmpty(request.getPath())) {
|
||||||
{
|
|
||||||
treeWalk.setFilter(PathFilter.create(request.getPath()));
|
treeWalk.setFilter(PathFilter.create(request.getPath()));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
if (!Strings.isNullOrEmpty(request.getAncestorChangeset()))
|
if (!Strings.isNullOrEmpty(request.getAncestorChangeset())) {
|
||||||
{
|
|
||||||
ObjectId otherRevision = repository.resolve(request.getAncestorChangeset());
|
ObjectId otherRevision = repository.resolve(request.getAncestorChangeset());
|
||||||
ObjectId ancestorId = GitUtil.computeCommonAncestor(repository, revision, otherRevision);
|
ObjectId ancestorId = GitUtil.computeCommonAncestor(repository, revision, otherRevision);
|
||||||
RevTree tree = walk.parseCommit(ancestorId).getTree();
|
RevTree tree = walk.parseCommit(ancestorId).getTree();
|
||||||
treeWalk.addTree(tree);
|
treeWalk.addTree(tree);
|
||||||
}
|
} else if (commit.getParentCount() > 0) {
|
||||||
else if (commit.getParentCount() > 0)
|
|
||||||
{
|
|
||||||
RevTree tree = commit.getParent(0).getTree();
|
RevTree tree = commit.getParent(0).getTree();
|
||||||
|
|
||||||
if (tree != null)
|
if (tree != null) {
|
||||||
{
|
|
||||||
treeWalk.addTree(tree);
|
treeWalk.addTree(tree);
|
||||||
}
|
} else {
|
||||||
else
|
|
||||||
{
|
|
||||||
treeWalk.addTree(new EmptyTreeIterator());
|
treeWalk.addTree(new EmptyTreeIterator());
|
||||||
}
|
}
|
||||||
}
|
} else {
|
||||||
else
|
|
||||||
{
|
|
||||||
treeWalk.addTree(new EmptyTreeIterator());
|
treeWalk.addTree(new EmptyTreeIterator());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -7,6 +7,7 @@ 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;
|
||||||
|
|
||||||
class GitFastForwardIfPossible extends GitMergeStrategy {
|
class GitFastForwardIfPossible extends GitMergeStrategy {
|
||||||
|
|
||||||
@@ -22,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();
|
return new MergeCommandResult(Collections.emptyList(), "");
|
||||||
} else {
|
} else {
|
||||||
return fallbackMerge.run();
|
return fallbackMerge.run();
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,10 +3,17 @@ package sonia.scm.repository.spi;
|
|||||||
import org.eclipse.jgit.api.Git;
|
import org.eclipse.jgit.api.Git;
|
||||||
import org.eclipse.jgit.api.MergeCommand;
|
import org.eclipse.jgit.api.MergeCommand;
|
||||||
import org.eclipse.jgit.api.MergeResult;
|
import org.eclipse.jgit.api.MergeResult;
|
||||||
|
import org.eclipse.jgit.revwalk.RevCommit;
|
||||||
|
import sonia.scm.ContextEntry;
|
||||||
|
import sonia.scm.repository.InternalRepositoryException;
|
||||||
import sonia.scm.repository.Repository;
|
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.GitMerger.evaluateRevisionFromMergeCommit;
|
||||||
|
|
||||||
class GitMergeCommit extends GitMergeStrategy {
|
class GitMergeCommit extends GitMergeStrategy {
|
||||||
|
|
||||||
@@ -21,11 +28,12 @@ class GitMergeCommit extends GitMergeStrategy {
|
|||||||
MergeResult result = doMergeInClone(mergeCommand);
|
MergeResult result = doMergeInClone(mergeCommand);
|
||||||
|
|
||||||
if (result.getMergeStatus().isSuccessful()) {
|
if (result.getMergeStatus().isSuccessful()) {
|
||||||
doCommit();
|
Optional<RevCommit> revCommit = doCommit();
|
||||||
push();
|
push();
|
||||||
return MergeCommandResult.success();
|
return new MergeCommandResult(Collections.emptyList(), evaluateRevisionFromMergeCommit(revCommit));
|
||||||
} else {
|
} else {
|
||||||
return analyseFailure(result);
|
return analyseFailure(result);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6,6 +6,7 @@ import org.eclipse.jgit.api.MergeCommand;
|
|||||||
import org.eclipse.jgit.api.MergeResult;
|
import org.eclipse.jgit.api.MergeResult;
|
||||||
import org.eclipse.jgit.api.errors.GitAPIException;
|
import org.eclipse.jgit.api.errors.GitAPIException;
|
||||||
import org.eclipse.jgit.lib.ObjectId;
|
import org.eclipse.jgit.lib.ObjectId;
|
||||||
|
import org.eclipse.jgit.revwalk.RevCommit;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
import sonia.scm.repository.InternalRepositoryException;
|
import sonia.scm.repository.InternalRepositoryException;
|
||||||
@@ -14,6 +15,7 @@ import sonia.scm.repository.api.MergeCommandResult;
|
|||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.text.MessageFormat;
|
import java.text.MessageFormat;
|
||||||
|
import java.util.Optional;
|
||||||
|
|
||||||
abstract class GitMergeStrategy extends AbstractGitCommand.GitCloneWorker<MergeCommandResult> {
|
abstract class GitMergeStrategy extends AbstractGitCommand.GitCloneWorker<MergeCommandResult> {
|
||||||
|
|
||||||
@@ -52,9 +54,9 @@ abstract class GitMergeStrategy extends AbstractGitCommand.GitCloneWorker<MergeC
|
|||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
void doCommit() {
|
Optional<RevCommit> doCommit() {
|
||||||
logger.debug("merged branch {} into {}", toMerge, target);
|
logger.debug("merged branch {} into {}", toMerge, target);
|
||||||
doCommit(MessageFormat.format(determineMessageTemplate(), toMerge, target), author);
|
return doCommit(MessageFormat.format(determineMessageTemplate(), toMerge, target), author);
|
||||||
}
|
}
|
||||||
|
|
||||||
private String determineMessageTemplate() {
|
private String determineMessageTemplate() {
|
||||||
|
|||||||
@@ -2,11 +2,16 @@ package sonia.scm.repository.spi;
|
|||||||
|
|
||||||
import org.eclipse.jgit.api.Git;
|
import org.eclipse.jgit.api.Git;
|
||||||
import org.eclipse.jgit.api.MergeResult;
|
import org.eclipse.jgit.api.MergeResult;
|
||||||
|
import org.eclipse.jgit.revwalk.RevCommit;
|
||||||
import sonia.scm.repository.Repository;
|
import sonia.scm.repository.Repository;
|
||||||
import sonia.scm.repository.api.MergeCommandResult;
|
import sonia.scm.repository.api.MergeCommandResult;
|
||||||
import org.eclipse.jgit.api.MergeCommand;
|
import org.eclipse.jgit.api.MergeCommand;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.Optional;
|
||||||
|
|
||||||
|
import static sonia.scm.repository.spi.GitMerger.evaluateRevisionFromMergeCommit;
|
||||||
|
|
||||||
class GitMergeWithSquash extends GitMergeStrategy {
|
class GitMergeWithSquash extends GitMergeStrategy {
|
||||||
|
|
||||||
@@ -21,9 +26,9 @@ class GitMergeWithSquash extends GitMergeStrategy {
|
|||||||
MergeResult result = doMergeInClone(mergeCommand);
|
MergeResult result = doMergeInClone(mergeCommand);
|
||||||
|
|
||||||
if (result.getMergeStatus().isSuccessful()) {
|
if (result.getMergeStatus().isSuccessful()) {
|
||||||
doCommit();
|
Optional<RevCommit> revCommit = doCommit();
|
||||||
push();
|
push();
|
||||||
return MergeCommandResult.success();
|
return new MergeCommandResult(Collections.emptyList(), evaluateRevisionFromMergeCommit(revCommit));
|
||||||
} else {
|
} else {
|
||||||
return analyseFailure(result);
|
return analyseFailure(result);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,17 @@
|
|||||||
|
package sonia.scm.repository.spi;
|
||||||
|
|
||||||
|
import org.eclipse.jgit.revwalk.RevCommit;
|
||||||
|
import sonia.scm.ContextEntry;
|
||||||
|
import sonia.scm.repository.InternalRepositoryException;
|
||||||
|
|
||||||
|
import java.util.Optional;
|
||||||
|
|
||||||
|
public class GitMerger {
|
||||||
|
|
||||||
|
static String evaluateRevisionFromMergeCommit(Optional<RevCommit> revCommit) {
|
||||||
|
if (revCommit.isPresent()) {
|
||||||
|
return revCommit.get().toString().split(" ")[1];
|
||||||
|
}
|
||||||
|
throw new InternalRepositoryException(ContextEntry.ContextBuilder.entity(GitMergeCommit.class, "merge commit failed"), "could not create commit on merge");
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -7,8 +7,11 @@ import parser from "gitdiff-parser";
|
|||||||
import Loading from "../Loading";
|
import Loading from "../Loading";
|
||||||
import Diff from "./Diff";
|
import Diff from "./Diff";
|
||||||
import { DiffObjectProps, File } from "./DiffTypes";
|
import { DiffObjectProps, File } from "./DiffTypes";
|
||||||
|
import { NotFoundError } from "../errors";
|
||||||
|
import { Notification } from "../index";
|
||||||
|
import {withTranslation, WithTranslation} from "react-i18next";
|
||||||
|
|
||||||
type Props = DiffObjectProps & {
|
type Props = WithTranslation & DiffObjectProps & {
|
||||||
url: string;
|
url: string;
|
||||||
defaultCollapse?: boolean;
|
defaultCollapse?: boolean;
|
||||||
};
|
};
|
||||||
@@ -43,7 +46,7 @@ class LoadingDiff extends React.Component<Props, State> {
|
|||||||
|
|
||||||
fetchDiff = () => {
|
fetchDiff = () => {
|
||||||
const { url } = this.props;
|
const { url } = this.props;
|
||||||
this.setState({loading: true});
|
this.setState({ loading: true });
|
||||||
apiClient
|
apiClient
|
||||||
.get(url)
|
.get(url)
|
||||||
.then(response => response.text())
|
.then(response => response.text())
|
||||||
@@ -66,6 +69,9 @@ class LoadingDiff extends React.Component<Props, State> {
|
|||||||
render() {
|
render() {
|
||||||
const { diff, loading, error } = this.state;
|
const { diff, loading, error } = this.state;
|
||||||
if (error) {
|
if (error) {
|
||||||
|
if (error instanceof NotFoundError) {
|
||||||
|
return <Notification type="info">{this.props.t("changesets.noChangesets")}</Notification>;
|
||||||
|
}
|
||||||
return <ErrorNotification error={error} />;
|
return <ErrorNotification error={error} />;
|
||||||
} else if (loading) {
|
} else if (loading) {
|
||||||
return <Loading />;
|
return <Loading />;
|
||||||
@@ -77,4 +83,4 @@ class LoadingDiff extends React.Component<Props, State> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export default LoadingDiff;
|
export default withTranslation("repos")(LoadingDiff);
|
||||||
|
|||||||
@@ -28,7 +28,7 @@ class ChangesetDiff extends React.Component<Props> {
|
|||||||
return <Notification type="danger">{t("changeset.diffNotSupported")}</Notification>;
|
return <Notification type="danger">{t("changeset.diffNotSupported")}</Notification>;
|
||||||
} else {
|
} else {
|
||||||
const url = this.createUrl(changeset);
|
const url = this.createUrl(changeset);
|
||||||
return <LoadingDiff url={url} defaultCollapse={defaultCollapse} />;
|
return <LoadingDiff url={url} defaultCollapse={defaultCollapse} sideBySide={false}/>;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -72,7 +72,7 @@
|
|||||||
"changesets": {
|
"changesets": {
|
||||||
"errorTitle": "Fehler",
|
"errorTitle": "Fehler",
|
||||||
"errorSubtitle": "Changesets konnten nicht abgerufen werden",
|
"errorSubtitle": "Changesets konnten nicht abgerufen werden",
|
||||||
"noChangesets": "Keine Changesets in diesem Branch gefunden.",
|
"noChangesets": "Keine Changesets in diesem Branch gefunden. Die Commits könnten gelöscht wurden sein.",
|
||||||
"branchSelectorLabel": "Branches",
|
"branchSelectorLabel": "Branches",
|
||||||
"collapseDiffs": "Auf-/Zuklappen"
|
"collapseDiffs": "Auf-/Zuklappen"
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -72,7 +72,7 @@
|
|||||||
"changesets": {
|
"changesets": {
|
||||||
"errorTitle": "Error",
|
"errorTitle": "Error",
|
||||||
"errorSubtitle": "Could not fetch changesets",
|
"errorSubtitle": "Could not fetch changesets",
|
||||||
"noChangesets": "No changesets found for this branch.",
|
"noChangesets": "No changesets found for this branch. The commits could have been removed.",
|
||||||
"branchSelectorLabel": "Branches",
|
"branchSelectorLabel": "Branches",
|
||||||
"collapseDiffs": "Collapse"
|
"collapseDiffs": "Collapse"
|
||||||
},
|
},
|
||||||
|
|||||||
Reference in New Issue
Block a user