aboutsummaryrefslogtreecommitdiff
path: root/t
diff options
context:
space:
mode:
authorKarthik Nayak <karthik.188@gmail.com>2026-02-25 10:40:46 +0100
committerJunio C Hamano <gitster@pobox.com>2026-02-25 09:40:00 -0800
commit53592d68e86814fcc4a8df6cc38340597e56fe5a (patch)
treeba40af5ae430ea79abd4f1898562c736f4f408b9 /t
parent01dc84594ee365ee7086fccc7f590ab527730531 (diff)
downloadgit-53592d68e86814fcc4a8df6cc38340597e56fe5a.tar.xz
refs: add GIT_REFERENCE_BACKEND to specify reference backend
Git allows setting a different object directory via 'GIT_OBJECT_DIRECTORY', but provides no equivalent for references. In the previous commit we extended the 'extensions.refStorage' config to also support an URI input for reference backend with location. Let's also add a new environment variable 'GIT_REFERENCE_BACKEND' that takes in the same input as the config variable. Having an environment variable allows us to modify the reference backend and location on the fly for individual Git commands. The environment variable also allows usage of alternate reference directories during 'git-clone(1)' and 'git-init(1)'. Add the config to the repository when created with the environment variable set. When initializing the repository with an alternate reference folder, create the required stubs in the repositories $GIT_DIR. The inverse, i.e. removal of the ref store doesn't clean up the stubs in the $GIT_DIR since that would render it unusable. Removal of ref store is only used when migrating between ref formats and cleanup of the $GIT_DIR doesn't make sense in such a situation. Helped-by: Jean-Noël Avila <jn.avila@free.fr> Signed-off-by: Karthik Nayak <karthik.188@gmail.com> Signed-off-by: Junio C Hamano <gitster@pobox.com>
Diffstat (limited to 't')
-rwxr-xr-xt/t1423-ref-backend.sh187
1 files changed, 154 insertions, 33 deletions
diff --git a/t/t1423-ref-backend.sh b/t/t1423-ref-backend.sh
index 82cccb7a65..fd47d77e8e 100755
--- a/t/t1423-ref-backend.sh
+++ b/t/t1423-ref-backend.sh
@@ -11,16 +11,25 @@ test_description='Test reference backend URIs'
# <backend> is the original ref storage of the repo.
# <uri> is the new URI to be set for the ref storage.
# <cmd> is the git subcommand to be run in the repository.
+# <via> if 'config', set the backend via the 'extensions.refStorage' config.
+# if 'env', set the backend via the 'GIT_REFERENCE_BACKEND' env.
run_with_uri () {
repo=$1 &&
backend=$2 &&
uri=$3 &&
cmd=$4 &&
+ via=$5 &&
- git -C "$repo" config set core.repositoryformatversion 1
- git -C "$repo" config set extensions.refStorage "$uri" &&
- git -C "$repo" $cmd &&
- git -C "$repo" config set extensions.refStorage "$backend"
+ git -C "$repo" config set core.repositoryformatversion 1 &&
+ if test "$via" = "env"
+ then
+ test_env GIT_REFERENCE_BACKEND="$uri" git -C "$repo" $cmd
+ elif test "$via" = "config"
+ then
+ git -C "$repo" config set extensions.refStorage "$uri" &&
+ git -C "$repo" $cmd &&
+ git -C "$repo" config set extensions.refStorage "$backend"
+ fi
}
# Test a repository with a given reference storage by running and comparing
@@ -30,44 +39,84 @@ run_with_uri () {
# <repo> is the relative path to the repo to run the command in.
# <backend> is the original ref storage of the repo.
# <uri> is the new URI to be set for the ref storage.
+# <via> if 'config', set the backend via the 'extensions.refStorage' config.
+# if 'env', set the backend via the 'GIT_REFERENCE_BACKEND' env.
# <err_msg> (optional) if set, check if 'git-refs(1)' failed with the provided msg.
test_refs_backend () {
repo=$1 &&
backend=$2 &&
uri=$3 &&
- err_msg=$4 &&
+ via=$4 &&
+ err_msg=$5 &&
+
- git -C "$repo" config set core.repositoryformatversion 1 &&
if test -n "$err_msg";
then
- git -C "$repo" config set extensions.refStorage "$uri" &&
- test_must_fail git -C "$repo" refs list 2>err &&
- test_grep "$err_msg" err
+ if test "$via" = "env"
+ then
+ test_env GIT_REFERENCE_BACKEND="$uri" test_must_fail git -C "$repo" refs list 2>err
+ elif test "$via" = "config"
+ then
+ git -C "$repo" config set extensions.refStorage "$uri" &&
+ test_must_fail git -C "$repo" refs list 2>err &&
+ test_grep "$err_msg" err
+ fi
else
git -C "$repo" refs list >expect &&
- run_with_uri "$repo" "$backend" "$uri" "refs list" >actual &&
+ run_with_uri "$repo" "$backend" "$uri" "refs list" "$via">actual &&
test_cmp expect actual
fi
}
-test_expect_success 'URI is invalid' '
+# Verify that the expected files are present in the gitdir and the refsdir.
+# Usage: verify_files_exist <gitdir> <refdir>
+# <gitdir> is the path for the gitdir.
+# <refdir> is the path for the refdir.
+verify_files_exist () {
+ gitdir=$1 &&
+ refdir=$2 &&
+
+ # verify that the stubs were added to the $GITDIR.
+ echo "repository uses alternate refs storage" >expect &&
+ test_cmp expect $gitdir/refs/heads &&
+ echo "ref: refs/heads/.invalid" >expect &&
+ test_cmp expect $gitdir/HEAD
+
+ # verify that backend specific files exist.
+ case "$GIT_DEFAULT_REF_FORMAT" in
+ files)
+ test_path_is_dir $refdir/refs/heads &&
+ test_path_is_file $refdir/HEAD;;
+ reftable)
+ test_path_is_dir $refdir/reftable &&
+ test_path_is_file $refdir/reftable/tables.list;;
+ *)
+ BUG "unhandled ref format $GIT_DEFAULT_REF_FORMAT";;
+ esac
+}
+
+methods="config env"
+for method in $methods
+do
+
+test_expect_success "$method: URI is invalid" '
test_when_finished "rm -rf repo" &&
git init repo &&
- test_refs_backend repo files "reftable@/home/reftable" \
+ test_refs_backend repo files "reftable@/home/reftable" "$method" \
"invalid value for ${SQ}extensions.refstorage${SQ}"
'
-test_expect_success 'URI ends with colon' '
+test_expect_success "$method: URI ends with colon" '
test_when_finished "rm -rf repo" &&
git init repo &&
- test_refs_backend repo files "reftable:" \
+ test_refs_backend repo files "reftable:" "$method" \
"invalid value for ${SQ}extensions.refstorage${SQ}"
'
-test_expect_success 'unknown reference backend' '
+test_expect_success "$method: unknown reference backend" '
test_when_finished "rm -rf repo" &&
git init repo &&
- test_refs_backend repo files "db://.git" \
+ test_refs_backend repo files "db://.git" "$method" \
"invalid value for ${SQ}extensions.refstorage${SQ}"
'
@@ -86,7 +135,7 @@ do
for dir in "$(pwd)/repo/.git" "."
do
- test_expect_success "read from $to_format backend, $dir dir" '
+ test_expect_success "$method: read from $to_format backend, $dir dir" '
test_when_finished "rm -rf repo" &&
git init --ref-format=$from_format repo &&
(
@@ -101,7 +150,7 @@ do
)
'
- test_expect_success "write to $to_format backend, $dir dir" '
+ test_expect_success "$method: write to $to_format backend, $dir dir" '
test_when_finished "rm -rf repo" &&
git init --ref-format=$from_format repo &&
(
@@ -113,20 +162,22 @@ do
git refs migrate --dry-run --ref-format=$to_format >out &&
BACKEND_PATH="$dir/$(sed "s/.* ${SQ}.git\/\(.*\)${SQ}/\1/" out)" &&
- test_refs_backend . $from_format "$to_format://$BACKEND_PATH" &&
+ test_refs_backend . $from_format "$to_format://$BACKEND_PATH" "$method" &&
git refs list >expect &&
- run_with_uri . "$from_format" "$to_format://$BACKEND_PATH" "tag -d 1" &&
+ run_with_uri . "$from_format" "$to_format://$BACKEND_PATH" \
+ "tag -d 1" "$method" &&
git refs list >actual &&
test_cmp expect actual &&
git refs list | grep -v "refs/tags/1" >expect &&
- run_with_uri . "$from_format" "$to_format://$BACKEND_PATH" "refs list" >actual &&
+ run_with_uri . "$from_format" "$to_format://$BACKEND_PATH" \
+ "refs list" "$method" >actual &&
test_cmp expect actual
)
'
- test_expect_success "with worktree and $to_format backend, $dir dir" '
+ test_expect_success "$method: with worktree and $to_format backend, $dir dir" '
test_when_finished "rm -rf repo wt" &&
git init --ref-format=$from_format repo &&
(
@@ -138,22 +189,92 @@ do
git refs migrate --dry-run --ref-format=$to_format >out &&
BACKEND_PATH="$dir/$(sed "s/.* ${SQ}.git\/\(.*\)${SQ}/\1/" out)" &&
- git config set core.repositoryformatversion 1 &&
- git config set extensions.refStorage "$to_format://$BACKEND_PATH" &&
-
- git worktree add ../wt 2
- ) &&
+ run_with_uri . "$from_format" "$to_format://$BACKEND_PATH" \
+ "worktree add ../wt 2" "$method" &&
- git -C repo for-each-ref --include-root-refs >expect &&
- git -C wt for-each-ref --include-root-refs >expect &&
- ! test_cmp expect actual &&
+ run_with_uri . "$from_format" "$to_format://$BACKEND_PATH" \
+ "for-each-ref --include-root-refs" "$method" >actual &&
+ run_with_uri ../wt "$from_format" "$to_format://$BACKEND_PATH" \
+ "for-each-ref --include-root-refs" "$method" >expect &&
+ ! test_cmp expect actual &&
- git -C wt rev-parse 2 >expect &&
- git -C wt rev-parse HEAD >actual &&
- test_cmp expect actual
+ run_with_uri . "$from_format" "$to_format://$BACKEND_PATH" \
+ "rev-parse 2" "$method" >actual &&
+ run_with_uri ../wt "$from_format" "$to_format://$BACKEND_PATH" \
+ "rev-parse HEAD" "$method" >expect &&
+ test_cmp expect actual
+ )
'
done # closes dir
+
+ test_expect_success "migrating repository to $to_format with alternate refs directory" '
+ test_when_finished "rm -rf repo refdir" &&
+ mkdir refdir &&
+ GIT_REFERENCE_BACKEND="${from_format}://$(pwd)/refdir" git init repo &&
+ (
+ cd repo &&
+
+ test_commit 1 &&
+ test_commit 2 &&
+ test_commit 3 &&
+
+ git refs migrate --ref-format=$to_format &&
+ git refs list >out &&
+ test_grep "refs/tags/1" out &&
+ test_grep "refs/tags/2" out &&
+ test_grep "refs/tags/3" out
+ )
+ '
+
done # closes to_format
done # closes from_format
+done # closes method
+
+test_expect_success 'initializing repository with alt ref directory' '
+ test_when_finished "rm -rf repo refdir" &&
+ mkdir refdir &&
+ BACKEND="$(test_detect_ref_format)://$(pwd)/refdir" &&
+ GIT_REFERENCE_BACKEND=$BACKEND git init repo &&
+ verify_files_exist repo/.git refdir &&
+ (
+ cd repo &&
+
+ git config get extensions.refstorage >actual &&
+ echo $BACKEND >expect &&
+ test_cmp expect actual &&
+
+ test_commit 1 &&
+ test_commit 2 &&
+ test_commit 3 &&
+ git refs list >out &&
+ test_grep "refs/tags/1" out &&
+ test_grep "refs/tags/2" out &&
+ test_grep "refs/tags/3" out
+ )
+'
+
+test_expect_success 'cloning repository with alt ref directory' '
+ test_when_finished "rm -rf source repo refdir" &&
+ mkdir refdir &&
+
+ git init source &&
+ test_commit -C source 1 &&
+ test_commit -C source 2 &&
+ test_commit -C source 3 &&
+
+ BACKEND="$(test_detect_ref_format)://$(pwd)/refdir" &&
+ GIT_REFERENCE_BACKEND=$BACKEND git clone source repo &&
+
+ git -C repo config get extensions.refstorage >actual &&
+ echo $BACKEND >expect &&
+ test_cmp expect actual &&
+
+ verify_files_exist repo/.git refdir &&
+
+ git -C source for-each-ref refs/tags/ >expect &&
+ git -C repo for-each-ref refs/tags/ >actual &&
+ test_cmp expect actual
+'
+
test_done