module CommandLine
class Option
attr_reader :name, :alias
attr_reader :parameter_count
attr_reader :default_value
# Rewrites a command line keyword by replacing the underscores with dashes
# sym The symbol to rewrite
def self.rewrite(sym)
sym.to_s.gsub(/_/, '-').to_sym
end
# Initialize new CommandLine::Option
# name The name of the flag
# definition The definition of the flag.
def initialize(name, definition = {})
@name = CommandLine::Option.rewrite(name)
@alias = definition[:alias] ? definition[:alias].to_sym : nil
@required = definition.has_key?(:required) && definition[:required] == true
@parameter_count = definition[:parameters] || 1
@multiple = definition[:multiple] || false
@default_value = definition[:default] || false
end
def parse(arguments_parser)
if @parameter_count == 0
return true
elsif @parameter_count == 1
parameter = arguments_parser.next_parameter
raise CommandLine::ParameterExpected, self if parameter.nil?
return parameter
elsif @parameter_count == :any
parameters = []
while parameter = arguments_parser.next_parameter && parameter != '--'
parameters << parameter
end
return parameters
else
parameters = []
@parameter_count.times do |n|
parameter = arguments_parser.next_parameter
raise CommandLine::ParameterExpected, self if parameter.nil?
parameters << parameter
end
return parameters
end
end
def =~(test)
[@name, @alias].include?(CommandLine::Option.rewrite(test))
end
# Argument representation of the flag (--fast)
def to_option
"--#{@name}"
end
# Argument alias representation of the flag (-f)
def to_alias
"-#{@alias}"
end
# Check if flag has an alias
def has_alias?
!@alias.nil?
end
# Check if flag is required
def required?
@required
end
# Check if flag is optional
def optional?
!@required
end
def multiple?
@multiple
end
def has_default?
!@default_value.nil?
end
end
class Arguments
class Definition
ENDLESS_PARAMETERS = 99999
attr_reader :commands, :options, :parameters
def initialize(parent)
@parent = parent
@options = {}
@commands = {}
@parameters = nil
end
def [](option_name)
option_symbol = CommandLine::Option.rewrite(option_name)
if the_option = @options.detect { |(_, odef)| odef =~ option_symbol }
the_option[1]
else
raise CommandLine::UnknownOption, option_name
end
end
def minimum_parameters=(count_specifier)
@parameters = count_specifier..ENDLESS_PARAMETERS
end
def parameters=(count_specifier)
@parameters = count_specifier
end
alias :files= :parameters=
def option(name, options = {})
clo = CommandLine::Option.new(name, options)
@options[clo.name] = clo
end
def switch(name, switch_alias = nil)
option(name, :alias => switch_alias, :parameters => 0)
end
def command(name, &block)
command_definition = Definition.new(self)
yield(command_definition) if block_given?
@commands[CommandLine::Option.rewrite(name)] = command_definition
end
def has_command?(command)
@commands[CommandLine::Option.rewrite(command)]
end
end
OPTION_REGEXP = /^\-\-([A-Za-z0-9-]+)$/;
ALIASES_REGEXP = /^\-([A-Aa-z0-9]+)$/
attr_reader :definition
attr_reader :tokens
attr_reader :command, :options, :parameters
def self.parse(tokens = $*, &block)
cla = Arguments.new
cla.define(&block)
return cla.parse!(tokens)
end
def initialize
@tokens = []
@definition = Definition.new(self)
@current_definition = @definition
end
def define(&block)
yield(@definition)
end
def [](option)
if the_option = @options.detect { |(key, _)| key =~ option }
the_option[1]
else
@current_definition[option].default_value
end
end
def next_token
@current_token = @tokens.shift
return @current_token
end
def next_parameter
parameter_candidate = @tokens.first
parameter = (parameter_candidate.nil? || OPTION_REGEXP =~ parameter_candidate || ALIASES_REGEXP =~ parameter_candidate) ? nil : @tokens.shift
return parameter
end
def parse!(tokens)
@current_definition = @definition
@first_token = true
@tokens = tokens.clone
@options = {}
@parameters = []
@command = nil
prepare_result!
while next_token
if @first_token && command_definition = @definition.has_command?(@current_token)
@current_definition = command_definition
@command = CommandLine::Option.rewrite(@current_token)
else
case @current_token
when ALIASES_REGEXP; handle_alias_expansion($1)
when OPTION_REGEXP; handle_option($1)
else; handle_other_parameter(@current_token)
end
@first_token = false
end
end
validate_arguments!
return self
end
protected
def prepare_result!
multiple_options = Hash[*@current_definition.options.select { |name, o| o.multiple? }.flatten]
multiple_options.each { |name, definition| @options[definition] = [] }
end
def validate_arguments!
if @current_definition.parameters && !(@current_definition.parameters === @parameters.length)
raise CommandLine::ParametersOutOfRange.new(@current_definition.parameters, @parameters.length)
end
required_options = Hash[*@current_definition.options.select { |name, o| o.required? }.flatten]
required_options.each do |name, definition|
raise CommandLine::RequiredOptionMissing, definition unless self[name]
end
end
def handle_alias_expansion(aliases)
aliases.reverse.scan(/./) do |alias_char|
if option_definition = @current_definition[alias_char]
@tokens.unshift(option_definition.to_option)
else
raise CommandLine::UnknownOption, alias_char
end
end
end
def handle_other_parameter(parameter)
@parameters << parameter
end
def handle_option(option_name)
option_definition = @current_definition[option_name]
raise CommandLine::UnknownOption, option_name if option_definition.nil?
if option_definition.multiple?
@options[option_definition] << option_definition.parse(self)
else
@options[option_definition] = option_definition.parse(self)
end
end
end
# Commandline parsing errors and exceptions
class Error < Exception
end
# Missing a required flag
class RequiredOptionMissing < CommandLine::Error
def initialize(option)
super("You have to provide the #{option.name} option!")
end
end
# Missing a required file
class ParametersOutOfRange < CommandLine::Error
def initialize(expected, actual)
if expected.kind_of?(Range)
if expected.end == CommandLine::Arguments::Definition::ENDLESS_PARAMETERS
super("The command expected at least #{expected.begin} parameters, but found #{actual}!")
else
super("The command expected between #{expected.begin} and #{expected.end} parameters, but found #{actual}!")
end
else
super("The command expected #{expected} parameters, but found #{actual}!")
end
end
end
# Missing a required flag argument
class ParameterExpected < CommandLine::Error
def initialize(option)
super("The option #{option.inspect} expects a parameter!")
end
end
# Encountered an unkown flag
class UnknownOption < CommandLine::Error
def initialize(option_identifier)
super("#{option_identifier.inspect} not recognized as a valid option!")
end
end
end