Skip to content

Commit 9a66904

Browse files
committed
Refactor to extract ResponseHandler class
1 parent 418d282 commit 9a66904

File tree

4 files changed

+109
-79
lines changed

4 files changed

+109
-79
lines changed

ext/js/lib/js/require_remote.rb

+13-54
Original file line numberDiff line numberDiff line change
@@ -1,80 +1,39 @@
11
require "singleton"
22
require "js"
3-
require_relative "./url_resolver"
3+
require_relative "./require_remote/url_resolver"
4+
require_relative "./require_remote/response_handler"
45

56
module JS
6-
ScriptLocation = Data.define(:url, :filename)
7-
87
class RequireRemote
98
include Singleton
109

1110
def initialize
12-
set_base_url
13-
14-
# The loaded URLs are saved as Ruby strings.
15-
@loaded_urls = Set.new
11+
base_url = JS.global[:URL].new(JS.global[:location][:href])
12+
@resolver = URLResolver.new(base_url)
13+
@handler = ResponseHandler.new(method(:record_url))
1614
end
1715

1816
# Load the given feature from remote.
1917
# The use_maps parameter is not used.
2018
# This is a parameter to keep the method name unchanged when supporting require methods.
2119
# Maps like import maps will be used.
2220
def load(relative_feature, use_maps: false)
23-
location = get_location(relative_feature)
21+
location = @resolver.get_location(relative_feature)
2422

2523
# Do not load the same URL twice.
26-
return false if @loaded_urls.include?(location.url[:href].to_s)
24+
return false if @handler.evaluated?(location)
2725

2826
response = JS.global.fetch(location.url).await
29-
30-
if response[:status].to_i == 200
31-
# Check if the redirected URL has already loaded.
32-
return false if @loaded_urls.include?(response[:url].to_s)
33-
34-
code = response.text().await.to_s
35-
eval_code(code, location)
36-
37-
# The redirect that occurred may have been temporary.
38-
# The original URL is not recorded.
39-
# Only the URL after the redirect is recorded.
40-
@loaded_urls << response[:url].to_s
41-
true
42-
else
43-
raise LoadError.new "cannot load such url -- #{location.url}"
44-
end
27+
@handler.execute_and_record(response, location)
4528
end
4629

4730
private
4831

49-
def set_base_url
50-
base_url = JS.global[:URL].new(JS.global[:location][:href])
51-
@resolver = URLResolver.new(base_url)
52-
end
53-
54-
def get_location(relative_feature)
55-
filename = filename_from(relative_feature)
56-
url = resolver.resolve(filename)
57-
ScriptLocation.new(url, filename)
58-
end
59-
60-
# Evaluate the given Ruby code with the given location and save the URL to the stack.
61-
def eval_code(code, location)
62-
begin
63-
resolver.push(location.url)
64-
Kernel.eval(code, ::Object::TOPLEVEL_BINDING, location.filename)
65-
ensure
66-
resolver.pop
67-
end
68-
end
69-
70-
def filename_from(relative_feature)
71-
if relative_feature.end_with?(".rb")
72-
relative_feature
73-
else
74-
"#{relative_feature}.rb"
75-
end
32+
def record_url(url)
33+
@resolver.push(url)
34+
yield
35+
ensure
36+
@resolver.pop
7637
end
77-
78-
attr_reader :resolver
7938
end
8039
end
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,51 @@
1+
module JS
2+
class RequireRemote
3+
# Execute the body of the response and record the URL.
4+
class ResponseHandler
5+
# The around_eval parameter is a block that is executed before and after the body of the response.
6+
# The block is passed the URL of the response.
7+
def initialize(around_eval)
8+
@around_eval = around_eval
9+
10+
# The evaluated URLs are saved as Ruby strings.
11+
@evaluated_urls = Set.new
12+
end
13+
14+
# Execute the body of the response and record the URL.
15+
def execute_and_record(response, location)
16+
if response[:status].to_i == 200
17+
# Check if the redirected URL has already evaluated.
18+
return false if url_evaluated?(response[:url].to_s)
19+
20+
code = response.text().await.to_s
21+
evaluate(code, location)
22+
23+
# The redirect that occurred may have been temporary.
24+
# The original URL is not recorded.
25+
# Only the URL after the redirect is recorded.
26+
@evaluated_urls << response[:url].to_s
27+
true
28+
else
29+
raise LoadError.new "cannot load such url -- #{location.url}"
30+
end
31+
end
32+
33+
def evaluated?(location)
34+
url_evaluated?(location.url[:href].to_s)
35+
end
36+
37+
private
38+
39+
def url_evaluated?(url)
40+
@evaluated_urls.include?(url)
41+
end
42+
43+
# Evaluate the given Ruby code with the given location and save the URL to the stack.
44+
def evaluate(code, location)
45+
@around_eval.call(location.url) do
46+
Kernel.eval(code, ::Object::TOPLEVEL_BINDING, location.filename)
47+
end
48+
end
49+
end
50+
end
51+
end
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,45 @@
1+
module JS
2+
class RequireRemote
3+
ScriptLocation = Data.define(:url, :filename)
4+
5+
# When require_relative is called within a running Ruby script,
6+
# the URL is resolved from a relative file path based on the URL of the running Ruby script.
7+
# It uses a stack to store URLs of running Ruby Script.
8+
# Push the URL onto the stack before executing the new script.
9+
# Then pop it when the script has finished executing.
10+
class URLResolver
11+
def initialize(base_url)
12+
@url_stack = [base_url]
13+
end
14+
15+
def get_location(relative_feature)
16+
filename = filename_from(relative_feature)
17+
url = resolve(filename)
18+
ScriptLocation.new(url, filename)
19+
end
20+
21+
def push(url)
22+
@url_stack.push url
23+
end
24+
25+
def pop()
26+
@url_stack.pop
27+
end
28+
29+
private
30+
31+
def filename_from(relative_feature)
32+
if relative_feature.end_with?(".rb")
33+
relative_feature
34+
else
35+
"#{relative_feature}.rb"
36+
end
37+
end
38+
39+
# Return a URL object of JavaScript.
40+
def resolve(relative_filepath)
41+
JS.global[:URL].new relative_filepath, @url_stack.last
42+
end
43+
end
44+
end
45+
end

ext/js/lib/js/url_resolver.rb

-25
This file was deleted.

0 commit comments

Comments
 (0)