Stashes are numbered where index 0 is the most recent. New stashes are added to the front of the list to become the new index 0.

Stash references

A stash can be referenced like this:

  • Long reference: stash@{0}, stash@{1}, etc.
  • Short reference: 0, 1, etc.

If you omit the reference, then the index 0 is implied. Which means the most recent stash.

Manage stashes

A workflow for creating, viewing, applying and deleting stashes.

Stash everything including untracked.

$ git stash -u

See what we stashed at stash index 0. First file names and then file changes.

$ git stash show

$ git stash show -p

List

Look at all the stashes.

$ git stash list

View

Look at the stash before it at stash 1. Or whatever index.

$ git stash show 1

Drop

Drop that stash:

$ git stash drop 1

Apply

Then view and apply the stash which was at 2 and which is now shifed to be at 1.

$ git apply 1

Create

Note you cannot stash on a repo with zero commits.

Quick reference

$ git stash
$ # OR
$ git stash push
$ git stash -m 'My message'

Default behavior is to stash staged and unstaged files, but not untracked. You can change this behavior as below.

Flags:

Flag Description
-p or --patch Stash interactively, as with git add --patch
-S or --staged Stash staged changes only. Similar to using git commit except saving to stash instead of a commit.
-k or --keep-index Stash unstaged changes and not staged changes.
-u or --include-untracked Stage untracked changes.
-a or --all Show all changes, including untracked and ignored files.

Note that git stash save is deprecated according to the docs.

Basic

Create stash.

$ git stash

Stash selectively

Based on SO.

Stash by path

From Git 2.13:

$ git stash push -m MESSAGE PATH

e.g.

$ git stash push -m welcome_cart abc.txt

Stash staged

Stage files to stash:

git add abc.txt def.txt

Then stash them, ignoring unstaged:

$ git stash --staged

Stage unstaged

Stage files to ignore:

git add abc.txt def.txt

Then stash everything that is not staged:

$ git stash --keep-index

Everything

By default, stashing will ignore files not tracked by Git.

Stash everything, tracked and not tracked:

$ git stash --include-untracked

Or use add so Git knows about the files (without committing them), then do a plain stash.

$ git add .
$ git stash

If you want to stash ignored files, do this:

$ git stash --all

Message

Give the stash a message.

$ git stash save MESSAGE
$ # e.g.
$ git stash save 'Description of stash'

Show

Basic

List stashes.

$ git stash list
stash@{0}: On foo-bar: My stash name
stash@{1}: On master: Another stash name
stash@{3}: WIP on master: a5a067af Commit message

Note the most recent one at the top of the output will have index 0.

The mesesage of the stash will be shown if possible, or the commit message.

File names

Show files names in a stash.

$ git stash show [STASH_REF]

Diff

Show the diff or “patch” of a stash.

$ git stash show [STASH_REF] -p

Apply

Apply and drop

Apply the stash, then remove from the stashes.

$ git stash pop [STASH_REF]

Apply and keep

Apply the stash, but keep it on the stash pile.

$ git stash apply [STASH_REF]

If you need to, you can get rid of your changes to get back to a clean state, then apply your stash again.

$ git reset --hard
$ git stash apply

Or you can commit your changes and drop the stash, if you no longer need it.

$ git commit .
$ git stash drop

Delete

Delete the stash without applying it.

$ git stash drop [STASH_REF]

Split changed files

If you have say 10 files changed and you want to stash just some of them, you can do this.

  1. Stage the files you want to stage.
     $ git add foo
    
  2. Stash only staged files using the --keep-index flag.
     $ git stash -k
    
  3. Optionally, you can now can stash the files which were not staged, so you’ll end up with two stashes.
     $ git stash