2021-01-08 11:10:24 -05:00
|
|
|
# typed: true
|
|
|
|
# frozen_string_literal: true
|
|
|
|
|
|
|
|
require "utils/link"
|
2021-01-12 16:27:25 -05:00
|
|
|
require "settings"
|
2021-01-05 22:44:39 -05:00
|
|
|
require "erb"
|
2021-01-08 11:10:24 -05:00
|
|
|
|
2021-01-13 11:16:09 -05:00
|
|
|
module Homebrew
|
|
|
|
# Helper functions for generating shell completions.
|
|
|
|
#
|
|
|
|
# @api private
|
|
|
|
module Completions
|
|
|
|
extend T::Sig
|
2021-01-08 11:10:24 -05:00
|
|
|
|
2021-01-13 11:16:09 -05:00
|
|
|
module_function
|
2021-01-08 11:10:24 -05:00
|
|
|
|
2021-01-05 22:44:39 -05:00
|
|
|
COMPLETIONS_DIR = (HOMEBREW_REPOSITORY/"completions").freeze
|
|
|
|
TEMPLATE_DIR = (HOMEBREW_LIBRARY_PATH/"completions").freeze
|
|
|
|
|
2021-01-13 11:16:09 -05:00
|
|
|
SHELLS = %w[bash fish zsh].freeze
|
2021-01-05 22:44:39 -05:00
|
|
|
COMPLETIONS_EXCLUSION_LIST = %w[
|
|
|
|
instal
|
|
|
|
uninstal
|
|
|
|
update-report
|
|
|
|
].freeze
|
|
|
|
|
|
|
|
BASH_NAMED_ARGS_COMPLETION_FUNCTION_MAPPING = {
|
|
|
|
formula: "__brew_complete_formulae",
|
|
|
|
installed_formula: "__brew_complete_installed_formulae",
|
|
|
|
outdated_formula: "__brew_complete_outdated_formulae",
|
|
|
|
cask: "__brew_complete_casks",
|
|
|
|
installed_cask: "__brew_complete_installed_casks",
|
|
|
|
outdated_cask: "__brew_complete_outdated_casks",
|
|
|
|
tap: "__brew_complete_tapped",
|
|
|
|
installed_tap: "__brew_complete_tapped",
|
|
|
|
command: "__brew_complete_commands",
|
|
|
|
diagnostic_check: '__brewcomp "$(brew doctor --list-checks)"',
|
2021-01-18 13:30:47 -05:00
|
|
|
file: "__brew_complete_files",
|
2021-01-05 22:44:39 -05:00
|
|
|
}.freeze
|
2021-01-13 11:16:09 -05:00
|
|
|
|
|
|
|
sig { void }
|
|
|
|
def link!
|
|
|
|
Settings.write :linkcompletions, true
|
|
|
|
Tap.each do |tap|
|
|
|
|
Utils::Link.link_completions tap.path, "brew completions link"
|
|
|
|
end
|
2021-01-11 12:24:48 -05:00
|
|
|
end
|
2021-01-08 11:10:24 -05:00
|
|
|
|
2021-01-13 11:16:09 -05:00
|
|
|
sig { void }
|
|
|
|
def unlink!
|
|
|
|
Settings.write :linkcompletions, false
|
|
|
|
Tap.each do |tap|
|
|
|
|
next if tap.official?
|
2021-01-11 12:24:48 -05:00
|
|
|
|
2021-01-13 11:16:09 -05:00
|
|
|
Utils::Link.unlink_completions tap.path
|
|
|
|
end
|
2021-01-11 12:24:48 -05:00
|
|
|
end
|
2021-01-08 11:10:24 -05:00
|
|
|
|
2021-01-13 11:16:09 -05:00
|
|
|
sig { returns(T::Boolean) }
|
|
|
|
def link_completions?
|
|
|
|
Settings.read(:linkcompletions) == "true"
|
|
|
|
end
|
2021-01-08 11:10:24 -05:00
|
|
|
|
2021-01-13 11:16:09 -05:00
|
|
|
sig { returns(T::Boolean) }
|
|
|
|
def completions_to_link?
|
|
|
|
Tap.each do |tap|
|
|
|
|
next if tap.official?
|
2021-01-11 12:24:48 -05:00
|
|
|
|
2021-01-13 11:16:09 -05:00
|
|
|
SHELLS.each do |shell|
|
|
|
|
return true if (tap.path/"completions/#{shell}").exist?
|
|
|
|
end
|
2021-01-11 12:24:48 -05:00
|
|
|
end
|
|
|
|
|
2021-01-13 11:16:09 -05:00
|
|
|
false
|
|
|
|
end
|
2021-01-11 12:24:48 -05:00
|
|
|
|
2021-01-13 11:16:09 -05:00
|
|
|
sig { void }
|
|
|
|
def show_completions_message_if_needed
|
|
|
|
return if Settings.read(:completionsmessageshown) == "true"
|
|
|
|
return unless completions_to_link?
|
2021-01-11 12:24:48 -05:00
|
|
|
|
2021-01-13 11:16:09 -05:00
|
|
|
ohai "Homebrew completions for external commands are unlinked by default!"
|
|
|
|
puts <<~EOS
|
|
|
|
To opt-in to automatically linking external tap shell competion files, run:
|
|
|
|
brew completions link
|
|
|
|
Then, follow the directions at #{Formatter.url("https://docs.brew.sh/Shell-Completion")}
|
|
|
|
EOS
|
2021-01-11 12:24:48 -05:00
|
|
|
|
2021-01-13 11:16:09 -05:00
|
|
|
Settings.write :completionsmessageshown, true
|
|
|
|
end
|
2021-01-05 22:44:39 -05:00
|
|
|
|
|
|
|
sig { void }
|
|
|
|
def update_shell_completions!
|
|
|
|
commands = Commands.commands(external: false, aliases: true).sort
|
|
|
|
|
|
|
|
(COMPLETIONS_DIR/"bash/brew").atomic_write generate_bash_completion_file(commands)
|
|
|
|
end
|
|
|
|
|
|
|
|
sig { params(command: String).returns(T::Boolean) }
|
|
|
|
def command_gets_completions?(command)
|
2021-01-18 03:09:57 -05:00
|
|
|
return false if command.start_with? "cask " # TODO: (2.8) remove when `brew cask` commands are removed
|
2021-01-05 22:44:39 -05:00
|
|
|
|
|
|
|
command_options(command).any?
|
|
|
|
end
|
|
|
|
|
|
|
|
sig { params(command: String).returns(T::Array[String]) }
|
|
|
|
def command_options(command)
|
|
|
|
options = []
|
|
|
|
Commands.command_options(command)&.each do |option|
|
|
|
|
next if option.blank?
|
|
|
|
|
|
|
|
name = option.first
|
|
|
|
if name.start_with? "--[no-]"
|
|
|
|
options << name.remove("[no-]")
|
|
|
|
options << name.sub("[no-]", "no-")
|
|
|
|
else
|
|
|
|
options << name
|
|
|
|
end
|
|
|
|
end&.compact
|
|
|
|
options.sort
|
|
|
|
end
|
|
|
|
|
|
|
|
sig { params(command: String).returns(T.nilable(String)) }
|
|
|
|
def generate_bash_subcommand_completion(command)
|
|
|
|
return unless command_gets_completions? command
|
|
|
|
|
|
|
|
named_completion_string = ""
|
|
|
|
if types = Commands.named_args_type(command)
|
|
|
|
named_args_strings, named_args_types = types.partition { |type| type.is_a? String }
|
|
|
|
|
|
|
|
named_args_types.each do |type|
|
|
|
|
next unless BASH_NAMED_ARGS_COMPLETION_FUNCTION_MAPPING.key? type
|
|
|
|
|
|
|
|
named_completion_string += "\n #{BASH_NAMED_ARGS_COMPLETION_FUNCTION_MAPPING[type]}"
|
|
|
|
end
|
|
|
|
|
|
|
|
named_completion_string += "\n __brewcomp \"#{named_args_strings.join(" ")}\"" if named_args_strings.any?
|
|
|
|
end
|
|
|
|
|
|
|
|
<<~COMPLETION
|
|
|
|
_brew_#{Commands.method_name command}() {
|
|
|
|
local cur="${COMP_WORDS[COMP_CWORD]}"
|
|
|
|
case "$cur" in
|
|
|
|
-*)
|
|
|
|
__brewcomp "
|
|
|
|
#{command_options(command).join("\n ")}
|
|
|
|
"
|
|
|
|
return
|
|
|
|
;;
|
|
|
|
esac#{named_completion_string}
|
|
|
|
}
|
|
|
|
COMPLETION
|
|
|
|
end
|
|
|
|
|
|
|
|
sig { params(commands: T::Array[String]).returns(T.nilable(String)) }
|
|
|
|
def generate_bash_completion_file(commands)
|
|
|
|
variables = OpenStruct.new
|
|
|
|
|
|
|
|
variables[:completion_functions] = commands.map do |command|
|
|
|
|
generate_bash_subcommand_completion command
|
|
|
|
end.compact
|
|
|
|
|
|
|
|
variables[:function_mappings] = commands.map do |command|
|
|
|
|
next unless command_gets_completions? command
|
|
|
|
|
|
|
|
"#{command}) _brew_#{Commands.method_name command} ;;"
|
|
|
|
end.compact
|
|
|
|
|
|
|
|
ERB.new((TEMPLATE_DIR/"bash.erb").read, trim_mode: ">").result(variables.instance_eval { binding })
|
|
|
|
end
|
2021-01-11 12:24:48 -05:00
|
|
|
end
|
2021-01-08 11:10:24 -05:00
|
|
|
end
|