208 lines
5.8 KiB
Ruby
Raw Normal View History

#: * `list`, `ls` [`--full-name`] [`-1`] [`-l`] [`-t`] [`-r`]:
#: List all installed formulae. If `--full-name` is passed, print formulae
#: with fully-qualified names. If `--full-name` is not passed, other
#: options (i.e. `-1`, `-l`, `-t` and `-r`) are passed to `ls` which produces the actual output.
2016-04-08 16:28:43 +02:00
#:
#: * `list`, `ls` `--unbrewed`:
#: List all files in the Homebrew prefix not installed by Homebrew.
#:
#: * `list`, `ls` [`--verbose`] [`--versions` [`--multiple`]] [`--pinned`] [<formulae>]:
2016-04-08 16:28:43 +02:00
#: List the installed files for <formulae>. Combined with `--verbose`, recursively
#: list the contents of all subdirectories in each <formula>'s keg.
#:
#: If `--versions` is passed, show the version number for installed formulae,
#: or only the specified formulae if <formulae> are given. With `--multiple`,
#: only show formulae with multiple versions installed.
#:
#: If `--pinned` is passed, show the versions of pinned formulae, or only the
#: specified (pinned) formulae if <formulae> are given.
#: See also `pin`, `unpin`.
2014-06-07 17:49:07 -05:00
require "metafiles"
require "formula"
require "cli_parser"
2014-06-07 17:49:07 -05:00
module Homebrew
2016-09-26 01:44:51 +02:00
module_function
def list
Homebrew::CLI::Parser.parse do
switch "--unbrewed"
switch "--pinned"
switch "--versions"
switch "--full-name"
2018-05-21 20:21:30 +00:00
switch "--multiple", depends_on: "--versions"
switch :verbose
# passed through to ls
switch "-1"
switch "-l"
switch "-t"
switch "-r"
end
2012-08-28 10:59:46 -07:00
# Use of exec means we don't explicitly exit
list_unbrewed if args.unbrewed?
2012-08-28 10:59:46 -07:00
# Unbrewed uses the PREFIX, which will exist
# Things below use the CELLAR, which doesn't until the first formula is installed.
unless HOMEBREW_CELLAR.exist?
2016-09-10 10:24:57 +01:00
raise NoSuchKegError, ARGV.named.first unless ARGV.named.empty?
2018-09-17 02:45:00 +02:00
return
end
if args.pinned? || args.versions?
filtered_list
elsif ARGV.named.empty?
if args.full_name?
2017-10-18 18:27:42 -04:00
full_names = Formula.installed.map(&:full_name).sort(&tap_and_name_comparison)
return if full_names.empty?
2018-09-17 02:45:00 +02:00
puts Formatter.columns(full_names)
else
ENV["CLICOLOR"] = nil
exec "ls", *ARGV.options_only << HOMEBREW_CELLAR
end
elsif args.verbose? || !$stdout.tty?
2014-06-26 14:08:17 -05:00
exec "find", *ARGV.kegs.map(&:to_s) + %w[-not -type d -print]
else
ARGV.kegs.each { |keg| PrettyListing.new keg }
end
end
2012-08-28 10:59:46 -07:00
2016-09-10 10:24:57 +01:00
UNBREWED_EXCLUDE_FILES = %w[.DS_Store].freeze
UNBREWED_EXCLUDE_PATHS = %w[
.github/*
bin/brew
2016-09-19 19:59:57 +01:00
completions/zsh/_brew
docs/*
lib/gdk-pixbuf-2.0/*
lib/gio/*
lib/node_modules/*
lib/python[23].[0-9]/*
lib/pypy/*
lib/pypy3/*
lib/ruby/gems/[12].*
lib/ruby/site_ruby/[12].*
lib/ruby/vendor_ruby/[12].*
2016-09-19 19:59:57 +01:00
manpages/brew.1
manpages/brew-cask.1
share/pypy/*
share/pypy3/*
share/info/dir
share/man/whatis
2016-09-10 10:24:57 +01:00
].freeze
2012-08-28 10:59:46 -07:00
def list_unbrewed
dirs = HOMEBREW_PREFIX.subdirs.map { |dir| dir.basename.to_s }
2012-08-28 10:59:46 -07:00
dirs -= %w[Library Cellar .git]
# Exclude cache, logs, and repository, if they are located under the prefix.
[HOMEBREW_CACHE, HOMEBREW_LOGS, HOMEBREW_REPOSITORY].each do |dir|
dirs.delete dir.relative_path_from(HOMEBREW_PREFIX).to_s
end
dirs.delete "etc"
dirs.delete "var"
arguments = dirs.sort + %w[-type f (]
arguments.concat UNBREWED_EXCLUDE_FILES.flat_map { |f| %W[! -name #{f}] }
arguments.concat UNBREWED_EXCLUDE_PATHS.flat_map { |d| %W[! -path #{d}] }
arguments.concat %w[)]
2012-08-28 10:59:46 -07:00
cd HOMEBREW_PREFIX
exec "find", *arguments
2012-08-28 10:59:46 -07:00
end
def filtered_list
names = if ARGV.named.empty?
Formula.racks
2012-08-28 10:59:46 -07:00
else
2016-11-16 02:08:38 +03:00
racks = ARGV.named.map { |n| Formulary.to_rack(n) }
racks.select do |rack|
Homebrew.failed = true unless rack.exist?
rack.exist?
end
2012-08-28 10:59:46 -07:00
end
if args.pinned?
pinned_versions = {}
names.sort.each do |d|
keg_pin = (HOMEBREW_PINNED_KEGS/d.basename.to_s)
if keg_pin.exist? || keg_pin.symlink?
pinned_versions[d] = keg_pin.readlink.basename.to_s
end
end
pinned_versions.each do |d, version|
puts d.basename.to_s.concat(args.versions? ? " #{version}" : "")
end
else # --versions without --pinned
names.sort.each do |d|
versions = d.subdirs.map { |pn| pn.basename.to_s }
next if args.multiple? && versions.length < 2
2018-09-17 02:45:00 +02:00
2017-06-01 16:06:51 +02:00
puts "#{d.basename} #{versions * " "}"
end
end
end
end
class PrettyListing
def initialize(path)
2014-07-01 13:32:53 -05:00
Pathname.new(path).children.sort_by { |p| p.to_s.downcase }.each do |pn|
case pn.basename.to_s
when "bin", "sbin"
pn.find { |pnn| puts pnn unless pnn.directory? }
when "lib"
print_dir pn do |pnn|
# dylibs have multiple symlinks and we don't care about them
(pnn.extname == ".dylib" || pnn.extname == ".pc") && !pnn.symlink?
end
when ".brew"
2016-11-13 23:37:51 +01:00
next # Ignore .brew
else
if pn.directory?
if pn.symlink?
puts "#{pn} -> #{pn.readlink}"
else
print_dir pn
end
2014-06-07 17:49:07 -05:00
elsif Metafiles.list?(pn.basename.to_s)
puts pn
end
end
end
end
def print_dir(root)
dirs = []
remaining_root_files = []
other = ""
root.children.sort.each do |pn|
if pn.directory?
dirs << pn
elsif block_given? && yield(pn)
puts pn
other = "other "
else
remaining_root_files << pn unless pn.basename.to_s == ".DS_Store"
end
end
dirs.each do |d|
files = []
d.find { |pn| files << pn unless pn.directory? }
print_remaining_files files, d
end
print_remaining_files remaining_root_files, root, other
end
def print_remaining_files(files, root, other = "")
2016-11-13 23:37:51 +01:00
if files.length == 1
puts files
2016-11-13 23:37:51 +01:00
elsif files.length > 1
puts "#{root}/ (#{files.length} #{other}files)"
end
end
end