Skip to main content

script/
module_loading.rs

1/* This Source Code Form is subject to the terms of the Mozilla Public
2 * License, v. 2.0. If a copy of the MPL was not distributed with this
3 * file, You can obtain one at https://mozilla.org/MPL/2.0/. */
4
5//! An implementation of ecma262's [LoadRequestedModules](https://tc39.es/ecma262/#sec-LoadRequestedModules)
6//! Partly inspired by mozjs implementation: <https://searchfox.org/firefox-main/source/js/src/vm/Modules.cpp#1450>
7//! Since we can't access ModuleObject internals (eg. ModuleRequest records), we deviate from the spec in some aspects.
8
9#![expect(unsafe_code)]
10
11use std::cell::{Cell, RefCell};
12use std::collections::HashSet;
13use std::rc::Rc;
14
15use js::context::JSContext;
16use js::conversions::jsstr_to_string;
17use js::jsapi::{HandleValue as RawHandleValue, IsCyclicModule, ModuleType};
18use js::jsval::{ObjectValue, UndefinedValue};
19use js::realm::{AutoRealm, CurrentRealm};
20use js::rust::wrappers2::{
21    GetModuleNamespace, GetRequestedModuleSpecifier, GetRequestedModuleType,
22    GetRequestedModulesCount, JS_GetModulePrivate, ModuleEvaluate, ModuleLink,
23};
24use js::rust::{HandleValue, IntoHandle};
25use net_traits::request::{Destination, Referrer};
26use script_bindings::reflector::DomObject;
27use script_bindings::settings_stack::run_a_callback;
28use servo_url::ServoUrl;
29
30use crate::DomTypeHolder;
31use crate::dom::bindings::error::Error;
32use crate::dom::bindings::root::DomRoot;
33use crate::dom::globalscope::GlobalScope;
34use crate::dom::promise::Promise;
35use crate::dom::promisenativehandler::{Callback, PromiseNativeHandler};
36use crate::realms::enter_auto_realm;
37use crate::script_module::{
38    ModuleFetchClient, ModuleHandler, ModuleObject, ModuleTree, RethrowError, ScriptFetchOptions,
39    fetch_a_single_module_script, gen_type_error, module_script_from_reference_private,
40};
41use crate::script_runtime::IntroductionType;
42
43#[derive(JSTraceable, MallocSizeOf)]
44struct OnRejectedHandler {
45    #[conditional_malloc_size_of]
46    promise: Rc<Promise>,
47}
48
49impl Callback for OnRejectedHandler {
50    fn callback(&self, cx: &mut CurrentRealm, v: HandleValue) {
51        // a. Perform ! Call(promiseCapability.[[Reject]], undefined, « reason »).
52        self.promise.reject_with_cx(cx, v);
53    }
54}
55
56pub(crate) enum Payload {
57    GraphRecord(Rc<GraphLoadingState>),
58    PromiseRecord(Rc<Promise>),
59}
60
61#[derive(JSTraceable)]
62pub(crate) struct LoadState {
63    pub(crate) error_to_rethrow: RefCell<Option<RethrowError>>,
64    #[no_trace]
65    pub(crate) destination: Destination,
66    #[no_trace]
67    pub(crate) fetch_client: ModuleFetchClient,
68}
69
70/// <https://tc39.es/ecma262/#graphloadingstate-record>
71pub(crate) struct GraphLoadingState {
72    /// [[PromiseCapability]]
73    promise: Rc<Promise>,
74    /// [[IsLoading]]
75    is_loading: Cell<bool>,
76    /// [[PendingModulesCount]]
77    pending_modules_count: Cell<u32>,
78    /// [[Visited]]
79    visited: RefCell<HashSet<ServoUrl>>,
80    /// [[HostDefined]]
81    load_state: Option<Rc<LoadState>>,
82}
83
84/// <https://tc39.es/ecma262/#sec-LoadRequestedModules>
85pub(crate) fn load_requested_modules(
86    cx: &mut CurrentRealm,
87    module: Rc<ModuleTree>,
88    load_state: Option<Rc<LoadState>>,
89) -> Rc<Promise> {
90    // Step 1. If hostDefined is not present, let hostDefined be empty.
91    //
92    // Not required, since we implement it as an `Option`
93
94    // Step 2. Let pc be ! NewPromiseCapability(%Promise%).
95    let mut realm = CurrentRealm::assert(cx);
96    let promise = Promise::new_in_realm(&mut realm);
97
98    // Step 3. Let state be the GraphLoadingState Record
99    // { [[IsLoading]]: true, [[PendingModulesCount]]: 1, [[Visited]]: « », [[PromiseCapability]]: pc, [[HostDefined]]: hostDefined }.
100    let state = GraphLoadingState {
101        promise: promise.clone(),
102        is_loading: Cell::new(true),
103        pending_modules_count: Cell::new(1),
104        visited: RefCell::new(HashSet::new()),
105        load_state,
106    };
107
108    // Step 4. Perform InnerModuleLoading(state, module).
109    inner_module_loading(cx, &Rc::new(state), module);
110
111    // Step 5. Return pc.[[Promise]].
112    promise
113}
114
115/// <https://tc39.es/ecma262/#sec-InnerModuleLoading>
116fn inner_module_loading(
117    cx: &mut CurrentRealm,
118    state: &Rc<GraphLoadingState>,
119    module: Rc<ModuleTree>,
120) {
121    // Step 1. Assert: state.[[IsLoading]] is true.
122    assert!(state.is_loading.get());
123
124    let module_handle = module.get_record().map(|module| module.handle()).unwrap();
125
126    let module_url = module.get_url();
127    let visited_contains_module = state.visited.borrow().contains(&module_url);
128
129    // Step 2. If module is a Cyclic Module Record, module.[[Status]] is new, and state.[[Visited]] does not contain module, then
130    // Note: mozjs doesn't expose a way to check the ModuleStatus of a ModuleObject.
131    if unsafe { IsCyclicModule(module_handle.get()) } && !visited_contains_module {
132        // a. Append module to state.[[Visited]].
133        state.visited.borrow_mut().insert(module_url);
134
135        // b. Let requestedModulesCount be the number of elements in module.[[RequestedModules]].
136        let requested_modules_count = unsafe { GetRequestedModulesCount(cx, module_handle) };
137
138        // c. Set state.[[PendingModulesCount]] to state.[[PendingModulesCount]] + requestedModulesCount.
139        let pending_modules_count = state.pending_modules_count.get();
140        state
141            .pending_modules_count
142            .set(pending_modules_count + requested_modules_count);
143
144        // d. For each ModuleRequest Record request of module.[[RequestedModules]], do
145        for index in 0..requested_modules_count {
146            // i. If AllImportAttributesSupported(request.[[Attributes]]) is false, then
147            // Note: Gecko will call hasFirstUnsupportedAttributeKey on each module request,
148            // GetRequestedModuleSpecifier will do it for us.
149            // In addition it will also check if specifier has an unknown module type.
150            let jsstr = unsafe { GetRequestedModuleSpecifier(cx, module_handle, index) };
151
152            if jsstr.is_null() {
153                // 1. Let error be ThrowCompletion(a newly created SyntaxError object).
154                let error = RethrowError::from_pending_exception(cx);
155
156                // See Step 7. of `host_load_imported_module`.
157                state.load_state.as_ref().inspect(|load_state| {
158                    load_state
159                        .error_to_rethrow
160                        .borrow_mut()
161                        .get_or_insert(error.clone());
162                });
163
164                // 2. Perform ContinueModuleLoading(state, error).
165                continue_module_loading(cx, state, Err(error));
166            } else {
167                let specifier =
168                    unsafe { jsstr_to_string(cx, std::ptr::NonNull::new(jsstr).unwrap()) };
169                let module_type = unsafe { GetRequestedModuleType(cx, module_handle, index) };
170
171                let realm = CurrentRealm::assert(cx);
172                let global = GlobalScope::from_current_realm(&realm);
173
174                // ii. Else if module.[[LoadedModules]] contains a LoadedModuleRequest Record record
175                // such that ModuleRequestsEqual(record, request) is true, then
176                let loaded_module =
177                    module.find_descendant_inside_module_map(&global, &specifier, module_type);
178
179                match loaded_module {
180                    // 1. Perform InnerModuleLoading(state, record.[[Module]]).
181                    Some(module) => inner_module_loading(cx, state, module),
182                    // iii. Else,
183                    None => {
184                        rooted!(&in(cx) let mut referrer = UndefinedValue());
185                        unsafe { JS_GetModulePrivate(module_handle.get(), referrer.handle_mut()) };
186
187                        // 1. Perform HostLoadImportedModule(module, request, state.[[HostDefined]], state).
188                        host_load_imported_module(
189                            cx,
190                            Some(module.clone()),
191                            referrer.handle().into_handle(),
192                            specifier,
193                            module_type,
194                            state.load_state.clone(),
195                            Payload::GraphRecord(state.clone()),
196                        );
197                    },
198                }
199            }
200
201            // iv. If state.[[IsLoading]] is false, return unused.
202            if !state.is_loading.get() {
203                return;
204            }
205        }
206    }
207
208    // Step 3. Assert: state.[[PendingModulesCount]] ≥ 1.
209    assert!(state.pending_modules_count.get() >= 1);
210
211    // Step 4. Set state.[[PendingModulesCount]] to state.[[PendingModulesCount]] - 1.
212    let pending_modules_count = state.pending_modules_count.get();
213    state.pending_modules_count.set(pending_modules_count - 1);
214
215    // Step 5. If state.[[PendingModulesCount]] = 0, then
216    if state.pending_modules_count.get() == 0 {
217        // a. Set state.[[IsLoading]] to false.
218        state.is_loading.set(false);
219
220        // b. For each Cyclic Module Record loaded of state.[[Visited]], do
221        // i. If loaded.[[Status]] is new, set loaded.[[Status]] to unlinked.
222        // Note: mozjs defaults to the unlinked status.
223
224        // c. Perform ! Call(state.[[PromiseCapability]].[[Resolve]], undefined, « undefined »).
225        state.promise.resolve_native_with_cx(cx, &());
226    }
227
228    // Step 6. Return unused.
229}
230
231/// <https://tc39.es/ecma262/#sec-ContinueModuleLoading>
232fn continue_module_loading(
233    cx: &mut CurrentRealm,
234    state: &Rc<GraphLoadingState>,
235    module_completion: Result<Rc<ModuleTree>, RethrowError>,
236) {
237    // Step 1. If state.[[IsLoading]] is false, return unused.
238    if !state.is_loading.get() {
239        return;
240    }
241
242    match module_completion {
243        // Step 2. If moduleCompletion is a normal completion, then
244        // a. Perform InnerModuleLoading(state, moduleCompletion.[[Value]]).
245        Ok(module) => inner_module_loading(cx, state, module),
246
247        // Step 3. Else,
248        Err(exception) => {
249            // a. Set state.[[IsLoading]] to false.
250            state.is_loading.set(false);
251
252            // b. Perform ! Call(state.[[PromiseCapability]].[[Reject]], undefined, « moduleCompletion.[[Value]] »).
253            state.promise.reject_with_cx(cx, exception.handle());
254        },
255    }
256
257    // Step 4. Return unused.
258}
259
260/// <https://tc39.es/ecma262/#sec-FinishLoadingImportedModule>
261fn finish_loading_imported_module(
262    cx: &mut CurrentRealm,
263    referrer_module: Option<Rc<ModuleTree>>,
264    module_request_specifier: String,
265    payload: Payload,
266    result: Result<Rc<ModuleTree>, RethrowError>,
267) {
268    match payload {
269        // Step 2. If payload is a GraphLoadingState Record, then
270        Payload::GraphRecord(state) => {
271            let module_tree =
272                referrer_module.expect("Module must not be None in non dynamic imports");
273
274            // Step 1. If result is a normal completion, then
275            if let Ok(ref module) = result {
276                module_tree.insert_module_dependency(module, module_request_specifier);
277            }
278
279            // a. Perform ContinueModuleLoading(payload, result).
280            continue_module_loading(cx, &state, result);
281        },
282
283        // Step 3. Else,
284        // a. Perform ContinueDynamicImport(payload, result).
285        Payload::PromiseRecord(promise) => continue_dynamic_import(cx, promise, result),
286    }
287
288    // 4. Return unused.
289}
290
291/// <https://tc39.es/ecma262/#sec-ContinueDynamicImport>
292fn continue_dynamic_import(
293    cx: &mut CurrentRealm,
294    promise: Rc<Promise>,
295    module_completion: Result<Rc<ModuleTree>, RethrowError>,
296) {
297    // Step 1. If moduleCompletion is an abrupt completion, then
298    if let Err(exception) = module_completion {
299        // a. Perform ! Call(promiseCapability.[[Reject]], undefined, « moduleCompletion.[[Value]] »).
300        promise.reject_with_cx(cx, exception.handle());
301
302        // b. Return unused.
303        return;
304    }
305
306    let realm = CurrentRealm::assert(cx);
307    let global = GlobalScope::from_current_realm(&realm);
308
309    // Step 2. Let module be moduleCompletion.[[Value]].
310    let module = module_completion.unwrap();
311    let record = ModuleObject::new(module.get_record().map(|module| module.handle()).unwrap());
312
313    // Step 3. Let loadPromise be module.LoadRequestedModules().
314    let load_promise = load_requested_modules(cx, module, None);
315
316    // Step 4. Let rejectedClosure be a new Abstract Closure with parameters (reason)
317    // that captures promiseCapability and performs the following steps when called:
318    // Step 5. Let onRejected be CreateBuiltinFunction(rejectedClosure, 1, "", « »).
319    // Note: implemented by OnRejectedHandler.
320
321    let global_scope = global.clone();
322    let inner_promise = promise.clone();
323    let fulfilled_promise = promise.clone();
324
325    // Step 6. Let linkAndEvaluateClosure be a new Abstract Closure with no parameters that captures
326    // module, promiseCapability, and onRejected and performs the following steps when called:
327    // Step 7. Let linkAndEvaluate be CreateBuiltinFunction(linkAndEvaluateClosure, 0, "", « »).
328    let link_and_evaluate = ModuleHandler::new_boxed(Box::new(
329        task!(link_and_evaluate: |cx, global_scope: DomRoot<GlobalScope>, inner_promise: Rc<Promise>, record: ModuleObject| {
330            let mut realm = AutoRealm::new(
331                cx,
332                std::ptr::NonNull::new(global_scope.reflector().get_jsobject().get()).unwrap(),
333            );
334            let cx = &mut realm.current_realm();
335            // a. Let link be Completion(module.Link()).
336            let link = unsafe { ModuleLink(cx, record.handle()) };
337
338            // b. If link is an abrupt completion, then
339            if !link {
340                // i. Perform ! Call(promiseCapability.[[Reject]], undefined, « link.[[Value]] »).
341                let exception = RethrowError::from_pending_exception(cx);
342                inner_promise.reject_with_cx(cx, exception.handle());
343
344                // ii. Return NormalCompletion(undefined).
345                return;
346            }
347
348            rooted!(&in(cx) let mut rval = UndefinedValue());
349
350            // c. Let evaluatePromise be module.Evaluate().
351            assert!(unsafe { ModuleEvaluate(cx, record.handle(), rval.handle_mut()) });
352
353            if !rval.is_object() {
354                let error = RethrowError::from_pending_exception(cx);
355                return inner_promise.reject_with_cx(cx, error.handle());
356            }
357
358            rooted!(&in(cx) let evaluate_promise = rval.to_object());
359            let evaluate_promise = Promise::new_with_js_promise(evaluate_promise.handle(), cx.into());
360
361            // d. Let fulfilledClosure be a new Abstract Closure with no parameters that captures
362            // module and promiseCapability and performs the following steps when called:
363            // e. Let onFulfilled be CreateBuiltinFunction(fulfilledClosure, 0, "", « »).
364            let on_fulfilled = ModuleHandler::new_boxed(Box::new(
365                task!(on_fulfilled: |cx, fulfilled_promise: Rc<Promise>, record: ModuleObject| {
366
367                    // i. Let namespace be GetModuleNamespace(module).
368                    rooted!(&in(cx) let rval = unsafe { GetModuleNamespace(cx, record.handle()) });
369                    rooted!(&in(cx) let namespace = ObjectValue(rval.get()));
370
371                    // ii. Perform ! Call(promiseCapability.[[Resolve]], undefined, « namespace »).
372                    fulfilled_promise.resolve_with_cx(cx, namespace.handle());
373
374                    // iii. Return NormalCompletion(undefined).
375            })));
376
377            // f. Perform PerformPromiseThen(evaluatePromise, onFulfilled, onRejected).
378            let handler = PromiseNativeHandler::new(cx, &global_scope,
379                Some(on_fulfilled),
380                Some(Box::new(OnRejectedHandler { promise: inner_promise })));
381            evaluate_promise.append_native_handler(cx, &handler);
382
383            // g. Return unused.
384        }),
385    ));
386
387    let mut realm = enter_auto_realm(cx, &*global);
388    let cx = &mut realm.current_realm();
389    run_a_callback::<DomTypeHolder, _>(&*global, || {
390        // Step 8. Perform PerformPromiseThen(loadPromise, linkAndEvaluate, onRejected).
391        let handler = PromiseNativeHandler::new(
392            cx,
393            &global,
394            Some(link_and_evaluate),
395            Some(Box::new(OnRejectedHandler { promise })),
396        );
397        load_promise.append_native_handler(cx, &handler);
398    });
399    // Step 9. Return unused.
400}
401
402/// <https://html.spec.whatwg.org/multipage/#hostloadimportedmodule>
403pub(crate) fn host_load_imported_module(
404    cx: &mut CurrentRealm,
405    referrer_module: Option<Rc<ModuleTree>>,
406    referrer: RawHandleValue,
407    specifier: String,
408    module_type: ModuleType,
409    load_state: Option<Rc<LoadState>>,
410    payload: Payload,
411) {
412    // Step 1. Let settingsObject be the current settings object.
413    let realm = CurrentRealm::assert(cx);
414    let mut global_scope = GlobalScope::from_current_realm(&realm);
415
416    // TODO Step 2. If settingsObject's global object implements WorkletGlobalScope or ServiceWorkerGlobalScope and loadState is undefined, then:
417
418    // Step 3. Let referencingScript be null.
419    // Step 6.1. Set referencingScript to referrer.[[HostDefined]].
420    let referencing_script = unsafe { module_script_from_reference_private(&referrer) };
421
422    // Step 6. If referrer is a Script Record or a Cyclic Module Record, then:
423    let (original_fetch_options, fetch_referrer) = match referencing_script {
424        Some(module) => (
425            // Step 6.4. Set originalFetchOptions to referencingScript's fetch options.
426            module.options.clone(),
427            // Step 6.3. Set fetchReferrer to referencingScript's base URL.
428            Referrer::ReferrerUrl(module.base_url.clone()),
429        ),
430        None => (
431            // Step 4. Let originalFetchOptions be the default script fetch options.
432            ScriptFetchOptions::default_classic_script(),
433            // Step 5. Let fetchReferrer be "client".
434            global_scope.get_referrer(),
435        ),
436    };
437
438    // TODO: investigate providing a `ModuleOwner` to classic scripts.
439    let script_owner = referencing_script.and_then(|script| script.owner.clone());
440
441    // Step 6.2. Set settingsObject to referencingScript's settings object.
442    if let Some(ref owner) = script_owner {
443        global_scope = owner.root();
444    }
445
446    let global = &global_scope.clone();
447
448    // Step 7 If referrer is a Cyclic Module Record and moduleRequest is equal to the first element of referrer.[[RequestedModules]], then:
449    // Note: These substeps are implemented by `GetRequestedModuleSpecifier`,
450    // setting loadState.[[ErrorToRethrow]] is done by `inner_module_loading`.
451
452    // Step 8 Let url be the result of resolving a module specifier given referencingScript and moduleRequest.[[Specifier]],
453    // catching any exceptions. If they throw an exception, let resolutionError be the thrown exception.
454    let url =
455        ModuleTree::resolve_module_specifier(global, referencing_script, specifier.clone().into());
456
457    // Step 9 If the previous step threw an exception, then:
458    if let Err(error) = url {
459        let resolution_error = gen_type_error(cx, &global_scope, error);
460
461        // Step 9.1. If loadState is not undefined and loadState.[[ErrorToRethrow]] is null,
462        // set loadState.[[ErrorToRethrow]] to resolutionError.
463        load_state.as_ref().inspect(|load_state| {
464            load_state
465                .error_to_rethrow
466                .borrow_mut()
467                .get_or_insert(resolution_error.clone());
468        });
469
470        // Step 9.2. Perform FinishLoadingImportedModule(referrer, moduleRequest, payload, ThrowCompletion(resolutionError)).
471        finish_loading_imported_module(
472            cx,
473            referrer_module,
474            specifier,
475            payload,
476            Err(resolution_error),
477        );
478
479        // Step 9.3. Return.
480        return;
481    };
482
483    let url = url.unwrap();
484
485    // Step 10. Let fetchOptions be the result of getting the descendant script fetch options given
486    // originalFetchOptions, url, and settingsObject.
487    let fetch_options = original_fetch_options.descendant_fetch_options(&url, &global_scope);
488
489    // Step 13. If loadState is not undefined, then:
490    // Note: loadState is undefined only in dynamic imports
491    let (destination, fetch_client) = match load_state.as_ref() {
492        // Step 13.1. Set destination to loadState.[[Destination]].
493        // Step 13.2. Set fetchClient to loadState.[[FetchClient]].
494        Some(load_state) => (load_state.destination, load_state.fetch_client.clone()),
495        None => (
496            // Step 11. Let destination be "script".
497            Destination::Script,
498            // Step 12. Let fetchClient be settingsObject.
499            ModuleFetchClient::from_global_scope(&global_scope),
500        ),
501    };
502
503    let on_single_fetch_complete =
504        move |cx: &mut JSContext, module_tree: Option<Rc<ModuleTree>>| {
505            let mut realm = CurrentRealm::assert(cx);
506            let cx = &mut realm;
507
508            // Step 1. Let completion be null.
509            let completion = match module_tree {
510                // Step 2. If moduleScript is null, then set completion to ThrowCompletion(a new TypeError).
511                None => Err(gen_type_error(
512                    cx,
513                    &global_scope,
514                    Error::Type(c"Module fetching failed".to_owned()),
515                )),
516                Some(module_tree) => {
517                    // Step 3. Otherwise, if moduleScript's parse error is not null, then:
518                    // Step 3.1 Let parseError be moduleScript's parse error.
519                    if let Some(parse_error) = module_tree.get_parse_error() {
520                        // Step 3.3 If loadState is not undefined and loadState.[[ErrorToRethrow]] is null,
521                        // set loadState.[[ErrorToRethrow]] to parseError.
522                        load_state.as_ref().inspect(|load_state| {
523                            load_state
524                                .error_to_rethrow
525                                .borrow_mut()
526                                .get_or_insert(parse_error.clone());
527                        });
528
529                        // Step 3.2 Set completion to ThrowCompletion(parseError).
530                        Err(parse_error.clone())
531                    } else {
532                        // Step 4. Otherwise, set completion to NormalCompletion(moduleScript's record).
533                        Ok(module_tree)
534                    }
535                },
536            };
537
538            // Step 5. Perform FinishLoadingImportedModule(referrer, moduleRequest, payload, completion).
539            finish_loading_imported_module(cx, referrer_module, specifier, payload, completion);
540        };
541
542    // Step 14 Fetch a single imported module script given url, fetchClient, destination, fetchOptions, settingsObject,
543    // fetchReferrer, moduleRequest, and onSingleFetchComplete as defined below.
544    // If loadState is not undefined and loadState.[[PerformFetch]] is not null, pass loadState.[[PerformFetch]] along as well.
545    // Note: we don't have access to the requested `ModuleObject`, so we pass only its type.
546    fetch_a_single_imported_module_script(
547        cx,
548        url,
549        fetch_client,
550        global,
551        destination,
552        fetch_options,
553        fetch_referrer,
554        module_type,
555        on_single_fetch_complete,
556    );
557}
558
559/// <https://html.spec.whatwg.org/multipage/#fetch-a-single-imported-module-script>
560#[expect(clippy::too_many_arguments)]
561fn fetch_a_single_imported_module_script(
562    cx: &mut JSContext,
563    url: ServoUrl,
564    fetch_client: ModuleFetchClient,
565    global: &GlobalScope,
566    destination: Destination,
567    options: ScriptFetchOptions,
568    referrer: Referrer,
569    module_type: ModuleType,
570    on_complete: impl FnOnce(&mut JSContext, Option<Rc<ModuleTree>>) + 'static,
571) {
572    // TODO Step 1. Assert: moduleRequest.[[Attributes]] does not contain any Record entry such that entry.[[Key]] is not "type",
573    // because we only asked for "type" attributes in HostGetSupportedImportAttributes.
574
575    // TODO Step 2. Let moduleType be the result of running the module type from module request steps given moduleRequest.
576
577    // Step 3. If the result of running the module type allowed steps given moduleType and settingsObject is false,
578    // then run onComplete given null, and return.
579    match module_type {
580        ModuleType::Unknown => return on_complete(cx, None),
581        ModuleType::JavaScript | ModuleType::JSON => (),
582    }
583
584    // Step 4. Fetch a single module script given url, fetchClient, destination, options, settingsObject, referrer,
585    // moduleRequest, false, and onComplete. If performFetch was given, pass it along as well.
586    fetch_a_single_module_script(
587        cx,
588        url,
589        fetch_client,
590        global,
591        destination,
592        options,
593        referrer,
594        Some(module_type),
595        false,
596        Some(IntroductionType::IMPORTED_MODULE),
597        on_complete,
598    );
599}