# frozen_string_literal: true module RaaP # sc = SymbolicCaller.new( # [:call, # [:call, C, :new, [], { # a: [:call, A, :new, [], {}, nil], # b: [:call, B, :new, [], {}, nil] }, nil], # :run, [], {}, nil] # sc.eval #=> 123 # # sc.to_lines # ↓ # a = A.new(1) # b = B.new(b: 'bbb') # c = C.new(a: a, b: b) # c.run() { } class SymbolicCaller class Var attr_reader :name def initialize(name) @name = name end def +(other) "#{self}#{other}" end def to_s @name end end attr_reader :symbolic_call attr_reader :allow_private def initialize(symbolic_call, allow_private: false) @symbolic_call = symbolic_call @allow_private = allow_private end def eval walk do |symbolic_call| eval_one(symbolic_call) end end def call_str symbolic_call => [:call, receiver, Symbol => method_name, Array => args, Hash => kwargs, block] receiver = try_eval(receiver) args, kwargs, block = try_eval([args, kwargs, block]) a = [] a << args.map(&BindCall.method(:inspect)).join(', ') if !args.empty? a << kwargs.map { |k, v| "#{k}: #{BindCall.inspect(v)}" }.join(', ') if !kwargs.empty? argument_str = a.join(', ') block_str = block ? "{ }" : nil "#{BindCall.inspect(receiver)}.#{method_name}(#{argument_str})#{block_str}" end def to_lines [].tap do |lines| walk do |symbolic_call, is_last| symbolic_call => [:call, receiver_value, method_name, args, kwargs, block] is_mod = receiver_value.is_a?(Module) case when receiver_value == Kernel var = Var.new(method_name.to_s.downcase) var_eq = "#{var} = " receiver = '' when BindCall.instance_of?(receiver_value, Var) var_eq = "#{receiver_value} = " var = Var.new(receiver_value.name) receiver = var + '.' when is_mod var = Var.new(var_name(receiver_value)) var_eq = "#{var} = " receiver = receiver_value.name + '.' else var_eq = '' receiver = Var.new(printable(receiver_value)) + '.' end var_eq = '' if is_last arguments = [] if !args.empty? # The reproduction code is kept short and executable. if [Value::Interface, Value::Intersection].include?(receiver_value) if args.all? { |a| a.respond_to?(:free_variables) } && !args.all? { |a| a.free_variables.empty? } # FIXME: Type arguments are not yet supported. args.each do |a| lines << "# Free variables: #{a.free_variables.to_a.join(', ')}" end end arguments << args.map { |a| printable(a.to_s) } else arguments << args.map { |a| printable(a) } end end arguments << kwargs.map { |k, v| "#{k}: #{printable(v)}" }.join(', ') if !kwargs.empty? block_str = block ? " { }" : "" line = "#{var_eq}#{receiver}#{method_name}(#{arguments.join(', ')})#{block_str}" lines << line var end end end private def try_eval(symbolic_call) SymbolicCaller.new(symbolic_call).eval rescue RuntimeError, NotImplementedError symbolic_call end def walk(&block) _walk(@symbolic_call, true, &block) end def _walk(symbolic_call, is_last, &block) return symbolic_call if BindCall::instance_of?(symbolic_call, BasicObject) return symbolic_call if !BindCall.respond_to?(symbolic_call, :deconstruct) && !BindCall.respond_to?(symbolic_call, :deconstruct_keys) case symbolic_call in [:call, receiver, Symbol => method_name, Array => args, Hash => kwargs, b] receiver = _walk(receiver, false, &block) args = _walk(args, false, &block) if !args.empty? kwargs = _walk(kwargs, false, &block) if !kwargs.empty? block.call [:call, receiver, method_name, args, kwargs, b], is_last in Array symbolic_call.map { |sc| _walk(sc, false, &block) } in Hash symbolic_call.transform_keys { |k| _walk(k, false, &block) } .transform_values! { |v| _walk(v, false, &block) } else symbolic_call end end def eval_one(symbolic_call) symbolic_call => [:call, receiver_value, method_name, args, kwargs, block] if @allow_private receiver_value.__send__(method_name, *args, **kwargs, &block) else BindCall.public_send(receiver_value, method_name, *args, **kwargs, &block) end end def var_name(mod) printable(mod).gsub('::', '_').downcase end def printable(obj) case obj when Var obj.name # Object from which it can get strings that can be eval with `#inspect` when Symbol, Integer, Float, Regexp, Range, nil, true, false obj.inspect when String obj.inspect.gsub('"', "'") or raise when Hash hash_body = obj.map do |k, v| ks = printable(k) vs = printable(v) ks.start_with?(':') ? "#{ks[1..-1]}: #{vs}" : "#{ks} => #{vs}" end "{#{hash_body.join(', ')}}" when Array "[#{obj.map { |o| printable(o) }.join(', ')}]" when Module BindCall.name(obj) or raise "`#class` method returns nil" else var_name(BindCall.class(obj)) end end end end