I wanted to stash untracked files, but I keep passing the wrong option. To me this sounds right:
git stash save [-a|--all]
but this in fact stashes ignored files as well. The correct one is:
git stash save [-u|--include-untracked]
When I run git stash save -a
and try to git stash pop
it, I get countless errors for all ignored files:
path/to/file1.ext already exists, no checkout
path/to/file1.ext already exists, no checkout
path/to/file1.ext already exists, no checkout
...
Could not restore untracked files from stash
so the command fails.
How do I get my tracked and untracked stashed changes back? git reflog
doesn't store stash commands.
Without fully understanding why the problem occurs, I found a quick solution:
The
--no-color
option removes any colors from the diff output, because they screw up thegit apply
command.However, it would be great if someone could edit this answer, providing the explanation why
git stash pop
fails.TL;DR version:
You need the directory to be clean (in
git clean
terms) for the stash to apply properly. This means runninggit clean -f
, or evengit clean -fdx
, which is kind of an ugly thing to have to do, since some of the untracked or untracked-and-ignored files/directories may be items you want to keep, rather than deleting entirely. (If so, you should move them outside your work-tree instead ofgit clean
-ing them away. Remember, the files thatgit clean
removes are precisely those that you can't get back from Git!)To see why, look at step 3 in the "apply" description. Note that there is no option to skip the untracked and/or ignored files in a stash.
Basic facts about the stash itself
When you use
git stash save
with either-u
or-a
, the stash script writes its "stash bag" as a three-parent commit rather than the usual two-parent commit.Diagrammatically, the "stash bag" normally looks like this, in terms of the commit graph:
The
o
s are any old ordinary commit nodes, as isC
. NodeC
(for Commit) has a letter so we can name it: it's where the "stash bag" hangs from.The stash bag itself is the little triangular bag hanging from
C
, and it contains two commits:w
is the work-tree commit andi
is the index commit. (Not shown, because it's just hard to diagram, is the fact thatw
's first parent isC
and its second parent isi
.)With
--untracked
or--all
there's a third parent forw
, so the diagram looks more like this:(these diagrams really need to be images so that they can have arrows, rather than ASCII-art where arrows are tough to include). In this case,
stash
is commitw
,stash^
is commitC
(still alsoHEAD
),stash^2
is commiti
, andstash^3
is commitu
, which contains the "untracked" or even "untracked and ignored" files. (It's not actually important, as far as I can tell, but I'll add here thati
hasC
as a parent commit, whileu
is a parentless, or root, commit. There seems to be no particular reason for this, it's just how the script does things, but it explains why the "arrows" (lines) are as they are in the diagram.)The various options at
save
timeAt save time, you can specify any or all of the following options:
-p
,--patch
-k
,--keep-index
,--no-keep-index
-q
,--quiet
-u
,--include-untracked
-a
,--all
Some of these imply, override, or disable others. Using
-p
, for instance, completely changes the algorithm the script uses to build the stash, and also turns on--keep-index
, forcing you to use--no-keep-index
to turn it off if you don't want that. It is incompatible with-a
and-u
and will error-out if any of those are given.Otherwise, between
-a
and-u
, whichever one is set last is retained.At this point the script creates either one or two commits:
C
-u
or-a
, a parentless commit containing (only) either untracked files, or all (untracked and ignored) files.The
stash
script then saves your current work tree. It does this with a temporary index file (basically, a fresh staging area). With-p
, the script reads out theHEAD
commit into the new staging area, then effectively1 runsgit add -i --patch
, so that this index winds up with the patches you select. Without-p
, it just diffs the work directory against the stashed index to find changed files.2 In either case it writes a tree object from the temporary index. This tree will be the tree for commitw
.As its last stash-creation step, the script uses the tree just saved, the parent commit
C
, the index commit, and the root commit for untracked files if it exists, to create the final stash commitw
. However, the script then takes several more steps that affect your work directory, depending on whether you're using-a
,-u
,-p
, and/or--keep-index
(and remember that-p
implies--keep-index
):With
-p
:"Reverse-patch" the work directory to remove the difference between
HEAD
and the stash. In essence, this leaves the work-directory with only those changes not stashed (specifically, those not in commitw
; everything in commiti
is ignored here).Only if you specified
--no-keep-index
: rungit reset
(with no options at all, i.e.,git reset --mixed
). This clears out the "to be committed" state for everything, without changing anything else. (Of course, any partial changes you had staged before runninggit stash save -p
, withgit add
orgit add -p
, are saved in commiti
.)Without
-p
:Run
git reset --hard
(with-q
if you specified that too). This sets the work tree back to the state in theHEAD
commit.Only if you specified
-a
or-u
: rungit clean --force --quiet -d
(with-x
if-a
, or without it if-u
). This removes all the untracked files, including untracked directories; with-x
(i.e., under-a
mode), it also removes all the ignored files.Only if you specified
-k
/--keep-index
: usegit read-tree --reset -u $i_tree
to "bring back" the stashed index as "changes to be committed" that also appear in the work tree. (The--reset
should have no effect since step 1 cleared out the work tree.)The various options at
apply
timeThe two main sub-commands that restore a stash are
apply
andpop
. Thepop
code just runsapply
and then, if theapply
succeeds, runsdrop
, so in effect, there's really justapply
. (Well, there is alsobranch
, which is a little more complicated—but in the end, it too usesapply
.)When you apply a stash—any "stash-like object", really, i.e., anything that the stash script can treat as a stash-bag—there are only two stash-specific options:
-q
,--quiet
--index
(not--keep-index
!)Other flags are accumulated, but are promptly ignored anyway. (The same parsing code is used for
show
, and here the other flags are passed on togit diff
.)Everything else is controlled by the contents of the stash-bag and the state of the work-tree and index. As above, I'll use the labels
w
,i
, andu
to denote the various commits in the stash, andC
to denote the commit from which the stash-bag hangs.The
apply
sequence goes like this, assuming all goes well (if something fails early, e.g., we are in the middle of a merge, orgit apply --cached
fails, the script errors-out at that point):--index
: diff commiti
against commitC
, pipe togit apply --cached
, save the resulting tree, and usegit reset
to unstage itu
exists: usegit read-tree
andgit checkout-index --all
with a temporary index, to recover theu
treegit merge-recursive
to merge the tree forC
(the "base") with that written in step 1 ("updated upstream") and the tree inw
("stashed changes")After this point it gets a bit complicated :-) as it depends on whether the merge in step 4 went well. But first let's expand the above a little.
Step 1 is pretty easy: the script just runs
git write-tree
, which fails if there are unmerged entries in the index. If the write-tree works the result is a tree ID ($c_tree
in the script).Step 2 is a more complicated as it checks not only the
--index
option but also that$b_tree != $i_tree
(i.e., that there is a difference between the tree forC
and the tree fori
), and that$c_tree
!=$i_tree
(i.e., that there is a difference between the tree written out in step 1, and the tree fori
). The test for$b_tree != $i_tree
makes sense: it's checking whether there's any change to apply. If there's no change—if the tree fori
matches that forC
—there's no index to restore, and--index
is not needed after all. However, if$i_tree
matches$c_tree
, that merely means that the current index already contains the changes to be restored via--index
. It's true that, in this case, we don't want togit apply
those changes; but we do want them to remain "restored". (Maybe that's the point of the code I don't quite understand below. It seems more likely that there is a slight bug here, though.)In any case, if step 2 needs to run
git apply --cached
, it also runsgit write-tree
to write the tree, saving this in the script's$unstashed_index_tree
variable. Otherwise$unstashed_index_tree
is left empty.Step 3 is where things go wrong in an "unclean" directory. If the
u
commit exists in the stash, the script insists on extracting it, butgit checkout-index --all
will fail if any of those files would be overwritten. (Note that this is done with a temporary index file, which is removed afterward: step 3 does not use the normal staging area at all.)(Step 4 uses three "magic" environment variables that I have not seen documented:
$GITHEAD_t
provides the "name" of the trees being merged. To rungit merge-recursive
, the script supplies four arguments:$b_tree
--
$c_tree
$w_tree
. As already noted these are the trees for the base commitC
, the index-at-start-of-apply
, and the stashed work commitw
. To get string-names for each of these trees,git merge-recursive
looks in the environment for names formed by prependingGITHEAD_
to the raw SHA-1 for each tree. The script does not pass any strategy arguments togit merge-recursive
, nor let you choose any strategy other thanrecursive
. Probably it should.)If the merge has a conflict, the stash script runs
git rerere
(q.v.) and, if--index
, tells you that the index was not restored and exits with the merge-conflict status. (As with other early exits, this prevents apop
from dropping the stash.)If the merge succeeds, though:
If we have a
$unstashed_index_tree
—i.e., we're doing--index
, and all those other tests in step 2 passed too—then we need to restore the index state created in step 2. In this case a simplegit read-tree $unstashed_index_tree
(with no options) does the trick.If we don't have something in
$unstashed_index_tree
, the script usesgit diff-index --cached --name-only --diff-filter=A $c_tree
to find files to add, runsgit read-tree --reset $c_tree
to do a single-tree merge against the original saved index, and thengit update-index --add
with the file names from the earlierdiff-index
. I'm not really sure why it goes to these lengths (there is a hint in thegit-read-tree
man page, about avoiding false hits for modified files, that might explain it), but that's what it does.Last, the script runs
git status
(with output sent to/dev/null
for-q
mode; not sure why it runs at all under-q
).A few words on
git stash branch
If you're having trouble applying a stash, you can turn it into a "real branch", which makes it guaranteed-to-restore (except, as usual, for the problem of a stash containing a commit
u
not applying unless you clean out unstaged and maybe even ignored files first).The trick here is to start by checking out commit
C
(e.g.,git checkout stash^
). This of course results in a "detached HEAD", so you need to create a new branch, which you can combine with the step that checks out commitC
:Now you can apply the stash, even with
--index
, and it should work since it will be applying to the same commit the stash-bag hangs from:At this point any earlier staged changes should be staged again, and any earlier unstaged (but tracked) files will have their unstaged-but-tracked changes in the work directory. It's safe to drop the stash now:
Using:
simply does the above sequence for you. It literally runs
git checkout -b
, and if that succeeds, applies the stash (with--index
) and then drops it.After this is done, you can commit the index (if you want to), then add and commit the remaining files, to make two (or one if you leave out the first, index, commit) "regular" commits on a "regular" branch:
and you've converted the stash-bag
i
andw
commits to ordinary, on-branch commitsI
andW
.1More correctly, it runs
git add-interactive --patch=stash --
, which directly invokes the perl script for interactive adding, with special magic set for stashing. There are a few other magic--patch
modes; see the script.2There's a very small bug here: git reads
$i_tree
, the committed index's tree, into the temporary index, but then diffs the work directory againstHEAD
. This means that if you changed some filef
in the index, then changed it back to match theHEAD
revision, the work-tree stored underw
in the stash-bag contains the index version off
instead of the work-tree version off
.