aboutsummaryrefslogtreecommitdiffstats
path: root/org.eclipse.jgit/src/org/eclipse/jgit/api
diff options
context:
space:
mode:
Diffstat (limited to 'org.eclipse.jgit/src/org/eclipse/jgit/api')
-rw-r--r--org.eclipse.jgit/src/org/eclipse/jgit/api/ResetCommand.java79
1 files changed, 78 insertions, 1 deletions
diff --git a/org.eclipse.jgit/src/org/eclipse/jgit/api/ResetCommand.java b/org.eclipse.jgit/src/org/eclipse/jgit/api/ResetCommand.java
index 1ddfd6d77a..206d4062b7 100644
--- a/org.eclipse.jgit/src/org/eclipse/jgit/api/ResetCommand.java
+++ b/org.eclipse.jgit/src/org/eclipse/jgit/api/ResetCommand.java
@@ -44,12 +44,17 @@ package org.eclipse.jgit.api;
import java.io.IOException;
import java.text.MessageFormat;
+import java.util.Collection;
+import java.util.LinkedList;
import org.eclipse.jgit.JGitText;
import org.eclipse.jgit.api.errors.JGitInternalException;
import org.eclipse.jgit.dircache.DirCache;
import org.eclipse.jgit.dircache.DirCacheBuilder;
import org.eclipse.jgit.dircache.DirCacheCheckout;
+import org.eclipse.jgit.dircache.DirCacheEditor;
+import org.eclipse.jgit.dircache.DirCacheEntry;
+import org.eclipse.jgit.dircache.DirCacheIterator;
import org.eclipse.jgit.lib.Constants;
import org.eclipse.jgit.lib.ObjectId;
import org.eclipse.jgit.lib.Ref;
@@ -58,6 +63,9 @@ import org.eclipse.jgit.lib.Repository;
import org.eclipse.jgit.lib.RepositoryState;
import org.eclipse.jgit.revwalk.RevCommit;
import org.eclipse.jgit.revwalk.RevWalk;
+import org.eclipse.jgit.treewalk.CanonicalTreeParser;
+import org.eclipse.jgit.treewalk.TreeWalk;
+import org.eclipse.jgit.treewalk.filter.PathFilterGroup;
/**
* A class used to execute a {@code Reset} command. It has setters for all
@@ -103,10 +111,12 @@ public class ResetCommand extends GitCommand<Ref> {
KEEP // TODO not implemented yet
}
- private String ref;
+ private String ref = Constants.HEAD;
private ResetType mode;
+ private Collection<String> filepaths = new LinkedList<String>();
+
/**
*
* @param repo
@@ -157,6 +167,13 @@ public class ResetCommand extends GitCommand<Ref> {
rw.release();
}
+ if (!filepaths.isEmpty()) {
+ // reset [commit] -- paths
+ resetIndexForPaths(commit);
+ setCallable(false);
+ return repo.getRef(Constants.HEAD);
+ }
+
// write the ref
final RefUpdate ru = repo.updateRef(Constants.HEAD);
ru.setNewObjectId(commitId);
@@ -217,10 +234,70 @@ public class ResetCommand extends GitCommand<Ref> {
* @return this instance
*/
public ResetCommand setMode(ResetType mode) {
+ if (!filepaths.isEmpty())
+ throw new JGitInternalException(MessageFormat.format(
+ JGitText.get().illegalCombinationOfArguments,
+ "[--mixed | --soft | --hard]", "<paths>..."));
this.mode = mode;
return this;
}
+ /**
+ * @param file
+ * the file to add
+ * @return this instance
+ */
+ public ResetCommand addPath(String file) {
+ if (mode != null)
+ throw new JGitInternalException(MessageFormat.format(
+ JGitText.get().illegalCombinationOfArguments, "<paths>...",
+ "[--mixed | --soft | --hard]"));
+ filepaths.add(file);
+ return this;
+ }
+
+ private void resetIndexForPaths(RevCommit commit) {
+ DirCache dc = null;
+ final DirCacheEditor edit;
+ try {
+ dc = repo.lockDirCache();
+ edit = dc.editor();
+
+ final TreeWalk tw = new TreeWalk(repo);
+ tw.addTree(new DirCacheIterator(dc));
+ tw.addTree(commit.getTree());
+ tw.setFilter(PathFilterGroup.createFromStrings(filepaths));
+
+ while (tw.next()) {
+ final String path = tw.getPathString();
+ // DirCacheIterator dci = tw.getTree(0, DirCacheIterator.class);
+ final CanonicalTreeParser tree = tw.getTree(1,
+ CanonicalTreeParser.class);
+ if (tree == null)
+ // file is not in the commit, remove from index
+ edit.add(new DirCacheEditor.DeletePath(path));
+ else {
+ // revert index to commit
+ edit.add(new DirCacheEditor.PathEdit(path) {
+ @Override
+ public void apply(DirCacheEntry ent) {
+ ent.setFileMode(tree.getEntryFileMode());
+ ent.setObjectId(tree.getEntryObjectId());
+ ent.setLastModified(0);
+ }
+ });
+ }
+ }
+
+ edit.commit();
+ } catch (IOException e) {
+ throw new RuntimeException(e);
+ } finally {
+ if (dc != null)
+ dc.unlock();
+ }
+ }
+
private void resetIndex(RevCommit commit) throws IOException {
DirCache dc = null;
try {