rev-parse: add '--absolute-git-dir' option
The output of 'git rev-parse --git-dir' can be either a relative or an absolute path, depending on whether the current working directory is at the top of the worktree or the .git directory or not, or how the path to the repository is specified via the '--git-dir=<path>' option or the $GIT_DIR environment variable. And if that output is a relative path, then it is relative to the directory where any 'git -C <path>' options might have led us. This doesn't matter at all for regular scripts, because the git wrapper automatically takes care of changing directories according to the '-C <path>' options, and the scripts can then simply follow any path returned by 'git rev-parse --git-dir', even if it's a relative path. Our Bash completion script, however, is unique in that it must run directly in the user's interactive shell environment. This means that it's not executed through the git wrapper and would have to take care of any '-C <path> options on its own, and it can't just change directories as it pleases. Consequently, adding support for taking any '-C <path>' options on the command line into account during completion turned out to be considerably more difficult, error prone and required more subshells and git processes when it had to cope with a relative path to the .git directory. Help this rather special use case and teach 'git rev-parse' a new '--absolute-git-dir' option which always outputs a canonicalized absolute path to the .git directory, regardless of whether the path is discovered automatically or is specified via $GIT_DIR or 'git --git-dir=<path>'. Signed-off-by: SZEDER Gábor <szeder.dev@gmail.com> Signed-off-by: Junio C Hamano <gitster@pobox.com>
This commit is contained in:
parent
336d694ce4
commit
a2f5a87626
@ -217,6 +217,10 @@ If `$GIT_DIR` is not defined and the current directory
|
||||
is not detected to lie in a Git repository or work tree
|
||||
print a message to stderr and exit with nonzero status.
|
||||
|
||||
--absolute-git-dir::
|
||||
Like `--git-dir`, but its output is always the canonicalized
|
||||
absolute path.
|
||||
|
||||
--git-common-dir::
|
||||
Show `$GIT_COMMON_DIR` if defined, else `$GIT_DIR`.
|
||||
|
||||
|
@ -802,17 +802,27 @@ int cmd_rev_parse(int argc, const char **argv, const char *prefix)
|
||||
putchar('\n');
|
||||
continue;
|
||||
}
|
||||
if (!strcmp(arg, "--git-dir")) {
|
||||
if (!strcmp(arg, "--git-dir") ||
|
||||
!strcmp(arg, "--absolute-git-dir")) {
|
||||
const char *gitdir = getenv(GIT_DIR_ENVIRONMENT);
|
||||
char *cwd;
|
||||
int len;
|
||||
if (gitdir) {
|
||||
puts(gitdir);
|
||||
continue;
|
||||
}
|
||||
if (!prefix) {
|
||||
puts(".git");
|
||||
continue;
|
||||
if (arg[2] == 'g') { /* --git-dir */
|
||||
if (gitdir) {
|
||||
puts(gitdir);
|
||||
continue;
|
||||
}
|
||||
if (!prefix) {
|
||||
puts(".git");
|
||||
continue;
|
||||
}
|
||||
} else { /* --absolute-git-dir */
|
||||
if (!gitdir && !prefix)
|
||||
gitdir = ".git";
|
||||
if (gitdir) {
|
||||
puts(real_path(gitdir));
|
||||
continue;
|
||||
}
|
||||
}
|
||||
cwd = xgetcwd();
|
||||
len = strlen(cwd);
|
||||
|
@ -3,7 +3,7 @@
|
||||
test_description='test git rev-parse'
|
||||
. ./test-lib.sh
|
||||
|
||||
# usage: [options] label is-bare is-inside-git is-inside-work prefix git-dir
|
||||
# usage: [options] label is-bare is-inside-git is-inside-work prefix git-dir absolute-git-dir
|
||||
test_rev_parse () {
|
||||
d=
|
||||
bare=
|
||||
@ -29,7 +29,8 @@ test_rev_parse () {
|
||||
--is-inside-git-dir \
|
||||
--is-inside-work-tree \
|
||||
--show-prefix \
|
||||
--git-dir
|
||||
--git-dir \
|
||||
--absolute-git-dir
|
||||
do
|
||||
test $# -eq 0 && break
|
||||
expect="$1"
|
||||
@ -62,26 +63,26 @@ test_expect_success 'setup' '
|
||||
cp -R .git repo.git
|
||||
'
|
||||
|
||||
test_rev_parse toplevel false false true '' .git
|
||||
test_rev_parse toplevel false false true '' .git "$ROOT/.git"
|
||||
|
||||
test_rev_parse -C .git .git/ false true false '' .
|
||||
test_rev_parse -C .git/objects .git/objects/ false true false '' "$ROOT/.git"
|
||||
test_rev_parse -C .git .git/ false true false '' . "$ROOT/.git"
|
||||
test_rev_parse -C .git/objects .git/objects/ false true false '' "$ROOT/.git" "$ROOT/.git"
|
||||
|
||||
test_rev_parse -C sub/dir subdirectory false false true sub/dir/ "$ROOT/.git"
|
||||
test_rev_parse -C sub/dir subdirectory false false true sub/dir/ "$ROOT/.git" "$ROOT/.git"
|
||||
|
||||
test_rev_parse -b t 'core.bare = true' true false false
|
||||
|
||||
test_rev_parse -b u 'core.bare undefined' false false true
|
||||
|
||||
|
||||
test_rev_parse -C work -g ../.git -b f 'GIT_DIR=../.git, core.bare = false' false false true ''
|
||||
test_rev_parse -C work -g ../.git -b f 'GIT_DIR=../.git, core.bare = false' false false true '' "../.git" "$ROOT/.git"
|
||||
|
||||
test_rev_parse -C work -g ../.git -b t 'GIT_DIR=../.git, core.bare = true' true false false ''
|
||||
|
||||
test_rev_parse -C work -g ../.git -b u 'GIT_DIR=../.git, core.bare undefined' false false true ''
|
||||
|
||||
|
||||
test_rev_parse -C work -g ../repo.git -b f 'GIT_DIR=../repo.git, core.bare = false' false false true ''
|
||||
test_rev_parse -C work -g ../repo.git -b f 'GIT_DIR=../repo.git, core.bare = false' false false true '' "../repo.git" "$ROOT/repo.git"
|
||||
|
||||
test_rev_parse -C work -g ../repo.git -b t 'GIT_DIR=../repo.git, core.bare = true' true false false ''
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user