Как обрабатывать отсутствующий обязательный аргумент в Ruby OptionParser?
В OptionParser я могу сделать опцию обязательной, но если я оставлю это значение, она примет название любой следующей опции в качестве значения, закручивая остальную часть синтаксического анализа командной строки.
Вот тестовый пример, отражающий значения параметров:
$ ./test_case.rb --input foo --output bar
output bar
input foo
Теперь оставьте значение для первой опции:
$ ./test_case.rb --input --output bar
input --output
Есть ли способ предотвратить использование другого имени параметра в качестве значения?
Спасибо!
Вот код тестового кода:
#!/usr/bin/env ruby
require 'optparse'
files = Hash.new
option_parser = OptionParser.new do |opts|
opts.on('-i', '--input FILENAME', 'Input filename - required') do |filename|
files[:input] = filename
end
opts.on('-o', '--output FILENAME', 'Output filename - required') do |filename|
files[:output] = filename
end
end
begin
option_parser.parse!(ARGV)
rescue OptionParser::ParseError
$stderr.print "Error: " + $! + "\n"
exit
end
files.keys.each do |key|
print "#{key} #{files[key]}\n"
end
Ответы
Ответ 1
Что вы хотите сделать, это не очень хорошая идея. Что делать, если у вас действительно есть файл с именем "--output"? Это абсолютно корректное имя файла в Unix. Каждый разбор параметров программы Unix работает так, как делает рубин, поэтому вы не должны его менять, потому что тогда ваша программа будет произвольно отличаться от всего остального, что запутывает и нарушает "принцип наименьшего удивления".
Реальный вопрос: почему у вас есть эта проблема в первую очередь? Возможно, вы запускаете свою программу из другой программы, а родительская программа предоставляет пустое имя файла как параметр для -input, что делает его видимым --output как параметр для -input. Вы можете обойти это, всегда цитируя имена файлов, которые вы передаете в командной строке:
./test_case.rb --input "" --output "bar"
Затем - вход будет пустым, и это легко обнаружить.
Также обратите внимание, что если для параметра --input установлено значение --output (и --output не является реальным файлом), вы можете просто попытаться открыть файл -input. Если это не удается, напечатайте сообщение, например:
can't open input file: --output: file not found
И это должно дать понять пользователю, что они сделали неправильно.
Ответ 2
В этом случае необязательная опция --output
отсутствует, так что сделайте это после вызова parse!
:
unless files[:input] && files[:output]
$stderr.puts "Error: you must specify both --input and --output options."
exit 1
end
Ответ 3
попробуйте следующее:
opts.on('-i', '--input FILENAME', 'Input filename - required') do |filename|
files[:input] = filename
end
opts.on('-o', '--output FILENAME', 'Output filename - required') do |filename|
files[:output] = filename
end
opts.on("-h", "--help", "Show this message") do
puts opts
exit
end
begin
ARGV << "-h" if ARGV.size != 2
option_parser.parse!(ARGV)
rescue OptionParser::ParseError
$stderr.print "Error: " + $! + "\n"
exit
end
Ответ 4
ОК - это работает - регулярное выражение в вызове on() допускает любую строку, если она не начинается с '-'
Если я не передаю аргумент в --input, и есть другой параметр downstream, тогда он примет этот ключ параметра в качестве аргумента для -input. (например, --input --output). Регулярное выражение улавливает это, а затем я проверяю сообщение об ошибке. Если аргумент, который он сообщает, начинается с "-", я вывожу правильное сообщение об ошибке, а именно, что отсутствует аргумент. Не очень, но, похоже, это работает.
Вот мой рабочий тестовый пример:
#!/usr/bin/env ruby
require 'optparse'
files = Hash.new
option_parser = OptionParser.new do |opts|
opts.on('-i FILENAME', '--input FILENAME', /\A[^\-]+/, 'Input filename - required') do |filename|
files[:input] = filename
end
opts.on('-o FILENAME', '--output FILENAME', /\A[^\-]+/, 'Output filename - required') do |filename|
files[:output] = filename
end
end
begin
option_parser.parse!(ARGV)
rescue OptionParser::ParseError
if $!.to_s =~ /invalid\s+argument\:\s+(\-\-\S+)\s+\-/
$stderr.print "Error: missing argument: #{$1}\n"
else
$stderr.print "Error: " + $! + "\n"
end
exit
end
files.keys.each do |key|
print "#{key} #{files[key]}\n"
end