# frozen_string_literal: true require 'sexp_processor' require 'ripper_ruby_parser/sexp_handlers' require 'ripper_ruby_parser/unescape' module RipperRubyParser # Processes the sexp created by Ripper to what RubyParser would produce. # # @api private class SexpProcessor < ::SexpProcessor include Unescape attr_reader :filename attr_reader :extra_compatible def initialize(filename: nil, extra_compatible: nil) super() @processors[:@int] = :process_at_int @processors[:@float] = :process_at_float @processors[:@rational] = :process_at_rational @processors[:@CHAR] = :process_at_CHAR @processors[:@label] = :process_at_label @processors[:@const] = :process_at_const @processors[:@ident] = :process_at_ident @processors[:@cvar] = :process_at_cvar @processors[:@gvar] = :process_at_gvar @processors[:@ivar] = :process_at_ivar @processors[:@kw] = :process_at_kw @processors[:@op] = :process_at_op @processors[:@backref] = :process_at_backref @processors[:@period] = :process_at_period @processors[:@tstring_content] = :process_at_tstring_content @filename = filename @extra_compatible = extra_compatible @errors = [] @in_method_body = false @kwrest = [] @block_kwrest = [] end include SexpHandlers def process_program(exp) _, content = exp.shift 2 process content end def process_module(exp) _, const_ref, body = exp.shift 3 const, line = const_ref_to_const_with_line_number const_ref with_line_number(line, s(:module, const, *class_or_module_body(body))) end def process_class(exp) _, const_ref, parent, body = exp.shift 4 const, line = const_ref_to_const_with_line_number const_ref parent = process(parent) with_line_number(line, s(:class, const, parent, *class_or_module_body(body))) end def process_sclass(exp) _, klass, block = exp.shift 3 s(:sclass, process(klass), *class_or_module_body(block)) end def process_stmts(exp) _, *statements = shift_all(exp) statements = map_process_list_compact statements case statements.count when 0 s(:void_stmt) when 1 statements.first else first = statements.shift if first.sexp_type == :block first.shift s(:block, *first, *statements) else s(:block, first, *statements) end end end def process_var_ref(exp) _, contents = exp.shift 2 process(contents) end def process_var_field(exp) _, contents = exp.shift 2 process(contents) end def process_var_alias(exp) _, left, right = exp.shift 3 s(:valias, left[1].to_sym, right[1].to_sym) end def process_const_path_ref(exp) _, left, right = exp.shift 3 s(:colon2, process(left), extract_node_symbol(process(right))) end def process_const_path_field(exp) s(:const, process_const_path_ref(exp)) end def process_const_ref(exp) _, ref = exp.shift 3 process(ref) end def process_top_const_ref(exp) _, ref = exp.shift 2 s(:colon3, extract_node_symbol(process(ref))) end def process_top_const_field(exp) s(:const, process_top_const_ref(exp)) end def process_paren(exp) _, body = exp.shift 2 result = process body if result.sexp_type == :void_stmt s(:nil) else result end end def process_comment(exp) _, comment, inner = exp.shift 3 sexp = process(inner) sexp.comments = comment sexp end def process_BEGIN(exp) _, body = exp.shift 2 body = reject_void_stmt map_process_list body.sexp_body s(:iter, s(:preexe), s(:args), *body) end def process_END(exp) _, body = exp.shift 2 body = map_process_list_compact body.sexp_body s(:iter, s(:postexe), 0, *body) end # number literals def process_at_int(exp) make_literal(exp) { |val| Integer(val) } end def process_at_float(exp) make_literal(exp, &:to_f) end def process_at_rational(exp) make_literal(exp, &:to_r) end # character literals def process_at_CHAR(exp) _, val, pos = exp.shift 3 with_position(pos, s(:str, unescape(val[1..-1]))) end def process_at_label(exp) make_literal(exp) { |val| val.chop.to_sym } end # symbol-like sexps def process_at_const(exp) make_identifier(:const, exp) end def process_at_cvar(exp) make_identifier(:cvar, exp) end def process_at_gvar(exp) make_identifier(:gvar, exp) end def process_at_ivar(exp) make_identifier(:ivar, exp) end def process_at_ident(exp) with_position_from_node_symbol(exp) do |ident| if replace_kwrest_arg_lvar? ident s(:call, nil, ident) else s(:lvar, ident) end end end def process_at_op(exp) make_identifier(:op, exp) end def process_at_kw(exp) sym, pos = extract_node_symbol_with_position(exp) result = case sym when :__ENCODING__ s(:colon2, s(:const, :Encoding), :UTF_8) when :__FILE__ s(:str, @filename) when :__LINE__ s(:lit, pos[0]) else s(sym) end with_position(pos, result) end def process_at_backref(exp) _, str, pos = exp.shift 3 name = str[1..-1] with_position pos do if name =~ /[0-9]/ s(:nth_ref, name.to_i) else s(:back_ref, name.to_sym) end end end def process_at_period(exp) _, period, = exp.shift 3 s(:period, period) end private def const_ref_to_const_with_line_number(const_ref) const = process(const_ref) line = const.line const = const[1] if const.sexp_type == :const return const, line end def class_or_module_body(exp) body = process(exp) return body if body.empty? if body.sexp_type == :block body.sexp_body else [body] end end def make_identifier(type, exp) with_position_from_node_symbol(exp) do |ident| s(type, ident) end end def make_literal(exp) _, val, pos = exp.shift 3 with_position(pos, s(:lit, yield(val))) end def replace_kwrest_arg_lvar?(ident) extra_compatible && @block_kwrest.include?(ident) end end end