module Steep module Services module HoverProvider class Ruby TypeContent = _ = Struct.new(:node, :type, :location, keyword_init: true) VariableContent = _ = Struct.new(:node, :name, :type, :location, keyword_init: true) TypeAssertionContent = _ = Struct.new(:node, :original_type, :asserted_type, :location, keyword_init: true) MethodCallContent = _ = Struct.new(:node, :method_call, :location, keyword_init: true) DefinitionContent = _ = Struct.new(:node, :method_name, :method_type, :definition, :location, keyword_init: true) ConstantContent = _ = Struct.new(:location, :full_name, :type, :decl, keyword_init: true) do # @implements ConstantContent def comments case when decl = class_decl decl.decls.map {|d| d.decl.comment } when decl = class_alias [decl.decl.comment] when decl = constant_decl [decl.decl.comment] else raise end.compact end def class_decl case decl when ::RBS::Environment::ClassEntry, ::RBS::Environment::ModuleEntry decl end end def class_alias case decl when ::RBS::Environment::ClassAliasEntry, ::RBS::Environment::ModuleAliasEntry decl end end def constant_decl if decl.is_a?(::RBS::Environment::ConstantEntry) decl end end def constant? constant_decl ? true : false end def class_or_module? (class_decl || class_alias) ? true : false end end attr_reader :service def initialize(service:) @service = service end def project service.project end def method_definition_for(factory, type_name, singleton_method: nil, instance_method: nil) case when instance_method factory.definition_builder.build_instance(type_name).methods[instance_method] when singleton_method methods = factory.definition_builder.build_singleton(type_name).methods if singleton_method == :new methods[:new] || methods[:initialize] else methods[singleton_method] end else raise "One of the instance_method or singleton_method is required" end end def typecheck(target, path:, content:, line:, column:) subtyping = service.signature_services[target.name].current_subtyping or return source = Source.parse(content, path: path, factory: subtyping.factory) source = source.without_unrelated_defs(line: line, column: column) resolver = ::RBS::Resolver::ConstantResolver.new(builder: subtyping.factory.definition_builder) pos = source.buffer.loc_to_pos([line, column]) Services::TypeCheckService.type_check(source: source, subtyping: subtyping, constant_resolver: resolver, cursor: pos) rescue nil end def method_name_from_method(context, builder:) context.method or raise defined_in = context.method.defined_in or raise method_name = context.name or raise case when defined_in.class? case when builder.build_instance(defined_in).methods.key?(method_name) InstanceMethodName.new(type_name: defined_in, method_name: method_name) when builder.build_singleton(defined_in).methods.key?(method_name) SingletonMethodName.new(type_name: defined_in, method_name: method_name) else raise end else InstanceMethodName.new(type_name: defined_in, method_name: method_name) end end def content_for(target:, path:, line:, column:) file = service.source_files[path] or return typing = typecheck(target, path: path, content: file.content, line: line, column: column) or return node, *parents = typing.source.find_nodes(line: line, column: column) if node && parents case node.type when :lvar var_name = node.children[0] context = typing.cursor_context.context or raise var_type = context.type_env[var_name] || AST::Types::Any.instance() return VariableContent.new( node: node, name: var_name, type: var_type, location: node.location.name # steep:ignore NoMethod ) when :lvasgn var_name, rhs = node.children context = typing.cursor_context.context or raise type = context.type_env[var_name] || typing.type_of(node: node) return VariableContent.new( node: node, name: var_name, type: type, location: node.location.name # steep:ignore NoMethod ) when :send, :csend result_node = case parents[0]&.type when :block, :numblock if node == parents[0].children[0] parents[0] else node end else node end case call = typing.call_of(node: result_node) when TypeInference::MethodCall::Typed, TypeInference::MethodCall::Error unless call.method_decls.empty? return MethodCallContent.new( node: result_node, method_call: call, location: node.location.selector # steep:ignore NoMethod ) end end when :def, :defs context = typing.cursor_context.context or raise method_context = context.method_context if method_context && method_context.method if method_context.method_type return DefinitionContent.new( node: node, method_name: method_name_from_method(method_context, builder: context.factory.definition_builder), method_type: method_context.method_type, definition: method_context.method, location: node.loc.name # steep:ignore NoMethod ) end end when :const, :casgn context = typing.cursor_context.context or raise type = typing.type_of(node: node) const_name = typing.source_index.reference(constant_node: node) if const_name entry = context.env.constant_entry(const_name) or return return ConstantContent.new( location: node.location.name, # steep:ignore NoMethod full_name: const_name, type: type, decl: entry ) end when :assertion original_node, _ = node.children original_type = typing.type_of(node: original_node) asserted_type = typing.type_of(node: node) if original_type != asserted_type return TypeAssertionContent.new(node: node, original_type: original_type, asserted_type: asserted_type, location: node.location.expression) end end TypeContent.new( node: node, type: typing.type_of(node: node), location: node.location.expression ) end end end end end end