263 lines
		
	
	
		
			7.8 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
			
		
		
	
	
			263 lines
		
	
	
		
			7.8 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
| # typed: false
 | |
| # frozen_string_literal: true
 | |
| 
 | |
| require "metafiles"
 | |
| require "formula"
 | |
| require "cli/parser"
 | |
| require "cask/cmd"
 | |
| 
 | |
| module Homebrew
 | |
|   extend T::Sig
 | |
| 
 | |
|   module_function
 | |
| 
 | |
|   sig { returns(CLI::Parser) }
 | |
|   def list_args
 | |
|     Homebrew::CLI::Parser.new do
 | |
|       usage_banner <<~EOS
 | |
|         `list`, `ls` [<options>] [<formula>|<cask>]
 | |
| 
 | |
|         List all installed formulae and casks.
 | |
| 
 | |
|         If <formula> is provided, summarise the paths within its current keg.
 | |
|       EOS
 | |
|       switch "--formula", "--formulae",
 | |
|              description: "List only formulae. This is the default when output is not to a terminal."
 | |
|       switch "--cask", "--casks",
 | |
|              description: "List only casks, or <cask> if provided."
 | |
|       switch "--unbrewed",
 | |
|              description: "List files in Homebrew's prefix not installed by Homebrew."
 | |
|       switch "--full-name",
 | |
|              depends_on:  "--formula",
 | |
|              description: "Print formulae with fully-qualified names. If `--full-name` is not "\
 | |
|                           "passed, other options (i.e. `-1`, `-l`, `-r` and `-t`) are passed to `ls`(1) "\
 | |
|                           "which produces the actual output."
 | |
|       switch "--versions",
 | |
|              description: "Show the version number for installed formulae, or only the specified "\
 | |
|                           "formulae if <formula> are provided."
 | |
|       switch "--multiple",
 | |
|              depends_on:  "--versions",
 | |
|              description: "Only show formulae with multiple versions installed."
 | |
|       switch "--pinned",
 | |
|              description: "Show the versions of pinned formulae, or only the specified (pinned) "\
 | |
|                           "formulae if <formula> are provided. See also `pin`, `unpin`."
 | |
|       # passed through to ls
 | |
|       switch "-1",
 | |
|              description: "Force output to be one entry per line. " \
 | |
|                           "This is the default when output is not to a terminal."
 | |
|       switch "-l",
 | |
|              description: "List formulae in long format. If the output is to a terminal, "\
 | |
|                           "a total sum for all the file sizes is printed before the long listing."
 | |
|       switch "-r",
 | |
|              description: "Reverse the order of the formulae sort to list the oldest entries first."
 | |
|       switch "-t",
 | |
|              description: "Sort formulae by time modified, listing most recently modified first."
 | |
| 
 | |
|       ["-1", "-l", "-r", "-t"].each do |flag|
 | |
|         conflicts "--full-name", flag
 | |
|         conflicts "--unbrewed", flag
 | |
|         conflicts "--pinned", flag
 | |
|         conflicts "--versions", flag
 | |
|       end
 | |
| 
 | |
|       ["--unbrewed", "--formula", "-l", "-r", "-t"].each do |flag|
 | |
|         conflicts "--cask", flag
 | |
|       end
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   def list
 | |
|     args = list_args.parse
 | |
| 
 | |
|     return list_casks(args: args) if args.cask?
 | |
| 
 | |
|     if args.unbrewed?
 | |
|       raise UsageError, "`--unbrewed` does not take a formula/cask argument." unless args.no_named?
 | |
| 
 | |
|       return list_unbrewed
 | |
|     end
 | |
| 
 | |
|     # 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?
 | |
|       raise NoSuchKegError, args.named.first if args.named.present?
 | |
| 
 | |
|       return
 | |
|     end
 | |
| 
 | |
|     if args.pinned? || args.versions?
 | |
|       filtered_list args: args
 | |
|     elsif args.no_named?
 | |
|       if args.full_name?
 | |
|         full_names = Formula.installed.map(&:full_name).sort(&tap_and_name_comparison)
 | |
|         return if full_names.empty?
 | |
| 
 | |
|         puts Formatter.columns(full_names)
 | |
|       else
 | |
|         ENV["CLICOLOR"] = nil
 | |
| 
 | |
|         ls_args = []
 | |
|         ls_args << "-1" if args.public_send(:'1?')
 | |
|         ls_args << "-l" if args.l?
 | |
|         ls_args << "-r" if args.r?
 | |
|         ls_args << "-t" if args.t?
 | |
| 
 | |
|         if !$stdout.tty? && !args.formula?
 | |
|           odeprecated "`brew list` to only list formulae", "`brew list --formula`"
 | |
|           safe_system "ls", *ls_args, HOMEBREW_CELLAR
 | |
|         else
 | |
|           safe_system "ls", *ls_args, HOMEBREW_CELLAR
 | |
|           list_casks(args: args) unless args.formula?
 | |
|         end
 | |
|       end
 | |
|     elsif args.verbose? || !$stdout.tty?
 | |
|       system_command! "find", args: args.named.to_kegs.map(&:to_s) + %w[-not -type d -print], print_stdout: true
 | |
|     else
 | |
|       args.named.to_kegs.each { |keg| PrettyListing.new keg }
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   UNBREWED_EXCLUDE_FILES = %w[.DS_Store].freeze
 | |
|   UNBREWED_EXCLUDE_PATHS = %w[
 | |
|     */.keepme
 | |
|     .github/*
 | |
|     bin/brew
 | |
|     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].*
 | |
|     manpages/brew.1
 | |
|     share/pypy/*
 | |
|     share/pypy3/*
 | |
|     share/info/dir
 | |
|     share/man/whatis
 | |
|   ].freeze
 | |
| 
 | |
|   def list_unbrewed
 | |
|     dirs  = HOMEBREW_PREFIX.subdirs.map { |dir| dir.basename.to_s }
 | |
|     dirs -= %w[Library Cellar Caskroom .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[)]
 | |
| 
 | |
|     cd HOMEBREW_PREFIX
 | |
|     safe_system "find", *arguments
 | |
|   end
 | |
| 
 | |
|   def filtered_list(args:)
 | |
|     names = if args.no_named?
 | |
|       Formula.racks
 | |
|     else
 | |
|       racks = args.named.map { |n| Formulary.to_rack(n) }
 | |
|       racks.select do |rack|
 | |
|         Homebrew.failed = true unless rack.exist?
 | |
|         rack.exist?
 | |
|       end
 | |
|     end
 | |
|     if args.pinned?
 | |
|       pinned_versions = {}
 | |
|       names.sort.each do |d|
 | |
|         keg_pin = (HOMEBREW_PINNED_KEGS/d.basename.to_s)
 | |
|         pinned_versions[d] = keg_pin.readlink.basename.to_s if keg_pin.exist? || keg_pin.symlink?
 | |
|       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
 | |
| 
 | |
|         puts "#{d.basename} #{versions * " "}"
 | |
|       end
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   def list_casks(args:)
 | |
|     Cask::Cmd::List.list_casks(
 | |
|       *args.named.to_casks,
 | |
|       one:       args.public_send(:'1?'),
 | |
|       full_name: args.full_name?,
 | |
|       versions:  args.versions?,
 | |
|       args:      args,
 | |
|     )
 | |
|   end
 | |
| end
 | |
| 
 | |
| class PrettyListing
 | |
|   def initialize(path)
 | |
|     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"
 | |
|         next # Ignore .brew
 | |
|       else
 | |
|         if pn.directory?
 | |
|           if pn.symlink?
 | |
|             puts "#{pn} -> #{pn.readlink}"
 | |
|           else
 | |
|             print_dir pn
 | |
|           end
 | |
|         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 = "")
 | |
|     if files.length == 1
 | |
|       puts files
 | |
|     elsif files.length > 1
 | |
|       puts "#{root}/ (#{files.length} #{other}files)"
 | |
|     end
 | |
|   end
 | |
| end
 | 
