lib/tins/go.rb
module Tins module GO module EnumerableExtension def push(argument) @arguments ||= [] @arguments.push argument self end alias << push def each(&block) @arguments.each(&block) self end include Enumerable end module_function # Parses the argument array _args_, according to the pattern _s_, to # retrieve the single character command line options from it. If _s_ is # 'xy:' an option '-x' without an option argument is searched, and an # option '-y foo' with an option argument ('foo'). To disable the '-x' # option, pass '~x'. # # The _defaults_ argument specifies default values for the options. # # An option hash is returned with all found options set to true or the # found option argument. def go(s, args = ARGV, defaults: {}) d = defaults || {} b, v = s.scan(/(.)(:?)/).inject([ {}, {} ]) { |t, (o, a)| a = a == ?: t[a ? 1 : 0][o] = a ? nil : false t } b.each_key do |k| d.key?(k) or next if [ 0, false, nil ].include?(d[k]) b[k] = false elsif d[k].respond_to?(:to_int) b[k] = d[k].to_int else b[k] = 1 end end v.each_key do |k| d.key?(k) or next if [ 0, false, nil ].include?(d[k]) v[k] = nil else v[k] = d[k].to_s end end r = [] while a = args.shift /\A-(?<p>.+)/ =~ a or (r << a; next) until p == '' o = p.slice!(0, 1) if v.key?(o) if p.empty? && args.empty? r << a break 1 elsif p == '' a = args.shift else a = p end if v[o].nil? || !(EnumerableExtension === v[o]) a = a.dup a.extend EnumerableExtension a << a v[o] = a else v[o] << a end break elsif b.key?(o) if b[o] b[o] += 1 else b[o] = 1 end else r << a end end && break end r.reject! { |a| (b[p] = false) || true if /\A~(?<p>.)/ =~ a } args.replace r b.merge(v) end end end require 'tins/alias'