# typed: strict # frozen_string_literal: true module RubyLsp class GlobalState extend T::Sig sig { returns(String) } attr_reader :test_library sig { returns(String) } attr_accessor :formatter sig { returns(T::Boolean) } attr_reader :has_type_checker sig { returns(RubyIndexer::Index) } attr_reader :index sig { returns(Encoding) } attr_reader :encoding sig { returns(T::Boolean) } attr_reader :supports_watching_files sig { returns(TypeInferrer) } attr_reader :type_inferrer sig { void } def initialize @workspace_uri = T.let(URI::Generic.from_path(path: Dir.pwd), URI::Generic) @encoding = T.let(Encoding::UTF_8, Encoding) @formatter = T.let("auto", String) @linters = T.let([], T::Array[String]) @test_library = T.let("minitest", String) @has_type_checker = T.let(true, T::Boolean) @index = T.let(RubyIndexer::Index.new, RubyIndexer::Index) @type_inferrer = T.let(TypeInferrer.new(@index), TypeInferrer) @supported_formatters = T.let({}, T::Hash[String, Requests::Support::Formatter]) @supports_watching_files = T.let(false, T::Boolean) end sig { params(identifier: String, instance: Requests::Support::Formatter).void } def register_formatter(identifier, instance) @supported_formatters[identifier] = instance end sig { returns(T.nilable(Requests::Support::Formatter)) } def active_formatter @supported_formatters[@formatter] end sig { returns(T::Array[Requests::Support::Formatter]) } def active_linters @linters.filter_map { |name| @supported_formatters[name] } end sig { params(options: T::Hash[Symbol, T.untyped]).void } def apply_options(options) direct_dependencies = gather_direct_dependencies all_dependencies = gather_direct_and_indirect_dependencies workspace_uri = options.dig(:workspaceFolders, 0, :uri) @workspace_uri = URI(workspace_uri) if workspace_uri specified_formatter = options.dig(:initializationOptions, :formatter) @formatter = specified_formatter if specified_formatter @formatter = detect_formatter(direct_dependencies, all_dependencies) if @formatter == "auto" specified_linters = options.dig(:initializationOptions, :linters) @linters = specified_linters || detect_linters(direct_dependencies) @test_library = detect_test_library(direct_dependencies) @has_type_checker = detect_typechecker(direct_dependencies) encodings = options.dig(:capabilities, :general, :positionEncodings) @encoding = if !encodings || encodings.empty? Encoding::UTF_16LE elsif encodings.include?(Constant::PositionEncodingKind::UTF8) Encoding::UTF_8 elsif encodings.include?(Constant::PositionEncodingKind::UTF16) Encoding::UTF_16LE else Encoding::UTF_32 end file_watching_caps = options.dig(:capabilities, :workspace, :didChangeWatchedFiles) if file_watching_caps&.dig(:dynamicRegistration) && file_watching_caps&.dig(:relativePatternSupport) @supports_watching_files = true end end sig { returns(String) } def workspace_path T.must(@workspace_uri.to_standardized_path) end sig { returns(String) } def encoding_name case @encoding when Encoding::UTF_8 Constant::PositionEncodingKind::UTF8 when Encoding::UTF_16LE Constant::PositionEncodingKind::UTF16 else Constant::PositionEncodingKind::UTF32 end end private sig { params(direct_dependencies: T::Array[String], all_dependencies: T::Array[String]).returns(String) } def detect_formatter(direct_dependencies, all_dependencies) # NOTE: Intentionally no $ at end, since we want to match rubocop-shopify, etc. return "rubocop" if direct_dependencies.any?(/^rubocop/) syntax_tree_is_direct_dependency = direct_dependencies.include?("syntax_tree") return "syntax_tree" if syntax_tree_is_direct_dependency rubocop_is_transitive_dependency = all_dependencies.include?("rubocop") return "rubocop" if dot_rubocop_yml_present && rubocop_is_transitive_dependency "none" end # Try to detect if there are linters in the project's dependencies. For auto-detection, we always only consider a # single linter. To have multiple linters running, the user must configure them manually sig { params(dependencies: T::Array[String]).returns(T::Array[String]) } def detect_linters(dependencies) linters = [] linters << "rubocop" if dependencies.any?(/^rubocop/) linters end sig { params(dependencies: T::Array[String]).returns(String) } def detect_test_library(dependencies) if dependencies.any?(/^rspec/) "rspec" # A Rails app may have a dependency on minitest, but we would instead want to use the Rails test runner provided # by ruby-lsp-rails. A Rails app doesn't need to depend on the rails gem itself, individual components like # activestorage may be added to the gemfile so that other components aren't downloaded. Check for the presence # of bin/rails to support these cases. elsif bin_rails_present "rails" # NOTE: Intentionally ends with $ to avoid mis-matching minitest-reporters, etc. in a Rails app. elsif dependencies.any?(/^minitest$/) "minitest" elsif dependencies.any?(/^test-unit/) "test-unit" else "unknown" end end sig { params(dependencies: T::Array[String]).returns(T::Boolean) } def detect_typechecker(dependencies) return false if ENV["RUBY_LSP_BYPASS_TYPECHECKER"] # We can't read the env from within `Bundle.with_original_env` so we need to set it here. ruby_lsp_env_is_test = (ENV["RUBY_LSP_ENV"] == "test") Bundler.with_original_env do sorbet_static_detected = dependencies.any?(/^sorbet-static/) # Don't show message while running tests, since it's noisy if sorbet_static_detected && !ruby_lsp_env_is_test $stderr.puts("Ruby LSP detected this is a Sorbet project so will defer to Sorbet LSP for some functionality") end sorbet_static_detected end rescue Bundler::GemfileNotFound false end sig { returns(T::Boolean) } def bin_rails_present File.exist?(File.join(workspace_path, "bin/rails")) end sig { returns(T::Boolean) } def dot_rubocop_yml_present File.exist?(File.join(workspace_path, ".rubocop.yml")) end sig { returns(T::Array[String]) } def gather_direct_dependencies Bundler.with_original_env { Bundler.default_gemfile } Bundler.locked_gems.dependencies.keys + gemspec_dependencies rescue Bundler::GemfileNotFound [] end sig { returns(T::Array[String]) } def gemspec_dependencies Bundler.locked_gems.sources .grep(Bundler::Source::Gemspec) .flat_map { _1.gemspec&.dependencies&.map(&:name) } end sig { returns(T::Array[String]) } def gather_direct_and_indirect_dependencies Bundler.with_original_env { Bundler.default_gemfile } Bundler.locked_gems.specs.map(&:name) rescue Bundler::GemfileNotFound [] end end end