1

Given the following code:

options = {}
optparse = OptionParser.new do |opts|
    opts.on('-t', '--thing [THING1,THING2]', Array, 'Set THING1, THING2') do |t|
      options[:things] = t
    end
end

If THING1 has a comma in it, how can I prevent OptionParser from splitting on it?

Sample case: ./scrit.rb -t 'foo,bar',baz. In this case I want options[:things] should be ['foo,bar', 'baz']

Is this even possible?

4

2 に答える 2

1

あなたの実行の場合:

./scrit.rb -t 'foo,bar',baz

シェルパス ARGV:

["-t", "foo,bar,baz"]

シェルは 'foo,bar',baz を foo,bar,baz に変換します。

$ strace -e trace=execve ./scrit.rb -t 'foo,bar',baz
execve("./scrit.rb", ["./scrit.rb", "-t", "foo,bar,baz"], [/* 52 vars */]) = 0
execve("/home/scuawn/bin/ruby", ["ruby", "./scrit.rb", "-t", "foo,bar,baz"], [/* 52 vars */]) = 0

他の区切り文字を使用できます。

  opts.on('-t', '--thing [THING1,THING2]', Array, 'Set THING1, THING2') do |t|
    options[:things] = t
    options[:things][0] = options[:things][0].split(":")
  end

$ ./scrit.rb -t foo:bar,baz
[["foo", "bar"], "baz"]

または:

  opts.on('-t', '--thing [THING1,THING2]', Array, 'Set THING1, THING2') do |t|
    options[:things] = t
    options[:things] = options[:things].length == 3 ? [[options[:things][0],options[:things][1]],options[:things][2]] : options[:things]
  end

$ ./scrit.rb -t foo,bar,baz
[["foo", "bar"], "baz"]
于 2011-07-20T02:17:43.957 に答える
0

まず、シェル1は、次のすべての引用バリエーションに対して同じ最終値を生成します。

./scrit.rb -t 'foo,bar',baz
./scrit.rb -t foo,'bar,baz'
./scrit.rb -t 'foo,bar,baz'
./scrit.rb -t foo,bar,baz
./scrit.rb -t fo"o,b"ar,baz
./scrit.rb -t foo,b\ar,baz
# obviously many more variations are possible

これは次のように確認できます。

ruby -e 'f=ARGV[0];ARGV.each_with_index{|a,i|puts "%u: %s <%s>\n" % [i,a==f,a]}'\
 'foo,bar',baz foo,'bar,baz' 'foo,bar,baz' foo,bar,baz fo"o,b"ar,baz foo,b\ar,baz

1 Bourne のようなシェル ( zshbashkshdashなどのshバリアント) を想定しています。


他のセパレーターに切り替えたい場合は、次のようにします。

split_on_semicolons = Object.new
OptionParser.accept split_on_semicolons do |s,|
  s.split ';'
end
⋮
opts.on('-t', '--thing [THING1;THING2]', split_on_semicolons, 'Set THING1, THING2 (semicolon must be quoted to protect it from the shell)') do |t|
  options[:things] = t
end

シェルはセミコロンに特別な意味を与えるため、エスケープするか引用符で囲む必要があります (そうしないと、無条件のコマンド セパレータとして機能します (例: echo foo; sleep 2; echo bar))。

./scrit.rb -t foo,bar\;baz
./scrit.rb -t foo,bar';'baz
./scrit.rb -t 'foo,bar;baz'
# et cetera

指定時に行われる「パース」Arrayはほぼ基本str.split(',')通り(空文字列の値も落とす)なので、エスケープ文字を直接指定する方法はありません。

コンマを使い続けたいが「エスケープ文字」を導入したい場合は、ブロック内の値を少し後処理して、OptionParser#on特定の値をつなぎ合わせることができます。

# use backslash as an after-the-fact escape character
# in a sequence of string values,
#   if a value ends with a odd number of backslashes, then
#     the last backslash should be replaced with
#     a command concatenated with the next value
#   a backslash before any other single character is removed
# 
# basic unsplit: (note doubled backslashes due to writing these as Ruby values)
#     %w[foo\\ bar baz] => %w[foo,bar baz]
#
# escaped, trailing backslash is not an unsplit:
#     %w[foo\\\\ bar baz] => %w[foo\\ bar baz]
#
# escaping [other, backslash, split], also consecutive unsplits
#     %w[f\\o\\\\o\\ \\\\\\bar\\\\\\ baz] => %w[fo\\o,\\bar\\,baz]

def unsplit_and_unescape(orig_values)
  values = []
  incompleteValue = nil
  orig_values.each do |val|
    incomplete = /\\*$/.match(val)[0].length.odd?
    val.gsub! /\\(.)/, '\1'
    val = incompleteValue + ',' + val if incompleteValue
    if incomplete
      incompleteValue = val[0..-2]
    else
      values << val
      incompleteValue = nil
    end
  end
  if incompleteValue
    raise ArgumentError, 'Incomplete final value'
  end
  values
end
⋮
opts.on('-t', '--thing [THING1,THING2]', Array, 'Set THING1, THING2 (use \\, to include a comma)') do |t|
  options[:things] = unsplit_and_unescape(t)
end

次に、次のようにシェルから実行できます(バックスラッシュもシェルにとって特別なので、エスケープするか引用符で囲む必要があります2):

./scrit.rb -t foo\\,bar,baz
./scrit.rb -t 'foo\,bar,baz'
./scrit.rb -t foo'\,'bar,baz
./scrit.rb -t "foo\\,bar,baz"
./scrit.rb -t fo"o\\,ba"r,baz
# et cetera

2 Ruby とは異なり、シェルの一重引用符は完全にリテラル (バックスラッシュは解釈されません) であるため、他のシェル特殊文字 (バックスラッシュや二重引用符など)を埋め込む必要がある場合に適しています。

于 2011-07-20T10:29:41.257 に答える