aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authornobu <nobu@b2dd03c8-39d4-4d8f-98ff-823fe69b080e>2015-12-31 20:06:22 +0000
committernobu <nobu@b2dd03c8-39d4-4d8f-98ff-823fe69b080e>2015-12-31 20:06:22 +0000
commit0b2d86c642991214a9361e348f0eeae0c7f47023 (patch)
tree51293019c4674da8db80696f7bf9a38b932a5517
parentce27945c2ce5d233a6325736a1a92d74a1048b8a (diff)
downloadruby-0b2d86c642991214a9361e348f0eeae0c7f47023.tar.gz
parse.y: single-quote indented heredoc
* parse.y (parser_here_document): update indent for each line in indented here document with single-quotes. [ruby-core:72479] [Bug #11871] git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@53398 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
-rw-r--r--ChangeLog6
-rw-r--r--parse.y53
-rw-r--r--test/ruby/test_syntax.rb116
3 files changed, 99 insertions, 76 deletions
diff --git a/ChangeLog b/ChangeLog
index 8d9da44de1..bce027f63e 100644
--- a/ChangeLog
+++ b/ChangeLog
@@ -1,3 +1,9 @@
+Fri Jan 1 05:06:20 2016 Nobuyoshi Nakada <nobu@ruby-lang.org>
+
+ * parse.y (parser_here_document): update indent for each line in
+ indented here document with single-quotes.
+ [ruby-core:72479] [Bug #11871]
+
Fri Jan 1 03:26:44 2016 Nobuyoshi Nakada <nobu@ruby-lang.org>
* lib/ostruct.rb (freeze): define deferred accessors before
diff --git a/parse.y b/parse.y
index 161ba02dbb..cfbef9c764 100644
--- a/parse.y
+++ b/parse.y
@@ -6215,6 +6215,32 @@ simple_re_meta(int c)
}
static int
+parser_update_heredoc_indent(struct parser_params *parser, int c)
+{
+ if (heredoc_line_indent == -1) {
+ if (c == '\n') heredoc_line_indent = 0;
+ }
+ else {
+ if (c == ' ') {
+ heredoc_line_indent++;
+ return TRUE;
+ }
+ else if (c == '\t') {
+ int w = (heredoc_line_indent / TAB_WIDTH) + 1;
+ heredoc_line_indent = w * TAB_WIDTH;
+ return TRUE;
+ }
+ else if (c != '\n') {
+ if (heredoc_indent > heredoc_line_indent) {
+ heredoc_indent = heredoc_line_indent;
+ }
+ heredoc_line_indent = -1;
+ }
+ }
+ return FALSE;
+}
+
+static int
parser_tokadd_string(struct parser_params *parser,
int func, int term, int paren, long *nest,
rb_encoding **encp)
@@ -6244,24 +6270,7 @@ parser_tokadd_string(struct parser_params *parser,
while ((c = nextc()) != -1) {
if (heredoc_indent > 0) {
- if (heredoc_line_indent == -1) {
- if (c == '\n') heredoc_line_indent = 0;
- }
- else {
- if (c == ' ') {
- heredoc_line_indent++;
- }
- else if (c == '\t') {
- int w = (heredoc_line_indent / TAB_WIDTH) + 1;
- heredoc_line_indent = w * TAB_WIDTH;
- }
- else if (c != '\n') {
- if (heredoc_indent > heredoc_line_indent) {
- heredoc_indent = heredoc_line_indent;
- }
- heredoc_line_indent = -1;
- }
- }
+ parser_update_heredoc_indent(parser, c);
}
if (paren && c == paren) {
@@ -6881,6 +6890,14 @@ parser_here_document(struct parser_params *parser, NODE *here)
--pend;
}
}
+
+ if (heredoc_indent > 0) {
+ long i = 0;
+ while (p + i < pend && parser_update_heredoc_indent(parser, p[i]))
+ i++;
+ heredoc_line_indent = 0;
+ }
+
if (str)
rb_str_cat(str, p, pend - p);
else
diff --git a/test/ruby/test_syntax.rb b/test/ruby/test_syntax.rb
index 63997f423f..1daf7b2d91 100644
--- a/test/ruby/test_syntax.rb
+++ b/test/ruby/test_syntax.rb
@@ -492,92 +492,92 @@ e"
assert_equal(expected, actual, "#{Bug7559}: ")
end
+ def assert_dedented_heredoc(expect, result, mesg = "")
+ %w[eos "eos" 'eos'].each do |eos|
+ assert_equal(eval("<<-#{eos}\n#{expect}eos\n"),
+ eval("<<~#{eos}\n#{result}eos\n"),
+ message(mesg) {"with #{eos}"})
+ end
+ end
+
def test_dedented_heredoc_without_indentation
- assert_equal(" y\nz\n", <<~eos)
- y
-z
- eos
+ result = " y\n" \
+ "z\n"
+ expect = result
+ assert_dedented_heredoc(expect, result)
end
def test_dedented_heredoc_with_indentation
- assert_equal(" a\nb\n", <<~eos)
- a
- b
- eos
+ result = " a\n" \
+ " b\n"
+ expect = " a\n" \
+ "b\n"
+ assert_dedented_heredoc(expect, result)
end
def test_dedented_heredoc_with_blank_less_indented_line
# the blank line has two leading spaces
- result = eval("<<~eos\n" \
- " a\n" \
- " \n" \
- " b\n" \
- " eos\n")
- assert_equal("a\n\nb\n", result)
+ result = " a\n" \
+ " \n" \
+ " b\n"
+ expect = "a\n" \
+ "\n" \
+ "b\n"
+ assert_dedented_heredoc(expect, result)
end
def test_dedented_heredoc_with_blank_less_indented_line_escaped
- result = eval("<<~eos\n" \
- " a\n" \
- "\\ \\ \n" \
- " b\n" \
- " eos\n")
- assert_equal(" a\n \n b\n", result)
+ result = " a\n" \
+ "\\ \\ \n" \
+ " b\n"
+ expect = result
+ assert_dedented_heredoc(expect, result)
end
def test_dedented_heredoc_with_blank_more_indented_line
# the blank line has six leading spaces
- result = eval("<<~eos\n" \
- " a\n" \
- " \n" \
- " b\n" \
- " eos\n")
- assert_equal("a\n \nb\n", result)
+ result = " a\n" \
+ " \n" \
+ " b\n"
+ expect = "a\n" \
+ " \n" \
+ "b\n"
+ assert_dedented_heredoc(expect, result)
end
def test_dedented_heredoc_with_blank_more_indented_line_escaped
- result = eval("<<~eos\n" \
- " a\n" \
- "\\ \\ \\ \\ \\ \\ \n" \
- " b\n" \
- " eos\n")
- assert_equal(" a\n \n b\n", result)
+ result = " a\n" \
+ "\\ \\ \\ \\ \\ \\ \n" \
+ " b\n"
+ expect = result
+ assert_dedented_heredoc(expect, result)
end
def test_dedented_heredoc_with_empty_line
-result = eval("<<~eos\n" \
- " This would contain specially formatted text.\n" \
- "\n" \
- " That might span many lines\n" \
- " eos\n")
- assert_equal(<<-eos, result)
-This would contain specially formatted text.
-
-That might span many lines
- eos
+ result = " This would contain specially formatted text.\n" \
+ "\n" \
+ " That might span many lines\n"
+ expect = 'This would contain specially formatted text.'"\n" \
+ ''"\n" \
+ 'That might span many lines'"\n"
+ assert_dedented_heredoc(expect, result)
end
def test_dedented_heredoc_with_interpolated_expression
- result = eval(" <<~eos\n" \
- " #{1}a\n" \
- " zy\n" \
- " eos\n")
- assert_equal(<<-eos, result)
- #{1}a
-zy
- eos
+ result = ' #{1}a'"\n" \
+ " zy\n"
+ expect = ' #{1}a'"\n" \
+ "zy\n"
+ assert_dedented_heredoc(expect, result)
end
def test_dedented_heredoc_with_interpolated_string
w = ""
- result = eval("<<~eos\n" \
- " \#{w} a\n" \
- " zy\n" \
- " eos\n")
- assert_equal(<<-eos, result)
-#{w} a
- zy
- eos
+ result = " \#{mesg} a\n" \
+ " zy\n"
+ expect = '#{mesg} a'"\n" \
+ ' zy'"\n"
+ assert_dedented_heredoc(expect, result)
end
def test_lineno_after_heredoc