require 'optparse' begin basic_object = Module.const_get('BasicObject') # We are 1.9.x rescue NameError BasicObject = Object end module Methadone # Include this module to gain access to the "canonical command-line app structure" # DSL. This is a *very* lightweight layer on top of what you might # normally write that gives you just a bit of help to keep your code structured # in a sensible way. You can use as much or as little as you want, though # you must at least use #main to get any benefits. # # You also get a more expedient interface to OptionParser as well # as checking for required arguments to your app. For example, if # we want our app to accept a negatable switch named "switch", a flag # named "flag", and two arguments "needed" (which is required) # and "maybe" which is optional, we can do the following: # # #!/usr/bin/env ruby -w # # require 'methadone' # # include Methadone::Main # # main do |needed, maybe| # options[:switch] => true or false, based on command line # options[:flag] => value of flag passed on command line # end # # # Proxy to an OptionParser instance's on method # on("--[no]-switch") # on("--flag VALUE") # # arg :needed # arg :maybe, :optional # # go! # # Our app then acts as follows: # # $ our_app # # => parse error: 'needed' is required # $ our_app foo # # => succeeds; "maybe" in main is nil # # This also includes Methadone::CLILogging to give you access to simple logging module Main include Methadone::CLILogging # Declare the main method for your app. # This allows you to specify the general logic of your # app at the top of your bin file, but can rely on any methods # or other code that you define later. # # For example, suppose you want to process a set of files, but # wish to determine that list from another method to keep your # code clean. # # #!/usr/bin/env ruby -w # # require 'methadone' # # include Methadone::Main # # main do # files_to_process.each do |file| # # process file # end # end # # def files_to_process # # return list of files # end # # go! # # The block can accept any parameters, and unparsed arguments # from the command line will be passed. # # To run this method, call #go! def main(&block) @main_block = block @options = {} @option_parser = OptionParserProxy.new(OptionParser.new,@options) end # Start your command-line app, exiting appropriately when # complete. # # This *will* exit your program when it completes. If your # #main block evaluates to an integer, that value will be sent # to Kernel#exit, otherwise, this will exit with 0 # # If the command-line options couldn't be parsed, this # will exit with 64 and whatever message OptionParser provided. # # If a required argument (see #arg) is not found, this exits with # 64 and a message about that missing argument. # def go! opts.parse! opts.check_args! result = call_main if result.kind_of? Fixnum exit result else exit 0 end rescue OptionParser::ParseError => ex error ex.message exit 64 # Linux standard for bad command line end # Call this to exit the program immediately # with the given error code and message. # # +exit_code+:: exit status you'd like to exit with # +message+:: message to display to the user explaining the problem def exit_now!(exit_code,message=nil) raise Methadone::Error.new(exit_code,message) end # Returns an OptionParser that you can use # to declare your command-line interface. Generally, you # won't use this and will use #on directly, but this allows # you to have complete control of option parsing. # # The object returned has # an additional feature that implements typical use of OptionParser. # # opts.on("--flag VALUE") # # Does this under the covers: # # opts.on("--flag VALUE") do |value| # options[:flag] = value # end # # Since, most of the time, this is all you want to do, # this makes it more expedient to do so. The key that is # is set in #options will be a symbol of the option name, without # the dashes. Note that if you use multiple option names, a key # will be generated for each. Further, if you use the negatable form, # only the positive key will be set, e.g. for --[no-]verbose, # only :verbose will be set (to true or false). def opts @option_parser end # Calls opts.on with the given arguments def on(*args,&block) opts.on(*args,&block) end # Sets the name of an arguments your app accepts. Note # that no sanity checking is done on the configuration # of your arguments you create via multiple calls to this method. # Namely, the last argument should be the only one that is # a :many or a :any, but the system here won't sanity check that. # # +arg_name+:: name of the argument to appear in documentation # This will be converted into a String and used to create # the banner (unless you have overridden the banner) # +options+:: list (not Hash) of options: # :required:: this arg is required (this is the default) # :optional:: this arg is optional # :one:: only one of this arg should be supplied (default) # :many:: many of this arg may be supplied, but at least one is required # :any:: any number, include zero, may be supplied def arg(arg_name,*options) opts.arg(arg_name,*options) end # Set the description of your app for inclusion in the help output. # +desc+:: a short, one-line description of your app def description(desc) opts.description(desc) end # Returns a Hash that you can use to store or retrieve options # parsed from the command line def options @options end private # Handle calling main and trapping any exceptions thrown def call_main @main_block.call(*ARGV) rescue Methadone::Error => ex error ex.message unless no_message? ex ex.exit_code rescue => ex error ex.message unless no_message? ex 70 # Linux sysexit code for internal software error end def no_message?(exception) exception.message.nil? || exception.message.strip.empty? end end # A proxy to OptionParser that intercepts #on # so that we can allow a simpler interface class OptionParserProxy < BasicObject # Create the proxy # # +option_parser+:: An OptionParser instance # +options+:: a hash that will store the options # set via automatic setting. The caller should # retain a reference to this def initialize(option_parser,options) @option_parser = option_parser @options = options @user_specified_banner = false @accept_options = false @args = [] @arg_options = {} @description = nil set_banner end def check_args! ::Hash[@args.zip(::ARGV)].each do |arg_name,arg_value| if @arg_options[arg_name].include? :required if arg_value.nil? message = "'#{arg_name.to_s}' is required" message = "at least one " + message if @arg_options[arg_name].include? :many raise ::OptionParser::ParseError,message end end end end # If invoked as with OptionParser, behaves the exact same way. # If invoked without a block, however, the options hash given # to the constructor will be used to store # the parsed command-line value. See #opts in the Main module # for how that works. def on(*args,&block) @accept_options = true if block @option_parser.on(*args,&block) else opt_names = option_names(*args) @option_parser.on(*args) do |value| opt_names.each do |name| @options[name] = value @options[name.to_s] = value end end end set_banner end # Proxies to underlying OptionParser def banner=(new_banner) @option_parser.banner=new_banner @user_specified_banner = true end # Sets the banner to include these arg names def arg(arg_name,*options) options << :optional if options.include?(:any) && !options.include?(:optional) options << :required unless options.include? :optional options << :one unless options.include?(:any) || options.include?(:many) @args << arg_name @arg_options[arg_name] = options set_banner end def description(desc) @description = desc set_banner end # Defers all calls save #on to # the underlying OptionParser instance def method_missing(sym,*args,&block) @option_parser.send(sym,*args,&block) end # Since we extend Object on 1.8.x, to_s is defined and thus not proxied by method_missing def to_s #::nodoc:: @option_parser.to_s end private def set_banner unless @user_specified_banner new_banner="Usage: #{::File.basename($0)}" new_banner += " [options]" if @accept_options unless @args.empty? new_banner += " " new_banner += @args.map { |arg| if @arg_options[arg].include? :any "[#{arg.to_s}...]" elsif @arg_options[arg].include? :optional "[#{arg.to_s}]" elsif @arg_options[arg].include? :many "#{arg.to_s}..." else arg.to_s end }.join(' ') end new_banner += "\n\n#{@description}" if @description new_banner += "\n\nOptions:" if @accept_options @option_parser.banner=new_banner end end def option_names(*opts_on_args,&block) opts_on_args.map { |arg| if arg =~ /^--\[no-\]([^-\s][^\s]*)/ $1.to_sym elsif arg =~ /^--([^-\s][^\s]*)/ $1.to_sym elsif arg =~ /^-([^-\s][^\s]*)/ $1.to_sym else nil end }.reject(&:nil?) end end end