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