module ReplTypeCompletor

Constants

VERSION

Attributes

last_completion_error[R]

Public Class Methods

analyze(code, binding:, filename: nil) click to toggle source
# File repl_type_completor-0.1.9/lib/repl_type_completor.rb, line 31
def analyze(code, binding:, filename: nil)
  verbose, $VERBOSE = $VERBOSE, nil
  result = analyze_code(code, binding)
  Result.new(result, binding, filename) if result
rescue Exception => e
  handle_exception(e)
  nil
ensure
  $VERBOSE = verbose
end
handle_exception(e) click to toggle source
# File repl_type_completor-0.1.9/lib/repl_type_completor.rb, line 42
def handle_exception(e)
  @last_completion_error = e
end
info() click to toggle source
# File repl_type_completor-0.1.9/lib/repl_type_completor.rb, line 46
def info
  require 'rbs'
  prism_info = "Prism: #{Prism::VERSION}"
  rbs_info = "RBS: #{RBS::VERSION}"
  if rbs_load_error
    rbs_info << " #{rbs_load_error.inspect}"
  elsif !rbs_load_started?
    rbs_info << ' signatures not loaded'
  elsif !rbs_loaded?
    rbs_info << ' signatures loading'
  end
  "ReplTypeCompletor: #{VERSION}, #{prism_info}, #{rbs_info}"
end
load_rbs() click to toggle source
# File repl_type_completor-0.1.9/lib/repl_type_completor.rb, line 23
def load_rbs
  Types.load_rbs_builder unless rbs_loaded?
end
preload_rbs() click to toggle source
# File repl_type_completor-0.1.9/lib/repl_type_completor.rb, line 27
def preload_rbs
  Types.preload_rbs_builder
end
rbs_load_error() click to toggle source
# File repl_type_completor-0.1.9/lib/repl_type_completor.rb, line 11
def rbs_load_error
  Types.rbs_load_error
end
rbs_load_started?() click to toggle source
# File repl_type_completor-0.1.9/lib/repl_type_completor.rb, line 15
def rbs_load_started?
  Types.rbs_load_started?
end
rbs_loaded?() click to toggle source
# File repl_type_completor-0.1.9/lib/repl_type_completor.rb, line 19
def rbs_loaded?
  !!Types.rbs_builder
end

Private Class Methods

analyze_code(code, binding = Object::TOPLEVEL_BINDING) click to toggle source
# File repl_type_completor-0.1.9/lib/repl_type_completor.rb, line 62
def analyze_code(code, binding = Object::TOPLEVEL_BINDING)
  ast = Prism.parse(code, scopes: [binding.local_variables]).value
  *parents, target_node = find_target ast, code.bytesize
  return unless target_node

  calculate_scope = -> { TypeAnalyzer.calculate_target_type_scope(binding, parents, target_node).last }
  calculate_type_scope = ->(node) { TypeAnalyzer.calculate_target_type_scope binding, [*parents, target_node], node }

  case target_node
  when Prism::StringNode
    return unless target_node.closing&.empty?

    call_node, args_node = parents.last(2)
    return unless call_node.is_a?(Prism::CallNode) && call_node.receiver.nil?
    return unless args_node.is_a?(Prism::ArgumentsNode) && args_node.arguments.size == 1

    if call_node.name == :require || call_node.name == :require_relative
      [call_node.name, target_node.content]
    end
  when Prism::SymbolNode
    return unless !target_node.closing || target_node.closing.empty?

    name = target_node.value.to_s
    if parents.last.is_a? Prism::BlockArgumentNode # method(&:target)
      receiver_type, _scope = calculate_type_scope.call target_node
      [:call, name, receiver_type, false]
    else
      [:symbol, name] unless name.empty?
    end
  when Prism::CallNode, Prism::CallTargetNode
    return if target_node.is_a?(Prism::CallNode) && target_node.opening

    name = target_node.message.to_s
    return [:lvar_or_method, name, calculate_scope.call] if target_node.receiver.nil?

    self_call = target_node.receiver.is_a? Prism::SelfNode
    op = target_node.call_operator
    receiver_type, _scope = calculate_type_scope.call target_node.receiver
    receiver_type = receiver_type.nonnillable if op == '&.'
    [op == '::' ? :call_or_const : :call, name, receiver_type, self_call]
  when Prism::LocalVariableReadNode, Prism::LocalVariableTargetNode
    [:lvar_or_method, target_node.name.to_s, calculate_scope.call]
  when Prism::ItLocalVariableReadNode
    [:lvar_or_method, 'it', calculate_scope.call]
  when Prism::ConstantPathNode, Prism::ConstantPathTargetNode
    name = target_node.name.to_s
    if target_node.parent # A::B
      receiver, scope = calculate_type_scope.call(target_node.parent)
      [:const, name, receiver, scope]
    else # ::A
      scope = calculate_scope.call
      [:const, name, Types::SingletonType.new(Object), scope]
    end
  when Prism::ConstantReadNode, Prism::ConstantTargetNode
    [:const, target_node.name.to_s, nil, calculate_scope.call]
  when Prism::GlobalVariableReadNode, Prism::GlobalVariableTargetNode
    [:gvar, target_node.name.to_s, calculate_scope.call]
  when Prism::InstanceVariableReadNode, Prism::InstanceVariableTargetNode
    [:ivar, target_node.name.to_s, calculate_scope.call]
  when Prism::ClassVariableReadNode, Prism::ClassVariableTargetNode
    [:cvar, target_node.name.to_s, calculate_scope.call]
  end
end
find_target(node, position) click to toggle source
# File repl_type_completor-0.1.9/lib/repl_type_completor.rb, line 126
def find_target(node, position)
  # Skip because location of these nodes gives location of whole block
  return if node.is_a?(Prism::NumberedParametersNode) || node.is_a?(Prism::ItParametersNode)

  node.compact_child_nodes.each do |n|
    match = find_target(n, position)
    next unless match

    match.unshift node
    return match
  end

  [node] if node.location.end_offset == position
end