Skip to content

Commit

Permalink
Fix reset when using the sparse-checkout feature.
Browse files Browse the repository at this point in the history
When using the sparse checkout feature the git reset command will add
entries to the index that will have the skip-worktree bit off but will
leave the working directory empty.  File data is lost because the index
version of the files has been changed but there is nothing that is in
the working directory.  This will cause the next status call to show
either deleted for files modified or deleting or nothing for files
added.  The added files should be shown as untracked and modified files
should be shown as modified.

To fix this when the reset is running if there is not a file in the
working directory and if it will be missing with the new index entry or
was not missing in the previous version, we create the previous index
version of the file in the working directory so that status will report
correctly and the files will be availble for the user to deal with.

2022-03-28: This commit was ejected during the 2.35.0 rebase, and I am
cherry-picking it back. We had a user report that "git reset HEAD~1"
would fail to update the index properly in a VFS for Git enlistment. No
VFS for Git functional test caught this, even though it appears like
that exact case should be included. I have locally failed to reproduce
the error in a test case, but have verified that the failing case in the
full reproduction is fixed by adding this commit again.

There is a big change in this version of the code: we condition the
check on core_virtualfilesystem instead of core_apply_sparse_checkout,
since this is _not_ the right thing to do in the sparse-checkout case,
but is the right thing to do for the virtual filesystem case.

Signed-off-by: Kevin Willford <kewillf@microsoft.com>
Signed-off-by: Derrick Stolee <derrickstolee@github.com>
  • Loading branch information
Kevin Willford authored and derrickstolee committed Mar 28, 2022
1 parent 2331939 commit 15b14a3
Show file tree
Hide file tree
Showing 2 changed files with 93 additions and 0 deletions.
35 changes: 35 additions & 0 deletions builtin/reset.c
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,8 @@
#include "dir.h"
#include "strbuf.h"
#include "quote.h"
#include "dir.h"
#include "entry.h"

#define REFRESH_INDEX_DELAY_WARNING_IN_MS (2 * 1000)

Expand Down Expand Up @@ -142,8 +144,41 @@ static void update_index_from_diff(struct diff_queue_struct *q,
for (i = 0; i < q->nr; i++) {
int pos;
struct diff_filespec *one = q->queue[i]->one;
struct diff_filespec *two = q->queue[i]->two;
int is_in_reset_tree = one->mode && !is_null_oid(&one->oid);
int is_missing = !(one->mode && !is_null_oid(&one->oid));
int was_missing = !two->mode && is_null_oid(&two->oid);
struct cache_entry *ce;
struct cache_entry *ceBefore;
struct checkout state = CHECKOUT_INIT;

/*
* When using a virtual filesystem, the cache entries that are
* added here will not have the skip-worktree bit set.
* Without this code there is data that is lost because the files that
* would normally be in the working directory are not there and show as
* deleted for the next status or in the case of added files just disappear.
* We need to create the previous version of the files in the working
* directory so that they will have the right content and the next
* status call will show modified or untracked files correctly.
*/
if (core_virtualfilesystem && !file_exists(two->path))
{
pos = cache_name_pos(two->path, strlen(two->path));
if ((pos >= 0 && ce_skip_worktree(active_cache[pos])) && (is_missing || !was_missing))
{
state.force = 1;
state.refresh_cache = 1;
state.istate = &the_index;
ceBefore = make_cache_entry(&the_index, two->mode, &two->oid, two->path,
0, 0);
if (!ceBefore)
die(_("make_cache_entry failed for path '%s'"),
two->path);

checkout_entry(ceBefore, &state, NULL, NULL);
}
}

if (!is_in_reset_tree && !intent_to_add) {
remove_file_from_cache(one->path);
Expand Down
58 changes: 58 additions & 0 deletions t/t7114-reset-sparse-checkout.sh
Original file line number Diff line number Diff line change
@@ -0,0 +1,58 @@
#!/bin/sh

test_description='reset when using a sparse-checkout'

. ./test-lib.sh

# reset using a sparse-checkout file

test_expect_success 'setup' '
test_tick &&
echo "checkout file" >c &&
echo "modify file" >m &&
echo "delete file" >d &&
git add . &&
git commit -m "initial commit" &&
echo "added file" >a &&
echo "modification of a file" >m &&
git rm d &&
git add . &&
git commit -m "second commit" &&
git checkout -b endCommit
'

test_expect_success 'reset when there is a sparse-checkout' '
echo "/c" >.git/info/sparse-checkout &&
test_config core.sparsecheckout true &&
git checkout -B resetBranch &&
test_path_is_missing m &&
test_path_is_missing a &&
test_path_is_missing d &&
git reset HEAD~1 &&
test "checkout file" = "$(cat c)" &&
test "modification of a file" = "$(cat m)" &&
test "added file" = "$(cat a)" &&
test_path_is_missing d
'

test_expect_success 'reset after deleting file without skip-worktree bit' '
git checkout -f endCommit &&
git clean -xdf &&
echo "/c
/m" >.git/info/sparse-checkout &&
test_config core.sparsecheckout true &&
git checkout -B resetAfterDelete &&
test_path_is_file m &&
test_path_is_missing a &&
test_path_is_missing d &&
rm -f m &&
git reset HEAD~1 &&
test "checkout file" = "$(cat c)" &&
test "added file" = "$(cat a)" &&
test_path_is_missing m &&
test_path_is_missing d
'



test_done

0 comments on commit 15b14a3

Please sign in to comment.