aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorRobin Stocker <robin@nibor.org>2013-03-09 14:09:01 +0100
committerMatthias Sohn <matthias.sohn@sap.com>2013-03-16 23:32:13 +0100
commit0e9f1cf57dac274f92a6db38197e14b55b3277af (patch)
treebbd8dfd7f787d4627b03d7016ce5fa7fb834a052
parentbba74ba2e0af10db4d75258819ca0a1b4307e377 (diff)
downloadjgit-0e9f1cf57dac274f92a6db38197e14b55b3277af.tar.gz
jgit-0e9f1cf57dac274f92a6db38197e14b55b3277af.zip
Support aborting non-interactive rebase started from C Git
Continuing is trickier, as .git/rebase-apply contains no message file and no git-rebase-todo. Bug: 336820 Change-Id: I4eb87c850078ca187b38b81cc91c92afb1176945 Signed-off-by: Matthias Sohn <matthias.sohn@sap.com>
-rw-r--r--org.eclipse.jgit.test/tst/org/eclipse/jgit/api/RebaseCommandTest.java24
-rw-r--r--org.eclipse.jgit/src/org/eclipse/jgit/api/RebaseCommand.java150
2 files changed, 120 insertions, 54 deletions
diff --git a/org.eclipse.jgit.test/tst/org/eclipse/jgit/api/RebaseCommandTest.java b/org.eclipse.jgit.test/tst/org/eclipse/jgit/api/RebaseCommandTest.java
index 27f41b50ff..94577a0d68 100644
--- a/org.eclipse.jgit.test/tst/org/eclipse/jgit/api/RebaseCommandTest.java
+++ b/org.eclipse.jgit.test/tst/org/eclipse/jgit/api/RebaseCommandTest.java
@@ -1,5 +1,5 @@
/*
- * Copyright (C) 2010, Mathias Kinzler <mathias.kinzler@sap.com>
+ * Copyright (C) 2010, 2013 Mathias Kinzler <mathias.kinzler@sap.com>
* and other copyright owners as documented in the project's IP log.
*
* This program and the accompanying materials are made available
@@ -79,6 +79,7 @@ import org.eclipse.jgit.merge.MergeStrategy;
import org.eclipse.jgit.merge.ResolveMerger.MergeFailureReason;
import org.eclipse.jgit.revwalk.RevCommit;
import org.eclipse.jgit.revwalk.RevWalk;
+import org.eclipse.jgit.util.FileUtils;
import org.junit.Before;
import org.junit.Test;
@@ -1507,6 +1508,27 @@ public class RebaseCommandTest extends RepositoryTestCase {
}
@Test
+ public void testAbortShouldAlsoAbortNonInteractiveRebaseWithRebaseApplyDir()
+ throws Exception {
+ writeTrashFile(FILE1, "initial file");
+ git.add().addFilepattern(FILE1).call();
+ git.commit().setMessage("initial commit").call();
+
+ File applyDir = new File(db.getDirectory(), "rebase-apply");
+ File headName = new File(applyDir, "head-name");
+ FileUtils.mkdir(applyDir);
+ write(headName, "master");
+ db.writeOrigHead(db.resolve(Constants.HEAD));
+
+ git.rebase().setOperation(Operation.ABORT).call();
+
+ assertFalse("Abort should clean up .git/rebase-apply",
+ applyDir.exists());
+ assertEquals(RepositoryState.SAFE, git.getRepository()
+ .getRepositoryState());
+ }
+
+ @Test
public void testRebaseShouldBeAbleToHandleEmptyLinesInRebaseTodoFile()
throws IOException {
String emptyLine = "\n";
diff --git a/org.eclipse.jgit/src/org/eclipse/jgit/api/RebaseCommand.java b/org.eclipse.jgit/src/org/eclipse/jgit/api/RebaseCommand.java
index a100c9d878..a206a3bc21 100644
--- a/org.eclipse.jgit/src/org/eclipse/jgit/api/RebaseCommand.java
+++ b/org.eclipse.jgit/src/org/eclipse/jgit/api/RebaseCommand.java
@@ -1,5 +1,5 @@
/*
- * Copyright (C) 2010, Mathias Kinzler <mathias.kinzler@sap.com>
+ * Copyright (C) 2010, 2013 Mathias Kinzler <mathias.kinzler@sap.com>
* and other copyright owners as documented in the project's IP log.
*
* This program and the accompanying materials are made available
@@ -108,11 +108,16 @@ import org.eclipse.jgit.util.RawParseUtils;
*/
public class RebaseCommand extends GitCommand<RebaseResult> {
/**
- * The name of the "rebase-merge" folder
+ * The name of the "rebase-merge" folder for interactive rebases.
*/
public static final String REBASE_MERGE = "rebase-merge"; //$NON-NLS-1$
/**
+ * The name of the "rebase-apply" folder for non-interactive rebases.
+ */
+ private static final String REBASE_APPLY = "rebase-apply"; //$NON-NLS-1$
+
+ /**
* The name of the "stopped-sha" file
*/
public static final String STOPPED_SHA = "stopped-sha"; //$NON-NLS-1$
@@ -177,7 +182,7 @@ public class RebaseCommand extends GitCommand<RebaseResult> {
private final RevWalk walk;
- private final File rebaseDir;
+ private final RebaseState rebaseState;
private InteractiveHandler interactiveHandler;
@@ -187,7 +192,7 @@ public class RebaseCommand extends GitCommand<RebaseResult> {
protected RebaseCommand(Repository repo) {
super(repo);
walk = new RevWalk(repo);
- rebaseDir = new File(repo.getDirectory(), REBASE_MERGE);
+ rebaseState = new RebaseState(repo.getDirectory());
}
/**
@@ -219,9 +224,9 @@ public class RebaseCommand extends GitCommand<RebaseResult> {
case SKIP:
// fall through
case CONTINUE:
- String upstreamCommitId = readFile(rebaseDir, ONTO);
+ String upstreamCommitId = rebaseState.readFile(ONTO);
try {
- upstreamCommitName = readFile(rebaseDir, ONTO_NAME);
+ upstreamCommitName = rebaseState.readFile(ONTO_NAME);
} catch (FileNotFoundException e) {
// Fall back to commit ID if file doesn't exist (e.g. rebase
// was started by C Git)
@@ -242,7 +247,7 @@ public class RebaseCommand extends GitCommand<RebaseResult> {
if (operation == Operation.CONTINUE) {
newHead = continueRebase();
- File amendFile = new File(rebaseDir, AMEND);
+ File amendFile = rebaseState.getFile(AMEND);
boolean amendExists = amendFile.exists();
if (amendExists) {
FileUtils.delete(amendFile);
@@ -265,9 +270,9 @@ public class RebaseCommand extends GitCommand<RebaseResult> {
List<Step> steps = loadSteps();
if (isInteractive()) {
interactiveHandler.prepareSteps(steps);
- BufferedWriter fw = new BufferedWriter(
- new OutputStreamWriter(new FileOutputStream(new File(
- rebaseDir, GIT_REBASE_TODO)),
+ BufferedWriter fw = new BufferedWriter(new OutputStreamWriter(
+ new FileOutputStream(
+ rebaseState.getFile(GIT_REBASE_TODO)),
Constants.CHARACTER_ENCODING));
fw.newLine();
try {
@@ -339,7 +344,7 @@ public class RebaseCommand extends GitCommand<RebaseResult> {
.setAmend(true).call();
continue;
case EDIT:
- createFile(rebaseDir, AMEND, commitToPick.name());
+ rebaseState.createFile(AMEND, commitToPick.name());
return stop(commitToPick);
}
} finally {
@@ -347,9 +352,9 @@ public class RebaseCommand extends GitCommand<RebaseResult> {
}
}
if (newHead != null) {
- String headName = readFile(rebaseDir, HEAD_NAME);
+ String headName = rebaseState.readFile(HEAD_NAME);
updateHead(headName, newHead);
- FileUtils.delete(rebaseDir, FileUtils.RECURSIVE);
+ FileUtils.delete(rebaseState.getDir(), FileUtils.RECURSIVE);
if (lastStepWasForward)
return RebaseResult.FAST_FORWARD_RESULT;
return RebaseResult.OK_RESULT;
@@ -458,7 +463,7 @@ public class RebaseCommand extends GitCommand<RebaseResult> {
if (needsCommit) {
CommitCommand commit = new Git(repo).commit();
- commit.setMessage(readFile(rebaseDir, MESSAGE));
+ commit.setMessage(rebaseState.readFile(MESSAGE));
commit.setAuthor(parseAuthor());
return commit.call();
}
@@ -466,7 +471,7 @@ public class RebaseCommand extends GitCommand<RebaseResult> {
}
private PersonIdent parseAuthor() throws IOException {
- File authorScriptFile = new File(rebaseDir, AUTHOR_SCRIPT);
+ File authorScriptFile = rebaseState.getFile(AUTHOR_SCRIPT);
byte[] raw;
try {
raw = IO.readFully(authorScriptFile);
@@ -479,15 +484,17 @@ public class RebaseCommand extends GitCommand<RebaseResult> {
private RebaseResult stop(RevCommit commitToPick) throws IOException {
PersonIdent author = commitToPick.getAuthorIdent();
String authorScript = toAuthorScript(author);
- createFile(rebaseDir, AUTHOR_SCRIPT, authorScript);
- createFile(rebaseDir, MESSAGE, commitToPick.getFullMessage());
+ rebaseState.createFile(AUTHOR_SCRIPT, authorScript);
+ rebaseState.createFile(MESSAGE, commitToPick.getFullMessage());
ByteArrayOutputStream bos = new ByteArrayOutputStream();
DiffFormatter df = new DiffFormatter(bos);
df.setRepository(repo);
df.format(commitToPick.getParent(0), commitToPick);
- createFile(rebaseDir, PATCH, new String(bos.toByteArray(),
+ rebaseState.createFile(PATCH, new String(bos.toByteArray(),
Constants.CHARACTER_ENCODING));
- createFile(rebaseDir, STOPPED_SHA, repo.newObjectReader().abbreviate(
+ rebaseState.createFile(STOPPED_SHA,
+ repo.newObjectReader()
+ .abbreviate(
commitToPick).name());
// Remove cherry pick state file created by CherryPickCommand, it's not
// needed for rebase
@@ -531,8 +538,8 @@ public class RebaseCommand extends GitCommand<RebaseResult> {
return;
List<String> todoLines = new ArrayList<String>();
List<String> poppedLines = new ArrayList<String>();
- File todoFile = new File(rebaseDir, GIT_REBASE_TODO);
- File doneFile = new File(rebaseDir, DONE);
+ File todoFile = rebaseState.getFile(GIT_REBASE_TODO);
+ File doneFile = rebaseState.getFile(DONE);
BufferedReader br = new BufferedReader(new InputStreamReader(
new FileInputStream(todoFile), Constants.CHARACTER_ENCODING));
try {
@@ -646,16 +653,16 @@ public class RebaseCommand extends GitCommand<RebaseResult> {
Collections.reverse(cherryPickList);
// create the folder for the meta information
- FileUtils.mkdir(rebaseDir);
+ FileUtils.mkdir(rebaseState.getDir());
repo.writeOrigHead(headId);
- createFile(rebaseDir, REBASE_HEAD, headId.name());
- createFile(rebaseDir, HEAD_NAME, headName);
- createFile(rebaseDir, ONTO, upstreamCommit.name());
- createFile(rebaseDir, ONTO_NAME, upstreamCommitName);
- createFile(rebaseDir, INTERACTIVE, ""); //$NON-NLS-1$
+ rebaseState.createFile(REBASE_HEAD, headId.name());
+ rebaseState.createFile(HEAD_NAME, headName);
+ rebaseState.createFile(ONTO, upstreamCommit.name());
+ rebaseState.createFile(ONTO_NAME, upstreamCommitName);
+ rebaseState.createFile(INTERACTIVE, ""); //$NON-NLS-1$
BufferedWriter fw = new BufferedWriter(new OutputStreamWriter(
- new FileOutputStream(new File(rebaseDir, GIT_REBASE_TODO)),
+ new FileOutputStream(rebaseState.getFile(GIT_REBASE_TODO)),
Constants.CHARACTER_ENCODING));
fw.write("# Created by EGit: rebasing " + upstreamCommit.name()
+ " onto " + headId.name());
@@ -687,7 +694,7 @@ public class RebaseCommand extends GitCommand<RebaseResult> {
checkoutOk = checkoutCommit(upstreamCommit);
} finally {
if (!checkoutOk)
- FileUtils.delete(rebaseDir, FileUtils.RECURSIVE);
+ FileUtils.delete(rebaseState.getDir(), FileUtils.RECURSIVE);
}
monitor.endTask();
@@ -799,18 +806,6 @@ public class RebaseCommand extends GitCommand<RebaseResult> {
}
}
- private void createFile(File parentDir, String name, String content)
- throws IOException {
- File file = new File(parentDir, name);
- FileOutputStream fos = new FileOutputStream(file);
- try {
- fos.write(content.getBytes(Constants.CHARACTER_ENCODING));
- fos.write('\n');
- } finally {
- fos.close();
- }
- }
-
private RebaseResult abort(RebaseResult result) throws IOException {
try {
ObjectId origHead = repo.readOrigHead();
@@ -840,7 +835,7 @@ public class RebaseCommand extends GitCommand<RebaseResult> {
monitor.endTask();
}
try {
- String headName = readFile(rebaseDir, HEAD_NAME);
+ String headName = rebaseState.readFile(HEAD_NAME);
if (headName.startsWith(Constants.R_REFS)) {
monitor.beginTask(MessageFormat.format(
JGitText.get().resettingHead, headName),
@@ -860,7 +855,7 @@ public class RebaseCommand extends GitCommand<RebaseResult> {
}
}
// cleanup the files
- FileUtils.delete(rebaseDir, FileUtils.RECURSIVE);
+ FileUtils.delete(rebaseState.getDir(), FileUtils.RECURSIVE);
repo.writeCherryPickHead(null);
return result;
@@ -869,15 +864,6 @@ public class RebaseCommand extends GitCommand<RebaseResult> {
}
}
- private String readFile(File directory, String fileName) throws IOException {
- byte[] content = IO.readFully(new File(directory, fileName));
- // strip off the last LF
- int end = content.length;
- while (0 < end && content[end - 1] == '\n')
- end--;
- return RawParseUtils.decode(content, 0, end);
- }
-
private boolean checkoutCommit(RevCommit commit) throws IOException,
CheckoutConflictException {
try {
@@ -911,7 +897,7 @@ public class RebaseCommand extends GitCommand<RebaseResult> {
}
List<Step> loadSteps() throws IOException {
- byte[] buf = IO.readFully(new File(rebaseDir, GIT_REBASE_TODO));
+ byte[] buf = IO.readFully(rebaseState.getFile(GIT_REBASE_TODO));
int ptr = 0;
int tokenBegin = 0;
ArrayList<Step> r = new ArrayList<Step>();
@@ -1225,4 +1211,62 @@ public class RebaseCommand extends GitCommand<RebaseResult> {
return new PersonIdent(name, email, when, tz);
return null;
}
+
+ private static class RebaseState {
+
+ private final File repoDirectory;
+ private File dir;
+
+ public RebaseState(File repoDirectory) {
+ this.repoDirectory = repoDirectory;
+ }
+
+ public File getDir() {
+ if (dir == null) {
+ File rebaseApply = new File(repoDirectory, REBASE_APPLY);
+ if (rebaseApply.exists()) {
+ dir = rebaseApply;
+ } else {
+ File rebaseMerge = new File(repoDirectory, REBASE_MERGE);
+ dir = rebaseMerge;
+ }
+ }
+ return dir;
+ }
+
+ public String readFile(String name) throws IOException {
+ return readFile(getDir(), name);
+ }
+
+ public void createFile(String name, String content) throws IOException {
+ createFile(getDir(), name, content);
+ }
+
+ public File getFile(String name) {
+ return new File(getDir(), name);
+ }
+
+ private static String readFile(File directory, String fileName)
+ throws IOException {
+ byte[] content = IO.readFully(new File(directory, fileName));
+ // strip off the last LF
+ int end = content.length;
+ while (0 < end && content[end - 1] == '\n')
+ end--;
+ return RawParseUtils.decode(content, 0, end);
+ }
+
+ private static void createFile(File parentDir, String name,
+ String content)
+ throws IOException {
+ File file = new File(parentDir, name);
+ FileOutputStream fos = new FileOutputStream(file);
+ try {
+ fos.write(content.getBytes(Constants.CHARACTER_ENCODING));
+ fos.write('\n');
+ } finally {
+ fos.close();
+ }
+ }
+ }
}