From: "D. Ben Knoble" <ben.knoble+github@gmail.com>
To: git@vger.kernel.org
Cc: "D. Ben Knoble" <ben.knoble+github@gmail.com>,
Junio C Hamano <gitster@pobox.com>,
Phillip Wood <phillip.wood123@gmail.com>
Subject: [PATCH v3 0/4] Teach git-stash to use --index from config
Date: Sun, 21 Sep 2025 21:39:02 -0400 [thread overview]
Message-ID: <cover.1758505011.git.ben.knoble+github@gmail.com> (raw)
In-Reply-To: <cover.1757982870.git.ben.knoble+github@gmail.com>
Changes from v2:
- use "rational" tests (check interactions between CLI, config; drop
duplicate tests) thanks to Phillip's review
Changes from v1:
- configure --index via config
- drop BreakingChanges related work
With stash.index=true, git-stash(1) command now tries to reinstate the
index by default in the "apply" and "pop" modes. Not doing so creates a
common trap: "git stash apply" is not the reverse of "git stash push"
because carefully staged indices are lost and have to be manually
recreated. OTOH, this mode is not always desirable and may create more
conflicts when applying stashes. Use "--no-index" to disable this behavior.
Cf. <CAPx1GvcxyDDQmCssMjEnt6JoV6qPc5ZUpgPLX3mpUC_4PNYA1w@mail.gmail.com>,
<c5a811ac-8cd3-c389-ac6d-29020a648c87@gmail.com>.
PS I've left some new t3903 tests as copy-pasta for now to get feedback
on the rest of the series; there are bits of that file that could use an
update to the modern style (e.g., not using "test 1 = $(cat file)").
Since some new tests are substantially similar to old tests that use
this style, such cleanup is /probably/ warranted but will delay eyeballs
on the core of this series.
Published-as: https://github.com/benknoble/git/tree/stash-apply-index
v1: https://lore.kernel.org/git/20250510183358.36806-1-ben.knoble+github@gmail.com/
v2: https://lore.kernel.org/git/cover.1757982870.git.ben.knoble+github@gmail.com/
D. Ben Knoble (4):
t3903: reduce dependencies on previous tests
t3905: remove unneeded blank line
stash: refactor private config globals
stash: honor stash.index in apply, pop modes
Documentation/config/stash.adoc | 5 ++++
builtin/stash.c | 17 ++++++++-----
t/t3903-stash.sh | 40 ++++++++++++++++++++++++++++++
t/t3905-stash-include-untracked.sh | 1 -
4 files changed, 56 insertions(+), 7 deletions(-)
Diff-intervalle contre v2 :
1: 1328eb8eac = 1: 1328eb8eac t3903: reduce dependencies on previous tests
2: 8ac06ad62d = 2: 8ac06ad62d t3905: remove unneeded blank line
3: bf0a561ce3 = 3: bf0a561ce3 stash: refactor private config globals
4: 585e124467 ! 4: 8e6cafbf3a stash: honor stash.index in apply, pop modes
@@ builtin/stash.c: static int git_stash_config(const char *var, const char *value,
## t/t3903-stash.sh ##
@@ t/t3903-stash.sh: setup_stash()
- test 1 = $(git show HEAD:file)
+ )
'
-+test_expect_success 'apply stashed changes with stash.index' '
-+ test_config stash.index true &&
-+ git reset --hard HEAD^ &&
-+ echo 5 >other-file &&
-+ git add other-file &&
-+ test_tick &&
-+ git commit -m other-file &&
-+ git stash apply &&
-+ test 3 = $(cat file) &&
-+ test 2 = $(git show :file) &&
-+ test 1 = $(git show HEAD:file)
-+'
-+
- test_expect_success 'apply stashed changes (including index)' '
- git reset --hard HEAD^ &&
- echo 6 >other-file &&
-@@ t/t3903-stash.sh: setup_stash()
- test 1 = $(git show HEAD:file)
- '
-
-+test_expect_success 'drop top stash with stash.index' '
-+ test_config stash.index true &&
-+ git reset --hard &&
-+ git stash list >expected &&
-+ echo 7 >file &&
-+ git stash &&
-+ git stash drop &&
-+ git stash list >actual &&
-+ test_cmp expected actual &&
-+ git stash apply &&
-+ test 3 = $(cat file) &&
-+ test 2 = $(git show :file) &&
-+ test 1 = $(git show HEAD:file)
-+'
-+
- test_expect_success 'drop middle stash' '
- git reset --hard &&
- echo 8 >file &&
-@@ t/t3903-stash.sh: setup_stash()
- test 1 = $(git show HEAD:file)
- '
-
-+test_expect_success 'drop middle stash with stash.index' '
-+ test_config stash.index true &&
-+ git reset --hard &&
-+ echo 8 >file &&
-+ git stash &&
-+ echo 9 >file &&
-+ git stash &&
-+ git stash drop stash@{1} &&
-+ test 2 = $(git stash list | wc -l) &&
-+ git stash apply &&
-+ test 9 = $(cat file) &&
-+ test 1 = $(git show :file) &&
-+ test 1 = $(git show HEAD:file) &&
-+ git reset --hard &&
-+ git stash drop &&
-+ git stash apply &&
-+ test 3 = $(cat file) &&
-+ test 2 = $(git show :file) &&
-+ test 1 = $(git show HEAD:file)
-+'
-+
- test_expect_success 'drop middle stash by index' '
- git reset --hard &&
- echo 8 >file &&
-@@ t/t3903-stash.sh: setup_stash()
- test 0 = $(git stash list | wc -l)
- '
-
-+test_expect_success 'stash pop with stash.index' '
-+ test_config stash.index true &&
-+ git reset --hard &&
-+ setup_stash &&
-+ git stash pop &&
-+ test 3 = $(cat file) &&
-+ test 2 = $(git show :file) &&
-+ test 1 = $(git show HEAD:file) &&
-+ test 0 = $(git stash list | wc -l)
-+'
-+
- cat >expect <<EOF
- diff --git a/file2 b/file2
- new file mode 100644
-@@ t/t3903-stash.sh: setup_stash()
- test_must_be_empty output.out
- '
-
-+test_expect_success 'pop -q works and is quiet with stash.index' '
-+ # Added file, deleted file, modified file all staged for commit
-+ echo foo >new-file &&
-+ echo test >file &&
-+ git add new-file file &&
-+ git rm other-file &&
++test_expect_success 'stash.index=true implies --index' '
++ # setup for a few related tests
++ test_commit file base &&
++ echo index >file &&
++ git add file &&
++ echo working >file &&
+ git stash &&
+
-+ test_config stash.index true &&
-+ git stash pop -q >output.out 2>&1 &&
-+ echo test >expect &&
-+ git show :file >actual &&
++ test_when_finished "git reset --hard" &&
++ git -c stash.index=true stash apply &&
++ echo index >expect &&
++ git show :0:file >actual &&
+ test_cmp expect actual &&
-+ test_must_be_empty output.out
++ echo working >expect &&
++ test_cmp expect file
+'
+
- test_expect_success 'pop -q --index works and is quiet' '
- echo foo >file &&
- git add file &&
-@@ t/t3903-stash.sh: setup_stash()
- test_path_is_file bar
- '
-
-+test_expect_success 'stash -- <pathspec> stashes and restores the file with stash.index' '
-+ test_config stash.index true &&
-+ >foo &&
-+ >bar &&
-+ git add foo bar &&
-+ git stash push -- foo &&
-+ test_path_is_file bar &&
-+ test_path_is_missing foo &&
-+ git stash pop --no-index &&
-+ test_path_is_file foo &&
-+ test_path_is_file bar
++test_expect_success 'stash.index=true overridden by --no-index' '
++ test_when_finished "git reset --hard" &&
++ git -c stash.index=true stash apply --no-index &&
++ echo base >expect &&
++ git show :0:file >actual &&
++ test_cmp expect actual &&
++ echo working >expect &&
++ test_cmp expect file
+'
+
- test_expect_success 'stash -- <pathspec> stashes in subdirectory' '
- mkdir sub &&
- >foo &&
-@@ t/t3903-stash.sh: setup_stash()
- test_path_is_file bar
- '
-
-+test_expect_success 'stash -- <pathspec> stashes in subdirectory with stash.index' '
-+ test_config stash.index true &&
-+ rm -r sub &&
-+ mkdir sub &&
-+ >foo &&
-+ >bar &&
-+ git add foo bar &&
-+ (
-+ cd sub &&
-+ git stash push -- ../foo
-+ ) &&
-+ test_path_is_file bar &&
-+ test_path_is_missing foo &&
-+ git stash pop --no-index &&
-+ test_path_is_file foo &&
-+ test_path_is_file bar
++test_expect_success 'stash.index=false overridden by --index' '
++ test_when_finished "git reset --hard" &&
++ git -c stash.index=false stash apply --index &&
++ echo index >expect &&
++ git show :0:file >actual &&
++ test_cmp expect actual &&
++ echo working >expect &&
++ test_cmp expect file
+'
+
- test_expect_success 'stash with multiple pathspec arguments' '
- >foo &&
- >bar &&
-@@ t/t3903-stash.sh: setup_stash()
- test_path_is_file extra
- '
-
-+test_expect_success 'stash with multiple pathspec arguments with stash.index' '
-+ test_config stash.index true &&
-+ >foo &&
-+ >bar &&
-+ >extra &&
-+ git add foo bar extra &&
-+ git stash push -- foo bar &&
-+ test_path_is_missing bar &&
-+ test_path_is_missing foo &&
-+ test_path_is_file extra &&
-+ git stash pop --no-index &&
-+ test_path_is_file foo &&
-+ test_path_is_file bar &&
-+ test_path_is_file extra
-+'
-+
- test_expect_success 'stash with file including $IFS character' '
- >"foo bar" &&
- >foo &&
-@@ t/t3903-stash.sh: setup_stash()
- test_path_is_file bar
- '
-
-+test_expect_success 'stash with file including $IFS character with stash.index' '
-+ test_config stash.index true &&
-+ >"foo bar" &&
-+ >foo &&
-+ >bar &&
-+ git add foo* &&
-+ git stash push -- "foo b*" &&
-+ test_path_is_missing "foo bar" &&
-+ test_path_is_file foo &&
-+ test_path_is_file bar &&
-+ git stash pop --no-index &&
-+ test_path_is_file "foo bar" &&
-+ test_path_is_file foo &&
-+ test_path_is_file bar
-+'
-+
- test_expect_success 'stash with pathspec matching multiple paths' '
- echo original >file &&
- echo original >other-file &&
-@@ t/t3903-stash.sh: setup_stash()
- test_path_is_file bar
- '
-
-+test_expect_success 'stash without verb with pathspec with stash.index' '
-+ test_config stash.index true &&
-+ >"foo bar" &&
-+ >foo &&
-+ >bar &&
-+ git add foo* &&
-+ git stash -- "foo b*" &&
-+ test_path_is_missing "foo bar" &&
-+ test_path_is_file foo &&
-+ test_path_is_file bar &&
-+ git stash pop --no-index &&
-+ test_path_is_file "foo bar" &&
-+ test_path_is_file foo &&
-+ test_path_is_file bar
-+'
-+
- test_expect_success 'stash -k -- <pathspec> leaves unstaged files intact' '
- git reset &&
- >foo &&
-
- ## t/t3904-stash-patch.sh ##
-@@
- verify_state dir/foo work head
- '
-
-+test_expect_success 'git stash -p with stash.index' '
-+ test_config stash.index true &&
-+ set_state HEAD HEADfile_work HEADfile_index &&
-+ set_state dir/foo work index &&
-+ test_write_lines y n y | git stash save -p &&
-+ git reset --hard &&
-+ git stash apply &&
-+ verify_state HEAD HEADfile_work HEADfile_index &&
-+ verify_state dir/foo head index
-+'
-+
- test_expect_success 'git stash -p --no-keep-index' '
- set_state HEAD HEADfile_work HEADfile_index &&
- set_state bar bar_work bar_index &&
-
- ## t/t3905-stash-include-untracked.sh ##
-@@
-
- . ./test-lib.sh
-
--test_expect_success 'stash save --include-untracked some dirty working directory' '
-+setup() {
- echo 1 >file &&
- git add file &&
- test_tick &&
-@@
- git stash --include-untracked &&
- git diff-files --quiet &&
- git diff-index --cached --quiet HEAD
-+}
-+
-+test_expect_success 'stash save --include-untracked some dirty working directory' '
-+ setup
- '
-
- test_expect_success 'stash save --include-untracked cleaned the untracked files' '
-@@
- test_cmp untracked_expect untracked/untracked
- '
-
-+test_expect_success 'stash pop after save --include-untracked leaves files untracked again with stash.index' '
-+ git init repo &&
-+ test_when_finished rm -r repo &&
-+ (
-+ cd repo &&
-+ git config stash.index true &&
-+ setup &&
-+ cat >expect <<-EOF &&
-+ MM file
-+ ?? HEAD
-+ ?? actual
-+ ?? expect
-+ ?? file2
-+ ?? untracked/
-+ EOF
-+
-+ git stash pop &&
-+ git status --porcelain >actual &&
-+ test_cmp expect actual &&
-+ echo 1 >expect_file2 &&
-+ test_cmp expect_file2 file2 &&
-+ echo untracked >untracked_expect &&
-+ test_cmp untracked_expect untracked/untracked
-+ )
-+'
-+
- test_expect_success 'clean up untracked/ directory to prepare for next tests' '
- git clean --force --quiet -d
- '
-@@
- test_path_is_file bar
- '
-
-+test_expect_success 'stash push with $IFS character with stash.index' '
-+ test_config stash.index true &&
-+ >"foo bar" &&
-+ >foo &&
-+ >bar &&
-+ git add foo* &&
-+ git stash push --include-untracked -- "foo b*" &&
-+ test_path_is_missing "foo bar" &&
-+ test_path_is_file foo &&
-+ test_path_is_file bar &&
-+ git stash pop --no-index &&
-+ test_path_is_file "foo bar" &&
-+ test_path_is_file foo &&
-+ test_path_is_file bar
-+'
-+
- test_expect_success 'stash previously ignored file' '
- cat >.gitignore <<-EOF &&
- ignored
+ test_done
base-commit: 1ee85f0e215f22b0878d0ad4b2445d12bbb63887
--
2.48.1
next prev parent reply other threads:[~2025-09-22 1:39 UTC|newest]
Thread overview: 36+ messages / expand[flat|nested] mbox.gz Atom feed top
2025-05-10 18:33 [PATCH 0/9] make stash apply with --index by default D. Ben Knoble
2025-05-10 18:33 ` [PATCH 1/9] t3903: reduce dependencies on previous tests D. Ben Knoble
2025-05-10 18:33 ` [PATCH 2/9] t3905: remove unneeded blank line D. Ben Knoble
2025-05-10 18:33 ` [PATCH 3/9] BreakingChanges: announce stash {apply,pop} will imply --index D. Ben Knoble
2025-05-10 18:33 ` [PATCH 4/9] stash: restore the index by default when breaking changes are enabled D. Ben Knoble
2025-05-10 18:33 ` [PATCH 5/9] t0450: mark stash documentation as a known discrepancy D. Ben Knoble
2025-05-10 18:33 ` [PATCH 6/9] t3903: adjust stash test to account for --[no-]index with breaking changes D. Ben Knoble
2025-05-10 18:33 ` [PATCH 7/9] t3904: adjust stash -p test to account for index states " D. Ben Knoble
2025-05-10 18:33 ` [PATCH 8/9] t3905: adjust stash -u tests for " D. Ben Knoble
2025-05-10 18:33 ` [PATCH 9/9] t3906: adjust stash submodule tests to account " D. Ben Knoble
2025-05-12 12:52 ` [PATCH 0/9] make stash apply with --index by default Junio C Hamano
2025-05-20 14:36 ` D. Ben Knoble
2025-05-20 14:39 ` D. Ben Knoble
2025-09-16 0:37 ` [PATCH v2 0/4] Teach git-stash to use --index from config D. Ben Knoble
2025-09-16 0:37 ` [PATCH v2 1/4] t3903: reduce dependencies on previous tests D. Ben Knoble
2025-09-16 0:37 ` [PATCH v2 2/4] t3905: remove unneeded blank line D. Ben Knoble
2025-09-16 0:37 ` [PATCH v2 3/4] stash: refactor private config globals D. Ben Knoble
2025-09-16 0:37 ` [PATCH v2 4/4] stash: honor stash.index in apply, pop modes D. Ben Knoble
2025-09-16 9:18 ` Phillip Wood
2025-09-16 17:07 ` D. Ben Knoble
2025-09-16 9:24 ` [PATCH v2 0/4] Teach git-stash to use --index from config Phillip Wood
2025-09-16 17:06 ` D. Ben Knoble
2025-09-16 16:49 ` Junio C Hamano
2025-09-22 1:39 ` D. Ben Knoble [this message]
2025-09-22 1:39 ` [PATCH v3 1/4] t3903: reduce dependencies on previous tests D. Ben Knoble
2025-09-22 1:39 ` [PATCH v3 2/4] t3905: remove unneeded blank line D. Ben Knoble
2025-09-22 1:39 ` [PATCH v3 3/4] stash: refactor private config globals D. Ben Knoble
2025-09-22 1:39 ` [PATCH v3 4/4] stash: honor stash.index in apply, pop modes D. Ben Knoble
2025-09-22 14:11 ` Phillip Wood
2025-09-24 20:40 ` D. Ben Knoble
2025-09-29 10:01 ` Phillip Wood
2025-10-06 12:59 ` [PATCH] doc: explain the impact of stash.index on --autostash options D. Ben Knoble
2025-10-09 22:54 ` Kristoffer Haugsbakk
2025-10-11 14:44 ` D. Ben Knoble
2025-10-11 17:28 ` Junio C Hamano
2025-10-12 18:04 ` Ben Knoble
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=cover.1758505011.git.ben.knoble+github@gmail.com \
--to=ben.knoble+github@gmail.com \
--cc=git@vger.kernel.org \
--cc=gitster@pobox.com \
--cc=phillip.wood123@gmail.com \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).