brew/Library/Homebrew/cmd/search.rb

171 lines
5.9 KiB
Ruby
Raw Normal View History

# typed: strict
# frozen_string_literal: true
2024-04-01 10:12:52 -07:00
require "abstract_command"
require "formula"
require "missing_formula"
require "descriptions"
require "search"
module Homebrew
2024-04-01 10:12:52 -07:00
module Cmd
class SearchCmd < AbstractCommand
PACKAGE_MANAGERS = T.let({
2024-04-01 10:12:52 -07:00
repology: ->(query) { "https://repology.org/projects/?search=#{query}" },
macports: ->(query) { "https://ports.macports.org/search/?q=#{query}" },
fink: ->(query) { "https://pdb.finkproject.org/pdb/browse.php?summary=#{query}" },
opensuse: ->(query) { "https://software.opensuse.org/search?q=#{query}" },
fedora: ->(query) { "https://packages.fedoraproject.org/search?query=#{query}" },
archlinux: ->(query) { "https://archlinux.org/packages/?q=#{query}" },
debian: lambda { |query|
"https://packages.debian.org/search?keywords=#{query}&searchon=names&suite=all&section=all"
},
ubuntu: lambda { |query|
"https://packages.ubuntu.com/search?keywords=#{query}&searchon=names&suite=all&section=all"
},
}.freeze, T::Hash[Symbol, T.proc.params(query: String).returns(String)])
2024-04-01 10:12:52 -07:00
cmd_args do
description <<~EOS
Perform a substring search of cask tokens and formula names for <text>. If <text>
is flanked by slashes, it is interpreted as a regular expression.
EOS
switch "--formula", "--formulae",
description: "Search for formulae."
switch "--cask", "--casks",
description: "Search for casks."
switch "--desc",
description: "Search for formulae with a description matching <text> and casks with " \
"a name or description matching <text>."
switch "--eval-all",
depends_on: "--desc",
description: "Evaluate all available formulae and casks, whether installed or not, to search their " \
"descriptions. Implied if `$HOMEBREW_EVAL_ALL` is set."
2024-04-01 10:12:52 -07:00
switch "--pull-request",
description: "Search for GitHub pull requests containing <text>."
switch "--open",
depends_on: "--pull-request",
description: "Search for only open GitHub pull requests."
switch "--closed",
depends_on: "--pull-request",
description: "Search for only closed GitHub pull requests."
package_manager_switches = PACKAGE_MANAGERS.keys.map { |name| "--#{name}" }
package_manager_switches.each do |s|
switch s,
description: "Search for <text> in the given database."
end
conflicts "--desc", "--pull-request"
conflicts "--open", "--closed"
conflicts(*package_manager_switches)
named_args :text_or_regex, min: 1
2018-06-01 14:19:00 +02:00
end
2020-07-30 18:40:10 +02:00
2024-04-01 10:12:52 -07:00
sig { override.void }
def run
return if search_package_manager
2021-01-10 14:26:40 -05:00
2024-04-01 10:12:52 -07:00
query = args.named.join(" ")
string_or_regex = Search.query_regexp(query)
2024-04-01 10:12:52 -07:00
if args.desc?
if !args.eval_all? && !Homebrew::EnvConfig.eval_all? && Homebrew::EnvConfig.no_install_from_api?
raise UsageError, "`brew search --desc` needs `--eval-all` passed or `$HOMEBREW_EVAL_ALL` set!"
2024-04-01 10:12:52 -07:00
end
2018-06-01 14:19:00 +02:00
2024-04-01 10:12:52 -07:00
Search.search_descriptions(string_or_regex, args)
elsif args.pull_request?
search_pull_requests(query)
else
formulae, casks = Search.search_names(string_or_regex, args)
print_results(formulae, casks, query)
end
2018-06-01 14:19:00 +02:00
2024-04-01 10:12:52 -07:00
puts "Use `brew desc` to list packages with a short description." if args.verbose?
2018-06-18 16:09:13 +02:00
2024-04-01 10:12:52 -07:00
print_regex_help
end
2024-04-01 10:12:52 -07:00
private
sig { void }
2024-04-01 10:12:52 -07:00
def print_regex_help
return unless $stdout.tty?
2024-04-01 10:12:52 -07:00
metacharacters = %w[\\ | ( ) [ ] { } ^ $ * + ?].freeze
return unless metacharacters.any? do |char|
args.named.any? do |arg|
arg.include?(char) && !arg.start_with?("/")
end
end
2018-09-17 02:45:00 +02:00
2024-04-01 10:12:52 -07:00
opoo <<~EOS
Did you mean to perform a regular expression search?
Surround your query with /slashes/ to search locally by regex.
EOS
end
2018-09-17 02:45:00 +02:00
sig { returns(T::Boolean) }
2024-04-01 10:12:52 -07:00
def search_package_manager
2024-11-30 13:52:46 -08:00
package_manager = PACKAGE_MANAGERS.find { |name,| args.public_send(:"#{name}?") }
2024-04-01 10:12:52 -07:00
return false if package_manager.nil?
2024-04-01 10:12:52 -07:00
_, url = package_manager
exec_browser url.call(URI.encode_www_form_component(args.named.join(" ")))
true
end
sig { params(query: String).returns(String) }
2024-04-01 10:12:52 -07:00
def search_pull_requests(query)
only = if args.open? && !args.closed?
"open"
elsif args.closed? && !args.open?
"closed"
end
2024-04-01 10:12:52 -07:00
GitHub.print_pull_requests_matching(query, only)
end
sig { params(all_formulae: T::Array[String], all_casks: T::Array[String], query: String).void }
2024-04-01 10:12:52 -07:00
def print_results(all_formulae, all_casks, query)
count = all_formulae.size + all_casks.size
if all_formulae.any?
if $stdout.tty?
ohai "Formulae", Formatter.columns(all_formulae)
else
puts all_formulae
end
end
puts if all_formulae.any? && all_casks.any?
if all_casks.any?
if $stdout.tty?
ohai "Casks", Formatter.columns(all_casks)
else
puts all_casks
end
end
print_missing_formula_help(query, count.positive?) if all_casks.exclude?(query)
odie "No formulae or casks found for #{query.inspect}." if count.zero?
end
sig { params(query: String, found_matches: T::Boolean).void }
2024-04-01 10:12:52 -07:00
def print_missing_formula_help(query, found_matches)
return unless $stdout.tty?
2024-04-01 10:12:52 -07:00
reason = MissingFormula.reason(query, silent: true)
return if reason.nil?
2024-04-01 10:12:52 -07:00
if found_matches
puts
puts "If you meant #{query.inspect} specifically:"
end
puts reason
end
end
end
end