Skip to content

Commit

Permalink
Handle unloading of deleted stimulus controllers
Browse files Browse the repository at this point in the history
  • Loading branch information
jorgemanrubia committed Dec 19, 2024
1 parent 1423b29 commit 5703b6c
Show file tree
Hide file tree
Showing 7 changed files with 96 additions and 15 deletions.
28 changes: 25 additions & 3 deletions app/assets/javascripts/hotwire_spark.js
Original file line number Diff line number Diff line change
Expand Up @@ -3538,13 +3538,18 @@ var HotwireSpark = (function () {
log("Reload Stimulus controllers...");
this.application.stop();
await this.#reloadStimulusControllers();
this.#unloadDeletedStimulusControllers();
this.application.start();
}
async #reloadStimulusControllers() {
await Promise.all(this.#stimulusControllerPaths.map(async moduleName => this.#reloadStimulusController(moduleName)));
await Promise.all(this.#stimulusControllerPathsToReload.map(async moduleName => this.#reloadStimulusController(moduleName)));
}
get #stimulusControllerPathsToReload() {
return this.#stimulusControllerPaths.filter(path => this.#shouldReloadController(path));
}
get #stimulusControllerPaths() {
return Object.keys(this.#stimulusPathsByModule).filter(path => path.endsWith("_controller") && this.#shouldReloadController(path));
this.controllerPaths = this.controllerPaths || Object.keys(this.#stimulusPathsByModule).filter(path => path.endsWith("_controller"));
return this.controllerPaths;
}
#shouldReloadController(path) {
return this.filePattern.test(path);
Expand All @@ -3564,16 +3569,33 @@ var HotwireSpark = (function () {
const module = await import(path);
this.#registerController(controllerName, module);
}
#unloadDeletedStimulusControllers() {
this.#controllersToUnload.forEach(controller => this.#deregisterController(controller.identifier));
}
get #controllersToUnload() {
if (this.#didChangeTriggerAReload) {
return [];
} else {
return this.application.controllers.filter(controller => this.filePattern.test(`${controller.identifier}_controller`));
}
}
get #didChangeTriggerAReload() {
return this.#stimulusControllerPaths.find(path => this.#shouldReloadController(path));
}
#pathForModuleName(moduleName) {
return this.#stimulusPathsByModule[moduleName];
}
#extractControllerName(path) {
return path.replace(/^.*\//, "").replace("_controller", "").replace(/\//g, "--").replace(/_/g, "-");
}
#registerController(name, module) {
this.application.unload(name);
this.#deregisterController(name);
this.application.register(name, module.default);
}
#deregisterController(name) {
log(`\tRemoving controller ${name}`);
this.application.unload(name);
}
}

class HtmlReloader {
Expand Down
2 changes: 1 addition & 1 deletion app/assets/javascripts/hotwire_spark.min.js

Large diffs are not rendered by default.

2 changes: 1 addition & 1 deletion app/assets/javascripts/hotwire_spark.min.js.map

Large diffs are not rendered by default.

35 changes: 32 additions & 3 deletions app/javascript/hotwire/spark/reloaders/stimulus_reloader.js
Original file line number Diff line number Diff line change
Expand Up @@ -18,18 +18,26 @@ export class StimulusReloader {
log("Reload Stimulus controllers...")

this.application.stop()

await this.#reloadStimulusControllers()
this.#unloadDeletedStimulusControllers()

this.application.start()
}

async #reloadStimulusControllers() {
await Promise.all(
this.#stimulusControllerPaths.map(async moduleName => this.#reloadStimulusController(moduleName))
this.#stimulusControllerPathsToReload.map(async moduleName => this.#reloadStimulusController(moduleName))
)
}

get #stimulusControllerPathsToReload() {
return this.#stimulusControllerPaths.filter(path => this.#shouldReloadController(path))
}

get #stimulusControllerPaths() {
return Object.keys(this.#stimulusPathsByModule).filter(path => path.endsWith("_controller") && this.#shouldReloadController(path))
this.controllerPaths = this.controllerPaths || Object.keys(this.#stimulusPathsByModule).filter(path => path.endsWith("_controller"))
return this.controllerPaths
}

#shouldReloadController(path) {
Expand Down Expand Up @@ -57,6 +65,22 @@ export class StimulusReloader {
this.#registerController(controllerName, module)
}

#unloadDeletedStimulusControllers() {
this.#controllersToUnload.forEach(controller => this.#deregisterController(controller.identifier))
}

get #controllersToUnload() {
if (this.#didChangeTriggerAReload) {
return []
} else {
return this.application.controllers.filter(controller => this.filePattern.test(`${controller.identifier}_controller`))
}
}

get #didChangeTriggerAReload() {
return this.#stimulusControllerPaths.find(path => this.#shouldReloadController(path))
}

#pathForModuleName(moduleName) {
return this.#stimulusPathsByModule[moduleName]
}
Expand All @@ -70,7 +94,12 @@ export class StimulusReloader {
}

#registerController(name, module) {
this.application.unload(name)
this.#deregisterController(name)
this.application.register(name, module.default)
}

#deregisterController(name) {
log(`\tRemoving controller ${name}`)
this.application.unload(name)
}
}
2 changes: 2 additions & 0 deletions test/dummy/app/javascript/controllers/dummy_controller.js
Original file line number Diff line number Diff line change
Expand Up @@ -8,9 +8,11 @@ export default class extends Controller {
connect() {
console.debug("Dummy controller connected ", this.version)
this.element.querySelector("#replace").textContent = "_REPLACE_"
this.element.setAttribute("data-dummy-version", this.version)
}

disconnect() {
console.debug("Dummy controller disconnected", this.version)
this.element.removeAttribute("data-dummy-version")
}
}
28 changes: 23 additions & 5 deletions test/helpers/files_helper.rb
Original file line number Diff line number Diff line change
Expand Up @@ -11,13 +11,13 @@ module FilesHelper
ORIGINAL_EXTENSION = ".original"

def edit_file(path, replace:, with:)
path = Rails.application.root.join(path).to_s
path = expand_path(path)

raise ArgumentError, "File at '#{path}' does not exist." unless File.exist?(path)

original_path = "#{path}#{ORIGINAL_EXTENSION}"
remember_path_to_restore original_path
system "cp", path, original_path
original_path = remember_original_path_to_restore path

FileUtils.cp path, original_path

content = File.read(path)
updated_content = content.gsub(replace, with)
Expand All @@ -27,7 +27,7 @@ def edit_file(path, replace:, with:)
end

def add_file(path, content)
path = Rails.application.root.join(path).to_s
path = expand_path(path)

raise ArgumentError, "File at '#{path}' already exists." if File.exist?(path)

Expand All @@ -37,7 +37,25 @@ def add_file(path, content)
reload_rails_reloader
end

def remove_file(path)
path = expand_path(path)

original_path = remember_original_path_to_restore path

FileUtils.mv path, original_path
end

private
def expand_path(path)
Rails.application.root.join(path).to_s
end

def remember_original_path_to_restore(path)
"#{path}#{ORIGINAL_EXTENSION}".tap do |original_path|
remember_path_to_restore original_path
end
end

def remember_path_to_restore(path)
paths_to_restore << path
end
Expand Down
14 changes: 12 additions & 2 deletions test/stimulus_reload_test.rb
Original file line number Diff line number Diff line change
@@ -1,8 +1,11 @@
require "application_system_test_case"

class StimulusReloadTest < ApplicationSystemTestCase
test "reload Stimulus controller changes" do
setup do
visit root_path
end

test "reload Stimulus controller changes" do
assert_no_text "This was replaced!"

edit_file "app/javascript/controllers/dummy_controller.js", replace: "_REPLACE_", with: "This was replaced!"
Expand All @@ -11,7 +14,6 @@ class StimulusReloadTest < ApplicationSystemTestCase
end

test "load new Stimulus controllers" do
visit root_path
assert_no_text "This was replaced!"

edit_file "app/views/home/show.html.erb", replace: "_REPLACE_CONTROLLER_", with: "other-dummy"
Expand All @@ -29,4 +31,12 @@ class StimulusReloadTest < ApplicationSystemTestCase

assert_text "This was replaced!"
end

test "unload removed Stimulus controllers" do
assert_css "[data-dummy-version]"

remove_file "app/javascript/controllers/dummy_controller.js"

assert_no_css "[data-dummy-version]"
end
end

0 comments on commit 5703b6c

Please sign in to comment.