summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authornobu <nobu@b2dd03c8-39d4-4d8f-98ff-823fe69b080e>2005-11-08 23:41:40 +0000
committernobu <nobu@b2dd03c8-39d4-4d8f-98ff-823fe69b080e>2005-11-08 23:41:40 +0000
commite8201d12db87546249e23871b959317580ff81ff (patch)
tree4fc7f31c32a0aa03abc6c185609dbe208dd4e80b
parent6ddd56a1308703c4ee26d942e3012531c7f80973 (diff)
* lib/shellwords.rb: refactored. [ruby-core:06581]
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@9519 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
-rw-r--r--ChangeLog4
-rw-r--r--lib/shellwords.rb33
-rw-r--r--test/test_shellwords.rb45
3 files changed, 60 insertions, 22 deletions
diff --git a/ChangeLog b/ChangeLog
index 97753c97f0..b004dcad27 100644
--- a/ChangeLog
+++ b/ChangeLog
@@ -1,3 +1,7 @@
+Wed Nov 9 08:39:38 2005 Nobuyoshi Nakada <nobu@ruby-lang.org>
+
+ * lib/shellwords.rb: refactored. [ruby-core:06581]
+
Tue Nov 8 17:35:53 2005 Hirokazu Yamamoto <ocean@m2.ccsnet.ne.jp>
* intern.h, eval.c (rb_thread_signal_raise): costified.
diff --git a/lib/shellwords.rb b/lib/shellwords.rb
index e87b9e656b..055a6a5be2 100644
--- a/lib/shellwords.rb
+++ b/lib/shellwords.rb
@@ -28,31 +28,20 @@ module Shellwords
def shellwords(line)
line = String.new(line) rescue
raise(ArgumentError, "Argument must be a string")
- line.lstrip!
words = []
- until line.empty?
- field = ''
- loop do
- if line.sub!(/\A"(([^"\\]|\\.)*)"/, '') then
- snippet = $1.gsub(/\\(.)/, '\1')
- elsif line =~ /\A"/ then
- raise ArgumentError, "Unmatched double quote: #{line}"
- elsif line.sub!(/\A'([^']*)'/, '') then
- snippet = $1
- elsif line =~ /\A'/ then
- raise ArgumentError, "Unmatched single quote: #{line}"
- elsif line.sub!(/\A\\(.)?/, '') then
- snippet = $1 || '\\'
- elsif line.sub!(/\A([^\s\\'"]+)/, '') then
- snippet = $1
- else
- line.lstrip!
- break
- end
- field.concat(snippet)
+ field = ''
+ last = 0
+ sep = nil
+ line.scan(/\G\s*(?:([^\s\\\'\"]+)|'([^\']*)'|"((?:[^\"\\]|\\.)*)"|(\\.?))(\s+|\z)?/m) do
+ last = $~.end(0)
+ sep = $~.begin(5)
+ field << ($1 || $2 || ($3 || $4).gsub(/\\(?=.)/, ''))
+ if sep
+ words << field
+ field = ''
end
- words.push(field)
end
+ raise ArgumentError, "Unmatched double quote: #{line}" if line[last]
words
end
diff --git a/test/test_shellwords.rb b/test/test_shellwords.rb
new file mode 100644
index 0000000000..1d914c5c0e
--- /dev/null
+++ b/test/test_shellwords.rb
@@ -0,0 +1,45 @@
+require 'test/unit'
+require 'shellwords'
+
+class TestShellwords < Test::Unit::TestCase
+
+ include Shellwords
+
+ def setup
+ @not_string = Class.new
+ @cmd = "ruby my_prog.rb | less"
+ end
+
+
+ def test_not_string
+ assert_raises ArgumentError do
+ shellwords(@not_string)
+ end
+ end
+
+ def test_string
+ assert_instance_of(Array, shellwords(@cmd))
+ assert_equal(4, shellwords(@cmd).length)
+ end
+
+ def test_unmatched_double_quote
+ bad_cmd = 'one two "three'
+ assert_raises ArgumentError do
+ shellwords(bad_cmd)
+ end
+ end
+
+ def test_unmatched_single_quote
+ bad_cmd = "one two 'three"
+ assert_raises ArgumentError do
+ shellwords(bad_cmd)
+ end
+ end
+
+ def test_unmatched_quotes
+ bad_cmd = "one '"'"''""'""
+ assert_raises ArgumentError do
+ shellwords(bad_cmd)
+ end
+ end
+end