|
1 | 1 | require "singleton"
|
2 | 2 | require "js"
|
3 |
| -require_relative "./url_resolver" |
| 3 | +require_relative "./require_remote/url_resolver" |
| 4 | +require_relative "./require_remote/response_handler" |
4 | 5 |
|
5 | 6 | module JS
|
6 |
| - ScriptLocation = Data.define(:url, :filename) |
7 |
| - |
8 | 7 | class RequireRemote
|
9 | 8 | include Singleton
|
10 | 9 |
|
11 | 10 | 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)) |
16 | 14 | end
|
17 | 15 |
|
18 | 16 | # Load the given feature from remote.
|
19 | 17 | # The use_maps parameter is not used.
|
20 | 18 | # This is a parameter to keep the method name unchanged when supporting require methods.
|
21 | 19 | # Maps like import maps will be used.
|
22 | 20 | def load(relative_feature, use_maps: false)
|
23 |
| - location = get_location(relative_feature) |
| 21 | + location = @resolver.get_location(relative_feature) |
24 | 22 |
|
25 | 23 | # 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) |
27 | 25 |
|
28 | 26 | 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) |
45 | 28 | end
|
46 | 29 |
|
47 | 30 | private
|
48 | 31 |
|
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 |
76 | 37 | end
|
77 |
| - |
78 |
| - attr_reader :resolver |
79 | 38 | end
|
80 | 39 | end
|
0 commit comments