# typed: strict # frozen_string_literal: true module RubyLsp module Listeners class Hover extend T::Sig include Requests::Support::Common ALLOWED_TARGETS = T.let( [ Prism::CallNode, Prism::ConstantReadNode, Prism::ConstantWriteNode, Prism::ConstantPathNode, Prism::InstanceVariableReadNode, Prism::InstanceVariableAndWriteNode, Prism::InstanceVariableOperatorWriteNode, Prism::InstanceVariableOrWriteNode, Prism::InstanceVariableTargetNode, Prism::InstanceVariableWriteNode, Prism::SymbolNode, Prism::StringNode, ], T::Array[T.class_of(Prism::Node)], ) ALLOWED_REMOTE_PROVIDERS = T.let( [ "https://github.com", "https://gitlab.com", ].freeze, T::Array[String], ) sig do params( response_builder: ResponseBuilders::Hover, global_state: GlobalState, uri: URI::Generic, node_context: NodeContext, dispatcher: Prism::Dispatcher, typechecker_enabled: T::Boolean, ).void end def initialize(response_builder, global_state, uri, node_context, dispatcher, typechecker_enabled) # rubocop:disable Metrics/ParameterLists @response_builder = response_builder @global_state = global_state @index = T.let(global_state.index, RubyIndexer::Index) @path = T.let(uri.to_standardized_path, T.nilable(String)) @node_context = node_context @typechecker_enabled = typechecker_enabled dispatcher.register( self, :on_constant_read_node_enter, :on_constant_write_node_enter, :on_constant_path_node_enter, :on_call_node_enter, :on_instance_variable_read_node_enter, :on_instance_variable_write_node_enter, :on_instance_variable_and_write_node_enter, :on_instance_variable_operator_write_node_enter, :on_instance_variable_or_write_node_enter, :on_instance_variable_target_node_enter, ) end sig { params(node: Prism::ConstantReadNode).void } def on_constant_read_node_enter(node) return if @typechecker_enabled name = constant_name(node) return if name.nil? generate_hover(name, node.location) end sig { params(node: Prism::ConstantWriteNode).void } def on_constant_write_node_enter(node) return if @global_state.typechecker generate_hover(node.name.to_s, node.name_loc) end sig { params(node: Prism::ConstantPathNode).void } def on_constant_path_node_enter(node) return if @global_state.typechecker name = constant_name(node) return if name.nil? generate_hover(name, node.location) end sig { params(node: Prism::CallNode).void } def on_call_node_enter(node) return unless self_receiver?(node) if @path && File.basename(@path) == GEMFILE_NAME && node.name == :gem generate_gem_hover(node) return end return if @typechecker_enabled message = node.message return unless message methods = @index.resolve_method(message, @node_context.fully_qualified_name) return unless methods categorized_markdown_from_index_entries(message, methods).each do |category, content| @response_builder.push(content, category: category) end end sig { params(node: Prism::InstanceVariableReadNode).void } def on_instance_variable_read_node_enter(node) handle_instance_variable_hover(node.name.to_s) end sig { params(node: Prism::InstanceVariableWriteNode).void } def on_instance_variable_write_node_enter(node) handle_instance_variable_hover(node.name.to_s) end sig { params(node: Prism::InstanceVariableAndWriteNode).void } def on_instance_variable_and_write_node_enter(node) handle_instance_variable_hover(node.name.to_s) end sig { params(node: Prism::InstanceVariableOperatorWriteNode).void } def on_instance_variable_operator_write_node_enter(node) handle_instance_variable_hover(node.name.to_s) end sig { params(node: Prism::InstanceVariableOrWriteNode).void } def on_instance_variable_or_write_node_enter(node) handle_instance_variable_hover(node.name.to_s) end sig { params(node: Prism::InstanceVariableTargetNode).void } def on_instance_variable_target_node_enter(node) handle_instance_variable_hover(node.name.to_s) end private sig { params(name: String).void } def handle_instance_variable_hover(name) entries = @index.resolve_instance_variable(name, @node_context.fully_qualified_name) return unless entries categorized_markdown_from_index_entries(name, entries).each do |category, content| @response_builder.push(content, category: category) end rescue RubyIndexer::Index::NonExistingNamespaceError # If by any chance we haven't indexed the owner, then there's no way to find the right declaration end sig { params(name: String, location: Prism::Location).void } def generate_hover(name, location) entries = @index.resolve(name, @node_context.nesting) return unless entries # We should only show hover for private constants if the constant is defined in the same namespace as the # reference first_entry = T.must(entries.first) return if first_entry.private? && first_entry.name != "#{@node_context.fully_qualified_name}::#{name}" categorized_markdown_from_index_entries(name, entries).each do |category, content| @response_builder.push(content, category: category) end end sig { params(node: Prism::CallNode).void } def generate_gem_hover(node) first_argument = node.arguments&.arguments&.first return unless first_argument.is_a?(Prism::StringNode) spec = Gem::Specification.find_by_name(first_argument.content) return unless spec info = T.let( [ spec.description, spec.summary, "This rubygem does not have a description or summary.", ].find { |text| !text.nil? && !text.empty? }, String, ) # Remove leading whitespace if a heredoc was used for the summary or description info = info.gsub(/^ +/, "") remote_url = [spec.homepage, spec.metadata["source_code_uri"]].compact.find do |page| page.start_with?(*ALLOWED_REMOTE_PROVIDERS) end @response_builder.push( "**#{spec.name}** (#{spec.version}) #{remote_url && " - [open remote](#{remote_url})"}", category: :title, ) @response_builder.push(info, category: :documentation) rescue Gem::MissingSpecError # Do nothing if the spec cannot be found end end end end