|
|
|
#!/bin/sh
|
|
|
|
|
|
|
|
test_description='parallel-checkout basics
|
|
|
|
|
|
|
|
Ensure that parallel-checkout basically works on clone and checkout, spawning
|
|
|
|
the required number of workers and correctly populating both the index and the
|
|
|
|
working tree.
|
|
|
|
'
|
|
|
|
|
|
|
|
TEST_NO_CREATE_REPO=1
|
|
|
|
. ./test-lib.sh
|
|
|
|
. "$TEST_DIRECTORY/lib-parallel-checkout.sh"
|
|
|
|
|
|
|
|
# Test parallel-checkout with a branch switch containing a variety of file
|
|
|
|
# creations, deletions, and modifications, involving different entry types.
|
|
|
|
# The branches B1 and B2 have the following paths:
|
|
|
|
#
|
|
|
|
# B1 B2
|
|
|
|
# a/a (file) a (file)
|
|
|
|
# b (file) b/b (file)
|
|
|
|
#
|
|
|
|
# c/c (file) c (symlink)
|
|
|
|
# d (symlink) d/d (file)
|
|
|
|
#
|
|
|
|
# e/e (file) e (submodule)
|
|
|
|
# f (submodule) f/f (file)
|
|
|
|
#
|
|
|
|
# g (submodule) g (symlink)
|
|
|
|
# h (symlink) h (submodule)
|
|
|
|
#
|
|
|
|
# Additionally, the following paths are present on both branches, but with
|
|
|
|
# different contents:
|
|
|
|
#
|
|
|
|
# i (file) i (file)
|
|
|
|
# j (symlink) j (symlink)
|
|
|
|
# k (submodule) k (submodule)
|
|
|
|
#
|
|
|
|
# And the following paths are only present in one of the branches:
|
|
|
|
#
|
|
|
|
# l/l (file) -
|
|
|
|
# - m/m (file)
|
|
|
|
#
|
|
|
|
test_expect_success 'setup repo for checkout with various types of changes' '
|
|
|
|
test_config_global protocol.file.allow always &&
|
|
|
|
|
|
|
|
git init sub &&
|
|
|
|
(
|
|
|
|
cd sub &&
|
|
|
|
git checkout -b B2 &&
|
|
|
|
echo B2 >file &&
|
|
|
|
git add file &&
|
|
|
|
git commit -m file &&
|
|
|
|
|
|
|
|
git checkout -b B1 &&
|
|
|
|
echo B1 >file &&
|
|
|
|
git add file &&
|
|
|
|
git commit -m file
|
|
|
|
) &&
|
|
|
|
|
|
|
|
git init various &&
|
|
|
|
(
|
|
|
|
cd various &&
|
|
|
|
|
|
|
|
git checkout -b B1 &&
|
|
|
|
mkdir a c e &&
|
|
|
|
echo a/a >a/a &&
|
|
|
|
echo b >b &&
|
|
|
|
echo c/c >c/c &&
|
|
|
|
test_ln_s_add c d &&
|
|
|
|
echo e/e >e/e &&
|
|
|
|
git submodule add ../sub f &&
|
|
|
|
git submodule add ../sub g &&
|
|
|
|
test_ln_s_add c h &&
|
|
|
|
|
|
|
|
echo "B1 i" >i &&
|
|
|
|
test_ln_s_add c j &&
|
|
|
|
git submodule add -b B1 ../sub k &&
|
|
|
|
mkdir l &&
|
|
|
|
echo l/l >l/l &&
|
|
|
|
|
|
|
|
git add . &&
|
|
|
|
git commit -m B1 &&
|
|
|
|
|
|
|
|
git checkout -b B2 &&
|
|
|
|
git rm -rf :^.gitmodules :^k &&
|
|
|
|
mkdir b d f &&
|
|
|
|
echo a >a &&
|
|
|
|
echo b/b >b/b &&
|
|
|
|
test_ln_s_add b c &&
|
|
|
|
echo d/d >d/d &&
|
|
|
|
git submodule add ../sub e &&
|
|
|
|
echo f/f >f/f &&
|
|
|
|
test_ln_s_add b g &&
|
|
|
|
git submodule add ../sub h &&
|
|
|
|
|
|
|
|
echo "B2 i" >i &&
|
|
|
|
test_ln_s_add b j &&
|
|
|
|
git -C k checkout B2 &&
|
|
|
|
mkdir m &&
|
|
|
|
echo m/m >m/m &&
|
|
|
|
|
|
|
|
git add . &&
|
|
|
|
git commit -m B2 &&
|
|
|
|
|
|
|
|
git checkout --recurse-submodules B1
|
|
|
|
)
|
|
|
|
'
|
|
|
|
|
|
|
|
for mode in sequential parallel sequential-fallback
|
|
|
|
do
|
|
|
|
case $mode in
|
|
|
|
sequential) workers=1 threshold=0 expected_workers=0 ;;
|
|
|
|
parallel) workers=2 threshold=0 expected_workers=2 ;;
|
|
|
|
sequential-fallback) workers=2 threshold=100 expected_workers=0 ;;
|
|
|
|
esac
|
|
|
|
|
|
|
|
test_expect_success "$mode checkout" '
|
|
|
|
repo=various_$mode &&
|
|
|
|
cp -R -P various $repo &&
|
|
|
|
|
|
|
|
# The just copied files have more recent timestamps than their
|
|
|
|
# associated index entries. So refresh the cached timestamps
|
|
|
|
# to avoid an "entry not up-to-date" error from `git checkout`.
|
|
|
|
# We only have to do this for the submodules as `git checkout`
|
|
|
|
# will already refresh the superproject index before performing
|
|
|
|
# the up-to-date check.
|
|
|
|
#
|
|
|
|
git -C $repo submodule foreach "git update-index --refresh" &&
|
|
|
|
|
|
|
|
set_checkout_config $workers $threshold &&
|
|
|
|
test_checkout_workers $expected_workers \
|
|
|
|
git -C $repo checkout --recurse-submodules B2 &&
|
|
|
|
verify_checkout $repo
|
|
|
|
'
|
|
|
|
done
|
|
|
|
|
|
|
|
for mode in parallel sequential-fallback
|
|
|
|
do
|
|
|
|
case $mode in
|
|
|
|
parallel) workers=2 threshold=0 expected_workers=2 ;;
|
|
|
|
sequential-fallback) workers=2 threshold=100 expected_workers=0 ;;
|
|
|
|
esac
|
|
|
|
|
|
|
|
test_expect_success "$mode checkout on clone" '
|
|
|
|
test_config_global protocol.file.allow always &&
|
|
|
|
repo=various_${mode}_clone &&
|
|
|
|
set_checkout_config $workers $threshold &&
|
|
|
|
test_checkout_workers $expected_workers \
|
|
|
|
git clone --recurse-submodules --branch B2 various $repo &&
|
|
|
|
verify_checkout $repo
|
|
|
|
'
|
|
|
|
done
|
|
|
|
|
|
|
|
# Just to be paranoid, actually compare the working trees' contents directly.
|
|
|
|
test_expect_success 'compare the working trees' '
|
|
|
|
rm -rf various_*/.git &&
|
|
|
|
rm -rf various_*/*/.git &&
|
|
|
|
|
|
|
|
# We use `git diff` instead of `diff -r` because the latter would
|
|
|
|
# follow symlinks, and not all `diff` implementations support the
|
|
|
|
# `--no-dereference` option.
|
|
|
|
#
|
|
|
|
git diff --no-index various_sequential various_parallel &&
|
|
|
|
git diff --no-index various_sequential various_parallel_clone &&
|
|
|
|
git diff --no-index various_sequential various_sequential-fallback &&
|
|
|
|
git diff --no-index various_sequential various_sequential-fallback_clone
|
|
|
|
'
|
|
|
|
|
|
|
|
# Currently, each submodule is checked out in a separated child process, but
|
|
|
|
# these subprocesses must also be able to use parallel checkout workers to
|
|
|
|
# write the submodules' entries.
|
|
|
|
test_expect_success 'submodules can use parallel checkout' '
|
|
|
|
set_checkout_config 2 0 &&
|
|
|
|
git init super &&
|
|
|
|
(
|
|
|
|
cd super &&
|
|
|
|
git init sub &&
|
|
|
|
test_commit -C sub A &&
|
|
|
|
test_commit -C sub B &&
|
|
|
|
git submodule add ./sub &&
|
|
|
|
git commit -m sub &&
|
|
|
|
rm sub/* &&
|
|
|
|
test_checkout_workers 2 git checkout --recurse-submodules .
|
|
|
|
)
|
|
|
|
'
|
|
|
|
|
|
|
|
test_expect_success 'parallel checkout respects --[no]-force' '
|
|
|
|
set_checkout_config 2 0 &&
|
|
|
|
git init dirty &&
|
|
|
|
(
|
|
|
|
cd dirty &&
|
|
|
|
mkdir D &&
|
|
|
|
test_commit D/F &&
|
|
|
|
test_commit F &&
|
|
|
|
|
|
|
|
rm -rf D &&
|
|
|
|
echo changed >D &&
|
|
|
|
echo changed >F.t &&
|
|
|
|
|
|
|
|
# We expect 0 workers because there is nothing to be done
|
|
|
|
test_checkout_workers 0 git checkout HEAD &&
|
|
|
|
test_path_is_file D &&
|
|
|
|
grep changed D &&
|
|
|
|
grep changed F.t &&
|
|
|
|
|
|
|
|
test_checkout_workers 2 git checkout --force HEAD &&
|
|
|
|
test_path_is_dir D &&
|
|
|
|
grep D/F D/F.t &&
|
|
|
|
grep F F.t
|
|
|
|
)
|
|
|
|
'
|
|
|
|
|
|
|
|
test_expect_success SYMLINKS 'parallel checkout checks for symlinks in leading dirs' '
|
|
|
|
set_checkout_config 2 0 &&
|
|
|
|
git init symlinks &&
|
|
|
|
(
|
|
|
|
cd symlinks &&
|
|
|
|
mkdir D untracked &&
|
|
|
|
# Commit 2 files to have enough work for 2 parallel workers
|
|
|
|
test_commit D/A &&
|
|
|
|
test_commit D/B &&
|
|
|
|
rm -rf D &&
|
|
|
|
ln -s untracked D &&
|
|
|
|
|
|
|
|
test_checkout_workers 2 git checkout --force HEAD &&
|
|
|
|
! test -h D &&
|
|
|
|
grep D/A D/A.t &&
|
|
|
|
grep D/B D/B.t
|
|
|
|
)
|
|
|
|
'
|
|
|
|
|
|
|
|
# This test is here (and not in e.g. t2022-checkout-paths.sh), because we
|
|
|
|
# check the final report including sequential, parallel, and delayed entries
|
|
|
|
# all at the same time. So we must have finer control of the parallel checkout
|
|
|
|
# variables.
|
|
|
|
test_expect_success '"git checkout ." report should not include failed entries' '
|
|
|
|
test_config_global filter.delay.process \
|
|
|
|
"test-tool rot13-filter --always-delay --log=delayed.log clean smudge delay" &&
|
|
|
|
test_config_global filter.delay.required true &&
|
|
|
|
test_config_global filter.cat.clean cat &&
|
|
|
|
test_config_global filter.cat.smudge cat &&
|
|
|
|
test_config_global filter.cat.required true &&
|
|
|
|
|
|
|
|
set_checkout_config 2 0 &&
|
|
|
|
git init failed_entries &&
|
|
|
|
(
|
|
|
|
cd failed_entries &&
|
|
|
|
cat >.gitattributes <<-EOF &&
|
|
|
|
*delay* filter=delay
|
|
|
|
parallel-ineligible* filter=cat
|
|
|
|
EOF
|
|
|
|
echo a >missing-delay.a &&
|
|
|
|
echo a >parallel-ineligible.a &&
|
|
|
|
echo a >parallel-eligible.a &&
|
|
|
|
echo b >success-delay.b &&
|
|
|
|
echo b >parallel-ineligible.b &&
|
|
|
|
echo b >parallel-eligible.b &&
|
|
|
|
git add -A &&
|
|
|
|
git commit -m files &&
|
|
|
|
|
|
|
|
a_blob="$(git rev-parse :parallel-ineligible.a)" &&
|
|
|
|
rm .git/objects/$(test_oid_to_path $a_blob) &&
|
|
|
|
rm *.a *.b &&
|
|
|
|
|
|
|
|
test_checkout_workers 2 test_must_fail git checkout . 2>err &&
|
|
|
|
|
|
|
|
# All *.b entries should succeed and all *.a entries should fail:
|
|
|
|
# - missing-delay.a: the delay filter will drop this path
|
|
|
|
# - parallel-*.a: the blob will be missing
|
|
|
|
#
|
|
|
|
grep "Updated 3 paths from the index" err &&
|
|
|
|
test_stdout_line_count = 3 ls *.b &&
|
|
|
|
! ls *.a
|
|
|
|
)
|
|
|
|
'
|
|
|
|
|
|
|
|
test_done
|