Skip to content

Commit bfee5d8

Browse files
Set up reloading after_initialize
Alternate fix for #336. We went with #347 instead because the solution in this commit didn't work with Rails 4.2. Since we are no longer supporting Rails 4.2, I think this is a better approach. The original problem looked like this: Rails runs all of the initializers 1. Run the ["factory_bot.set_factory_paths"][set_factory_paths] initializer 2. Run the ["factory_bot.register_reloader"][register_reloader] initializer, which sets up a [prepare callback][] 3. Run the [`:run_prepare_callbacks`][] initializer 4. This triggers the factory_bot [prepare callback][], which causes factory\_bot to [reload][] Rails runs `after_initialize` callbacks 1. [I18n initializes] 2. factory\_bot [reloads again][] as described in #334 Triggering the first factory_bot reload before initializing I18n could cause an error in some cases. We avoided the problem in #347 by adding a conditional to skip reloading factory_bot before the application has initialized. This commit, on the other hand, moves factory_bot reloading from a prepare callback into an `after_initialize` callback. The initialization process is now simplified to: Rails runs all of the initializers 1. Run the ["factory_bot.set_factory_paths"][set_factory_paths] 2. Run the [`:run_prepare_callbacks`][] initializer, which no longer involves factory_bot Rails runs `after_intialize` callbacks 1. [I18n initializes] 2. factory_bot loads definitions for the first time. It then runs the reloader to set up the prepare callback to reload factory_bot when the application reloads (for example by calling `reload!` in the console), and to register the reloader to trigger reloads when any factory_bot definition files change. [set_factory_paths]: https://github.com/thoughtbot/factory_bot_rails/blob/3815aae2b9e4a5c5c3027a2ee8851f44b7c6a4da/lib/factory_bot_rails/railtie.rb#L17-L19 [register_reloader]: https://github.com/thoughtbot/factory_bot_rails/blob/3815aae2b9e4a5c5c3027a2ee8851f44b7c6a4da/lib/factory_bot_rails/railtie.rb#L21-L23 [prepare callback]: https://github.com/thoughtbot/factory_bot_rails/blob/master/lib/factory_bot_rails/reloader.rb#L34-L36 [`:run_prepare_callbacks`]: https://github.com/rails/rails/blob/5-2-stable/railties/lib/rails/application/finisher.rb#L62-L64 [reload]: https://github.com/thoughtbot/factory_bot_rails/blob/master/lib/factory_bot_rails/reloader.rb#L24-L26 [I18n initializes]: https://github.com/rails/rails/blob/13e2102517fafc8f8736fce5d57de901067202d0/activesupport/lib/active_support/i18n_railtie.rb#L16-L20 [reloads again]: https://github.com/thoughtbot/factory_bot_rails/blob/master/lib/factory_bot_rails/railtie.rb#L25-L27
1 parent 870bf2a commit bfee5d8

File tree

3 files changed

+53
-29
lines changed

3 files changed

+53
-29
lines changed

lib/factory_bot_rails/railtie.rb

+3-6
Original file line numberDiff line numberDiff line change
@@ -18,12 +18,9 @@ class Railtie < Rails::Railtie
1818
FactoryBot.definition_file_paths = definition_file_paths
1919
end
2020

21-
initializer "factory_bot.register_reloader" do |app|
22-
Reloader.new(app, config).run
23-
end
24-
25-
config.after_initialize do
26-
FactoryBot.reload
21+
config.after_initialize do |app|
22+
FactoryBot.find_definitions
23+
Reloader.new(app).run
2724
end
2825

2926
private

lib/factory_bot_rails/reloader.rb

+4-9
Original file line numberDiff line numberDiff line change
@@ -4,9 +4,8 @@
44

55
module FactoryBotRails
66
class Reloader
7-
def initialize(app, config)
7+
def initialize(app)
88
@app = app
9-
@config = config
109
@paths = DefinitionFilePaths.new(FactoryBot.definition_file_paths)
1110
end
1211

@@ -18,7 +17,7 @@ def run
1817

1918
private
2019

21-
attr_reader :app, :config
20+
attr_reader :app
2221

2322
def build_reloader
2423
reloader_class.new(@paths.files, @paths.directories) do
@@ -31,12 +30,8 @@ def reloader_class
3130
end
3231

3332
def register_reloader(reloader)
34-
closed_over_app = app
35-
36-
config.to_prepare do
37-
if closed_over_app.initialized?
38-
reloader.execute
39-
end
33+
app.reloader.to_prepare do
34+
reloader.execute
4035
end
4136

4237
app.reloaders << reloader

spec/factory_bot_rails/reloader_spec.rb

+46-14
Original file line numberDiff line numberDiff line change
@@ -12,42 +12,74 @@
1212

1313
context "when a definition file paths exist" do
1414
it "registers a reloader" do
15-
allow(reloader_class).to receive(:new)
15+
file_watcher = file_watcher_double
1616

17-
run_reloader(["spec/fixtures/factories", "not_exist_directory"])
17+
run_reloader(
18+
["spec/fixtures/factories", "not_exist_directory"],
19+
file_watcher
20+
)
1821

19-
expect(reloader_class).to have_received(:new)
22+
expect(file_watcher).to have_received(:new)
2023
end
2124
end
2225

2326
context "when a file exists but not a directory" do
2427
it "registers a reloader" do
25-
allow(reloader_class).to receive(:new)
28+
file_watcher = file_watcher_double
2629

27-
run_reloader(["spec/fake_app", "not_exist_directory"])
30+
run_reloader(
31+
["spec/fake_app", "not_exist_directory"],
32+
file_watcher
33+
)
2834

29-
expect(reloader_class).to have_received(:new)
35+
expect(file_watcher).to have_received(:new)
3036
end
3137
end
3238

3339
context "when a definition file paths NOT exist" do
3440
it "does NOT register a reloader" do
35-
allow(reloader_class).to receive(:new)
41+
file_watcher = file_watcher_double
3642

37-
run_reloader(["not_exist_directory"])
43+
run_reloader(["not_exist_directory"], file_watcher)
3844

39-
expect(reloader_class).not_to have_received(:new)
45+
expect(file_watcher).not_to have_received(:new)
4046
end
4147
end
4248

43-
def run_reloader(definition_file_paths)
49+
def run_reloader(definition_file_paths, file_watcher)
4450
FactoryBot.definition_file_paths = definition_file_paths
45-
FactoryBotRails::Reloader
46-
.new(Rails.application, Rails.application.config).run
51+
app = app_double(file_watcher)
52+
FactoryBotRails::Reloader.new(app).run
4753
end
4854

49-
def reloader_class
50-
Rails.application.config.file_watcher
55+
def file_watcher_double
56+
class_double(
57+
Rails.application.config.file_watcher,
58+
new: double(:reloader, execute: nil)
59+
)
60+
end
61+
62+
def app_double(file_watcher)
63+
instance_double(
64+
Rails.application.class,
65+
config: app_config_double(file_watcher),
66+
reloader: reloader_double,
67+
reloaders: []
68+
)
69+
end
70+
71+
def app_config_double(file_watcher)
72+
instance_double(
73+
Rails.application.config.class,
74+
file_watcher: file_watcher
75+
)
76+
end
77+
78+
def reloader_double
79+
class_double(
80+
Rails.application.reloader,
81+
to_prepare: nil
82+
)
5183
end
5284
end
5385
end

0 commit comments

Comments
 (0)