aboutsummaryrefslogtreecommitdiffhomepage
path: root/scripts/bazel-complete-template.bash
diff options
context:
space:
mode:
authorGravatar Damien Martin-Guillerez <dmarting@google.com>2015-04-09 21:10:33 +0000
committerGravatar Ulf Adams <ulfjack@google.com>2015-04-10 08:02:42 +0000
commit5f9c6babed7a941cf23557fc152692a79b90395a (patch)
treea386009780f6d26ea5d9e481e415a158ddf53bc0 /scripts/bazel-complete-template.bash
parent29728d4c37721aa88621a8578a7c2d7bfe8b5c68 (diff)
Open-source Bazel bash completion script
This script enable shell completion for Bazel inside the Bourne-again shell. Fixes #38 -- MOS_MIGRATED_REVID=90745457
Diffstat (limited to 'scripts/bazel-complete-template.bash')
-rw-r--r--scripts/bazel-complete-template.bash469
1 files changed, 469 insertions, 0 deletions
diff --git a/scripts/bazel-complete-template.bash b/scripts/bazel-complete-template.bash
new file mode 100644
index 0000000000..e1df891b89
--- /dev/null
+++ b/scripts/bazel-complete-template.bash
@@ -0,0 +1,469 @@
+# -*- sh -*- (Bash only)
+#
+# Copyright 2015 Google Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+#
+#
+# Bash completion of Bazel commands.
+#
+# The template is expanded at build time using tables of commands/options
+# derived from the bazel executable built in the same client; the expansion is
+# written to bazel-complete.bash.
+#
+# Provides command-completion for:
+# - bazel prefix options (e.g. --host_jvm_args)
+# - blaze command-set (e.g. build, test)
+# - blaze command-specific options (e.g. --copts)
+# - values for enum-valued options
+# - package-names, exploring all package-path roots.
+# - targets within packages.
+
+# The package path used by the completion routines. Unfortunately
+# this isn't necessarily the same as the actual package path used by
+# Bazel, but that's ok. (It's impossible for us to reliably know what
+# the relevant package-path, so this is just a good guess. Users can
+# override it if they want.)
+#
+# Don't use it directly. Generate the final script with
+# bazel build //scripts:bash_completion instead.
+#
+: ${BAZEL_COMPLETION_PACKAGE_PATH:=%workspace%}
+
+
+# If true, Bazel query is used for autocompletion. This is more
+# accurate than the heuristic grep, especially for strangely-formatted
+# BUILD files. But it can be slower, especially if the Bazel server
+# is busy, and more brittle, if the BUILD file contains serious
+# errors. This is an experimental feature.
+: ${BAZEL_COMPLETION_USE_QUERY:=false}
+
+
+# If true, Bazel run allows autocompletion for test targets. This is convenient
+# for users who run a lot of tests/benchmarks locally with blaze run.
+: ${BAZEL_COMPLETION_ALLOW_TESTS_FOR_RUN:=false}
+
+# Some commands might interfer with the important one, so don't complete them
+: ${BAZEL_IGNORED_COMMAND_REGEX:="__none__"}
+
+# Bazel command
+: ${BAZEL:=bazel}
+
+# Pattern to match for looking for a target
+# BAZEL_BUILD_MATCH_PATTERN__* give the pattern for label-*
+# when looking in the the build file.
+# BAZEL_QUERY_MATCH_PATTERN__* give the pattern for label-*
+# when using 'bazel query'.
+# _RUNTEST are special case when BAZEL_COMPLETION_ALLOW_TESTS_FOR_RUN
+# is on.
+: ${BAZEL_BUILD_MATCH_PATTERN__test:='(.*_test|test_suite)'}
+: ${BAZEL_QUERY_MATCH_PATTERN__test:='(test|test_suite)'}
+: ${BAZEL_BUILD_MATCH_PATTERN__bin:='.*_binary'}
+: ${BAZEL_QUERY_MATCH_PATTERN__bin:='(binary)'}
+: ${BAZEL_BUILD_MATCH_PATTERN_RUNTEST__bin:='(.*_(binary|test)|test_suite)'}
+: ${BAZEL_QUERY_MATCH_PATTERN_RUNTEST__bin:='(binary|test)'}
+: ${BAZEL_BUILD_MATCH_PATTERN__:='.*'}
+: ${BAZEL_QUERY_MATCH_PATTERN__:=''}
+
+# Usage: _bazel__get_rule_match_pattern <command>
+# Determine what kind of rules to match, based on command.
+_bazel__get_rule_match_pattern() {
+ local var_name pattern
+ if _bazel__is_true "$BAZEL_COMPLETION_USE_QUERY"; then
+ var_name="BAZEL_QUERY_MATCH_PATTERN"
+ else
+ var_name="BAZEL_BUILD_MATCH_PATTERN"
+ fi
+ if [[ "$1" =~ ^label-?([a-z]*)$ ]]; then
+ pattern=${BASH_REMATCH[1]:-}
+ if _bazel__is_true "$BAZEL_COMPLETION_ALLOW_TESTS_FOR_RUN"; then
+ eval "echo \"\${${var_name}_RUNTEST__${pattern}:-\$${var_name}__${pattern}}\""
+ else
+ eval "echo \"\$${var_name}__${pattern}\""
+ fi
+ fi
+}
+
+# Compute workspace directory. Search for the innermost
+# enclosing directory with a WORKSPACE file.
+_bazel__get_workspace_path() {
+ local workspace=$PWD
+ while true; do
+ if [ -f "${workspace}/WORKSPACE" ]; then
+ break
+ elif [ -z "$workspace" -o "$workspace" = "/" ]; then
+ workspace=$PWD
+ break;
+ fi
+ workspace=${workspace%/*}
+ done
+ echo $workspace
+}
+
+
+# Find the current piece of the line to complete, but only do word breaks at
+# certain characters. In particular, ignore these: "':=
+# This method also takes into account the current cursor position.
+#
+# Works with both bash 3 and 4! Bash 3 and 4 perform different word breaks when
+# computing the COMP_WORDS array. We need this here because Bazel options are of
+# the form --a=b, and labels of the form //some/label:target.
+_bazel__get_cword() {
+ local cur=${COMP_LINE:0:$COMP_POINT}
+ # This expression finds the last word break character, as defined in the
+ # COMP_WORDBREAKS variable, but without '=' or ':', which is not preceeded by
+ # a slash. Quote characters are also excluded.
+ local wordbreaks="$COMP_WORDBREAKS"
+ wordbreaks="${wordbreaks//\'/}"
+ wordbreaks="${wordbreaks//\"/}"
+ wordbreaks="${wordbreaks//:/}"
+ wordbreaks="${wordbreaks//=/}"
+ local word_start=$(expr "$cur" : '.*[^\]['"${wordbreaks}"']')
+ echo "${cur:$word_start}"
+}
+
+
+# Usage: _bazel__package_path <workspace> <displacement>
+#
+# Prints a list of package-path root directories, displaced using the
+# current displacement from the workspace. All elements have a
+# trailing slash.
+_bazel__package_path() {
+ local workspace=$1 displacement=$2 root
+ IFS=:
+ for root in ${BAZEL_COMPLETION_PACKAGE_PATH//\%workspace\%/$workspace}; do
+ unset IFS
+ echo "$root/$displacement"
+ done
+}
+
+# Usage: _bazel__options_for <command>
+#
+# Prints the set of options for a given Bazel command, e.g. "build".
+_bazel__options_for() {
+ local options
+ if [[ "${BAZEL_COMMAND_LIST}" =~ ^(.* )?$1( .*)?$ ]]; then
+ local option_name=$(echo $1 | perl -ne 'print uc' | tr "-" "_")
+ eval "echo \${BAZEL_COMMAND_${option_name}_FLAGS}" | tr " " "\n"
+ fi
+}
+# Usage: _bazel__expansion_for <command>
+#
+# Prints the completion pattern for a given Bazel command, e.g. "build".
+_bazel__expansion_for() {
+ local options
+ if [[ "${BAZEL_COMMAND_LIST}" =~ ^(.* )?$1( .*)?$ ]]; then
+ local option_name=$(echo $1 | perl -ne 'print uc' | tr "-" "_")
+ eval "echo \${BAZEL_COMMAND_${option_name}_ARGUMENT}"
+ fi
+}
+
+# Usage: _bazel__matching_targets <kind> <prefix>
+#
+# Prints target names of kind <kind> and starting with <prefix> in the BUILD
+# file given as standard input. <kind> is a basic regex (BRE) used to match the
+# bazel rule kind and <prefix> is the prefix of the target name.
+_bazel__matching_targets() {
+ local kind_pattern="$1"
+ local target_prefix="$2"
+ # The following commands do respectively:
+ # Remove BUILD file comments
+ # Replace \n by spaces to have the BUILD file in a single line
+ # Extract all rule types and target names
+ # Grep the kind pattern and the target prefix
+ # Returns the target name
+ sed 's/#.*$//' \
+ | tr "\n" " " \
+ | sed 's/\([a-zA-Z0-9_]*\) *(\([^)]* \)\{0,1\}name *= *['\''"]\([a-zA-Z0-9_/.+=,@~-]*\)['\''"][^)]*)/\
+type:\1 name:\3\
+/g' \
+ | grep -E "^type:$kind_pattern name:$target_prefix" \
+ | cut -d ':' -f 3
+}
+
+
+# Usage: _bazel__is_true <string>
+#
+# Returns true or false based on the input string. The following are
+# valid true values (the rest are false): "1", "true".
+_bazel__is_true() {
+ local str="$1"
+ [[ "$str" == "1" || "$str" == "true" ]]
+}
+
+# Usage: _bazel__expand_rules_in_package <workspace> <displacement>
+# <current> <label-type>
+#
+# Expands rules in specified packages, exploring all roots of
+# $BAZEL_COMPLETION_PACKAGE_PATH, not just $(pwd). Only rules
+# appropriate to the command are printed. Sets $COMPREPLY array to
+# result.
+#
+# If $BAZEL_COMPLETION_USE_QUERY is true, 'bazel query' is used
+# instead, with the actual Bazel package path;
+# $BAZEL_COMPLETION_PACKAGE_PATH is ignored in this case, since the
+# actual Bazel value is likely to be more accurate.
+_bazel__expand_rules_in_package() {
+ local workspace=$1 displacement=$2 current=$3 label_type=$4
+ local package_name=$(echo "$current" | cut -f1 -d:)
+ local rule_prefix=$(echo "$current" | cut -f2 -d:)
+ local root buildfile rule_pattern r result
+
+ result=
+ pattern=$(_bazel__get_rule_match_pattern "$label_type")
+ if _bazel__is_true "$BAZEL_COMPLETION_USE_QUERY"; then
+ package_name=$(echo "$package_name" | tr -d "'\"") # remove quotes
+ result=$(${BAZEL} --output_base=/tmp/${BAZEL}-completion-$USER query \
+ --keep_going --noshow_progress \
+ "kind('$pattern rule', '$package_name:*')" 2>/dev/null |
+ cut -f2 -d: | grep "^$rule_prefix")
+ else
+ for root in $(_bazel__package_path "$workspace" "$displacement"); do
+ buildfile="$root/$package_name/BUILD"
+ if [ -f "$buildfile" ]; then
+ result=$(_bazel__matching_targets \
+ "$pattern" "$rule_prefix" <"$buildfile")
+ break
+ fi
+ done
+ fi
+
+ index=$(echo $result | wc -w)
+ if [ -n "$result" ]; then
+ echo "$result" | tr " " "\n" | sed 's|$| |'
+ fi
+ # Include ":all" wildcard if there was no unique match. (The zero
+ # case is tricky: we need to include "all" in that case since
+ # otherwise we won't expand "a" to "all" in the absence of rules
+ # starting with "a".)
+ if [ $index -ne 1 ] && expr all : "\\($rule_prefix\\)" >/dev/null; then
+ echo "all "
+ fi
+}
+
+# Usage: _bazel__expand_package_name <workspace> <displacement> <current-word>
+# <label-type>
+#
+# Expands directories, but explores all roots of
+# BAZEL_COMPLETION_PACKAGE_PATH, not just $(pwd). When a directory is
+# a bazel package, the completion offers "pkg:" so you can expand
+# inside the package.
+# Sets $COMPREPLY array to result.
+_bazel__expand_package_name() {
+ local workspace=$1 displacement=$2 current=$3 type=${4:-} root dir index
+ for root in $(_bazel__package_path "$workspace" "$displacement"); do
+ found=0
+ for dir in $(compgen -d $root$current); do
+ [ -L "$dir" ] && continue # skip symlinks (e.g. bazel-bin)
+ [[ "$dir" =~ ^(.*/)?\.[^/]*$ ]] && continue # skip dotted dir (e.g. .git)
+ found=1
+ echo "${dir#$root}/"
+ if [ -f $dir/BUILD ]; then
+ if [ "${type}" = "label-package" ]; then
+ echo "${dir#$root} "
+ else
+ echo "${dir#$root}:"
+ fi
+ fi
+ done
+ [ $found -gt 0 ] && break # Stop searching package path upon first match.
+ done
+}
+
+# Usage: _bazel__expand_target_pattern <workspace> <displacement>
+# <word> <label-syntax>
+#
+# Expands "word" to match target patterns, using the current workspace
+# and displacement from it. "command" is used to filter rules.
+# Sets $COMPREPLY array to result.
+_bazel__expand_target_pattern() {
+ local workspace=$1 displacement=$2 current=$3 label_syntax=$4
+ case "$current" in
+ //*:*) # Expand rule names within package, no displacement.
+ if [ "${label_syntax}" = "label-package" ]; then
+ compgen -S " " -W "BUILD" "$(echo current | cut -f ':' -d2)"
+ else
+ _bazel__expand_rules_in_package "$workspace" "" "$current" "$label_syntax"
+ fi
+ ;;
+ *:*) # Expand rule names within package, displaced.
+ if [ "${label_syntax}" = "label-package" ]; then
+ compgen -S " " -W "BUILD" "$(echo current | cut -f ':' -d2)"
+ else
+ _bazel__expand_rules_in_package \
+ "$workspace" "$displacement" "$current" "$label_syntax"
+ fi
+ ;;
+ //*) # Expand filenames using package-path, no displacement
+ _bazel__expand_package_name "$workspace" "" "$current" "$label_syntax"
+ ;;
+ *) # Expand filenames using package-path, displaced.
+ if [ -n "$current" ]; then
+ _bazel__expand_package_name "$workspace" "$displacement" "$current" "$label_syntax"
+ fi
+ ;;
+ esac
+}
+
+_bazel__get_command() {
+ for word in "${COMP_WORDS[@]:1:COMP_CWORD-1}"; do
+ if echo "$BAZEL_COMMAND_LIST" | grep -wsq -e "$word"; then
+ echo $word
+ break
+ fi
+ done
+}
+
+# Returns the displacement to the workspace given in $1
+_bazel__get_displacement() {
+ if [[ "$PWD" =~ ^$1/.*$ ]]; then
+ echo ${PWD##$1/}/
+ fi
+}
+
+
+# Usage: _bazel__complete_pattern <workspace> <displacement> <current>
+# <type>
+#
+# Expand a word according to a type. The currently supported types are:
+# - {a,b,c}: an enum that can take value a, b or c
+# - label: a label of any kind
+# - label-bin: a label to a runnable rule (basically to a _binary rule)
+# - label-test: a label to a test rule
+# - info-key: an info key as listed by `bazel help info-keys`
+# - command: the name of a command
+# - path: a file path
+# - combinaison of previous type using | as separator
+_bazel__complete_pattern() {
+ local workspace=$1 displacement=$2 current=$3 types=$4
+ for type in $(echo $types | tr "|" "\n"); do
+ case "$type" in
+ label*)
+ _bazel__expand_target_pattern "$workspace" "$displacement" \
+ "$current" "$type"
+ ;;
+ info-key)
+ compgen -S " " -W "${BAZEL_INFO_KEYS}" -- "$current"
+ ;;
+ "command")
+ local commands=$(echo "${BAZEL_COMMAND_LIST}" \
+ | tr " " "\n" | grep -v "^${BAZEL_IGNORED_COMMAND_REGEX}$")
+ compgen -S " " -W "${commands}" -- "$current"
+ ;;
+ path)
+ compgen -f -- "$current"
+ ;;
+ *)
+ compgen -S " " -W "$type" -- "$current"
+ ;;
+ esac
+ done
+}
+
+# Usage: _bazel__expand_options <workspace> <displacement> <current-word>
+# <options>
+#
+# Expands options, making sure that if current-word contains an equals sign,
+# it is handled appropriately.
+_bazel__expand_options() {
+ local workspace="$1" displacement="$2" cur="$3" options="$4"
+ if [[ $cur =~ = ]]; then
+ # also expands special labels
+ current=$(echo "$cur" | cut -f2 -d=)
+ _bazel__complete_pattern "$workspace" "$displacement" "$current" \
+ "$(compgen -W "$options" -- "$cur" | cut -f2 -d=)" \
+ | sort -u
+ else
+ compgen -W "$(echo "$options" | sed 's|=.*$|=|')" -- "$cur" \
+ | sed 's|\([^=]\)$|\1 |'
+ fi
+}
+
+
+_bazel__complete_stdout() {
+ local cur=$(_bazel__get_cword) word command displacement workspace
+
+ # Determine command: "" (startup-options) or one of $BAZEL_COMMAND_LIST.
+ command="$(_bazel__get_command)"
+
+ workspace="$(_bazel__get_workspace_path)"
+ displacement="$(_bazel__get_displacement ${workspace})"
+
+ case "$command" in
+ "") # Expand startup-options or commands
+ local commands=$(echo "${BAZEL_COMMAND_LIST}" \
+ | tr " " "\n" | grep -v "^${BAZEL_IGNORED_COMMAND_REGEX}$")
+ _bazel__expand_options "$workspace" "$displacement" "$cur" \
+ "${commands}\
+ ${BAZEL_STARTUP_OPTIONS}"
+ ;;
+
+ *)
+ case "$cur" in
+ -*) # Expand options:
+ _bazel__expand_options "$workspace" "$displacement" "$cur" \
+ "$(_bazel__options_for $command)"
+ ;;
+ *) # Expand target pattern
+ expansion_pattern="$(_bazel__expansion_for $command)"
+ NON_QUOTE_REGEX="^[\"']"
+ if [[ $command = query && $cur =~ $NON_QUOTE_REGEX ]]; then
+ : # Ideally we would expand query expressions---it's not
+ # that hard, conceptually---but readline is just too
+ # damn complex when it comes to quotation. Instead,
+ # for query, we just expand target patterns, unless
+ # the first char is a quote.
+ elif [ -n "$expansion_pattern" ]; then
+ _bazel__complete_pattern \
+ "$workspace" "$displacement" "$cur" "$expansion_pattern"
+ fi
+ ;;
+ esac
+ ;;
+ esac
+}
+
+_bazel__to_compreply() {
+ local replies="$1"
+ COMPREPLY=()
+ # Trick to preserve whitespaces
+ while IFS="" read -r reply; do
+ COMPREPLY+=("${reply}")
+ done < <(echo "${replies}")
+}
+
+_bazel__complete() {
+ _bazel__to_compreply "$(_bazel__complete_stdout)"
+}
+
+# Some users have aliases such as bt="bazel test" or bb="bazel build", this
+# completion function allows them to have auto-completion for these aliases.
+_bazel__complete_target_stdout() {
+ local cur=$(_bazel__get_cword) word command displacement workspace
+
+ # Determine command: "" (startup-options) or one of $BAZEL_COMMAND_LIST.
+ command="$1"
+
+ workspace="$(_bazel__get_workspace_path)"
+ displacement="$(_bazel__get_displacement ${workspace})"
+
+ _bazel__to_compreply "$(_bazel__expand_target_pattern "$workspace" "$displacement" \
+ "$cur" "$command")"
+}
+
+# default completion for bazel
+complete -F _bazel__complete -o nospace "${BAZEL}"
+
+########################################################################
+## DO NOT EDIT BELOW THIS LINE.
+## This section is automatically generated by update-completion.sh.