]> git.ipfire.org Git - thirdparty/git.git/commitdiff
mv: refresh stat info for moved entry
authorVictoria Dye <vdye@github.com>
Tue, 29 Mar 2022 01:07:07 +0000 (01:07 +0000)
committerJunio C Hamano <gitster@pobox.com>
Tue, 29 Mar 2022 16:45:02 +0000 (09:45 -0700)
Update the stat info of the moved index entry in 'rename_index_entry_at()'
if the entry is up-to-date with the index. Internally, 'git mv' uses
'rename_index_entry_at()' to move the source index entry to the destination.
However, it directly copies the stat info of the original cache entry, which
will not reflect the 'ctime' of the file renaming operation that happened as
part of the move. If a file is otherwise up-to-date with the index, that
difference in 'ctime' will make the entry appear out-of-date until the next
index-refreshing operation (e.g., 'git status').

Some commands, such as 'git reset', use the cached stat information to
determine whether a file is up-to-date; if this information is incorrect,
the command will fail when it should pass. In order to ensure a moved entry
is evaluated as 'up-to-date' when appropriate, refresh the destination index
entry's stat info in 'git mv' if and only if the file is up-to-date.

Note that the test added in 't7001-mv.sh' requires a "sleep 1" to ensure the
'ctime' of the file creation will be definitively older than the 'ctime' of
the renamed file in 'git mv'.

Reported-by: Maximilian Reichel <reichemn@icloud.com>
Signed-off-by: Victoria Dye <vdye@github.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
read-cache.c
t/t7001-mv.sh

index cbe73f14e5e7efc63b20e80a2702fb5c0dea9a5d..7961383c3ad1b2743203770fc2dc8b3a2a2ecea3 100644 (file)
@@ -133,7 +133,7 @@ static void replace_index_entry(struct index_state *istate, int nr, struct cache
 
 void rename_index_entry_at(struct index_state *istate, int nr, const char *new_name)
 {
-       struct cache_entry *old_entry = istate->cache[nr], *new_entry;
+       struct cache_entry *old_entry = istate->cache[nr], *new_entry, *refreshed;
        int namelen = strlen(new_name);
 
        new_entry = make_empty_cache_entry(istate, namelen);
@@ -146,7 +146,20 @@ void rename_index_entry_at(struct index_state *istate, int nr, const char *new_n
        cache_tree_invalidate_path(istate, old_entry->name);
        untracked_cache_remove_from_index(istate, old_entry->name);
        remove_index_entry_at(istate, nr);
-       add_index_entry(istate, new_entry, ADD_CACHE_OK_TO_ADD|ADD_CACHE_OK_TO_REPLACE);
+
+       /*
+        * Refresh the new index entry. Using 'refresh_cache_entry' ensures
+        * we only update stat info if the entry is otherwise up-to-date (i.e.,
+        * the contents/mode haven't changed). This ensures that we reflect the
+        * 'ctime' of the rename in the index without (incorrectly) updating
+        * the cached stat info to reflect unstaged changes on disk.
+        */
+       refreshed = refresh_cache_entry(istate, new_entry, CE_MATCH_REFRESH);
+       if (refreshed && refreshed != new_entry) {
+               add_index_entry(istate, refreshed, ADD_CACHE_OK_TO_ADD|ADD_CACHE_OK_TO_REPLACE);
+               discard_cache_entry(new_entry);
+       } else
+               add_index_entry(istate, new_entry, ADD_CACHE_OK_TO_ADD|ADD_CACHE_OK_TO_REPLACE);
 }
 
 void fill_stat_data(struct stat_data *sd, struct stat *st)
index 963356ba5f9257c3691e67267ff1d0cfb86355ad..a402908142d0d34a7145df03b6ea863d513a7200 100755 (executable)
@@ -4,6 +4,25 @@ test_description='git mv in subdirs'
 . ./test-lib.sh
 . "$TEST_DIRECTORY"/lib-diff-data.sh
 
+test_expect_success 'mv -f refreshes updated index entry' '
+       echo test >bar &&
+       git add bar &&
+       git commit -m test &&
+
+       echo foo >foo &&
+       git add foo &&
+
+       # Wait one second to ensure ctime of rename will differ from original
+       # file creation ctime.
+       sleep 1 &&
+       git mv -f foo bar &&
+       git reset --merge HEAD &&
+
+       # Verify the index has been reset
+       git diff-files >out &&
+       test_must_be_empty out
+'
+
 test_expect_success 'prepare reference tree' '
        mkdir path0 path1 &&
        COPYING_test_data >path0/COPYING &&