Beginner 7 min · March 30, 2026
Git Stash: Save and Restore Work in Progress

Git Stash: Untracked Files Follow Branches – CI Failures

git stash ignores untracked files.

N
Naren Founder & Principal Engineer

20+ years shipping production infrastructure and CI/CD at scale. Written from production experience, not tutorials.

Follow
Production
production tested
May 24, 2026
last updated
1,663
articles · all by Naren
 ● Production Incident 🔎 Debug Guide ⚙ Triage Commands
Quick Answer
  • git stash push -m 'description': saves tracked changes with a label
  • git stash -u: includes untracked files (new files not yet git add'd)
  • git stash pop: restores and removes from stack — use when confident no conflicts
  • git stash apply: restores but keeps in stack — use when conflicts are possible
✦ Definition~90s read
What is Git Stash?

Git stash is a local-only mechanism that temporarily shelves uncommitted changes in your working directory, letting you switch branches or contexts without committing half-baked work. It solves the immediate problem of needing to pivot—say, to fix a production bug while mid-feature—without losing your progress or creating messy WIP commits.

Git stash is a clipboard for your uncommitted work.

The stash lives as a stack of commits in .git/refs/stash and its reflog, not in any remote, meaning it's purely local and ephemeral by design. The trap? By default, git stash only stashes tracked files—untracked and ignored files stay put, silently following you across branches and causing CI failures when stale build artifacts or configs pollute a different branch's environment.

Alternatives like git worktree let you maintain multiple working directories simultaneously, avoiding the stash entirely, while git commit --allow-empty or temporary branches offer more durable context switches. For untracked files, you need git stash -u or git stash --include-untracked to capture them, or git stash --all to also grab ignored files—but use the latter with caution, as it can wipe build caches.

The stash is a convenience tool, not a safety net; it's best for short-lived interruptions, not long-term storage, and it's a common source of subtle bugs when developers assume it captures everything.

Plain-English First

Git stash is a clipboard for your uncommitted work. You're mid-feature when someone drops an urgent bug on your desk. Instead of committing half-finished code or losing your changes, you stash them — Git packs up your working directory changes into a temporary storage area, restores you to a clean HEAD, and waits. Fix the bug, deploy the hotfix, then come back and unstash. Your work resumes exactly where you left it.

git stash shelves uncommitted working directory and index changes into a stack. It is the standard tool for context switching — you stash your current work, switch to a hotfix branch, then unstash when you return. The core failure mode: stash pop deletes the stash even on merge conflicts, leaving you with no backup. Use apply + manual drop when conflicts are possible.

The second failure mode: stash does not include untracked files by default. New files you have not git add'd remain in the working directory when you switch branches. They get committed to the wrong branch. Use stash -u to include them.

Stashes are global to the repository, not scoped to a branch. You can pop a stash made on feature/x onto main. This is sometimes intentional and sometimes a very bad day. Always use stash push -m 'description' to label your stashes — unnamed stashes become mystery boxes within 24 hours.

Git Stash: The Hidden Trap of Untracked Files

Git stash temporarily shelves changes in your working directory, letting you switch branches without committing half-finished work. The core mechanic: git stash records tracked file modifications and staged changes, then reverts your working tree to match HEAD. But by default, it ignores untracked and ignored files — a detail that routinely causes CI failures.

When you stash, Git creates a stash entry as a special commit-like object reachable via stash@{0}. Crucially, the stash is not branch-specific; it lives on the refs/stash reflog. Applying a stash merges its changes into the current working tree, which can produce conflicts if the branch has diverged. Untracked files remain untouched unless you explicitly use git stash -u or git stash --include-untracked.

The practical danger: a developer stashes changes, switches to a feature branch, builds, and commits — but untracked files (like generated configs or IDE artifacts) from the original branch remain in the working directory. Those files get committed, polluting the branch and breaking CI builds that expect a clean environment. Use git stash -u when you need a truly clean slate, or git stash --all to also clear ignored files.

Stash ≠ Clean Slate
Default git stash leaves untracked files behind. Always use git stash -u if you need to guarantee no stray files follow you to another branch.
Production Insight
A developer stashes work on main, switches to a release branch, and builds — untracked .env files from main leak into the release branch.
CI fails with 'unexpected environment configuration' because the build script detects files that shouldn't exist in that branch.
Rule: before switching branches, run git status and explicitly stash untracked files with -u if any are present.
Key Takeaway
Default git stash does not touch untracked or ignored files — they persist across branches.
Always use git stash -u (or --all) when you need a fully clean working directory.
Stash entries are global, not branch-scoped — applying to a different branch can cause merge conflicts.
Git Stash: Untracked Files Follow Branches THECODEFORGE.IO Git Stash: Untracked Files Follow Branches Flow from stashing to restoring with untracked file trap Basic Stash: Save Work Stash includes tracked changes by default Stash Untracked Files Use 'git stash -u' to include untracked Stash Pop vs Apply Pop removes stash; apply keeps it Stash Partial Changes Use -p to interactively select hunks Stash Lives in .git/refs/stash Stored as a ref, not a branch ⚠ Untracked files follow branches when stashed Always use -u or --include-untracked to avoid CI failures THECODEFORGE.IO
thecodeforge.io
Git Stash: Untracked Files Follow Branches
Git Stash

Basic Stash: Save and Restore Work

The core workflow is three commands. git stash shelves everything in your working directory and index that differs from HEAD. git stash pop restores the most recent stash and removes it from the stack. git stash apply restores without removing — useful when you want to apply the same stash to multiple branches.

One thing that surprises developers coming from other VCS systems: git stash by default only stashes tracked files. If you created a new file that hasn't been git add-ed yet, it won't be stashed. It'll sit there in your working directory when you switch branches, which can lead to unintended commits on the wrong branch. Use git stash -u (or --include-untracked) to include new files.

The stash is a stack — each git stash pushes a new entry. git stash pop pops the top (most recent). This is correct behaviour for the common case but becomes confusion when you stash, do some more work, stash again, and then try to figure out which stash has what.

io/thecodeforge/git/StashBasics.shBASH
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
# io.thecodeforge — Git Stash Basics

# ─────────────────────────────────────────────────────────────
# Save current work to stash (tracked files only)
# ─────────────────────────────────────────────────────────────
git stash

# Save with a descriptive message — always do this
git stash push -m "WIP: payment retry exponential backoff"

# Include untracked files (new files not yet git add'd)
git stash push -u -m "WIP: new PaymentRetryService class"

# ─────────────────────────────────────────────────────────────
# Restore most recent stash AND remove it from the stack
# ─────────────────────────────────────────────────────────────
git stash pop

# Restore most recent stash but KEEP it in the stack
git stash apply

# ─────────────────────────────────────────────────────────────
# List all stashes
# ─────────────────────────────────────────────────────────────
git stash list
# Output:
# stash@{0}: On feature/payment-retry: WIP: payment retry exponential backoff
# stash@{1}: On main: WIP: hotfix debug logging

# ─────────────────────────────────────────────────────────────
# Restore a specific stash by index
# ─────────────────────────────────────────────────────────────
git stash pop stash@{1}
git stash apply stash@{1}

# ─────────────────────────────────────────────────────────────
# See what's in a stash before applying
# ─────────────────────────────────────────────────────────────
git stash show stash@{0}
git stash show -p stash@{0}   # Full diff
Output
Saved working directory and index state On feature/payment-retry: WIP: payment retry exponential backoff
# git stash list:
stash@{0}: On feature/payment-retry: WIP: payment retry exponential backoff
stash@{1}: On main: WIP: hotfix debug logging
# git stash show stash@{0}:
src/main/java/io/thecodeforge/payment/PaymentRetryService.java | 47 +++++++++
1 file changed, 47 insertions(+)
Stash Is a Stack, Not a Branch
  • Stash is a stack — push adds, pop removes the top
  • Stashes are global — not attached to any branch
  • You can pop a stash made on feature/x onto main (or any branch)
  • This flexibility is powerful but dangerous — always check which branch you are on before popping
Production Insight
The biggest production failure with basic stash: developers stash without -u, then run git clean on the wrong branch thinking it will remove build artifacts. git clean removes all untracked files — including the untracked files that were not stashed. The work is gone. There is no reflog for untracked files. The fix: always use stash -u when you have new files. And never run git clean -f without first running git clean -n (dry run) to see what will be deleted.
Key Takeaway
git stash saves tracked changes by default. Use -u to include untracked files. Always label stashes with push -m 'description'. Stashes are global to the repo — not scoped to any branch.

Stash Pop vs Apply: When to Use Each

The difference matters more than most tutorials let on.

git stash pop = apply + delete from stack. Use this 95% of the time. The moment you successfully restore your work, the stash entry is gone. Clean, no accumulation.

git stash apply = apply but leave in stack. Use this when: you want to apply the same changes to multiple branches (e.g. a config change you need on both a feature branch and a hotfix branch), or when you're not 100% sure the stash applies cleanly and you want to keep it as a backup while you verify.

There's one critical behaviour difference: if git stash pop encounters a conflict, it still removes the stash from the stack before throwing the conflict. You're now in a conflicted state with the stash gone. This is a Git footgun — you want git stash apply when there's any chance of conflicts, so the stash survives even if the apply fails.

io/thecodeforge/git/StashPopVsApply.shBASH
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
# io.thecodeforge — Stash Pop vs Apply

# ─────────────────────────────────────────────────────────────
# Scenario: you might have conflicts — use apply, not pop
# ─────────────────────────────────────────────────────────────
git stash apply stash@{0}

# If it applied cleanly with no conflicts, manually drop the stash
git stash drop stash@{0}

# If there ARE conflicts, resolve them, then drop manually
# The stash is still in the stack — you haven't lost anything
git status                # Shows conflicted files
# ... resolve conflicts ...
git add .
git stash drop stash@{0}  # Now safe to remove

# ─────────────────────────────────────────────────────────────
# Apply same stash to two branches
# ─────────────────────────────────────────────────────────────
git checkout feature/branch-a
git stash apply stash@{0}    # Applied to branch A — stash still exists
git checkout feature/branch-b
git stash apply stash@{0}    # Applied to branch B too
git stash drop stash@{0}     # Now clean up

# ─────────────────────────────────────────────────────────────
# Clear ALL stashes (nuclear — use with care)
# ─────────────────────────────────────────────────────────────
git stash clear
Output
# Conflict during apply:
Auto-merging src/main/java/io/thecodeforge/payment/PaymentService.java
CONFLICT (content): Merge conflict in src/main/java/io/thecodeforge/payment/PaymentService.java
# Stash is still in stack — safe to resolve then drop
git stash pop deletes the stash even on conflict
  • pop deletes the stash from the stack before you resolve conflicts
  • If the apply fails, you have no stash backup — the entry is gone
  • Use apply when conflicts are possible. Then drop manually after resolving.
  • This is one of the most common Git footguns in production.
Production Insight
The pop-vs-apply decision is about risk tolerance. pop is faster (one command) but has a catastrophic failure mode: if the apply fails, the stash is already deleted from the stack. You can recover via reflog (stash entries are commits internally), but most developers do not know this. The production rule: use apply when you are stashing across branches, when the target branch has changed significantly since you stashed, or when you are not 100% sure the stash applies cleanly. Use pop only when you are confident the apply will succeed.
Key Takeaway
pop = apply + delete. apply = apply + keep. Use apply when conflicts are possible — the stash survives a bad merge. Use pop only when confident the apply will succeed. Always apply, verify, then drop manually for safety.

Stashing Partial Changes and Specific Files

Sometimes you only want to stash part of your changes — for example, you've been working on two unrelated things in the same branch (happens to everyone, don't judge) and you want to commit one thing cleanly before stashing the other.

git stash push -- <path> stashes only the specified files. Everything else stays in your working directory.

git stash push -p (patch mode) drops you into an interactive hunk-by-hunk selection — exactly like git add -p. You choose yes/no for each chunk. This is powerful but slow for large changesets.

io/thecodeforge/git/StashPartial.shBASH
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
# io.thecodeforge — Stashing Partial Changes and Specific Files

# ─────────────────────────────────────────────────────────────
# Stash only specific files
# ─────────────────────────────────────────────────────────────
git stash push -m "WIP: retry config" -- src/main/java/io/thecodeforge/payment/RetryConfig.java

# Stash multiple specific files
git stash push -m "WIP: two related files" \
  -- src/main/java/io/thecodeforge/payment/RetryConfig.java \
  -- src/main/java/io/thecodeforge/payment/RetryPolicy.java

# ─────────────────────────────────────────────────────────────
# Interactive patch mode — choose which hunks to stash
# ─────────────────────────────────────────────────────────────
git stash push -p -m "WIP: partial changes"
# Git shows each change hunk and asks: stash this hunk? [y/n/q/a/d/?]

# ─────────────────────────────────────────────────────────────
# Create a branch from a stash (useful for long-lived stashes)
# ─────────────────────────────────────────────────────────────
git stash branch feature/retry-config stash@{0}
# Creates new branch at the commit where you stashed, applies the stash, drops it
Output
Saved working directory and index state On feature/payment-retry: WIP: retry config
# Only RetryConfig.java was stashed
# RetryPolicy.java remains in working directory
git stash branch: Convert a Stash into a Branch
  • git stash branch <name> <stash>: creates branch at stash point, applies stash, drops it
  • Use when a stash has been sitting for days and has become important work
  • Branches are visible in git branch -a. Stashes are only visible in git stash list.
  • If a stash survives for weeks, it should have been a branch from the start.
Production Insight
Partial stashing is the fix for the 'I committed two unrelated changes in one commit' problem — but in reverse. Instead of committing everything and trying to split later, you stash the changes you do not want to commit right now, commit the rest cleanly, then unstash. This produces cleaner commits than interactive rebase for the common case of 'I was working on feature X and accidentally started working on feature Y in the same branch'.
Key Takeaway
git stash push -- <path> stashes only specific files. git stash push -p gives hunk-by-hunk control. git stash branch converts a long-lived stash into a proper branch. Use partial stashing to separate unrelated changes before committing.

Git Stash vs Git Reset: The Nuclear Option vs The Snooze Button

Junior devs treat git stash like a magical undo button. It's not. Let's kill this confusion right now.

Git reset --hard throws away your working directory changes permanently. That's the nuclear option. Git stash temporarily shelves changes so you can pick them up later. That's the snooze button.

When you're mid-refactor and a critical production bug drops in your lap, you stash your messy working tree, fix the bug, then pop the stash. If you had used reset, those refactored files are gone. Forever.

Here's the real-world distinction: reset moves the HEAD pointer and optionally overwrites the index and working tree. Stash is a stack of compressed delta objects stored in .git/refs/stash. Reset is destructive by design. Stash is a save-state for your incomplete work.

If you're using git stash as a habit because you're scared of commits, you're doing it wrong. Commit early, commit often. Only stash when you cannot commit — dirty debug code, half-failed experiments, or when you absolutely must switch branches without a second commit.

ResetVsStash.ymlYAML
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
// io.thecodeforge — devops tutorial

// Scenario: You're refactoring payment-api and need to fix P1 bug

# Wrong way — permanent loss
git reset --hard
# All refactoring work gone, customer data models deleted

# Right way — safe shelving
git stash push -m "payment-api refactor WIP"
# Saved as stash@{0}: On payment-api-wip: payment-api refactor WIP

git checkout main
git pull origin main
# Fix the emergency bug, commit, push

git checkout payment-api-wip
git stash pop
# Back to work, zero lost changes
Output
Saved working directory and index state On payment-api-wip: payment-api refactor WIP
HEAD is now at a1b2c3d... checkpoint before refactoring
Production Trap:
git reset --hard with uncommitted work = debugging your own carelessness at 2 AM. If you need to stash across multiple branches, name your stashes. Unnamed stashes are how you lose 3 hours of work.
Key Takeaway
Reset is for committed history surgery. Stash is for uncommitted work-in-progress. Never use one where the other belongs.

Where Does Git Stash Actually Live? (.git/refs/stash Exposed)

Every stash is a commit object. Not a temp file. Not a config entry. A real, bonafide Git commit with a parent tree.

Look inside .git/refs/stash and you'll find nothing but a SHA. Run 'git log --oneline stash@{0}' and you'll see a commit history of your stashes. That's because stash creates a merge commit with three parents: the original HEAD, the index state, and the working tree state. This is why you can branch off a stash, diff it, or even merge it.

Multiple stashes stack on .git/refs/stash as a reflog. 'git stash list' shows you the stack. 'git stash drop' removes the top commit from that reflog. 'git stash clear' nukes the entire reflog.

There's no magic repository in the sky. It's all refs and commits in your local .git directory. That also means stashing is purely local — no remote, no backup, no safety net. If your laptop catches fire, so do all your stashes.

Knowing this changes how you use stash. You can cherry-pick from a stash. You can merge stash changes. You can even push a branch from a stash when you realize it's a feature, not a quick fix.

StashInternals.ymlYAML
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
// io.thecodeforge — devops tutorial

// Visualize stash as real commits

# See stash commits with full history
git log --oneline --graph stash@{0}

# Output:
# *   3a4b5c6 WIP on payment-api-wip
# |\  
# | * 2b3c4d5 index on payment-api-wip: checkpoint
# |/  
# * 1a2b3c4 checkpoint before refactoring

# Diff a specific file from stash
git diff stash@{0} -- src/payment/api/validators.go

# Create a branch from a stash (saves it permanently)
git stash branch production-hotfix/payment-fix stash@{2}
Output
* 3a4b5c6 WIP on payment-api-wip
|\
| * 2b3c4d5 index on payment-api-wip: checkpoint
|/
* 1a2b3c4 checkpoint before refactoring
Senior Shortcut:
Stale stashes older than 30 days are tech debt. Review your 'git stash list' weekly. If a stash survives two sprints, create a branch from it or delete it. Dead stashes hide active bugs.
Key Takeaway
Stashes are local commits with a three-parent structure in .git/refs/stash. No remote backup means treat them as ephemeral — branch them if they matter.

Git Stash Best Practices

Stashing is a safety net, not a workflow crutch. The primary reason to stash is to temporarily set aside uncommitted changes when you need to switch branches, pull updates, or fix an urgent bug. Never stash as a substitute for meaningful commits. Before stashing, commit your work if it's logically complete—stashes are ephemeral and easy to lose. Use descriptive stash messages: git stash push -m "WIP: refactor auth module" saves hours of confusion later. Avoid piling up multiple stashes; each stash is a hidden breadcrumb that accumulates technical debt. Instead, pop and integrate each stash immediately after the interruption passes. Untracked files are not stashed by default unless you add -u or --include-untracked. This hidden trap causes accidental data loss when switching branches expecting all changes to travel with you. Finally, never keep a stash older than a few days—it signals that either the work should have been committed or it's dead code waiting to bite.

stash-best-practices.ymlYAML
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
// io.thecodeforge — devops tutorial

stash_rules:
  messages: required
  max_age_hours: 48
  pop_before_new_work: true
  untracked_flag: -u

commands:
  save_work:
    cmd: git stash push -m "refactor: extract billing service"
  restore:
    cmd: git stash pop
  check_stash_list:
    cmd: git stash list
Production Trap:
If you stash without -u and later drop the stash, untracked files are gone forever. Always verify with git status before stashing.
Key Takeaway
Stash only temporary interruptions, label everything, pop immediately, and never rely on stashes for long-term history.

Git Fundamentals

Git's core architecture is a directed acyclic graph of snapshots, not diffs. Every commit is a full snapshot of your tracked files at that moment—this is why operations like checkout and reset are fast. The three main areas working directory, staging index, and commit history define your workflow. Changes move from working tree to staging with git add, then to a commit with git commit. Git stash is a fourth temporary area: it saves the dirty state of your working directory and index as two separate commits stored under .git/refs/stash. Understanding this triad prevents confusion: git reset --soft only moves HEAD, --mixed resets the index, and --hard wipes working directory changes. Branches are lightweight pointers to commits; merging creates a new commit with two parents, rebasing rewrites history by replaying commits. Git's distributed nature means every clone has the full history. This makes local operations instant but adds complexity with remotes. Use git fetch before merging to inspect upstream changes without danger. The golden rule: never rewrite public history with rebase or reset --hard.

git-fundamentals.ymlYAML
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
// io.thecodeforge — devops tutorial

areas:
  working_directory:
    state: dirty or clean
  staging_index:
    command: git add
  commit_history:
    command: git commit

stash_structure:
  path: .git/refs/stash
  type: two parent commits

key_rules:
  fetch_before_merge: true
  no_rewrite_public: true
Production Trap:
Resetting a published branch with --hard strips commits from everyone else's view. Prefer git revert for safety.
Key Takeaway
Git is snapshot-based, versioned via pointers, and your local stash is just hidden commits—never rewrite shared history.
● Production incidentPOST-MORTEMseverity: high

Stash Pop on Wrong Branch: Untracked Files Committed to Main

Symptom
The CI pipeline on main failed after a hotfix merge. The build included a new file — PaymentRetryService.java — that was not part of the hotfix. The file contained incomplete feature code with compilation errors. No developer had intentionally committed this file to main.
Assumption
The developer assumed that git stash would save everything in their working directory. They did not realize that untracked files (files not yet git add'd) are excluded from stash by default. When they switched to main, the untracked PaymentRetryService.java remained in the working directory. During the hotfix, they ran git add . which staged the orphaned file. It was committed to main with the hotfix.
Root cause
1. Developer was working on feature/payment-retry and created PaymentRetryService.java but had not run git add on it yet. 2. They ran git stash to save their work — but stash only saves tracked files by default. 3. PaymentRetryService.java was untracked, so it was NOT stashed. It remained in the working directory. 4. Developer switched to main: git checkout main. 5. PaymentRetryService.java (the untracked file) followed them to main. 6. During the hotfix, they ran git add . which staged everything — including PaymentRetryService.java. 7. The file was committed to main with incomplete, non-compiling feature code. 8. CI built from main and failed on the compilation error.
Fix
1. Immediate: revert the offending commit on main: git revert <hotfix-commit-hash>. 2. The developer switched back to feature/payment-retry and stashed correctly: git stash push -u -m 'WIP: PaymentRetryService'. 3. Team rule: always use git stash push -u when stashing with untracked files. 4. Team rule: never run git add . on main — always add files individually or use git add -p. 5. Added pre-commit hook to warn when new files are staged on main that do not exist on the previous commit.
Key lesson
  • git stash does not include untracked files by default. Use git stash -u to include them.
  • Untracked files follow you across branch switches. They are not part of the stash or the commit — they are just files in the working directory.
  • Never run git add . on main. Stage files individually or use git add -p to review each hunk.
  • Always label stashes with git stash push -m 'description'. Unnamed stashes become mystery boxes.
Production debug guideSystematic recovery paths for lost stashes, wrong-branch pops, and stash conflicts.5 entries
Symptom · 01
Ran git stash pop and hit a merge conflict — stash is gone from the stack
Fix
1. git stash pop deletes the stash even on conflict. The stash is no longer in git stash list. 2. Check reflog: git reflog — find the stash application entry. 3. The stash commit object still exists: git fsck --no-reflogs | grep dangling — find dangling commit objects. 4. Recover: git stash apply <hash> using a hash that looks like your stash. 5. Prevention: use git stash apply when conflicts are possible. Then git stash drop manually after resolving.
Symptom · 02
Stashed changes applied to the wrong branch — working directory is now wrong
Fix
1. Do NOT commit. The changes are staged/unstaged but not committed. 2. Undo: git reset --hard HEAD to restore the branch to its pre-pop state. 3. Switch to the correct branch: git checkout <correct-branch>. 4. Apply there: git stash apply stash@{0} (the stash is still in the stack if you used apply, or recover via reflog if you used pop).
Symptom · 03
Cannot find which stash has the changes I need
Fix
1. List all stashes: git stash list — read the messages (this is why you always use -m). 2. If messages are unhelpful: git stash show -p stash@{0} — full diff of each stash. 3. Search stashes for a specific change: git stash show -p stash@{N} | grep 'search-term' for each N. 4. Or: git log --all --oneline --grep='search-term' — stash entries appear in the log.
Symptom · 04
Untracked files committed to the wrong branch after stash + switch
Fix
1. The files were never stashed — they are untracked and followed you across branches. 2. Check: git log --oneline -3 — find the commit that includes the wrong files. 3. Revert: git revert <commit-hash> to undo the commit on the wrong branch. 4. Switch to correct branch: git checkout <correct-branch>. 5. Stash correctly: git stash push -u -m 'WIP: new files' to include untracked files.
Symptom · 05
git stash clear ran accidentally — all stashes are gone
Fix
1. git stash clear removes all stashes with no confirmation. 2. Recovery: git fsck --no-reflogs | grep dangling — find dangling commit objects. 3. Each stash entry is a commit internally. Filter by date: look for commits created around the time you stashed. 4. Apply: git stash apply <hash> for each recovered stash commit. 5. Prevention: never run git stash clear without checking git stash list first.
★ Git Stash Triage Cheat SheetFast recovery for stash-related failures, lost changes, and wrong-branch applications.
Stash pop hit a merge conflict — stash is gone from the stack
Immediate action
Find the stash commit in reflog before it expires (30 days).
Commands
git reflog | grep stash (find the stash application entry)
git fsck --no-reflogs | grep dangling (find dangling commit objects)
Fix now
git stash apply <hash> to recover. Prevention: use apply instead of pop when conflicts are possible.
Stashed changes on wrong branch — working directory has wrong files+
Immediate action
Do not commit. Reset the branch to clean state.
Commands
git reset --hard HEAD (undo the stash application)
git checkout <correct-branch> (switch to where you need the stash)
Fix now
git stash apply stash@{0} on the correct branch. If stash is gone: recover via reflog.
Cannot find which stash has my changes+
Immediate action
Search stash diffs for the specific change.
Commands
git stash list (read messages — this is why you always use -m)
git stash show -p stash@{N} | grep 'search-term' (search each stash diff)
Fix now
Always label stashes: git stash push -m 'description'. Unnamed stashes are mystery boxes.
Untracked files ended up on the wrong branch after stash + switch+
Immediate action
The files were never stashed. Revert the commit that included them.
Commands
git log --oneline -3 (find the commit with the wrong files)
git revert <commit-hash> (undo the commit on the wrong branch)
Fix now
Prevention: always use git stash -u when stashing with untracked files.
git stash clear ran accidentally — all stashes gone+
Immediate action
Find dangling commit objects in the object store.
Commands
git fsck --no-reflogs | grep dangling (find lost stash commits)
git stash apply <hash> (recover each stash commit)
Fix now
Prevention: never run git stash clear without checking git stash list first.
Git Stash Commands Compared
CommandEffectStash Removed?Best For
git stashStash tracked filesN/AQuick context switch
git stash -uStash tracked + untrackedN/AWhen you have new files
git stash popApply + remove from stackYesNormal restore — 95% of cases
git stash applyApply, keep in stackNoRisky merges or multi-branch apply
git stash dropRemove without applyingYesDiscard stash you no longer need
git stash clearRemove ALL stashesYes (all)Spring cleaning — careful
git stash branch <name>Create branch from stashYesLong-lived stashes that deserve a branch

Key takeaways

1
git stash push -m 'description'
always add a message. Unnamed stashes become mystery boxes within 24 hours.
2
Use git stash -u to include untracked files; without it, new files stay in your working directory when you switch branches.
3
Prefer git stash apply over pop when conflicts are possible
pop deletes the stash before you resolve conflicts, leaving you with no backup.
4
git stash branch <name> <stash> converts an old stash into a proper branch
the right move when a stash has grown into more work than you expected.
5
Stashes are global to the repo, not scoped to a branch
applying a stash on the wrong branch is an easy mistake with sometimes messy consequences.
6
Stashes are commits internally
recoverable via git fsck --no-reflogs | grep dangling even after deletion. But do not rely on this as a primary safety net.
INTERVIEW PREP · PRACTICE MODE

Interview Questions on This Topic

FAQ · 6 QUESTIONS

Frequently Asked Questions

01
What is the difference between git stash pop and git stash apply?
02
Does git stash save untracked files?
03
How do I see what's in a stash before applying it?
04
Can I recover a stash I accidentally dropped?
05
How do I recover a stash after git stash pop hit a merge conflict?
06
What happens to stashes after 90 days?
N
Naren Founder & Principal Engineer

20+ years shipping production infrastructure and CI/CD at scale. Written from production experience, not tutorials.

Follow
Verified
production tested
May 24, 2026
last updated
1,663
articles · all by Naren
🔥

That's Git. Mark it forged?

7 min read · try the examples if you haven't

Previous
Git Squash Commits: Combine Multiple Commits into One
15 / 19 · Git
Next
Git Cherry Pick: Apply Commits Across Branches