diff options
| author | Kevin Newton <kddnewton@gmail.com> | 2024-03-05 16:58:24 -0500 |
|---|---|---|
| committer | git <svn-admin@ruby-lang.org> | 2024-03-06 16:42:46 +0000 |
| commit | a0674a5755f0429ce435432ae2bad7bf696b7314 (patch) | |
| tree | 94ecb04ee68d47f61b915b0343dd904e9788e717 /lib | |
| parent | cb116420816163bd7b271313b7b4a18d3d5b0638 (diff) | |
[ruby/prism] Fix implementation of [] and []= for ripper translation
https://github.com/ruby/prism/commit/d31cf63d45
Diffstat (limited to 'lib')
| -rw-r--r-- | lib/prism/translation/ripper.rb | 84 |
1 files changed, 63 insertions, 21 deletions
diff --git a/lib/prism/translation/ripper.rb b/lib/prism/translation/ripper.rb index 339b42c003..136ccad44c 100644 --- a/lib/prism/translation/ripper.rb +++ b/lib/prism/translation/ripper.rb @@ -469,12 +469,71 @@ module Prism # foo.bar() {} # ^^^^^^^^^^^^ def visit_call_node(node) - return visit_aref_node(node) if node.name == :[] - return visit_aref_field_node(node) if node.name == :[]= + case node.name + when :[] + if node.opening == "[" + receiver = visit(node.receiver) + arguments = node.arguments&.arguments || [] + block = node.block + + if block.is_a?(BlockArgumentNode) + arguments << block + block = nil + end - if node.variable_call? - raise NoMethodError, __method__ unless node.receiver.nil? + arguments = + if arguments.any? + args = visit_arguments(arguments) + + if node.block.is_a?(BlockArgumentNode) + args + else + bounds(arguments.first.location) + on_args_add_block(args, false) + end + end + + bounds(node.location) + call = on_aref(receiver, arguments) + if block.nil? + return call + else + block = visit(block) + + bounds(node.location) + return on_method_add_block(call, block) + end + end + when :[]= + if node.opening == "[" + receiver = visit(node.receiver) + + *arguments, last_argument = node.arguments.arguments + arguments << node.block if !node.block.nil? + + arguments = + if arguments.any? + args = visit_arguments(arguments) + + if !node.block.nil? + args + else + bounds(arguments.first.location) + on_args_add_block(args, false) + end + end + + bounds(node.location) + call = on_aref_field(receiver, arguments) + + value = visit(last_argument) + bounds(last_argument.location) + return on_assign(call, value) + end + end + + if node.variable_call? bounds(node.message_loc) return on_vcall(on_ident(node.message)) end @@ -2432,23 +2491,6 @@ module Prism end end - # In Prism this is a CallNode with :[] as the operator. - # In Ripper it's an :aref. - def visit_aref_node(node) - first_arg_val = visit(node.arguments.arguments[0]) - args_val = on_args_add_block(on_args_add(on_args_new, first_arg_val), false) - on_aref(visit(node.receiver), args_val) - end - - # In Prism this is a CallNode with :[]= as the operator. - # In Ripper it's an :aref_field. - def visit_aref_field_node(node) - first_arg_val = visit(node.arguments.arguments[0]) - args_val = on_args_add_block(on_args_add(on_args_new, first_arg_val), false) - assign_val = visit(node.arguments.arguments[1]) - on_assign(on_aref_field(visit(node.receiver), args_val), assign_val) - end - # Ripper has several methods of emitting a symbol literal. Inside an alias # sometimes it suppresses the [:symbol] wrapper around ident. If the symbol # is also the name of a keyword (e.g. :if) it will emit a :@kw wrapper, not |
