mirror of
https://github.com/Homebrew/brew.git
synced 2025-07-14 16:09:03 +08:00
172 lines
5.9 KiB
Ruby
172 lines
5.9 KiB
Ruby
# typed: strict
|
|
# frozen_string_literal: true
|
|
|
|
require "abstract_command"
|
|
require "formula"
|
|
require "missing_formula"
|
|
require "descriptions"
|
|
require "search"
|
|
|
|
module Homebrew
|
|
module Cmd
|
|
class SearchCmd < AbstractCommand
|
|
PACKAGE_MANAGERS = T.let({
|
|
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§ion=all"
|
|
},
|
|
ubuntu: lambda { |query|
|
|
"https://packages.ubuntu.com/search?keywords=#{query}&searchon=names&suite=all§ion=all"
|
|
},
|
|
}.freeze, T::Hash[Symbol, T.proc.params(query: String).returns(String)])
|
|
|
|
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.",
|
|
env: :eval_all
|
|
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
|
|
end
|
|
|
|
sig { override.void }
|
|
def run
|
|
return if search_package_manager
|
|
|
|
query = args.named.join(" ")
|
|
string_or_regex = Search.query_regexp(query)
|
|
|
|
if args.desc?
|
|
if !args.eval_all? && Homebrew::EnvConfig.no_install_from_api?
|
|
raise UsageError, "`brew search --desc` needs `--eval-all` passed or `$HOMEBREW_EVAL_ALL` set!"
|
|
end
|
|
|
|
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
|
|
|
|
puts "Use `brew desc` to list packages with a short description." if args.verbose?
|
|
|
|
print_regex_help
|
|
end
|
|
|
|
private
|
|
|
|
sig { void }
|
|
def print_regex_help
|
|
return unless $stdout.tty?
|
|
|
|
metacharacters = %w[\\ | ( ) [ ] { } ^ $ * + ?].freeze
|
|
return unless metacharacters.any? do |char|
|
|
args.named.any? do |arg|
|
|
arg.include?(char) && !arg.start_with?("/")
|
|
end
|
|
end
|
|
|
|
opoo <<~EOS
|
|
Did you mean to perform a regular expression search?
|
|
Surround your query with /slashes/ to search locally by regex.
|
|
EOS
|
|
end
|
|
|
|
sig { returns(T::Boolean) }
|
|
def search_package_manager
|
|
package_manager = PACKAGE_MANAGERS.find { |name,| args.public_send(:"#{name}?") }
|
|
return false if package_manager.nil?
|
|
|
|
_, url = package_manager
|
|
exec_browser url.call(URI.encode_www_form_component(args.named.join(" ")))
|
|
true
|
|
end
|
|
|
|
sig { params(query: String).returns(String) }
|
|
def search_pull_requests(query)
|
|
only = if args.open? && !args.closed?
|
|
"open"
|
|
elsif args.closed? && !args.open?
|
|
"closed"
|
|
end
|
|
|
|
GitHub.print_pull_requests_matching(query, only)
|
|
end
|
|
|
|
sig { params(all_formulae: T::Array[String], all_casks: T::Array[String], query: String).void }
|
|
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 }
|
|
def print_missing_formula_help(query, found_matches)
|
|
return unless $stdout.tty?
|
|
|
|
reason = MissingFormula.reason(query, silent: true)
|
|
return if reason.nil?
|
|
|
|
if found_matches
|
|
puts
|
|
puts "If you meant #{query.inspect} specifically:"
|
|
end
|
|
puts reason
|
|
end
|
|
end
|
|
end
|
|
end
|