This series is the updated version of 'sg/completion-refs-speedup'. It speeds up refs completion for large number of refs, partly by giving up disambiguating ambiguous refs and partly by eliminating most of the shell processing between 'git for-each-ref' and 'ls-remote' and Bash's completion facility. The rest is a bit of preparatory reorganization, cleanup and bugfixes.
Changes since v1: - Patch 8 (let 'for-each-ref' and 'ls-remote' filter matching refs; it was patch 7 in v1) was modified in two ways: * __git_refs() now does that filtering only when the ref to match was explicitly given as parameter, as opposed to falling back to the current word to be completed. The current word might be something like '--opt=maste', and in the fallback case we would then list only refs matching '--opt=maste', which is of course wrong. Most of the subsequent patches had to be adjusted because of conflicts. * patch 11 (list only matching symbolic and pseudorefs when completing refs) was squashed into patch 8. There was no reason to keep the two patches separate, and the docstring was inconsistent between the two patches. - Patch 12 now incorporates the squash! patch I sent out earlier [1]. - Patch 4 (support completing fully qualified non-fast-forward refspecs) is new, to fix a bug that is similar in nature to the one fixed in patch 3. - Patches 13 and 14 are new and make use of the new and faster __gitcomp_direct() for branches, tags, and fetch refspecs. - Some new tests run 'sed s/Z$//g'. Remove that 'g', because there is no point to ask to replace all instances of the match, when it matches only at the end of line. - A teardown test forgot to delete a branch. [1] - http://public-inbox.org/git/20170206181545.12869-1-szeder....@gmail.com/ SZEDER Gábor (14): completion: remove redundant __gitcomp_nl() options from _git_commit() completion: wrap __git_refs() for better option parsing completion: support completing full refs after '--option=refs/<TAB>' completion: support completing fully qualified non-fast-forward refspecs completion: support excluding full refs completion: don't disambiguate tags and branches completion: don't disambiguate short refs completion: let 'for-each-ref' and 'ls-remote' filter matching refs completion: let 'for-each-ref' strip the remote name from remote branches completion: let 'for-each-ref' filter remote branches for 'checkout' DWIMery completion: let 'for-each-ref' sort remote branches for 'checkout' DWIMery completion: fill COMPREPLY directly when completing refs completion: fill COMPREPLY directly when completing fetch refspecs completion: speed up branch and tag completion contrib/completion/git-completion.bash | 252 +++++++++++++++------ contrib/completion/git-completion.zsh | 9 + t/t9902-completion.sh | 387 +++++++++++++++++++++++++++++++++ 3 files changed, 577 insertions(+), 71 deletions(-) -- 2.12.1.485.g1616aa492 diff --git a/contrib/completion/git-completion.bash b/contrib/completion/git-completion.bash index 56ededb09..bd07d9a74 100644 --- a/contrib/completion/git-completion.bash +++ b/contrib/completion/git-completion.bash @@ -352,14 +352,27 @@ __git_index_files () done | sort | uniq } +# Lists branches from the local repository. +# 1: A prefix to be added to each listed branch (optional). +# 2: List only branches matching this word (optional; list all branches if +# unset or empty). +# 3: A suffix to be appended to each listed branch (optional). __git_heads () { - __git for-each-ref --format='%(refname:strip=2)' refs/heads + local pfx="${1-}" cur_="${2-}" sfx="${3-}" + + __git for-each-ref --format="${pfx//\%/%%}%(refname:strip=2)$sfx" \ + "refs/heads/$cur_*" "refs/heads/$cur_*/**" } +# Lists tags from the local repository. +# Accepts the same positional parameters as __git_heads() above. __git_tags () { - __git for-each-ref --format='%(refname:strip=2)' refs/tags + local pfx="${1-}" cur_="${2-}" sfx="${3-}" + + __git for-each-ref --format="${pfx//\%/%%}%(refname:strip=2)$sfx" \ + "refs/tags/$cur_*" "refs/tags/$cur_*/**" } # Lists refs from the local (by default) or from a remote repository. @@ -369,8 +382,8 @@ __git_tags () # 2: In addition to local refs, list unique branches from refs/remotes/ for # 'git checkout's tracking DWIMery (optional; ignored, if set but empty). # 3: A prefix to be added to each listed ref (optional). -# 4: List only refs matching this word instead of the current word being -# completed (optional; NOT ignored, if empty, but lists all refs). +# 4: List only refs matching this word (optional; list all refs if unset or +# empty). # 5: A suffix to be appended to each listed ref (optional; ignored, if set # but empty). # @@ -381,7 +394,8 @@ __git_refs () local list_refs_from=path remote="${1-}" local format refs local pfx="${3-}" cur_="${4-$cur}" sfx="${5-}" - local fer_pfx="${pfx//\%/%%}" + local match="${4-}" + local fer_pfx="${pfx//\%/%%}" # "escape" for-each-ref format specifiers __git_find_repo_path dir="$__git_repo_path" @@ -409,26 +423,28 @@ __git_refs () pfx="$pfx^" fer_pfx="$fer_pfx^" cur_=${cur_#^} + match=${match#^} fi case "$cur_" in refs|refs/*) format="refname" - refs=("$cur_*" "$cur_*/**") + refs=("$match*" "$match*/**") track="" ;; *) for i in HEAD FETCH_HEAD ORIG_HEAD MERGE_HEAD; do case "$i" in - $cur_*) if [ -e "$dir/$i" ]; then + $match*) + if [ -e "$dir/$i" ]; then echo "$pfx$i$sfx" fi ;; esac done format="refname:strip=2" - refs=("refs/tags/$cur_*" "refs/tags/$cur_*/**" - "refs/heads/$cur_*" "refs/heads/$cur_*/**" - "refs/remotes/$cur_*" "refs/remotes/$cur_*/**") + refs=("refs/tags/$match*" "refs/tags/$match*/**" + "refs/heads/$match*" "refs/heads/$match*/**" + "refs/remotes/$match*" "refs/remotes/$match*/**") ;; esac __git_dir="$dir" __git for-each-ref --format="$fer_pfx%($format)$sfx" \ @@ -439,14 +455,14 @@ __git_refs () # but only output if the branch name is unique __git for-each-ref --format="$fer_pfx%(refname:strip=3)$sfx" \ --sort="refname:strip=3" \ - "refs/remotes/*/$cur_*" "refs/remotes/*/$cur_*/**" | \ + "refs/remotes/*/$match*" "refs/remotes/*/$match*/**" | \ uniq -u fi return fi case "$cur_" in refs|refs/*) - __git ls-remote "$remote" "$cur_*" | \ + __git ls-remote "$remote" "$match*" | \ while read -r hash i; do case "$i" in *^{}) ;; @@ -457,19 +473,19 @@ __git_refs () *) if [ "$list_refs_from" = remote ]; then case "HEAD" in - $cur_*) echo "${pfx}HEAD$sfx" ;; + $match*) echo "${pfx}HEAD$sfx" ;; esac __git for-each-ref --format="$fer_pfx%(refname:strip=3)$sfx" \ - "refs/remotes/$remote/$cur_*" \ - "refs/remotes/$remote/$cur_*/**" + "refs/remotes/$remote/$match*" \ + "refs/remotes/$remote/$match*/**" else local query_symref case "HEAD" in - $cur_*) query_symref="HEAD" ;; + $match*) query_symref="HEAD" ;; esac __git ls-remote "$remote" $query_symref \ - "refs/tags/$cur_*" "refs/heads/$cur_*" \ - "refs/remotes/$cur_*" | + "refs/tags/$match*" "refs/heads/$match*" \ + "refs/remotes/$match*" | while read -r hash i; do case "$i" in *^{}) ;; @@ -513,6 +529,7 @@ __git_complete_refs () } # __git_refs2 requires 1 argument (to pass to __git_refs) +# Deprecated: use __git_complete_fetch_refspecs() instead. __git_refs2 () { local i @@ -521,6 +538,24 @@ __git_refs2 () done } +# Completes refspecs for fetching from a remote repository. +# 1: The remote repository. +# 2: A prefix to be added to each listed refspec (optional). +# 3: The ref to be completed as a refspec instead of the current word to be +# completed (optional) +# 4: A suffix to be appended to each listed refspec instead of the default +# space (optional). +__git_complete_fetch_refspecs () +{ + local i remote="$1" pfx="${2-}" cur_="${3-$cur}" sfx="${4- }" + + __gitcomp_direct "$( + for i in $(__git_refs "$remote" "" "" "$cur_") ; do + echo "$pfx$i:$i$sfx" + done + )" +} + # __git_refs_remotes requires 1 argument (to pass to ls-remote) __git_refs_remotes () { @@ -713,7 +748,7 @@ __git_complete_remote_or_refspec () case "$cmd" in fetch) if [ $lhs = 1 ]; then - __gitcomp_nl "$(__git_refs2 "$remote")" "$pfx" "$cur_" + __git_complete_fetch_refspecs "$remote" "$pfx" "$cur_" else __git_complete_refs --pfx="$pfx" --cur="$cur_" fi @@ -1161,7 +1196,7 @@ _git_branch () ;; *) if [ $only_local_ref = "y" -a $has_r = "n" ]; then - __gitcomp_nl "$(__git_heads)" + __gitcomp_direct "$(__git_heads "" "$cur" " ")" else __git_complete_refs fi @@ -2156,7 +2191,7 @@ _git_config () ;; branch.*) local pfx="${cur%.*}." cur_="${cur#*.}" - __gitcomp_nl "$(__git_heads)" "$pfx" "$cur_" "." + __gitcomp_direct "$(__git_heads "$pfx" "$cur_" ".")" __gitcomp_nl_append $'autosetupmerge\nautosetuprebase\n' "$pfx" "$cur_" return ;; @@ -2802,7 +2837,7 @@ _git_tag () i="${words[c]}" case "$i" in -d|-v) - __gitcomp_nl "$(__git_tags)" + __gitcomp_direct "$(__git_tags "" "$cur" " ")" return ;; -f) @@ -2817,7 +2852,7 @@ _git_tag () ;; -*|tag) if [ $f = 1 ]; then - __gitcomp_nl "$(__git_tags)" + __gitcomp_direct "$(__git_tags "" "$cur" " ")" fi ;; *) diff --git a/t/t9902-completion.sh b/t/t9902-completion.sh index be584c069..5ed28135b 100755 --- a/t/t9902-completion.sh +++ b/t/t9902-completion.sh @@ -571,6 +571,9 @@ test_expect_success '__git_refs - full refs' ' cat >expected <<-EOF && refs/heads/master refs/heads/matching-branch + refs/remotes/other/branch-in-other + refs/remotes/other/master-in-other + refs/tags/matching-tag EOF ( cur=refs/heads/ && @@ -636,6 +639,7 @@ test_expect_success '__git_refs - configured remote' ' test_expect_success '__git_refs - configured remote - full refs' ' cat >expected <<-EOF && + HEAD refs/heads/branch-in-other refs/heads/master-in-other refs/tags/tag-in-other @@ -664,6 +668,7 @@ test_expect_success '__git_refs - configured remote - repo given on the command test_expect_success '__git_refs - configured remote - full refs - repo given on the command line' ' cat >expected <<-EOF && + HEAD refs/heads/branch-in-other refs/heads/master-in-other refs/tags/tag-in-other @@ -708,6 +713,7 @@ test_expect_success '__git_refs - URL remote' ' test_expect_success '__git_refs - URL remote - full refs' ' cat >expected <<-EOF && + HEAD refs/heads/branch-in-other refs/heads/master-in-other refs/tags/tag-in-other @@ -861,6 +867,25 @@ test_expect_success 'setup for filtering matching refs' ' rm -f .git/FETCH_HEAD ' +test_expect_success '__git_refs - dont filter refs unless told so' ' + cat >expected <<-EOF && + HEAD + master + matching-branch + matching/branch + other/branch-in-other + other/master-in-other + other/matching/branch-in-other + matching-tag + matching/tag + EOF + ( + cur=master && + __git_refs >"$actual" + ) && + test_cmp expected "$actual" +' + test_expect_success '__git_refs - only matching refs' ' cat >expected <<-EOF && matching-branch @@ -870,7 +895,7 @@ test_expect_success '__git_refs - only matching refs' ' EOF ( cur=mat && - __git_refs >"$actual" + __git_refs "" "" "" "$cur" >"$actual" ) && test_cmp expected "$actual" ' @@ -882,7 +907,7 @@ test_expect_success '__git_refs - only matching refs - full refs' ' EOF ( cur=refs/heads/mat && - __git_refs >"$actual" + __git_refs "" "" "" "$cur" >"$actual" ) && test_cmp expected "$actual" ' @@ -894,7 +919,7 @@ test_expect_success '__git_refs - only matching refs - remote on local file syst EOF ( cur=ma && - __git_refs otherrepo >"$actual" + __git_refs otherrepo "" "" "$cur" >"$actual" ) && test_cmp expected "$actual" ' @@ -906,7 +931,7 @@ test_expect_success '__git_refs - only matching refs - configured remote' ' EOF ( cur=ma && - __git_refs other >"$actual" + __git_refs other "" "" "$cur" >"$actual" ) && test_cmp expected "$actual" ' @@ -918,7 +943,7 @@ test_expect_success '__git_refs - only matching refs - remote - full refs' ' EOF ( cur=refs/heads/ma && - __git_refs other >"$actual" + __git_refs other "" "" "$cur" >"$actual" ) && test_cmp expected "$actual" ' @@ -940,7 +965,7 @@ test_expect_success '__git_refs - only matching refs - checkout DWIMery' ' done && ( cur=mat && - __git_refs "" 1 >"$actual" + __git_refs "" 1 "" "$cur" >"$actual" ) && test_cmp expected "$actual" ' @@ -948,7 +973,8 @@ test_expect_success '__git_refs - only matching refs - checkout DWIMery' ' test_expect_success 'teardown after filtering matching refs' ' git branch -d matching/branch && git tag -d matching/tag && - git update-ref -d refs/remotes/other/matching/branch-in-other + git update-ref -d refs/remotes/other/matching/branch-in-other && + git -C otherrepo branch -D matching/branch-in-other ' test_expect_success '__git_refs - for-each-ref format specifiers in prefix' ' @@ -963,7 +989,7 @@ test_expect_success '__git_refs - for-each-ref format specifiers in prefix' ' ' test_expect_success '__git_complete_refs - simple' ' - sed -e "s/Z$//g" >expected <<-EOF && + sed -e "s/Z$//" >expected <<-EOF && HEAD Z master Z matching-branch Z @@ -980,7 +1006,7 @@ test_expect_success '__git_complete_refs - simple' ' ' test_expect_success '__git_complete_refs - matching' ' - sed -e "s/Z$//g" >expected <<-EOF && + sed -e "s/Z$//" >expected <<-EOF && matching-branch Z matching-tag Z EOF @@ -993,7 +1019,7 @@ test_expect_success '__git_complete_refs - matching' ' ' test_expect_success '__git_complete_refs - remote' ' - sed -e "s/Z$//g" >expected <<-EOF && + sed -e "s/Z$//" >expected <<-EOF && HEAD Z branch-in-other Z master-in-other Z @@ -1007,7 +1033,7 @@ test_expect_success '__git_complete_refs - remote' ' ' test_expect_success '__git_complete_refs - track' ' - sed -e "s/Z$//g" >expected <<-EOF && + sed -e "s/Z$//" >expected <<-EOF && HEAD Z master Z matching-branch Z @@ -1026,7 +1052,7 @@ test_expect_success '__git_complete_refs - track' ' ' test_expect_success '__git_complete_refs - current word' ' - sed -e "s/Z$//g" >expected <<-EOF && + sed -e "s/Z$//" >expected <<-EOF && matching-branch Z matching-tag Z EOF @@ -1039,7 +1065,7 @@ test_expect_success '__git_complete_refs - current word' ' ' test_expect_success '__git_complete_refs - prefix' ' - sed -e "s/Z$//g" >expected <<-EOF && + sed -e "s/Z$//" >expected <<-EOF && v1.0..matching-branch Z v1.0..matching-tag Z EOF @@ -1068,6 +1094,74 @@ test_expect_success '__git_complete_refs - suffix' ' test_cmp expected out ' +test_expect_success '__git_complete_fetch_refspecs - simple' ' + sed -e "s/Z$//" >expected <<-EOF && + HEAD:HEAD Z + branch-in-other:branch-in-other Z + master-in-other:master-in-other Z + EOF + ( + cur= && + __git_complete_fetch_refspecs other && + print_comp + ) && + test_cmp expected out +' + +test_expect_success '__git_complete_fetch_refspecs - matching' ' + sed -e "s/Z$//" >expected <<-EOF && + branch-in-other:branch-in-other Z + EOF + ( + cur=br && + __git_complete_fetch_refspecs other "" br && + print_comp + ) && + test_cmp expected out +' + +test_expect_success '__git_complete_fetch_refspecs - prefix' ' + sed -e "s/Z$//" >expected <<-EOF && + +HEAD:HEAD Z + +branch-in-other:branch-in-other Z + +master-in-other:master-in-other Z + EOF + ( + cur="+" && + __git_complete_fetch_refspecs other "+" "" && + print_comp + ) && + test_cmp expected out +' + +test_expect_success '__git_complete_fetch_refspecs - fully qualified' ' + sed -e "s/Z$//" >expected <<-EOF && + refs/heads/branch-in-other:refs/heads/branch-in-other Z + refs/heads/master-in-other:refs/heads/master-in-other Z + refs/tags/tag-in-other:refs/tags/tag-in-other Z + EOF + ( + cur=refs/ && + __git_complete_fetch_refspecs other "" refs/ && + print_comp + ) && + test_cmp expected out +' + +test_expect_success '__git_complete_fetch_refspecs - fully qualified & prefix' ' + sed -e "s/Z$//" >expected <<-EOF && + +refs/heads/branch-in-other:refs/heads/branch-in-other Z + +refs/heads/master-in-other:refs/heads/master-in-other Z + +refs/tags/tag-in-other:refs/tags/tag-in-other Z + EOF + ( + cur=+refs/ && + __git_complete_fetch_refspecs other + refs/ && + print_comp + ) && + test_cmp expected out +' + test_expect_success 'teardown after ref completion' ' git branch -d matching-branch && git tag -d matching-tag &&