#: * `search`, `-S`: #: Display all locally available formulae for brewing (including tapped ones). #: No online search is performed if called without arguments. #: #: * `search` [`--desc`] (|`/``/`): #: Perform a substring search of formula names for . If is #: surrounded with slashes, then it is interpreted as a regular expression. #: The search for is extended online to some popular taps. #: #: If `--desc` is passed, browse available packages matching including a #: description for each. #: #: * `search` (`--debian`|`--fedora`|`--fink`|`--macports`|`--opensuse`|`--ubuntu`) : #: Search for in the given package manager's list. require "formula" require "blacklist" require "utils" require "thread" require "official_taps" require "descriptions" module Homebrew module_function SEARCH_ERROR_QUEUE = Queue.new def search if ARGV.empty? puts Formatter.columns(Formula.full_names) elsif ARGV.include? "--macports" exec_browser "https://www.macports.org/ports.php?by=name&substr=#{ARGV.next}" elsif ARGV.include? "--fink" exec_browser "http://pdb.finkproject.org/pdb/browse.php?summary=#{ARGV.next}" elsif ARGV.include? "--debian" exec_browser "https://packages.debian.org/search?keywords=#{ARGV.next}&searchon=names&suite=all§ion=all" elsif ARGV.include? "--opensuse" exec_browser "https://software.opensuse.org/search?q=#{ARGV.next}" elsif ARGV.include? "--fedora" exec_browser "https://admin.fedoraproject.org/pkgdb/packages/%2A#{ARGV.next}%2A/" elsif ARGV.include? "--ubuntu" exec_browser "http://packages.ubuntu.com/search?keywords=#{ARGV.next}&searchon=names&suite=all§ion=all" elsif ARGV.include? "--desc" query = ARGV.next regex = query_regexp(query) Descriptions.search(regex, :desc).print elsif ARGV.first =~ HOMEBREW_TAP_FORMULA_REGEX query = ARGV.first user, repo, name = query.split("/", 3) begin result = Formulary.factory(query).name rescue FormulaUnavailableError result = search_tap(user, repo, name) end results = Array(result) puts Formatter.columns(results) unless results.empty? else query = ARGV.first regex = query_regexp(query) local_results = search_formulae(regex) puts Formatter.columns(local_results) unless local_results.empty? tap_results = search_taps(regex) puts Formatter.columns(tap_results) unless tap_results.empty? if $stdout.tty? count = local_results.length + tap_results.length if msg = blacklisted?(query) if count > 0 puts puts "If you meant #{query.inspect} precisely:" puts end puts msg elsif count.zero? puts "No formula found for #{query.inspect}." begin GitHub.print_pull_requests_matching(query) rescue GitHub::Error => e SEARCH_ERROR_QUEUE << e end end end end if $stdout.tty? metacharacters = %w[\\ | ( ) [ ] { } ^ $ * + ?] bad_regex = metacharacters.any? do |char| ARGV.any? do |arg| arg.include?(char) && !arg.start_with?("/") end end if !ARGV.empty? && bad_regex ohai "Did you mean to perform a regular expression search?" ohai "Surround your query with /slashes/ to search by regex." end end raise SEARCH_ERROR_QUEUE.pop unless SEARCH_ERROR_QUEUE.empty? end SEARCHABLE_TAPS = OFFICIAL_TAPS.map { |tap| ["Homebrew", tap] } + [ %w[Caskroom cask], %w[Caskroom versions], ] def query_regexp(query) case query when %r{^/(.*)/$} then Regexp.new($1) else /.*#{Regexp.escape(query)}.*/i end rescue RegexpError odie "#{query} is not a valid regex" end def search_taps(regex_or_string) SEARCHABLE_TAPS.map do |user, repo| Thread.new { search_tap(user, repo, regex_or_string) } end.inject([]) do |results, t| results.concat(t.value) end end def search_tap(user, repo, regex_or_string) regex = regex_or_string.is_a?(String) ? /^#{Regexp.escape(regex_or_string)}$/ : regex_or_string if (HOMEBREW_LIBRARY/"Taps/#{user.downcase}/homebrew-#{repo.downcase}").directory? && \ user != "Caskroom" return [] end remote_tap_formulae = Hash.new do |cache, key| user, repo = key.split("/", 2) tree = {} GitHub.open "https://api.github.com/repos/#{user}/homebrew-#{repo}/git/trees/HEAD?recursive=1" do |json| json["tree"].each do |object| next unless object["type"] == "blob" subtree, file = File.split(object["path"]) if File.extname(file) == ".rb" tree[subtree] ||= [] tree[subtree] << file end end end paths = tree["Formula"] || tree["HomebrewFormula"] || tree["."] || [] paths += tree["Casks"] || [] cache[key] = paths.map { |path| File.basename(path, ".rb") } end names = remote_tap_formulae["#{user}/#{repo}"] user = user.downcase if user == "Homebrew" # special handling for the Homebrew organization names.select { |name| name =~ regex }.map { |name| "#{user}/#{repo}/#{name}" } rescue GitHub::HTTPNotFoundError opoo "Failed to search tap: #{user}/#{repo}. Please run `brew update`" [] rescue GitHub::Error => e SEARCH_ERROR_QUEUE << e [] end def search_formulae(regex) aliases = Formula.alias_full_names results = (Formula.full_names+aliases).grep(regex).sort results.map do |name| begin formula = Formulary.factory(name) canonical_name = formula.name canonical_full_name = formula.full_name rescue canonical_name = canonical_full_name = name end # Ignore aliases from results when the full name was also found next if aliases.include?(name) && results.include?(canonical_full_name) if (HOMEBREW_CELLAR/canonical_name).directory? pretty_installed(name) else name end end.compact end end