| 
									
										
										
										
											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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-24 01:59:31 -05:00
										 |  |  |     ZSH_NAMED_ARGS_COMPLETION_FUNCTION_MAPPING = { | 
					
						
							|  |  |  |       formula:           "__brew_formulae", | 
					
						
							|  |  |  |       installed_formula: "__brew_installed_formulae", | 
					
						
							|  |  |  |       outdated_formula:  "__brew_outdated_formulae", | 
					
						
							|  |  |  |       cask:              "__brew_casks", | 
					
						
							|  |  |  |       installed_cask:    "__brew_installed_casks", | 
					
						
							|  |  |  |       outdated_cask:     "__brew_outdated_casks", | 
					
						
							|  |  |  |       tap:               "__brew_any_tap", | 
					
						
							|  |  |  |       installed_tap:     "__brew_installed_taps", | 
					
						
							|  |  |  |       command:           "__brew_commands", | 
					
						
							|  |  |  |       diagnostic_check:  "__brew_diagnostic_checks", | 
					
						
							|  |  |  |       file:              "__brew_formulae_or_ruby_files", | 
					
						
							|  |  |  |     }.freeze | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-26 10:47:56 -05:00
										 |  |  |     FISH_NAMED_ARGS_COMPLETION_FUNCTION_MAPPING = { | 
					
						
							|  |  |  |       formula:           "__fish_brew_suggest_formulae_all", | 
					
						
							|  |  |  |       installed_formula: "__fish_brew_suggest_formulae_installed", | 
					
						
							|  |  |  |       outdated_formula:  "__fish_brew_suggest_formulae_outdated", | 
					
						
							|  |  |  |       cask:              "__fish_brew_suggest_casks_all", | 
					
						
							|  |  |  |       installed_cask:    "__fish_brew_suggest_casks_installed", | 
					
						
							|  |  |  |       outdated_cask:     "__fish_brew_suggest_casks_outdated", | 
					
						
							|  |  |  |       tap:               "__fish_brew_suggest_taps_installed", | 
					
						
							|  |  |  |       installed_tap:     "__fish_brew_suggest_taps_installed", | 
					
						
							|  |  |  |       command:           "__fish_brew_suggest_commands", | 
					
						
							|  |  |  |       diagnostic_check:  "__fish_brew_suggest_diagnostic_checks", | 
					
						
							|  |  |  |     }.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 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-24 16:42:28 -05:00
										 |  |  |       puts "Writing completions to #{COMPLETIONS_DIR}" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-05 22:44:39 -05:00
										 |  |  |       (COMPLETIONS_DIR/"bash/brew").atomic_write generate_bash_completion_file(commands) | 
					
						
							| 
									
										
										
										
											2021-01-24 01:59:31 -05:00
										 |  |  |       (COMPLETIONS_DIR/"zsh/_brew").atomic_write generate_zsh_completion_file(commands) | 
					
						
							| 
									
										
										
										
											2021-01-26 10:47:56 -05:00
										 |  |  |       (COMPLETIONS_DIR/"fish/brew.fish").atomic_write generate_fish_completion_file(commands) | 
					
						
							| 
									
										
										
										
											2021-01-05 22:44:39 -05:00
										 |  |  |     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 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-26 10:47:56 -05:00
										 |  |  |     sig { params(description: String, fish: T::Boolean).returns(String) } | 
					
						
							|  |  |  |     def format_description(description, fish: false) | 
					
						
							|  |  |  |       description = if fish | 
					
						
							|  |  |  |         description.gsub("'", "\\\\'") | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  |         description.gsub("'", "'\\\\''") | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |       description.gsub(/[<>]/, "").tr("\n", " ").chomp(".") | 
					
						
							| 
									
										
										
										
											2021-01-24 01:59:31 -05:00
										 |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-24 03:12:52 -05:00
										 |  |  |     sig { params(command: String).returns(T::Hash[String, String]) } | 
					
						
							| 
									
										
										
										
											2021-01-05 22:44:39 -05:00
										 |  |  |     def command_options(command) | 
					
						
							| 
									
										
										
										
											2021-01-24 03:12:52 -05:00
										 |  |  |       options = {} | 
					
						
							| 
									
										
										
										
											2021-01-05 22:44:39 -05:00
										 |  |  |       Commands.command_options(command)&.each do |option| | 
					
						
							|  |  |  |         next if option.blank? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         name = option.first | 
					
						
							| 
									
										
										
										
											2021-01-24 16:42:28 -05:00
										 |  |  |         desc = option.second | 
					
						
							| 
									
										
										
										
											2021-01-05 22:44:39 -05:00
										 |  |  |         if name.start_with? "--[no-]" | 
					
						
							| 
									
										
										
										
											2021-01-24 03:12:52 -05:00
										 |  |  |           options[name.remove("[no-]")] = desc | 
					
						
							|  |  |  |           options[name.sub("[no-]", "no-")] = desc | 
					
						
							| 
									
										
										
										
											2021-01-05 22:44:39 -05:00
										 |  |  |         else | 
					
						
							| 
									
										
										
										
											2021-01-24 03:12:52 -05:00
										 |  |  |           options[name] = desc | 
					
						
							| 
									
										
										
										
											2021-01-05 22:44:39 -05:00
										 |  |  |         end | 
					
						
							| 
									
										
										
										
											2021-01-24 03:12:52 -05:00
										 |  |  |       end | 
					
						
							|  |  |  |       options | 
					
						
							| 
									
										
										
										
											2021-01-05 22:44:39 -05:00
										 |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     sig { params(command: String).returns(T.nilable(String)) } | 
					
						
							|  |  |  |     def generate_bash_subcommand_completion(command) | 
					
						
							|  |  |  |       return unless command_gets_completions? command | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       named_completion_string = "" | 
					
						
							| 
									
										
										
										
											2021-02-12 18:33:37 +05:30
										 |  |  |       if (types = Commands.named_args_type(command)) | 
					
						
							| 
									
										
										
										
											2021-01-05 22:44:39 -05:00
										 |  |  |         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 "
 | 
					
						
							| 
									
										
										
										
											2021-01-24 03:12:52 -05:00
										 |  |  |               #{command_options(command).keys.sort.join("\n      ")} | 
					
						
							| 
									
										
										
										
											2021-01-05 22:44:39 -05:00
										 |  |  |               "
 | 
					
						
							|  |  |  |               return | 
					
						
							|  |  |  |               ;; | 
					
						
							|  |  |  |           esac#{named_completion_string} | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       COMPLETION | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-24 01:59:31 -05:00
										 |  |  |     sig { params(commands: T::Array[String]).returns(String) } | 
					
						
							| 
									
										
										
										
											2021-01-05 22:44:39 -05:00
										 |  |  |     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-24 01:59:31 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     sig { params(command: String).returns(T.nilable(String)) } | 
					
						
							|  |  |  |     def generate_zsh_subcommand_completion(command) | 
					
						
							|  |  |  |       return unless command_gets_completions? command | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-24 03:12:52 -05:00
										 |  |  |       options = command_options(command).sort.map do |opt, desc| | 
					
						
							| 
									
										
										
										
											2021-01-24 01:59:31 -05:00
										 |  |  |         next opt if desc.blank? | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-27 13:45:11 +11:00
										 |  |  |         conflicts = generate_zsh_option_exclusions(command, opt) | 
					
						
							|  |  |  |         "#{conflicts}#{opt}[#{format_description desc}]" | 
					
						
							| 
									
										
										
										
											2021-01-24 01:59:31 -05:00
										 |  |  |       end | 
					
						
							| 
									
										
										
										
											2021-02-12 18:33:37 +05:30
										 |  |  |       if (types = Commands.named_args_type(command)) | 
					
						
							| 
									
										
										
										
											2021-01-24 01:59:31 -05:00
										 |  |  |         named_args_strings, named_args_types = types.partition { |type| type.is_a? String } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         named_args_types.each do |type| | 
					
						
							|  |  |  |           next unless ZSH_NAMED_ARGS_COMPLETION_FUNCTION_MAPPING.key? type | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           options << "::#{type}:#{ZSH_NAMED_ARGS_COMPLETION_FUNCTION_MAPPING[type]}" | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         options << "::subcommand:(#{named_args_strings.join(" ")})" if named_args_strings.any? | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       <<~COMPLETION | 
					
						
							|  |  |  |         # brew #{command} | 
					
						
							|  |  |  |         _brew_#{Commands.method_name command}() { | 
					
						
							|  |  |  |           _arguments \\ | 
					
						
							|  |  |  |             #{options.map! { |opt| "'#{opt}'" }.join(" \\\n    ")} | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       COMPLETION | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-27 13:45:11 +11:00
										 |  |  |     def generate_zsh_option_exclusions(command, option) | 
					
						
							|  |  |  |       conflicts = Commands.option_conflicts(command, option.gsub(/^--/, "")) | 
					
						
							|  |  |  |       return "" unless conflicts.presence | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       "(#{conflicts.map { |conflict| "--#{conflict}" }.join(" ")})" | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-24 01:59:31 -05:00
										 |  |  |     sig { params(commands: T::Array[String]).returns(String) } | 
					
						
							|  |  |  |     def generate_zsh_completion_file(commands) | 
					
						
							|  |  |  |       variables = OpenStruct.new | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       variables[:aliases] = Commands::HOMEBREW_INTERNAL_COMMAND_ALIASES.map do |alias_command, command| | 
					
						
							|  |  |  |         alias_command = "'#{alias_command}'" if alias_command.start_with? "-" | 
					
						
							|  |  |  |         command = "'#{command}'" if command.start_with? "-" | 
					
						
							|  |  |  |         "#{alias_command} #{command}" | 
					
						
							|  |  |  |       end.compact | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       variables[:builtin_command_descriptions] = commands.map do |command| | 
					
						
							|  |  |  |         next if Commands::HOMEBREW_INTERNAL_COMMAND_ALIASES.key? command | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-24 16:42:28 -05:00
										 |  |  |         description = Commands.command_description(command, short: true) | 
					
						
							| 
									
										
										
										
											2021-01-24 01:59:31 -05:00
										 |  |  |         next if description.blank? | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-24 16:42:28 -05:00
										 |  |  |         description = format_description description | 
					
						
							| 
									
										
										
										
											2021-01-24 01:59:31 -05:00
										 |  |  |         "'#{command}:#{description}'" | 
					
						
							|  |  |  |       end.compact | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       variables[:completion_functions] = commands.map do |command| | 
					
						
							|  |  |  |         generate_zsh_subcommand_completion command | 
					
						
							|  |  |  |       end.compact | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       ERB.new((TEMPLATE_DIR/"zsh.erb").read, trim_mode: ">").result(variables.instance_eval { binding }) | 
					
						
							|  |  |  |     end | 
					
						
							| 
									
										
										
										
											2021-01-26 10:47:56 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     sig { params(command: String).returns(T.nilable(String)) } | 
					
						
							|  |  |  |     def generate_fish_subcommand_completion(command) | 
					
						
							|  |  |  |       return unless command_gets_completions? command | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       command_description = format_description Commands.command_description(command, short: true), fish: true | 
					
						
							|  |  |  |       lines = ["__fish_brew_complete_cmd '#{command}' '#{command_description}'"] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       options = command_options(command).sort.map do |opt, desc| | 
					
						
							|  |  |  |         arg_line = "__fish_brew_complete_arg '#{command}' -l #{opt.sub(/^-+/, "")}" | 
					
						
							|  |  |  |         arg_line += " -d '#{format_description desc, fish: true}'" if desc.present? | 
					
						
							|  |  |  |         arg_line | 
					
						
							|  |  |  |       end.compact | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       subcommands = [] | 
					
						
							|  |  |  |       named_args = [] | 
					
						
							| 
									
										
										
										
											2021-02-12 18:33:37 +05:30
										 |  |  |       if (types = Commands.named_args_type(command)) | 
					
						
							| 
									
										
										
										
											2021-01-26 10:47:56 -05:00
										 |  |  |         named_args_strings, named_args_types = types.partition { |type| type.is_a? String } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         named_args_types.each do |type| | 
					
						
							|  |  |  |           next unless FISH_NAMED_ARGS_COMPLETION_FUNCTION_MAPPING.key? type | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           named_arg_function = FISH_NAMED_ARGS_COMPLETION_FUNCTION_MAPPING[type] | 
					
						
							|  |  |  |           named_args << "__fish_brew_complete_arg '#{command}' -a '(#{named_arg_function})'" | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         named_args_strings.each do |subcommand| | 
					
						
							|  |  |  |           subcommands << "__fish_brew_complete_sub_cmd '#{command}' '#{subcommand}'" | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       lines += subcommands + options + named_args | 
					
						
							|  |  |  |       <<~COMPLETION | 
					
						
							|  |  |  |         #{lines.join("\n").chomp} | 
					
						
							|  |  |  |       COMPLETION | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     sig { params(commands: T::Array[String]).returns(String) } | 
					
						
							|  |  |  |     def generate_fish_completion_file(commands) | 
					
						
							|  |  |  |       variables = OpenStruct.new | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       variables[:completion_functions] = commands.map do |command| | 
					
						
							|  |  |  |         generate_fish_subcommand_completion command | 
					
						
							|  |  |  |       end.compact | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       ERB.new((TEMPLATE_DIR/"fish.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 |