Krótkie rozwiązania, które działają z podmodułami, hakami i wewnątrz .git
katalogu
Oto krótka odpowiedź, której większość będzie chciała:
r=$(git rev-parse --git-dir) && r=$(cd "$r" && pwd)/ && echo "${r%%/.git/*}"
Będzie to działać w dowolnym miejscu drzewa roboczego git (w tym w .git
katalogu), ale zakłada, że katalog (y) repozytorium są nazywane.git
(co jest domyślne). W przypadku podmodułów nastąpi przejście do katalogu głównego najbardziej zewnętrznego repozytorium.
Jeśli chcesz dostać się do katalogu głównego bieżącego submodułu użyj:
echo $(r=$(git rev-parse --show-toplevel) && ([[ -n $r ]] && echo "$r" || (cd $(git rev-parse --git-dir)/.. && pwd) ))
Aby łatwo wykonać polecenie w katalogu głównym podmodułu, [alias]
w swoim .gitconfig
, dodaj:
sh = "!f() { root=$(pwd)/ && cd ${root%%/.git/*} && git rev-parse && exec \"$@\"; }; f"
To pozwala łatwo robić takie rzeczy jak git sh ag <string>
Solidne rozwiązanie, które obsługuje różne nazwy lub zewnętrzne .git
lub$GIT_DIR
katalogi .
Zauważ, że $GIT_DIR
może wskazywać gdzieś na zewnątrz (i nie można go nazwać.git
), stąd potrzeba dalszego sprawdzania.
Umieść to w .bashrc
:
# Print the name of the git working tree's root directory
function git_root() {
local root first_commit
# git displays its own error if not in a repository
root=$(git rev-parse --show-toplevel) || return
if [[ -n $root ]]; then
echo $root
return
elif [[ $(git rev-parse --is-inside-git-dir) = true ]]; then
# We're inside the .git directory
# Store the commit id of the first commit to compare later
# It's possible that $GIT_DIR points somewhere not inside the repo
first_commit=$(git rev-list --parents HEAD | tail -1) ||
echo "$0: Can't get initial commit" 2>&1 && false && return
root=$(git rev-parse --git-dir)/.. &&
# subshell so we don't change the user's working directory
( cd "$root" &&
if [[ $(git rev-list --parents HEAD | tail -1) = $first_commit ]]; then
pwd
else
echo "$FUNCNAME: git directory is not inside its repository" 2>&1
false
fi
)
else
echo "$FUNCNAME: Can't determine repository root" 2>&1
false
fi
}
# Change working directory to git repository root
function cd_git_root() {
local root
root=$(git_root) || return 1 # git_root will print any errors
cd "$root"
}
Wykonać go przez pisanie git_root
(po ponownym uruchomieniu powłoki: exec bash
)