getopt 1.3.6 → 1.3.7
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- data/CHANGES +61 -53
- data/MANIFEST +11 -12
- data/Rakefile +22 -23
- data/examples/example_long.rb +65 -0
- data/examples/example_std.rb +38 -0
- data/getopt.gemspec +24 -0
- data/lib/getopt/long.rb +234 -232
- data/lib/getopt/std.rb +79 -79
- data/test/{tc_getopt_long.rb → test_getopt_long.rb} +264 -264
- data/test/{tc_getopt_std.rb → test_getopt_std.rb} +82 -82
- metadata +53 -40
- data/test/ts_all.rb +0 -5
data/CHANGES
CHANGED
@@ -1,53 +1,61 @@
|
|
1
|
-
==
|
2
|
-
*
|
3
|
-
|
4
|
-
|
5
|
-
* Added
|
6
|
-
* Removed the
|
7
|
-
|
8
|
-
|
9
|
-
|
10
|
-
|
11
|
-
*
|
12
|
-
|
13
|
-
|
14
|
-
|
15
|
-
|
16
|
-
|
17
|
-
|
18
|
-
|
19
|
-
* Added
|
20
|
-
|
21
|
-
|
22
|
-
|
23
|
-
|
24
|
-
*
|
25
|
-
|
26
|
-
|
27
|
-
*
|
28
|
-
|
29
|
-
|
30
|
-
==
|
31
|
-
*
|
32
|
-
*
|
33
|
-
|
34
|
-
|
35
|
-
|
36
|
-
*
|
37
|
-
|
38
|
-
|
39
|
-
* Added
|
40
|
-
*
|
41
|
-
*
|
42
|
-
|
43
|
-
==
|
44
|
-
*
|
45
|
-
|
46
|
-
|
47
|
-
|
48
|
-
*
|
49
|
-
*
|
50
|
-
|
51
|
-
|
52
|
-
|
53
|
-
|
1
|
+
== 27-Jul-2008 - 1.3.7
|
2
|
+
* Fixed a potential infinite hash recursion bug in ARGV processing. This
|
3
|
+
was smoked out as the result of the alternate hash implementations in
|
4
|
+
JRuby and Ruby 1.9.
|
5
|
+
* Added the example programs to the gemspec.
|
6
|
+
* Removed the ts_all.rb file, and renamed the other test files. The Rakefile
|
7
|
+
test task was updated accordingly.
|
8
|
+
|
9
|
+
== 8-Aug-2007 - 1.3.6
|
10
|
+
* The Getopt::StdError class is now Getopt::Std::Error.
|
11
|
+
* The Getopt::LongError class is now Getopt::Long::Error.
|
12
|
+
* Added some inline rdoc documentation to the source code.
|
13
|
+
* Added a Rakefile with tasks for installation and testing.
|
14
|
+
* Removed the install.rb file - use the 'rake install' task instead.
|
15
|
+
|
16
|
+
== 5-Jul-2006 - 1.3.5
|
17
|
+
* Fixed a bug where multiple long switches with the same first character
|
18
|
+
could cause invalid results. Thanks go to Michael Campbell for the spot.
|
19
|
+
* Added documentation to the README file that explains what happens if you
|
20
|
+
specify multiple long switches with the same first character and no short
|
21
|
+
switch alias.
|
22
|
+
|
23
|
+
== 7-Mar-2006 - 1.3.4
|
24
|
+
* Fixed Getopt::Long so that it can handle embedded hyphens in the long
|
25
|
+
form, e.g. --foo-bar. Thanks go to Mark Meves for the spot.
|
26
|
+
* Corresponding test suite additions.
|
27
|
+
* Added example to the 'example_long.rb' file that uses long form with
|
28
|
+
embedded hyphens.
|
29
|
+
|
30
|
+
== 22-Feb-2006 - 1.3.3
|
31
|
+
* Bug fix for the two argument form of Getopt::Long.getopts.
|
32
|
+
* Corresponding test suite additions.
|
33
|
+
|
34
|
+
== 13-Feb-2006 - 1.3.2
|
35
|
+
* Improved error message if an option is passed without a preceding switch.
|
36
|
+
* Minor documentation fixes and clarifications.
|
37
|
+
|
38
|
+
== 18-Nov-2005 - 1.3.1
|
39
|
+
* Added support for compressed switches with getopt/long.
|
40
|
+
* More tests.
|
41
|
+
* Fixed a bug in the gemspec.
|
42
|
+
|
43
|
+
== 4-Nov-2005 - 1.3.0
|
44
|
+
* Added the Getopt::Long class (long.rb). This is a complete revamp of the
|
45
|
+
old getoptlong package, with ideas tossed in from Perl's Getopt::Long
|
46
|
+
package. See the README and example script for more detail.
|
47
|
+
* Added an example, and renamed the "test_std.rb" example to "example_std.rb".
|
48
|
+
* Added lots of documentation to the README file.
|
49
|
+
* Updated the MANIFEST, test suite, etc.
|
50
|
+
|
51
|
+
== 24-Oct-2005 - 1.2.0
|
52
|
+
* Altered the way multiple occurrences of the same switch are handled, for
|
53
|
+
those switches that accept arguments.
|
54
|
+
|
55
|
+
== 7-Oct-2005 - 1.1.0
|
56
|
+
* Changed parser, added a bit stricter enforcement
|
57
|
+
* Now handles squished arguments properly, e.g. "-ID" as well as "-I -D"
|
58
|
+
* Some test suite changes
|
59
|
+
|
60
|
+
== 5-Oct-2005 - 1.0.0
|
61
|
+
* Initial commit
|
data/MANIFEST
CHANGED
@@ -1,12 +1,11 @@
|
|
1
|
-
* CHANGES
|
2
|
-
* MANIFEST
|
3
|
-
* README
|
4
|
-
* Rakefile
|
5
|
-
* getopt.gemspec
|
6
|
-
* examples/example_std.rb
|
7
|
-
* examples/example_long.rb
|
8
|
-
* lib/getopt/std.rb
|
9
|
-
* lib/getopt/long.rb
|
10
|
-
* test/
|
11
|
-
* test/
|
12
|
-
* test/ts_all.rb
|
1
|
+
* CHANGES
|
2
|
+
* MANIFEST
|
3
|
+
* README
|
4
|
+
* Rakefile
|
5
|
+
* getopt.gemspec
|
6
|
+
* examples/example_std.rb
|
7
|
+
* examples/example_long.rb
|
8
|
+
* lib/getopt/std.rb
|
9
|
+
* lib/getopt/long.rb
|
10
|
+
* test/test_getopt_std.rb
|
11
|
+
* test/test_getopt_long.rb
|
data/Rakefile
CHANGED
@@ -1,23 +1,22 @@
|
|
1
|
-
require 'rake'
|
2
|
-
require 'rake/testtask'
|
3
|
-
|
4
|
-
desc "Install the getopt package (non-gem)"
|
5
|
-
task :install do
|
6
|
-
dest = File.join(Config::CONFIG['sitelibdir'], 'getopt')
|
7
|
-
Dir.mkdir(dest) unless File.exists? dest
|
8
|
-
cp 'lib/getopt/std.rb', dest, :verbose => true
|
9
|
-
cp 'lib/getopt/long.rb', dest, :verbose => true
|
10
|
-
end
|
11
|
-
|
12
|
-
desc "Install the getopt package as a gem"
|
13
|
-
task :install_gem do
|
14
|
-
ruby 'getopt.gemspec'
|
15
|
-
file = Dir["*.gem"].first
|
16
|
-
sh "gem install #{file}"
|
17
|
-
end
|
18
|
-
|
19
|
-
Rake::TestTask.new do |t|
|
20
|
-
t.
|
21
|
-
t.
|
22
|
-
|
23
|
-
end
|
1
|
+
require 'rake'
|
2
|
+
require 'rake/testtask'
|
3
|
+
|
4
|
+
desc "Install the getopt package (non-gem)"
|
5
|
+
task :install do
|
6
|
+
dest = File.join(Config::CONFIG['sitelibdir'], 'getopt')
|
7
|
+
Dir.mkdir(dest) unless File.exists? dest
|
8
|
+
cp 'lib/getopt/std.rb', dest, :verbose => true
|
9
|
+
cp 'lib/getopt/long.rb', dest, :verbose => true
|
10
|
+
end
|
11
|
+
|
12
|
+
desc "Install the getopt package as a gem"
|
13
|
+
task :install_gem do
|
14
|
+
ruby 'getopt.gemspec'
|
15
|
+
file = Dir["*.gem"].first
|
16
|
+
sh "gem install #{file}"
|
17
|
+
end
|
18
|
+
|
19
|
+
Rake::TestTask.new do |t|
|
20
|
+
t.warning = true
|
21
|
+
t.verbose = true
|
22
|
+
end
|
@@ -0,0 +1,65 @@
|
|
1
|
+
##########################################################################
|
2
|
+
# example_long.rb
|
3
|
+
#
|
4
|
+
# A series of examples to demonstrate the different ways that you can
|
5
|
+
# handle command line options.
|
6
|
+
##########################################################################
|
7
|
+
require "getopt/long"
|
8
|
+
|
9
|
+
# The truly lazy way. This creates two valid command line switches,
|
10
|
+
# automatically creates single letter switches (-f, -b), and sets each
|
11
|
+
# switch type to BOOLEAN.
|
12
|
+
opts = Getopt::Long.getopts("--foo --bar")
|
13
|
+
|
14
|
+
# Here's a comprehensive example that uses all types and options.
|
15
|
+
opts = Getopt::Long.getopts(
|
16
|
+
["--foo"], # --foo, -f, BOOLEAN
|
17
|
+
["--bar", "-z"], # --bar, -z, BOOLEAN
|
18
|
+
["--baz", "-b", OPTIONAL], # --baz, -b, OPTIONAL
|
19
|
+
["--name", "-n", REQUIRED], # --name, -n, REQUIRED
|
20
|
+
["--more", "-m", INCREMENT], # --more, -m, INCREMENT
|
21
|
+
["--verbose", "-v", BOOLEAN], # --verbose, -v, BOOLEAN
|
22
|
+
["--my-name", "-x", REQUIRED] # --my-name, -x, REQUIRED
|
23
|
+
)
|
24
|
+
|
25
|
+
# Using the above example:
|
26
|
+
|
27
|
+
# User passes "-f"
|
28
|
+
# opts -> { "f" => true, "foo" => true }
|
29
|
+
|
30
|
+
# User passes "-z"
|
31
|
+
# opts -> { "z" => true, "bar" => true }
|
32
|
+
|
33
|
+
# User passes "--verbose"
|
34
|
+
# opts -> { "v" => true, "verbose" => true }
|
35
|
+
|
36
|
+
# User passes "-m"
|
37
|
+
# opts -> { "m" => 1, "more" => 1 }
|
38
|
+
|
39
|
+
# User passes "-m -m"
|
40
|
+
# opts -> { "m" => 2, "more" => 2 }
|
41
|
+
|
42
|
+
# User passes "--name Dan" or "--name=Dan" or "-n Dan" or "-nDan"
|
43
|
+
# opts -> { "n" => "Dan", "name" => "Dan" }
|
44
|
+
|
45
|
+
# User passes "--my-name Dan" or "--my-name=Dan" or "-x Dan" or "-xDan"
|
46
|
+
# opts -> { "x" => "Dan", "my-name" => "Dan" }
|
47
|
+
|
48
|
+
# User passes "--name Dan --name Matz"
|
49
|
+
# opts -> { "n" => ["Dan","Matz"], "name" => ["Dan","Matz"] }
|
50
|
+
|
51
|
+
# User passes "--baz" with no argument
|
52
|
+
# opts -> { "b" => nil, "baz" => nil }
|
53
|
+
|
54
|
+
# User passes "--baz hello"
|
55
|
+
# opts =-> { "b" => "hello", "baz" => "hello" }
|
56
|
+
|
57
|
+
# User passes "-n" with no argument
|
58
|
+
# Getopt::LongError is raised, since an argument is REQUIRED.
|
59
|
+
|
60
|
+
# User passes "-f hello"
|
61
|
+
# Getopt::LongError is raised, since a BOOLEAN switch does not take an argument
|
62
|
+
|
63
|
+
# User passes "--warning"
|
64
|
+
# Getopt::LongError is raised, since "--warning" was not specified as a valid
|
65
|
+
# switch in the call to Getopt::Long.getopts
|
@@ -0,0 +1,38 @@
|
|
1
|
+
####################################################
|
2
|
+
# example_std.rb
|
3
|
+
#
|
4
|
+
# Some samples of how to use the Getopt::Std class.
|
5
|
+
#####################################################
|
6
|
+
base = File.basename(Dir.pwd)
|
7
|
+
|
8
|
+
if base == "examples" || base =~ /getopt/
|
9
|
+
Dir.chdir("..") if base == "examples"
|
10
|
+
$LOAD_PATH.unshift(Dir.pwd + "/lib")
|
11
|
+
Dir.chdir("examples") rescue nil
|
12
|
+
end
|
13
|
+
|
14
|
+
require "getopt/std"
|
15
|
+
include Getopt
|
16
|
+
|
17
|
+
# Try passing different switches to this script to see what happens
|
18
|
+
opts = Std.getopts("o:ID")
|
19
|
+
p opts
|
20
|
+
|
21
|
+
# User passes "-o hello -I"
|
22
|
+
# Result: {"o" => "hello", "I" => true}
|
23
|
+
|
24
|
+
# User passes "-I -D"
|
25
|
+
# Result: {"I" => true, "D" => true}
|
26
|
+
|
27
|
+
# User passes nothing
|
28
|
+
# Result: {}
|
29
|
+
|
30
|
+
# User passes "-o hello -o world -I"
|
31
|
+
# Result: {"I" => true, "o" => ["hello", "world"]}
|
32
|
+
|
33
|
+
# User passes "-o -I"
|
34
|
+
# Result: Getopt::StdError, because -o requires an argument (and does not
|
35
|
+
# accept -I as an argument, since it is a valid switch)
|
36
|
+
|
37
|
+
# User passes "-I -X"
|
38
|
+
# Result: Getopt::StdError, because -X was not listed as a valid switch.
|
data/getopt.gemspec
ADDED
@@ -0,0 +1,24 @@
|
|
1
|
+
require "rubygems"
|
2
|
+
|
3
|
+
spec = Gem::Specification.new do |gem|
|
4
|
+
gem.name = "getopt"
|
5
|
+
gem.version = "1.3.7"
|
6
|
+
gem.author = "Daniel J. Berger"
|
7
|
+
gem.email = "djberg96@gmail.com"
|
8
|
+
gem.homepage = "http://www.rubyforge.org/projects/shards"
|
9
|
+
gem.platform = Gem::Platform::RUBY
|
10
|
+
gem.summary = "Getopt::Std and Getopt::Long option parsers for Ruby"
|
11
|
+
gem.description = "Getopt::Std and Getopt::Long option parsers for Ruby"
|
12
|
+
gem.test_files = Dir["test/*.rb"]
|
13
|
+
gem.has_rdoc = true
|
14
|
+
gem.files = Dir['lib/**/*.rb'] + Dir['[A-Z]*'] + Dir['test/*'] + Dir['examples/*.rb']
|
15
|
+
gem.files.reject! { |fn| fn.include? "CVS" }
|
16
|
+
gem.require_path = "lib"
|
17
|
+
gem.extra_rdoc_files = ["README", "CHANGES", "MANIFEST"]
|
18
|
+
gem.rubyforge_project = 'shards'
|
19
|
+
end
|
20
|
+
|
21
|
+
if $0 == __FILE__
|
22
|
+
Gem.manage_gems
|
23
|
+
Gem::Builder.new(spec).build
|
24
|
+
end
|
data/lib/getopt/long.rb
CHANGED
@@ -1,232 +1,234 @@
|
|
1
|
-
module Getopt
|
2
|
-
|
3
|
-
REQUIRED = 0
|
4
|
-
BOOLEAN = 1
|
5
|
-
OPTIONAL = 2
|
6
|
-
INCREMENT = 3
|
7
|
-
NEGATABLE = 4
|
8
|
-
NUMERIC = 5
|
9
|
-
|
10
|
-
class Long
|
11
|
-
class Error < StandardError; end
|
12
|
-
|
13
|
-
VERSION = '1.3.
|
14
|
-
|
15
|
-
# Takes an array of switches. Each array consists of up to three
|
16
|
-
# elements that indicate the name and type of switch. Returns a hash
|
17
|
-
# containing each switch name, minus the '-', as a key. The value
|
18
|
-
# for each key depends on the type of switch and/or the value provided
|
19
|
-
# by the user.
|
20
|
-
#
|
21
|
-
# The long switch _must_ be provided. The short switch defaults to the
|
22
|
-
# first letter of the short switch. The default type is BOOLEAN.
|
23
|
-
#
|
24
|
-
# Example:
|
25
|
-
#
|
26
|
-
# opts = Getopt::Long.getopts(
|
27
|
-
# ["--debug"],
|
28
|
-
# ["--verbose", "-v"],
|
29
|
-
# ["--level", "-l", NUMERIC]
|
30
|
-
# )
|
31
|
-
#
|
32
|
-
# See the README file for more information.
|
33
|
-
#
|
34
|
-
def self.getopts(*switches)
|
35
|
-
if switches.empty?
|
36
|
-
raise ArgumentError, "no switches provided"
|
37
|
-
end
|
38
|
-
|
39
|
-
hash = {} # Hash returned to user
|
40
|
-
valid = [] # Tracks valid switches
|
41
|
-
types = {} # Tracks argument types
|
42
|
-
syns = {} # Tracks long and short arguments, or multiple shorts
|
43
|
-
|
44
|
-
# If a string is passed, split it and convert it to an array of arrays
|
45
|
-
if switches.first.kind_of?(String)
|
46
|
-
switches = switches.join.split
|
47
|
-
switches.map!{ |switch| switch = [switch] }
|
48
|
-
end
|
49
|
-
|
50
|
-
# Set our list of valid switches, and proper types for each switch
|
51
|
-
switches.each{ |switch|
|
52
|
-
valid.push(switch[0]) # Set valid long switches
|
53
|
-
|
54
|
-
# Set type for long switch, default to BOOLEAN.
|
55
|
-
if switch[1].kind_of?(Fixnum)
|
56
|
-
switch[2] = switch[1]
|
57
|
-
types[switch[0]] = switch[2]
|
58
|
-
switch[1] = switch[0][1..2]
|
59
|
-
else
|
60
|
-
switch[2] ||= BOOLEAN
|
61
|
-
types[switch[0]] = switch[2]
|
62
|
-
switch[1] ||= switch[0][1..2]
|
63
|
-
end
|
64
|
-
|
65
|
-
# Create synonym hash. Default to first char of long switch for
|
66
|
-
# short switch, e.g. "--verbose" creates a "-v" synonym. The same
|
67
|
-
# synonym can only be used once - first one wins.
|
68
|
-
syns[switch[0]] = switch[1] unless syns[switch[1]]
|
69
|
-
syns[switch[1]] = switch[0] unless syns[switch[1]]
|
70
|
-
|
71
|
-
switch[1].each{ |char|
|
72
|
-
types[char] = switch[2] # Set type for short switch
|
73
|
-
valid.push(char) # Set valid short switches
|
74
|
-
}
|
75
|
-
}
|
76
|
-
|
77
|
-
re_long = /^(--\w+[-\w+]*)?$/
|
78
|
-
re_short = /^(-\w)$/
|
79
|
-
re_long_eq = /^(--\w+[-\w+]*)?=(.*?)$|(-\w?)=(.*?)$/
|
80
|
-
re_short_sq = /^(-\w)(\S+?)$/
|
81
|
-
|
82
|
-
ARGV.each_with_index{ |opt, index|
|
83
|
-
|
84
|
-
# Allow either -x -v or -xv style for single char args
|
85
|
-
if re_short_sq.match(opt)
|
86
|
-
chars = opt.split("")[1..-1].map{ |s| s = "-#{s}" }
|
87
|
-
|
88
|
-
chars.each_with_index{ |char, i|
|
89
|
-
unless valid.include?(char)
|
90
|
-
raise Error, "invalid switch '#{char}'"
|
91
|
-
end
|
92
|
-
|
93
|
-
# Grab the next arg if the switch takes a required arg
|
94
|
-
if types[char] == REQUIRED
|
95
|
-
# Deal with a argument squished up against switch
|
96
|
-
if chars[i+1]
|
97
|
-
arg = chars[i+1..-1].join.tr("-","")
|
98
|
-
ARGV.push(char, arg)
|
99
|
-
break
|
100
|
-
else
|
101
|
-
arg = ARGV.delete_at(index+1)
|
102
|
-
if arg.nil? || valid.include?(arg) # Minor cheat here
|
103
|
-
err = "no value provided for required argument '#{char}'"
|
104
|
-
raise Error, err
|
105
|
-
end
|
106
|
-
ARGV.push(char, arg)
|
107
|
-
end
|
108
|
-
elsif types[char] == OPTIONAL
|
109
|
-
if chars[i+1] && !valid.include?(chars[i+1])
|
110
|
-
arg = chars[i+1..-1].join.tr("-","")
|
111
|
-
ARGV.push(char, arg)
|
112
|
-
break
|
113
|
-
elsif
|
114
|
-
if ARGV[index+1] && !valid.include?(ARGV[index+1])
|
115
|
-
arg = ARGV.delete_at(index+1)
|
116
|
-
ARGV.push(char, arg)
|
117
|
-
end
|
118
|
-
else
|
119
|
-
ARGV.push(char)
|
120
|
-
end
|
121
|
-
else
|
122
|
-
ARGV.push(char)
|
123
|
-
end
|
124
|
-
}
|
125
|
-
next
|
126
|
-
end
|
127
|
-
|
128
|
-
if match = re_long.match(opt) || match = re_short.match(opt)
|
129
|
-
switch = match.captures.first
|
130
|
-
end
|
131
|
-
|
132
|
-
if match = re_long_eq.match(opt)
|
133
|
-
switch, value = match.captures.compact
|
134
|
-
ARGV.push(switch, value)
|
135
|
-
next
|
136
|
-
end
|
137
|
-
|
138
|
-
# Make sure that all the switches are valid. If 'switch' isn't
|
139
|
-
# defined at this point, it means an option was passed without
|
140
|
-
# a preceding switch, e.g. --option foo bar.
|
141
|
-
unless valid.include?(switch)
|
142
|
-
switch ||= opt
|
143
|
-
raise Error, "invalid switch '#{switch}'"
|
144
|
-
end
|
145
|
-
|
146
|
-
# Required arguments
|
147
|
-
if types[switch] == REQUIRED
|
148
|
-
nextval = ARGV[index+1]
|
149
|
-
|
150
|
-
# Make sure there's a value for mandatory arguments
|
151
|
-
if nextval.nil?
|
152
|
-
err = "no value provided for required argument '#{switch}'"
|
153
|
-
raise Error, err
|
154
|
-
end
|
155
|
-
|
156
|
-
# If there is a value, make sure it's not another switch
|
157
|
-
if valid.include?(nextval)
|
158
|
-
err = "cannot pass switch '#{nextval}' as an argument"
|
159
|
-
raise Error, err
|
160
|
-
end
|
161
|
-
|
162
|
-
# If the same option appears more than once, put the values
|
163
|
-
# in array.
|
164
|
-
if hash[switch]
|
165
|
-
hash[switch] = [hash[switch], nextval].flatten
|
166
|
-
else
|
167
|
-
hash[switch] = nextval
|
168
|
-
end
|
169
|
-
ARGV.delete_at(index+1)
|
170
|
-
end
|
171
|
-
|
172
|
-
# For boolean arguments set the switch's value to true.
|
173
|
-
if types[switch] == BOOLEAN
|
174
|
-
if hash.has_key?(switch)
|
175
|
-
raise Error, "boolean switch already set"
|
176
|
-
end
|
177
|
-
hash[switch] = true
|
178
|
-
end
|
179
|
-
|
180
|
-
# For increment arguments, set the switch's value to 0, or
|
181
|
-
# increment it by one if it already exists.
|
182
|
-
if types[switch] == INCREMENT
|
183
|
-
if hash.has_key?(switch)
|
184
|
-
hash[switch] += 1
|
185
|
-
else
|
186
|
-
hash[switch] = 1
|
187
|
-
end
|
188
|
-
end
|
189
|
-
|
190
|
-
# For optional argument, there may be an argument. If so, it
|
191
|
-
# cannot be another switch. If not, it is set to true.
|
192
|
-
if types[switch] == OPTIONAL
|
193
|
-
nextval = ARGV[index+1]
|
194
|
-
if valid.include?(nextval)
|
195
|
-
hash[switch] = true
|
196
|
-
else
|
197
|
-
hash[switch] = nextval
|
198
|
-
ARGV.delete_at(index+1)
|
199
|
-
end
|
200
|
-
end
|
201
|
-
}
|
202
|
-
|
203
|
-
# Set synonymous switches to the same value, e.g. if -t is a synonym
|
204
|
-
# for --test, and the user passes "--test", then set "-t" to the same
|
205
|
-
# value that "--test" was set to.
|
206
|
-
#
|
207
|
-
# This allows users to refer to the long or short switch and get
|
208
|
-
# the same value
|
209
|
-
hash.each{ |switch, val|
|
210
|
-
if syns.keys.include?(switch)
|
211
|
-
syns[switch].each{ |key|
|
212
|
-
hash[key] = val
|
213
|
-
}
|
214
|
-
end
|
215
|
-
}
|
216
|
-
|
217
|
-
# Get rid of leading "--" and "-" to make it easier to reference
|
218
|
-
hash.each{ |key, value|
|
219
|
-
if key
|
220
|
-
|
221
|
-
|
222
|
-
|
223
|
-
|
224
|
-
|
225
|
-
|
226
|
-
|
227
|
-
|
228
|
-
|
229
|
-
|
230
|
-
|
231
|
-
|
232
|
-
|
1
|
+
module Getopt
|
2
|
+
|
3
|
+
REQUIRED = 0
|
4
|
+
BOOLEAN = 1
|
5
|
+
OPTIONAL = 2
|
6
|
+
INCREMENT = 3
|
7
|
+
NEGATABLE = 4
|
8
|
+
NUMERIC = 5
|
9
|
+
|
10
|
+
class Long
|
11
|
+
class Error < StandardError; end
|
12
|
+
|
13
|
+
VERSION = '1.3.7'
|
14
|
+
|
15
|
+
# Takes an array of switches. Each array consists of up to three
|
16
|
+
# elements that indicate the name and type of switch. Returns a hash
|
17
|
+
# containing each switch name, minus the '-', as a key. The value
|
18
|
+
# for each key depends on the type of switch and/or the value provided
|
19
|
+
# by the user.
|
20
|
+
#
|
21
|
+
# The long switch _must_ be provided. The short switch defaults to the
|
22
|
+
# first letter of the short switch. The default type is BOOLEAN.
|
23
|
+
#
|
24
|
+
# Example:
|
25
|
+
#
|
26
|
+
# opts = Getopt::Long.getopts(
|
27
|
+
# ["--debug"],
|
28
|
+
# ["--verbose", "-v"],
|
29
|
+
# ["--level", "-l", NUMERIC]
|
30
|
+
# )
|
31
|
+
#
|
32
|
+
# See the README file for more information.
|
33
|
+
#
|
34
|
+
def self.getopts(*switches)
|
35
|
+
if switches.empty?
|
36
|
+
raise ArgumentError, "no switches provided"
|
37
|
+
end
|
38
|
+
|
39
|
+
hash = {} # Hash returned to user
|
40
|
+
valid = [] # Tracks valid switches
|
41
|
+
types = {} # Tracks argument types
|
42
|
+
syns = {} # Tracks long and short arguments, or multiple shorts
|
43
|
+
|
44
|
+
# If a string is passed, split it and convert it to an array of arrays
|
45
|
+
if switches.first.kind_of?(String)
|
46
|
+
switches = switches.join.split
|
47
|
+
switches.map!{ |switch| switch = [switch] }
|
48
|
+
end
|
49
|
+
|
50
|
+
# Set our list of valid switches, and proper types for each switch
|
51
|
+
switches.each{ |switch|
|
52
|
+
valid.push(switch[0]) # Set valid long switches
|
53
|
+
|
54
|
+
# Set type for long switch, default to BOOLEAN.
|
55
|
+
if switch[1].kind_of?(Fixnum)
|
56
|
+
switch[2] = switch[1]
|
57
|
+
types[switch[0]] = switch[2]
|
58
|
+
switch[1] = switch[0][1..2]
|
59
|
+
else
|
60
|
+
switch[2] ||= BOOLEAN
|
61
|
+
types[switch[0]] = switch[2]
|
62
|
+
switch[1] ||= switch[0][1..2]
|
63
|
+
end
|
64
|
+
|
65
|
+
# Create synonym hash. Default to first char of long switch for
|
66
|
+
# short switch, e.g. "--verbose" creates a "-v" synonym. The same
|
67
|
+
# synonym can only be used once - first one wins.
|
68
|
+
syns[switch[0]] = switch[1] unless syns[switch[1]]
|
69
|
+
syns[switch[1]] = switch[0] unless syns[switch[1]]
|
70
|
+
|
71
|
+
switch[1].each{ |char|
|
72
|
+
types[char] = switch[2] # Set type for short switch
|
73
|
+
valid.push(char) # Set valid short switches
|
74
|
+
}
|
75
|
+
}
|
76
|
+
|
77
|
+
re_long = /^(--\w+[-\w+]*)?$/
|
78
|
+
re_short = /^(-\w)$/
|
79
|
+
re_long_eq = /^(--\w+[-\w+]*)?=(.*?)$|(-\w?)=(.*?)$/
|
80
|
+
re_short_sq = /^(-\w)(\S+?)$/
|
81
|
+
|
82
|
+
ARGV.each_with_index{ |opt, index|
|
83
|
+
|
84
|
+
# Allow either -x -v or -xv style for single char args
|
85
|
+
if re_short_sq.match(opt)
|
86
|
+
chars = opt.split("")[1..-1].map{ |s| s = "-#{s}" }
|
87
|
+
|
88
|
+
chars.each_with_index{ |char, i|
|
89
|
+
unless valid.include?(char)
|
90
|
+
raise Error, "invalid switch '#{char}'"
|
91
|
+
end
|
92
|
+
|
93
|
+
# Grab the next arg if the switch takes a required arg
|
94
|
+
if types[char] == REQUIRED
|
95
|
+
# Deal with a argument squished up against switch
|
96
|
+
if chars[i+1]
|
97
|
+
arg = chars[i+1..-1].join.tr("-","")
|
98
|
+
ARGV.push(char, arg)
|
99
|
+
break
|
100
|
+
else
|
101
|
+
arg = ARGV.delete_at(index+1)
|
102
|
+
if arg.nil? || valid.include?(arg) # Minor cheat here
|
103
|
+
err = "no value provided for required argument '#{char}'"
|
104
|
+
raise Error, err
|
105
|
+
end
|
106
|
+
ARGV.push(char, arg)
|
107
|
+
end
|
108
|
+
elsif types[char] == OPTIONAL
|
109
|
+
if chars[i+1] && !valid.include?(chars[i+1])
|
110
|
+
arg = chars[i+1..-1].join.tr("-","")
|
111
|
+
ARGV.push(char, arg)
|
112
|
+
break
|
113
|
+
elsif
|
114
|
+
if ARGV[index+1] && !valid.include?(ARGV[index+1])
|
115
|
+
arg = ARGV.delete_at(index+1)
|
116
|
+
ARGV.push(char, arg)
|
117
|
+
end
|
118
|
+
else
|
119
|
+
ARGV.push(char)
|
120
|
+
end
|
121
|
+
else
|
122
|
+
ARGV.push(char)
|
123
|
+
end
|
124
|
+
}
|
125
|
+
next
|
126
|
+
end
|
127
|
+
|
128
|
+
if match = re_long.match(opt) || match = re_short.match(opt)
|
129
|
+
switch = match.captures.first
|
130
|
+
end
|
131
|
+
|
132
|
+
if match = re_long_eq.match(opt)
|
133
|
+
switch, value = match.captures.compact
|
134
|
+
ARGV.push(switch, value)
|
135
|
+
next
|
136
|
+
end
|
137
|
+
|
138
|
+
# Make sure that all the switches are valid. If 'switch' isn't
|
139
|
+
# defined at this point, it means an option was passed without
|
140
|
+
# a preceding switch, e.g. --option foo bar.
|
141
|
+
unless valid.include?(switch)
|
142
|
+
switch ||= opt
|
143
|
+
raise Error, "invalid switch '#{switch}'"
|
144
|
+
end
|
145
|
+
|
146
|
+
# Required arguments
|
147
|
+
if types[switch] == REQUIRED
|
148
|
+
nextval = ARGV[index+1]
|
149
|
+
|
150
|
+
# Make sure there's a value for mandatory arguments
|
151
|
+
if nextval.nil?
|
152
|
+
err = "no value provided for required argument '#{switch}'"
|
153
|
+
raise Error, err
|
154
|
+
end
|
155
|
+
|
156
|
+
# If there is a value, make sure it's not another switch
|
157
|
+
if valid.include?(nextval)
|
158
|
+
err = "cannot pass switch '#{nextval}' as an argument"
|
159
|
+
raise Error, err
|
160
|
+
end
|
161
|
+
|
162
|
+
# If the same option appears more than once, put the values
|
163
|
+
# in array.
|
164
|
+
if hash[switch]
|
165
|
+
hash[switch] = [hash[switch], nextval].flatten
|
166
|
+
else
|
167
|
+
hash[switch] = nextval
|
168
|
+
end
|
169
|
+
ARGV.delete_at(index+1)
|
170
|
+
end
|
171
|
+
|
172
|
+
# For boolean arguments set the switch's value to true.
|
173
|
+
if types[switch] == BOOLEAN
|
174
|
+
if hash.has_key?(switch)
|
175
|
+
raise Error, "boolean switch already set"
|
176
|
+
end
|
177
|
+
hash[switch] = true
|
178
|
+
end
|
179
|
+
|
180
|
+
# For increment arguments, set the switch's value to 0, or
|
181
|
+
# increment it by one if it already exists.
|
182
|
+
if types[switch] == INCREMENT
|
183
|
+
if hash.has_key?(switch)
|
184
|
+
hash[switch] += 1
|
185
|
+
else
|
186
|
+
hash[switch] = 1
|
187
|
+
end
|
188
|
+
end
|
189
|
+
|
190
|
+
# For optional argument, there may be an argument. If so, it
|
191
|
+
# cannot be another switch. If not, it is set to true.
|
192
|
+
if types[switch] == OPTIONAL
|
193
|
+
nextval = ARGV[index+1]
|
194
|
+
if valid.include?(nextval)
|
195
|
+
hash[switch] = true
|
196
|
+
else
|
197
|
+
hash[switch] = nextval
|
198
|
+
ARGV.delete_at(index+1)
|
199
|
+
end
|
200
|
+
end
|
201
|
+
}
|
202
|
+
|
203
|
+
# Set synonymous switches to the same value, e.g. if -t is a synonym
|
204
|
+
# for --test, and the user passes "--test", then set "-t" to the same
|
205
|
+
# value that "--test" was set to.
|
206
|
+
#
|
207
|
+
# This allows users to refer to the long or short switch and get
|
208
|
+
# the same value
|
209
|
+
hash.each{ |switch, val|
|
210
|
+
if syns.keys.include?(switch)
|
211
|
+
syns[switch].each{ |key|
|
212
|
+
hash[key] = val
|
213
|
+
}
|
214
|
+
end
|
215
|
+
}
|
216
|
+
|
217
|
+
# Get rid of leading "--" and "-" to make it easier to reference
|
218
|
+
hash.each{ |key, value|
|
219
|
+
if key =~ /^-/
|
220
|
+
if key[0,2] == '--'
|
221
|
+
nkey = key.sub('--', '')
|
222
|
+
else
|
223
|
+
nkey = key.sub('-', '')
|
224
|
+
end
|
225
|
+
hash.delete(key)
|
226
|
+
hash[nkey] = value
|
227
|
+
end
|
228
|
+
}
|
229
|
+
|
230
|
+
hash
|
231
|
+
end
|
232
|
+
|
233
|
+
end
|
234
|
+
end
|