Browse Source

CheckoutCommand: Support checking out ours and theirs

The checkoutPaths body is split into two implementations, depending on
whether we are checking out the index or a branch. This improves
readability, as in the index case we now also need to have access to
DirCacheIterator.

Bug: 390147
Change-Id: I99fd599b25b2ace9bdd84535a56565286a3cb7f1
Signed-off-by: Chris Aniszczyk <zx@twitter.com>
tags/v2.2.0.201212191850-r
Robin Stocker 11 years ago
parent
commit
cdaded26b0

+ 45
- 3
org.eclipse.jgit.test/tst/org/eclipse/jgit/api/PathCheckoutCommandTest.java View File

@@ -47,6 +47,7 @@ import static org.junit.Assert.assertEquals;
import java.io.File;
import java.io.IOException;

import org.eclipse.jgit.api.CheckoutCommand.Stage;
import org.eclipse.jgit.api.errors.JGitInternalException;
import org.eclipse.jgit.dircache.DirCache;
import org.eclipse.jgit.dircache.DirCacheEntry;
@@ -246,10 +247,41 @@ public class PathCheckoutCommandTest extends RepositoryTestCase {
assertEquals("a", read(test2));
}


@Test(expected = JGitInternalException.class)
public void testCheckoutOfConflictingFileShouldThrow()
throws Exception {
// Setup
setupConflictingState();

git.checkout().addPath(FILE1).call();
}

@Test
public void testCheckoutOurs() throws Exception {
setupConflictingState();

git.checkout().setStage(Stage.OURS).addPath(FILE1).call();

assertEquals("3", read(FILE1));
assertStageOneToThree(FILE1);
}

@Test
public void testCheckoutTheirs() throws Exception {
setupConflictingState();

git.checkout().setStage(Stage.THEIRS).addPath(FILE1).call();

assertEquals("Conflicting", read(FILE1));
assertStageOneToThree(FILE1);
}

@Test(expected = IllegalStateException.class)
public void testStageNotPossibleWithBranch() throws Exception {
git.checkout().setStage(Stage.OURS).setStartPoint("master").call();
}

private void setupConflictingState() throws Exception {
git.checkout().setCreateBranch(true).setName("conflict")
.setStartPoint(initialCommit).call();
writeTrashFile(FILE1, "Conflicting");
@@ -260,8 +292,18 @@ public class PathCheckoutCommandTest extends RepositoryTestCase {

git.merge().include(conflict).call();
assertEquals(RepositoryState.MERGING, db.getRepositoryState());
assertStageOneToThree(FILE1);
}

// Now check out the conflicting path
git.checkout().addPath(FILE1).call();
private void assertStageOneToThree(String name) throws Exception {
DirCache cache = DirCache.read(db.getIndexFile(), db.getFS());
int i = cache.findEntry(name);
DirCacheEntry stage1 = cache.getEntry(i);
DirCacheEntry stage2 = cache.getEntry(i + 1);
DirCacheEntry stage3 = cache.getEntry(i + 2);

assertEquals(DirCacheEntry.STAGE_1, stage1.getStage());
assertEquals(DirCacheEntry.STAGE_2, stage2.getStage());
assertEquals(DirCacheEntry.STAGE_3, stage3.getStage());
}
}

+ 134
- 42
org.eclipse.jgit/src/org/eclipse/jgit/api/CheckoutCommand.java View File

@@ -125,6 +125,33 @@ import org.eclipse.jgit.util.FileUtils;
* >Git documentation about Checkout</a>
*/
public class CheckoutCommand extends GitCommand<Ref> {

/**
* Stage to check out, see {@link CheckoutCommand#setStage(Stage)}.
*/
public static enum Stage {
/**
* Base stage (#1)
*/
BASE(DirCacheEntry.STAGE_1),

/**
* Ours stage (#2)
*/
OURS(DirCacheEntry.STAGE_2),

/**
* Theirs stage (#3)
*/
THEIRS(DirCacheEntry.STAGE_3);

private final int number;

private Stage(int number) {
this.number = number;
}
}

private String name;

private boolean force = false;
@@ -137,6 +164,8 @@ public class CheckoutCommand extends GitCommand<Ref> {

private RevCommit startCommit;

private Stage checkoutStage = null;

private CheckoutResult status;

private List<String> paths;
@@ -327,52 +356,19 @@ public class CheckoutCommand extends GitCommand<Ref> {
RevWalk revWalk = new RevWalk(repo);
DirCache dc = repo.lockDirCache();
try {
DirCacheEditor editor = dc.editor();
TreeWalk startWalk = new TreeWalk(revWalk.getObjectReader());
startWalk.setRecursive(true);
TreeWalk treeWalk = new TreeWalk(revWalk.getObjectReader());
treeWalk.setRecursive(true);
if (!checkoutAllPaths)
startWalk.setFilter(PathFilterGroup.createFromStrings(paths));
final boolean checkoutIndex = startCommit == null
&& startPoint == null;
if (!checkoutIndex)
startWalk.addTree(revWalk.parseCommit(getStartPoint())
.getTree());
else
startWalk.addTree(new DirCacheIterator(dc));

final File workTree = repo.getWorkTree();
final ObjectReader r = repo.getObjectDatabase().newReader();
treeWalk.setFilter(PathFilterGroup.createFromStrings(paths));
try {
while (startWalk.next()) {
final ObjectId blobId = startWalk.getObjectId(0);
final FileMode mode = startWalk.getFileMode(0);
editor.add(new PathEdit(startWalk.getPathString()) {
public void apply(DirCacheEntry ent) {
if (checkoutIndex
&& ent.getStage() > DirCacheEntry.STAGE_0) {
UnmergedPathException e = new UnmergedPathException(ent);
throw new JGitInternalException(e.getMessage(), e);
}
ent.setObjectId(blobId);
ent.setFileMode(mode);
File file = new File(workTree, ent.getPathString());
File parentDir = file.getParentFile();
try {
FileUtils.mkdirs(parentDir, true);
DirCacheCheckout.checkoutEntry(repo, file, ent, r);
} catch (IOException e) {
throw new JGitInternalException(
MessageFormat.format(
JGitText.get().checkoutConflictWithFile,
ent.getPathString()), e);
}
}
});
if (isCheckoutIndex())
checkoutPathsFromIndex(treeWalk, dc);
else {
RevCommit commit = revWalk.parseCommit(getStartPoint());
checkoutPathsFromCommit(treeWalk, dc, commit);
}
editor.commit();
} finally {
startWalk.release();
r.release();
treeWalk.release();
}
} finally {
dc.unlock();
@@ -381,6 +377,75 @@ public class CheckoutCommand extends GitCommand<Ref> {
return this;
}

private void checkoutPathsFromIndex(TreeWalk treeWalk, DirCache dc)
throws IOException {
DirCacheIterator dci = new DirCacheIterator(dc);
treeWalk.addTree(dci);

final ObjectReader r = treeWalk.getObjectReader();
DirCacheEditor editor = dc.editor();
while (treeWalk.next()) {
DirCacheEntry entry = dci.getDirCacheEntry();
// Only add one edit per path
if (entry != null && entry.getStage() > DirCacheEntry.STAGE_1)
continue;
editor.add(new PathEdit(treeWalk.getPathString()) {
public void apply(DirCacheEntry ent) {
int stage = ent.getStage();
if (stage > DirCacheEntry.STAGE_0) {
if (checkoutStage != null) {
if (stage == checkoutStage.number)
checkoutPath(ent, r);
} else {
UnmergedPathException e = new UnmergedPathException(
ent);
throw new JGitInternalException(e.getMessage(), e);
}
} else {
checkoutPath(ent, r);
}
}
});
}
editor.commit();
}

private void checkoutPathsFromCommit(TreeWalk treeWalk, DirCache dc,
RevCommit commit) throws IOException {
treeWalk.addTree(commit.getTree());
final ObjectReader r = treeWalk.getObjectReader();
DirCacheEditor editor = dc.editor();
while (treeWalk.next()) {
final ObjectId blobId = treeWalk.getObjectId(0);
final FileMode mode = treeWalk.getFileMode(0);
editor.add(new PathEdit(treeWalk.getPathString()) {
public void apply(DirCacheEntry ent) {
ent.setObjectId(blobId);
ent.setFileMode(mode);
checkoutPath(ent, r);
}
});
}
editor.commit();
}

private void checkoutPath(DirCacheEntry entry, ObjectReader reader) {
File file = new File(repo.getWorkTree(), entry.getPathString());
File parentDir = file.getParentFile();
try {
FileUtils.mkdirs(parentDir, true);
DirCacheCheckout.checkoutEntry(repo, file, entry, reader);
} catch (IOException e) {
throw new JGitInternalException(MessageFormat.format(
JGitText.get().checkoutConflictWithFile,
entry.getPathString()), e);
}
}

private boolean isCheckoutIndex() {
return startCommit == null && startPoint == null;
}

private ObjectId getStartPoint() throws AmbiguousObjectException,
RefNotFoundException, IOException {
if (startCommit != null)
@@ -483,6 +548,7 @@ public class CheckoutCommand extends GitCommand<Ref> {
checkCallable();
this.startPoint = startPoint;
this.startCommit = null;
checkOptions();
return this;
}

@@ -503,6 +569,7 @@ public class CheckoutCommand extends GitCommand<Ref> {
checkCallable();
this.startCommit = startCommit;
this.startPoint = null;
checkOptions();
return this;
}

@@ -522,6 +589,24 @@ public class CheckoutCommand extends GitCommand<Ref> {
return this;
}

/**
* When checking out the index, check out the specified stage (ours or
* theirs) for unmerged paths.
* <p>
* This can not be used when checking out a branch, only when checking out
* the index.
*
* @param stage
* the stage to check out
* @return this
*/
public CheckoutCommand setStage(Stage stage) {
checkCallable();
this.checkoutStage = stage;
checkOptions();
return this;
}

/**
* @return the result, never <code>null</code>
*/
@@ -530,4 +615,11 @@ public class CheckoutCommand extends GitCommand<Ref> {
return CheckoutResult.NOT_TRIED_RESULT;
return status;
}

private void checkOptions() {
if (checkoutStage != null && !isCheckoutIndex())
throw new IllegalStateException(
"Checking out ours/theirs is only possible when checking out index, "
+ "not when switching branches.");
}
}

Loading…
Cancel
Save