Skip to main content

script/dom/abort/
abortsignal.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
5use std::cell::{Cell, RefCell};
6use std::rc::Rc;
7use std::sync::{Arc, Mutex};
8
9use dom_struct::dom_struct;
10use indexmap::IndexSet;
11use js::context::JSContext;
12use js::jsapi::{ExceptionStackBehavior, Heap};
13use js::jsval::{JSVal, UndefinedValue};
14use js::realm::CurrentRealm;
15use js::rust::wrappers2::JS_SetPendingException;
16use js::rust::{HandleObject, HandleValue, MutableHandleValue};
17use script_bindings::cell::DomRefCell;
18use script_bindings::inheritance::Castable;
19use script_bindings::reflector::reflect_dom_object_with_proto_and_cx;
20use script_bindings::weakref::WeakRef;
21
22use crate::dom::bindings::codegen::Bindings::AbortSignalBinding::AbortSignalMethods;
23use crate::dom::bindings::codegen::Bindings::EventListenerBinding::EventListener;
24use crate::dom::bindings::codegen::Bindings::EventTargetBinding::EventListenerOptions;
25use crate::dom::bindings::error::{Error, ErrorToJsval, Fallible};
26use crate::dom::bindings::refcounted::Trusted;
27use crate::dom::bindings::reflector::DomGlobal;
28use crate::dom::bindings::root::{Dom, DomRoot};
29use crate::dom::bindings::str::DOMString;
30use crate::dom::eventtarget::EventTarget;
31use crate::dom::globalscope::GlobalScope;
32use crate::dom::readablestream::PipeTo;
33use crate::fetch::{DeferredFetchRecordId, FetchContext};
34use crate::realms::enter_auto_realm;
35use crate::script_runtime::JSContext as SafeJSContext;
36
37impl js::gc::Rootable for AbortAlgorithm {}
38
39/// <https://dom.spec.whatwg.org/#abortcontroller-api-integration>
40/// TODO: implement algorithms at call point,
41/// in order to integrate the abort signal with its various use cases.
42#[derive(Clone, JSTraceable, MallocSizeOf)]
43#[cfg_attr(crown, crown::unrooted_must_root_lint::must_root)]
44pub(crate) enum AbortAlgorithm {
45    /// <https://dom.spec.whatwg.org/#add-an-event-listener>
46    DomEventListener(RemovableDomEventListener),
47    /// <https://streams.spec.whatwg.org/#readable-stream-pipe-to>
48    StreamPiping(PipeTo),
49    /// <https://fetch.spec.whatwg.org/#dom-global-fetch>
50    Fetch(
51        #[no_trace]
52        #[conditional_malloc_size_of]
53        Arc<Mutex<Option<FetchContext>>>,
54    ),
55    /// <https://fetch.spec.whatwg.org/#dom-window-fetchlater>
56    FetchLater(#[no_trace] DeferredFetchRecordId),
57}
58
59#[derive(Clone, JSTraceable, MallocSizeOf)]
60#[cfg_attr(crown, crown::unrooted_must_root_lint::must_root)]
61pub(crate) struct RemovableDomEventListener {
62    pub(crate) event_target: Dom<EventTarget>,
63    pub(crate) ty: DOMString,
64    #[conditional_malloc_size_of]
65    pub(crate) listener: Option<Rc<EventListener>>,
66    pub(crate) options: EventListenerOptions,
67}
68
69/// <https://dom.spec.whatwg.org/#abortsignal>
70#[dom_struct]
71pub(crate) struct AbortSignal {
72    eventtarget: EventTarget,
73
74    /// <https://dom.spec.whatwg.org/#abortsignal-abort-reason>
75    #[ignore_malloc_size_of = "mozjs"]
76    abort_reason: Heap<JSVal>,
77
78    /// <https://dom.spec.whatwg.org/#abortsignal-abort-algorithms>
79    abort_algorithms: RefCell<Vec<AbortAlgorithm>>,
80
81    /// <https://dom.spec.whatwg.org/#abortsignal-dependent>
82    dependent: Cell<bool>,
83
84    /// <https://dom.spec.whatwg.org/#abortsignal-source-signals>
85    #[no_trace]
86    #[ignore_malloc_size_of = "WeakRef"]
87    source_signals: DomRefCell<IndexSet<WeakRef<AbortSignal>>>,
88
89    /// <https://dom.spec.whatwg.org/#abortsignal-dependent-signals>
90    #[no_trace]
91    #[ignore_malloc_size_of = "WeakRef"]
92    dependent_signals: DomRefCell<IndexSet<WeakRef<AbortSignal>>>,
93}
94
95impl AbortSignal {
96    fn new_inherited() -> AbortSignal {
97        AbortSignal {
98            eventtarget: EventTarget::new_inherited(),
99            abort_reason: Default::default(),
100            abort_algorithms: Default::default(),
101            dependent: Default::default(),
102            source_signals: Default::default(),
103            dependent_signals: Default::default(),
104        }
105    }
106
107    pub(crate) fn new_with_proto(
108        cx: &mut JSContext,
109        global: &GlobalScope,
110        proto: Option<HandleObject>,
111    ) -> DomRoot<AbortSignal> {
112        reflect_dom_object_with_proto_and_cx(
113            Box::new(AbortSignal::new_inherited()),
114            global,
115            proto,
116            cx,
117        )
118    }
119
120    /// <https://dom.spec.whatwg.org/#abortsignal-signal-abort>
121    pub(crate) fn signal_abort(&self, cx: &mut CurrentRealm, reason: HandleValue) {
122        let global = self.global();
123
124        // Step 1. If signal is aborted, then return.
125        if self.Aborted() {
126            return;
127        }
128
129        // Step 2. Set signal’s abort reason to reason if it is given;
130        // otherwise to a new "AbortError" DOMException.
131        let abort_reason = reason.get();
132        if !abort_reason.is_undefined() {
133            self.abort_reason.set(abort_reason);
134        } else {
135            rooted!(&in(cx) let mut rooted_error = UndefinedValue());
136            Error::Abort(None).to_jsval(cx, &global, rooted_error.handle_mut());
137            self.abort_reason.set(rooted_error.get())
138        }
139
140        // Step 3. Let dependentSignalsToAbort be a new list.
141        let mut dependent_signals_to_abort = vec![];
142
143        // Step 4. For each dependentSignal of signal’s dependent signals:
144        for weak in self.dependent_signals.borrow().iter() {
145            if let Some(dependent_signal) = weak.root() {
146                // Step 4.1. If dependentSignal is not aborted:
147                if !dependent_signal.aborted() {
148                    // Step 4.1.1. Set dependentSignal’s abort reason to signal’s abort reason.
149                    dependent_signal.abort_reason.set(self.abort_reason.get());
150                    // Step 4.1.2. Append dependentSignal to dependentSignalsToAbort.
151                    dependent_signals_to_abort.push(dependent_signal);
152                }
153            }
154        }
155
156        // Step 5. Run the abort steps for signal.
157        self.run_the_abort_steps(cx, &global);
158
159        // Step 6. For each dependentSignal of dependentSignalsToAbort, run the abort steps for dependentSignal.
160        for dependent_signal in dependent_signals_to_abort.iter() {
161            dependent_signal.run_the_abort_steps(cx, &global);
162        }
163    }
164
165    /// <https://dom.spec.whatwg.org/#abortsignal-add>
166    pub(crate) fn add(&self, algorithm: &AbortAlgorithm) {
167        // Step 1. If signal is aborted, then return.
168        if self.aborted() {
169            return;
170        }
171
172        // Step 2. Append algorithm to signal’s abort algorithms.
173        self.abort_algorithms.borrow_mut().push(algorithm.clone());
174
175        // if this is a dependent signal, it may now need to be kept alive.
176        if self.dependent.get() {
177            self.global().register_dependent_abort_signal(self);
178        }
179    }
180
181    /// Run a specific abort algorithm.
182    pub(crate) fn run_abort_algorithm(
183        &self,
184        cx: &mut CurrentRealm,
185        global: &GlobalScope,
186        algorithm: &AbortAlgorithm,
187    ) {
188        match algorithm {
189            AbortAlgorithm::StreamPiping(pipe) => {
190                rooted!(&in(cx) let mut reason = UndefinedValue());
191                reason.set(self.abort_reason.get());
192                pipe.abort_with_reason(cx, global, reason.handle());
193            },
194            AbortAlgorithm::Fetch(fetch_context) => {
195                rooted!(&in(cx) let mut reason = UndefinedValue());
196                reason.set(self.abort_reason.get());
197                if let Some(fetch_context) = &mut *fetch_context.lock().unwrap() {
198                    fetch_context.abort_fetch(reason.handle(), cx);
199                }
200            },
201            AbortAlgorithm::FetchLater(deferred_fetch_record_id) => {
202                global
203                    .deferred_fetch_record_for_id(deferred_fetch_record_id)
204                    .abort();
205            },
206            AbortAlgorithm::DomEventListener(removable_listener) => {
207                removable_listener.event_target.remove_event_listener(
208                    removable_listener.ty.clone(),
209                    &removable_listener.listener,
210                    &removable_listener.options,
211                );
212            },
213        }
214    }
215
216    /// <https://dom.spec.whatwg.org/#run-the-abort-steps>
217    fn run_the_abort_steps(&self, cx: &mut CurrentRealm, global: &GlobalScope) {
218        // Step 1. For each algorithm of signal’s abort algorithms: run algorithm.
219        for algo in self.abort_algorithms.borrow().iter() {
220            self.run_abort_algorithm(cx, global, algo);
221        }
222        // Step 2. Empty signal’s abort algorithms.
223        self.abort_algorithms.borrow_mut().clear();
224
225        // Step 3. Fire an event named abort at signal.
226        self.upcast::<EventTarget>().fire_event(cx, atom!("abort"));
227    }
228
229    /// <https://dom.spec.whatwg.org/#abortsignal-aborted>
230    pub(crate) fn aborted(&self) -> bool {
231        // An AbortSignal object is aborted when its abort reason is not undefined.
232        !self.abort_reason.get().is_undefined()
233    }
234
235    /// <https://dom.spec.whatwg.org/#create-a-dependent-abort-signal>
236    pub(crate) fn create_dependent_abort_signal(
237        cx: &mut JSContext,
238        signals: Vec<DomRoot<AbortSignal>>,
239        global: &GlobalScope,
240    ) -> DomRoot<AbortSignal> {
241        // Step 1. Let resultSignal be a new object implementing signalInterface using realm.
242        let result_signal = Self::new_with_proto(cx, global, None);
243        // Step 2. For each signal of signals: if signal is aborted,
244        // then set resultSignal’s abort reason to signal’s abort reason and return resultSignal.
245        for signal in signals.iter() {
246            if signal.aborted() {
247                result_signal.abort_reason.set(signal.abort_reason.get());
248                return result_signal;
249            }
250        }
251        // Step 3. Set resultSignal’s dependent to true.
252        result_signal.dependent.set(true);
253        // Step 4. For each signal of signals:
254        for signal in signals.iter() {
255            // Step 4.1. If signal’s dependent is false:
256            if !signal.dependent.get() {
257                // Step 4.1.1. Append signal to resultSignal’s source signals.
258                result_signal
259                    .source_signals
260                    .borrow_mut()
261                    .insert(WeakRef::new(signal));
262                // Step 4.1.2. Append resultSignal to signal’s dependent signals.
263                signal
264                    .dependent_signals
265                    .borrow_mut()
266                    .insert(WeakRef::new(&*result_signal));
267            } else {
268                // Step 4.2. Otherwise, for each sourceSignal of signal’s source signals:
269                for source_signal_weak in signal.source_signals.borrow().iter() {
270                    if let Some(source_signal) = source_signal_weak.root() {
271                        // Step 4.2.1. Assert: sourceSignal is not aborted and not dependent.
272                        assert!(!source_signal.aborted() && !source_signal.dependent.get());
273                        // Step 4.2.2. Append sourceSignal to resultSignal’s source signals.
274                        result_signal
275                            .source_signals
276                            .borrow_mut()
277                            .insert(WeakRef::new(&*source_signal));
278                        // Step 4.2.3. Append resultSignal to sourceSignal’s dependent signals.
279                        source_signal
280                            .dependent_signals
281                            .borrow_mut()
282                            .insert(WeakRef::new(&*result_signal));
283                    }
284                }
285            }
286        }
287        // Step 5. Return resultSignal.
288        global.register_dependent_abort_signal(&result_signal);
289        result_signal
290    }
291
292    /// Remove weak references whose target has been collected.
293    fn prune_dead_weak_refs(&self) {
294        self.source_signals.borrow_mut().retain(|w| w.is_alive());
295        self.dependent_signals.borrow_mut().retain(|w| w.is_alive());
296    }
297
298    /// Whether this signal still has abort algorithms registered.
299    fn has_abort_algorithms(&self) -> bool {
300        !self.abort_algorithms.borrow().is_empty()
301    }
302
303    /// Whether this signal has any listeners for its "abort" event.
304    fn has_abort_listeners(&self) -> bool {
305        self.upcast::<EventTarget>()
306            .has_listeners_for(&atom!("abort"))
307    }
308
309    /// <https://dom.spec.whatwg.org/#abort-signal-garbage-collection>
310    /// A non-aborted dependent AbortSignal object must not be garbage
311    /// collected while its source signals is non-empty and it has
312    /// registered event listeners for its abort event or its abort
313    /// algorithms is non-empty.
314    pub(crate) fn must_keep_alive_for_gc(&self) -> bool {
315        // Treat dead weak entries as removed.
316        self.prune_dead_weak_refs();
317
318        // Only care about non-aborted signals.
319        if self.aborted() {
320            return false;
321        }
322
323        // If it has no live source signals left, it doesn’t need to be kept.
324        if self.source_signals.borrow().is_empty() {
325            return false;
326        }
327
328        let has_algos = self.has_abort_algorithms();
329        let has_listeners = self.has_abort_listeners();
330
331        has_algos || has_listeners
332    }
333}
334
335impl AbortSignalMethods<crate::DomTypeHolder> for AbortSignal {
336    /// <https://dom.spec.whatwg.org/#dom-abortsignal-aborted>
337    fn Aborted(&self) -> bool {
338        // The aborted getter steps are to return true if this is aborted; otherwise false.
339        self.aborted()
340    }
341
342    /// <https://dom.spec.whatwg.org/#dom-abortsignal-abort>
343    fn Abort(
344        cx: &mut JSContext,
345        global: &GlobalScope,
346        reason: HandleValue,
347    ) -> DomRoot<AbortSignal> {
348        // Step 1. Let signal be a new AbortSignal object.
349        let signal = AbortSignal::new_with_proto(cx, global, None);
350
351        // Step 2. Set signal’s abort reason to reason if it is given;
352        // otherwise to a new "AbortError" DOMException.
353        let abort_reason = reason.get();
354        if !abort_reason.is_undefined() {
355            signal.abort_reason.set(abort_reason);
356        } else {
357            rooted!(&in(cx) let mut rooted_error = UndefinedValue());
358            Error::Abort(None).to_jsval(cx, global, rooted_error.handle_mut());
359            signal.abort_reason.set(rooted_error.get())
360        }
361
362        // Step 3. Return signal.
363        signal
364    }
365
366    /// <https://dom.spec.whatwg.org/#dom-abortsignal-timeout>
367    fn Timeout(
368        cx: &mut JSContext,
369        global: &GlobalScope,
370        milliseconds: u64,
371    ) -> DomRoot<AbortSignal> {
372        // Step 1. Let signal be a new AbortSignal object.
373        let signal = AbortSignal::new_with_proto(cx, global, None);
374
375        // Step 2. Let global be signal’s relevant global object.
376        // We already have `global`.
377
378        let signal_keepalive: Trusted<AbortSignal> = Trusted::new(&signal);
379
380        let ms_i64 = if milliseconds > i64::MAX as u64 {
381            i64::MAX
382        } else {
383            milliseconds as i64
384        };
385
386        // Step 3. Run steps after a timeout given global, "AbortSignal-timeout", milliseconds, and the following step:
387        global.run_steps_after_a_timeout(
388            DOMString::from("AbortSignal-timeout"),
389            ms_i64,
390            move |_cx, global| {
391                let task_source = global.task_manager().timer_task_source().to_sendable();
392
393                // Step 3.1. Queue a global task on the timer task source given global to signal abort given signal and a new "TimeoutError" DOMException.
394                // For the duration of this timeout, if signal has any event listeners registered for its abort event,
395                // there must be a strong reference from global to signal.
396                task_source.queue(task!(abortsignal_timeout: move |cx| {
397                    let signal_for_task = signal_keepalive.root();
398
399                    rooted!(&in(cx) let mut reason = UndefinedValue());
400                    Error::Timeout(None).to_jsval(
401                        cx,
402                        &signal_for_task.global(),
403                        reason.handle_mut(),
404                    );
405
406                    let mut realm = enter_auto_realm(cx, &*signal_for_task.global());
407                    let mut realm = realm.current_realm();
408
409                    // “signal abort given signal and a new "TimeoutError" DOMException.”
410                    signal_for_task.signal_abort(
411                        &mut realm,
412                        reason.handle(),
413                    );
414                }));
415            },
416        );
417
418        // Step 4. Return signal.
419        signal
420    }
421
422    /// <https://dom.spec.whatwg.org/#dom-abortsignal-any>
423    fn Any(
424        cx: &mut JSContext,
425        global: &GlobalScope,
426        signals: Vec<DomRoot<AbortSignal>>,
427    ) -> DomRoot<AbortSignal> {
428        // The static any(signals) method steps are to return the result
429        // of creating a dependent abort signal from signals using AbortSignal and the current realm.
430        Self::create_dependent_abort_signal(cx, signals, global)
431    }
432
433    /// <https://dom.spec.whatwg.org/#dom-abortsignal-reason>
434    fn Reason(&self, _cx: SafeJSContext, mut rval: MutableHandleValue) {
435        // The reason getter steps are to return this’s abort reason.
436        rval.set(self.abort_reason.get());
437    }
438
439    /// <https://dom.spec.whatwg.org/#dom-abortsignal-throwifaborted>
440    #[expect(unsafe_code)]
441    fn ThrowIfAborted(&self, cx: &mut JSContext) -> Fallible<()> {
442        // The throwIfAborted() method steps are to throw this’s abort reason, if this is aborted.
443        if self.aborted() {
444            unsafe {
445                JS_SetPendingException(
446                    cx,
447                    HandleValue::from_raw(self.abort_reason.handle()),
448                    ExceptionStackBehavior::Capture,
449                )
450            };
451            return Err(Error::JSFailed);
452        }
453        Ok(())
454    }
455
456    // <https://dom.spec.whatwg.org/#dom-abortsignal-onabort>
457    event_handler!(abort, GetOnabort, SetOnabort);
458}