Skip to content

Commit 2d5c93f

Browse files
louimEarlopain
andcommitted
Make Rails app detection based on Rails::Application superclass
This change makes the Rails app detection based on the superclass of the main class being `Rails::Application`. This is a more reliable way to detect Rails apps than looking for the presence of the `rails` gem in the Gemfile. Application can require some of the Rails components without requiring the `rails` gem itself. co-authored-by: Earlopain <[email protected]>
1 parent 4abb4e1 commit 2d5c93f

File tree

3 files changed

+91
-19
lines changed

3 files changed

+91
-19
lines changed

lib/ruby_lsp/listeners/rails_app.rb

+28
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
1+
# typed: strict
2+
# frozen_string_literal: true
3+
4+
module RubyLsp
5+
module Listeners
6+
class RailsApp
7+
extend T::Sig
8+
9+
sig { returns(T::Boolean) }
10+
attr_reader :rails_app
11+
12+
sig { params(dispatcher: Prism::Dispatcher).void }
13+
def initialize(dispatcher)
14+
@rails_app = T.let(false, T::Boolean)
15+
dispatcher.register(self, :on_class_node_enter)
16+
end
17+
18+
sig { params(node: Prism::ClassNode).void }
19+
def on_class_node_enter(node)
20+
superclass = node.superclass
21+
case superclass
22+
when Prism::ConstantPathNode
23+
@rails_app = true if superclass.full_name == "Rails::Application"
24+
end
25+
end
26+
end
27+
end
28+
end

lib/ruby_lsp/setup_bundler.rb

+29-4
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,8 @@
77
require "pathname"
88
require "digest"
99
require "time"
10+
require "prism"
11+
require "ruby_lsp/listeners/rails_app"
1012

1113
# This file is a script that will configure a custom bundle for the Ruby LSP. The custom bundle allows developers to use
1214
# the Ruby LSP without including the gem in their application's Gemfile while at the same time giving us access to the
@@ -50,6 +52,7 @@ def initialize(project_path, **options)
5052
@last_updated_path = T.let(@custom_dir + "last_updated", Pathname)
5153

5254
@dependencies = T.let(load_dependencies, T::Hash[String, T.untyped])
55+
@rails_app = T.let(rails_app?, T::Boolean)
5356
@retry = T.let(false, T::Boolean)
5457
end
5558

@@ -62,7 +65,7 @@ def setup!
6265
# Do not set up a custom bundle if LSP dependencies are already in the Gemfile
6366
if @dependencies["ruby-lsp"] &&
6467
@dependencies["debug"] &&
65-
(@dependencies["rails"] ? @dependencies["ruby-lsp-rails"] : true)
68+
(@rails_app ? @dependencies["ruby-lsp-rails"] : true)
6669
$stderr.puts(
6770
"Ruby LSP> Skipping custom bundle setup since LSP dependencies are already in #{@gemfile}",
6871
)
@@ -148,7 +151,7 @@ def write_custom_gemfile
148151
parts << 'gem "debug", require: false, group: :development, platforms: :mri'
149152
end
150153

151-
if @dependencies["rails"] && !@dependencies["ruby-lsp-rails"]
154+
if @rails_app && !@dependencies["ruby-lsp-rails"]
152155
parts << 'gem "ruby-lsp-rails", require: false, group: :development'
153156
end
154157

@@ -209,7 +212,7 @@ def run_bundle_install(bundle_gemfile = @gemfile)
209212
command << " && bundle update "
210213
command << "ruby-lsp " unless @dependencies["ruby-lsp"]
211214
command << "debug " unless @dependencies["debug"]
212-
command << "ruby-lsp-rails " if @dependencies["rails"] && !@dependencies["ruby-lsp-rails"]
215+
command << "ruby-lsp-rails " if @rails_app && !@dependencies["ruby-lsp-rails"]
213216
command << "--pre" if @experimental
214217
command.delete_suffix!(" ")
215218
command << ")"
@@ -244,7 +247,7 @@ def run_bundle_install(bundle_gemfile = @gemfile)
244247
def should_bundle_update?
245248
# If `ruby-lsp`, `ruby-lsp-rails` and `debug` are in the Gemfile, then we shouldn't try to upgrade them or else it
246249
# will produce version control changes
247-
if @dependencies["rails"]
250+
if @rails_app
248251
return false if @dependencies.values_at("ruby-lsp", "ruby-lsp-rails", "debug").all?
249252

250253
# If the custom lockfile doesn't include `ruby-lsp`, `ruby-lsp-rails` or `debug`, we need to run bundle install
@@ -280,5 +283,27 @@ def correct_relative_remote_paths
280283

281284
@custom_lockfile.write(content)
282285
end
286+
287+
# Detects if the project is a Rails app by looking if the superclass of the main class is `Rails::Application`
288+
sig { returns(T::Boolean) }
289+
def rails_app?
290+
config = rails_app_config
291+
return false unless config
292+
293+
result = Prism.parse(config)
294+
return false unless result
295+
296+
dispatcher = Prism::Dispatcher.new
297+
listener = Listeners::RailsApp.new(dispatcher)
298+
dispatcher.dispatch(result.value)
299+
300+
listener.rails_app
301+
end
302+
303+
sig { returns(T.nilable(String)) }
304+
def rails_app_config
305+
config = Pathname.new("config/application.rb").expand_path(Dir.pwd)
306+
config.read if config.exist?
307+
end
283308
end
284309
end

test/setup_bundler_test.rb

+34-15
Original file line numberDiff line numberDiff line change
@@ -75,22 +75,33 @@ def test_creates_custom_bundle
7575
end
7676

7777
def test_creates_custom_bundle_for_a_rails_app
78-
Object.any_instance.expects(:system).with(
79-
bundle_env(".ruby-lsp/Gemfile"),
80-
"(bundle check || bundle install) 1>&2",
81-
).returns(true)
82-
Bundler::LockfileParser.any_instance.expects(:dependencies).returns({ "rails" => true }).at_least_once
83-
run_script
78+
Dir.mktmpdir do |dir|
79+
Dir.chdir(dir) do
80+
FileUtils.mkdir(File.join(dir, "config"))
81+
File.write(File.join(dir, "config", "application.rb"), <<~RUBY)
82+
module MyApp
83+
class Application < Rails::Application
84+
end
85+
end
86+
RUBY
8487

85-
assert_path_exists(".ruby-lsp")
86-
assert_path_exists(".ruby-lsp/Gemfile")
87-
assert_path_exists(".ruby-lsp/Gemfile.lock")
88-
assert_path_exists(".ruby-lsp/main_lockfile_hash")
89-
assert_match("ruby-lsp", File.read(".ruby-lsp/Gemfile"))
90-
assert_match("debug", File.read(".ruby-lsp/Gemfile"))
91-
assert_match("ruby-lsp-rails", File.read(".ruby-lsp/Gemfile"))
92-
ensure
93-
FileUtils.rm_r(".ruby-lsp") if Dir.exist?(".ruby-lsp")
88+
Object.any_instance.expects(:system).with(
89+
bundle_env(".ruby-lsp/Gemfile"),
90+
"(bundle check || bundle install) 1>&2",
91+
).returns(true)
92+
Bundler::LockfileParser.any_instance.expects(:dependencies).returns({ "rails" => true }).at_least_once
93+
run_script
94+
95+
assert_path_exists(".ruby-lsp")
96+
assert_path_exists(".ruby-lsp/Gemfile")
97+
assert_path_exists(".ruby-lsp/Gemfile.lock")
98+
assert_path_exists(".ruby-lsp/main_lockfile_hash")
99+
gemfile_content = File.read(".ruby-lsp/Gemfile")
100+
assert_match("ruby-lsp", gemfile_content)
101+
assert_match("debug", gemfile_content)
102+
assert_match("ruby-lsp-rails", gemfile_content)
103+
end
104+
end
94105
end
95106

96107
def test_changing_lockfile_causes_custom_bundle_to_be_rebuilt
@@ -486,6 +497,14 @@ def test_ruby_lsp_rails_is_automatically_included_in_rails_apps
486497
gem "rails"
487498
GEMFILE
488499

500+
FileUtils.mkdir(File.join(dir, "config"))
501+
File.write(File.join(dir, "config", "application.rb"), <<~RUBY)
502+
module MyApp
503+
class Application < Rails::Application
504+
end
505+
end
506+
RUBY
507+
489508
capture_subprocess_io do
490509
Bundler.with_unbundled_env do
491510
# Run bundle install to generate the lockfile

0 commit comments

Comments
 (0)