require 'sexp_processor' require 'ripper_ruby_parser/sexp_handlers' module RipperRubyParser # Processes the sexp created by Ripper to what RubyParser would produce. class SexpProcessor < ::SexpProcessor attr_accessor :filename attr_accessor :extra_compatible def initialize super # TODO: Find these automatically @processors[:@int] = :process_at_int @processors[:@float] = :process_at_float @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[:@tstring_content] = :process_at_tstring_content @errors = [] @in_method_body = false end def process exp return nil if exp.nil? result = super trickle_up_line_numbers result trickle_down_line_numbers result end include SexpHandlers def process_program exp _, content = exp.shift 2 statements = content.map { |sub_exp| process(sub_exp) } safe_wrap_in_block statements 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_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(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(ref)) end def process_top_const_field exp s(:const, process_top_const_ref(exp)) end def process_paren exp _, body = exp.shift 2 if body.empty? s() elsif body.first.is_a? Symbol process body else body.map! { |it| convert_void_stmt_to_nil process it } safe_wrap_in_block body 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 s(:iter, s(:preexe), s(:args), *map_body(body)) end def process_END exp _, body = exp.shift 2 s(:iter, s(:postexe), 0, *map_body(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 # 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 make_identifier(:lvar, exp) 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 :__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 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) if body.length == 1 && body.first.sexp_type == :block body = body.first body.shift end body 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 trickle_up_line_numbers exp exp.each do |sub_exp| if sub_exp.is_a? Sexp trickle_up_line_numbers sub_exp exp.line ||= sub_exp.line end end end def trickle_down_line_numbers exp exp.each do |sub_exp| if sub_exp.is_a? Sexp sub_exp.line ||= exp.line trickle_down_line_numbers sub_exp end end end def convert_void_stmt_to_nil sexp if sexp == s(:void_stmt) s(:nil) else sexp end end end end