#: * `search`, `-S`: #: Display all locally available formulae (including tapped ones). #: No online search is performed. #: #: * `search` `--casks` #: Display all locally available casks (including tapped ones). #: No online search is performed. #: #: * `search` [`--desc`] (|`/``/`): #: Perform a substring search of cask tokens and formula names for . If #: is surrounded with slashes, then it is interpreted as a regular expression. #: The search for is extended online to official taps. #: #: If `--desc` is passed, search formulae with a description matching and #: casks with a name matching . #: #: * `search` (`--debian`|`--fedora`|`--fink`|`--macports`|`--opensuse`|`--ubuntu`) : #: Search for in the given package manager's list. require "formula" require "missing_formula" require "descriptions" require "cli_parser" require "search" module Homebrew module_function extend Search PACKAGE_MANAGERS = { macports: ->(query) { "https://www.macports.org/ports.php?by=name&substr=#{query}" }, fink: ->(query) { "http://pdb.finkproject.org/pdb/browse.php?summary=#{query}" }, debian: ->(query) { "https://packages.debian.org/search?keywords=#{query}&searchon=names&suite=all§ion=all" }, opensuse: ->(query) { "https://software.opensuse.org/search?q=#{query}" }, fedora: ->(query) { "https://apps.fedoraproject.org/packages/s/#{query}" }, ubuntu: ->(query) { "https://packages.ubuntu.com/search?keywords=#{query}&searchon=names&suite=all§ion=all" }, }.freeze def search(argv = ARGV) CLI::Parser.parse(argv) do switch "--desc" package_manager_switches = PACKAGE_MANAGERS.keys.map { |name| "--#{name}" } package_manager_switches.each do |s| switch s end switch "--casks" conflicts(*package_manager_switches) end if package_manager = PACKAGE_MANAGERS.detect { |name,| args[:"#{name}?"] } _, url = package_manager exec_browser url.call(URI.encode_www_form_component(args.remaining.join(" "))) return end if args.remaining.empty? if args.casks? puts Formatter.columns(Hbc::Cask.to_a.map(&:full_name).sort) else puts Formatter.columns(Formula.full_names.sort) end return end query = args.remaining.join(" ") string_or_regex = query_regexp(query) if args.desc? search_descriptions(string_or_regex) else remote_results = search_taps(query, silent: true) local_formulae = search_formulae(string_or_regex) remote_formulae = remote_results[:formulae] all_formulae = local_formulae + remote_formulae local_casks = search_casks(string_or_regex) remote_casks = remote_results[:casks] all_casks = local_casks + remote_casks if all_formulae.any? ohai "Formulae" puts Formatter.columns(all_formulae) end if all_casks.any? puts if all_formulae.any? ohai "Casks" puts Formatter.columns(all_casks) end if $stdout.tty? count = all_formulae.count + all_casks.count if reason = MissingFormula.reason(query, silent: true) if count.positive? puts puts "If you meant #{query.inspect} specifically:" end puts reason elsif count.zero? puts "No formula or cask found for #{query.inspect}." GitHub.print_pull_requests_matching(query) end end end return unless $stdout.tty? return if args.remaining.empty? metacharacters = %w[\\ | ( ) [ ] { } ^ $ * + ?].freeze return unless metacharacters.any? do |char| args.remaining.any? do |arg| arg.include?(char) && !arg.start_with?("/") end end ohai <<~EOS Did you mean to perform a regular expression search? Surround your query with /slashes/ to search locally by regex. EOS end end