homebrew-cask/lib/hbc/cli.rb

273 lines
7.9 KiB
Ruby

class Hbc::CLI; end
require 'optparse'
require 'shellwords'
require 'hbc/cli/base'
require 'hbc/cli/alfred'
require 'hbc/cli/audit'
require 'hbc/cli/cat'
require 'hbc/cli/cleanup'
require 'hbc/cli/create'
require 'hbc/cli/doctor'
require 'hbc/cli/edit'
require 'hbc/cli/fetch'
require 'hbc/cli/home'
require 'hbc/cli/info'
require 'hbc/cli/install'
require 'hbc/cli/list'
require 'hbc/cli/search'
require 'hbc/cli/uninstall'
require 'hbc/cli/update'
require 'hbc/cli/zap'
require 'hbc/cli/internal_use_base'
require 'hbc/cli/internal_checkurl'
require 'hbc/cli/internal_dump'
require 'hbc/cli/internal_help'
require 'hbc/cli/internal_stanza'
class Hbc::CLI
ALIASES = {
'ls' => 'list',
'homepage' => 'home',
'-S' => 'search', # verb starting with "-" is questionable
'up' => 'update',
'instal' => 'install', # gem does the same
'rm' => 'uninstall',
'remove' => 'uninstall',
'abv' => 'info',
'dr' => 'doctor',
# aliases from Homebrew that we don't (yet) support
# 'ln' => 'link',
# 'configure' => 'diy',
# '--repo' => '--repository',
# 'environment' => '--env',
# '-c1' => '--config',
}
def self.command_classes
@@command_classes ||= Hbc::CLI.constants.
map { |sym| Hbc::CLI.const_get sym }.
select { |sym| sym.respond_to?(:run) }
end
def self.commands
@@commands ||= command_classes.map { |sym| sym.command_name }
end
def self.lookup_command(command_string)
@@lookup ||= Hash[commands.zip(command_classes)]
command_string = ALIASES.fetch(command_string, command_string)
@@lookup.fetch(command_string, command_string)
end
# modified from Homebrew
def self.require? path
require path
true # OK if already loaded
rescue LoadError => e
# HACK :( because we should raise on syntax errors but
# not if the file doesn't exist. TODO make robust!
raise unless e.to_s.include? path
end
def self.run_command(command, *rest)
if command.respond_to?(:run)
# usual case: built-in command verb
command.run(*rest)
elsif require? Hbc::Utils.which("brewcask-#{command}.rb").to_s
# external command as Ruby library on PATH, Homebrew-style
exit 0
elsif command.to_s.include?('/') and require? command.to_s
# external command as Ruby library with literal path, useful
# for development and troubleshooting
sym = Pathname.new(command.to_s).basename('.rb').to_s.capitalize
klass = begin
Hbc::CLI.const_get(sym)
rescue NameError => e
nil
end
if klass.respond_to?(:run)
# invoke "run" on a Ruby library which follows our coding conventions
klass.run(*rest)
else
# other Ruby libraries must do everything via "require"
exit 0
end
elsif Hbc::Utils.which "brewcask-#{command}"
# arbitrary external executable on PATH, Homebrew-style
exec "brewcask-#{command}", *ARGV[1..-1]
elsif Pathname.new(command.to_s).executable? and
command.to_s.include?('/') and
not command.to_s.match(%{\.rb$})
# arbitrary external executable with literal path, useful
# for development and troubleshooting
exec command, *ARGV[1..-1]
else
# failure
Hbc::CLI::NullCommand.new(command).run
end
end
def self.process(arguments)
command_string, *rest = *arguments
rest = process_options(rest)
Hbc.init
command = lookup_command(command_string)
run_command(command, *rest)
rescue Hbc::CaskError, Hbc::CaskSha256MismatchError => e
onoe e
$stderr.puts e.backtrace if Hbc.debug
exit 1
rescue StandardError, ScriptError, NoMemoryError => e
onoe e
puts Hbc::Utils.error_message_with_suggestions
puts e.backtrace
exit 1
end
def self.nice_listing(cask_list)
cask_taps = {}
cask_list.each do |c|
user, repo, token = c.split '/'
repo.sub!(/^homebrew-/i, '')
cask_taps[token] ||= []
cask_taps[token].push "#{user}/#{repo}"
end
list = []
cask_taps.each do |token,taps|
if taps.length == 1
list.push token
else
taps.each { |r| list.push [r,token].join '/' }
end
end
list.sort
end
def self.parser
# If you modify these arguments, please update USAGE.md
@parser ||= OptionParser.new do |opts|
opts.on("--caskroom=MANDATORY") do |v|
Hbc.caskroom = Pathname(v).expand_path
end
opts.on("--appdir=MANDATORY") do |v|
Hbc.appdir = Pathname(v).expand_path
end
opts.on("--colorpickerdir=MANDATORY") do |v|
Hbc.colorpickerdir = Pathname(v).expand_path
end
opts.on("--prefpanedir=MANDATORY") do |v|
Hbc.prefpanedir = Pathname(v).expand_path
end
opts.on("--qlplugindir=MANDATORY") do |v|
Hbc.qlplugindir = Pathname(v).expand_path
end
opts.on("--fontdir=MANDATORY") do |v|
Hbc.fontdir = Pathname(v).expand_path
end
opts.on("--servicedir=MANDATORY") do |v|
Hbc.servicedir = Pathname(v).expand_path
end
opts.on("--binarydir=MANDATORY") do |v|
Hbc.binarydir = Pathname(v).expand_path
end
opts.on("--input_methoddir=MANDATORY") do |v|
Hbc.input_methoddir = Pathname(v).expand_path
end
opts.on("--internet_plugindir=MANDATORY") do |v|
Hbc.internet_plugindir = Pathname(v).expand_path
end
opts.on("--screen_saverdir=MANDATORY") do |v|
Hbc.screen_saverdir = Pathname(v).expand_path
end
opts.on("--no-binaries") do |v|
Hbc.no_binaries = true
end
opts.on("--debug") do |v|
Hbc.debug = true
end
opts.on("--verbose") do |v|
Hbc.verbose = true
end
opts.on("--outdated") do |v|
Hbc.cleanup_outdated = true
end
end
end
def self.process_options(args)
all_args = Shellwords.shellsplit(ENV['HOMEBREW_CASK_OPTS'] || "") + args
remaining = []
until all_args.empty?
begin
head = all_args.shift
remaining.concat(parser.parse([head]))
rescue OptionParser::InvalidOption
remaining << head
retry
rescue OptionParser::MissingArgument
raise Hbc::CaskError.new("The option '#{head}' requires an argument")
rescue OptionParser::AmbiguousOption
raise Hbc::CaskError.new(
"There is more than one possible option that starts with '#{head}'")
end
end
# for compat with Homebrew, not certain if this is desirable
Hbc.verbose = true if !ENV['VERBOSE'].nil? or !ENV['HOMEBREW_VERBOSE'].nil?
remaining
end
class NullCommand
def initialize(attempted_verb)
@attempted_verb = attempted_verb
end
def run(*args)
if args.include?('--version') or @attempted_verb == '--version'
puts HBC_VERSION
else
purpose
if @attempted_verb and @attempted_verb != "help"
puts "!! "
puts "!! no command verb: #{@attempted_verb}"
puts "!! \n\n"
end
usage
end
end
def purpose
puts <<-PURPOSE.undent
brew-cask provides a friendly homebrew-style CLI workflow for the
administration of Mac applications distributed as binaries.
PURPOSE
end
def usage
max_command_len = Hbc::CLI.commands.map(&:length).max
puts "Commands:\n\n"
Hbc::CLI.command_classes.each do |klass|
next unless klass.visible
puts " #{klass.command_name.ljust(max_command_len)} #{_help_for(klass)}"
end
puts %Q{\nSee also "man brew-cask"}
end
def help
''
end
def _help_for(klass)
klass.respond_to?(:help) ? klass.help : nil
end
end
end