script/dom/
abortsignal.rs1use std::cell::RefCell;
6
7use dom_struct::dom_struct;
8use js::jsapi::{ExceptionStackBehavior, Heap, JS_SetPendingException};
9use js::jsval::{JSVal, UndefinedValue};
10use js::rust::{HandleObject, HandleValue, MutableHandleValue};
11use script_bindings::inheritance::Castable;
12
13use crate::dom::bindings::codegen::Bindings::AbortSignalBinding::AbortSignalMethods;
14use crate::dom::bindings::error::{Error, ErrorToJsval};
15use crate::dom::bindings::reflector::{DomGlobal, reflect_dom_object_with_proto};
16use crate::dom::bindings::root::DomRoot;
17use crate::dom::eventtarget::EventTarget;
18use crate::dom::globalscope::GlobalScope;
19use crate::dom::readablestream::PipeTo;
20use crate::realms::InRealm;
21use crate::script_runtime::{CanGc, JSContext as SafeJSContext};
22
23impl js::gc::Rootable for AbortAlgorithm {}
24
25#[derive(Clone, JSTraceable, MallocSizeOf)]
29#[cfg_attr(crown, crown::unrooted_must_root_lint::must_root)]
30#[allow(dead_code)]
31pub(crate) enum AbortAlgorithm {
32 DomEventLister,
34 StreamPiping(PipeTo),
36 Fetch,
38}
39
40#[dom_struct]
42pub(crate) struct AbortSignal {
43 eventtarget: EventTarget,
44
45 #[ignore_malloc_size_of = "mozjs"]
47 abort_reason: Heap<JSVal>,
48
49 abort_algorithms: RefCell<Vec<AbortAlgorithm>>,
51}
52
53impl AbortSignal {
54 fn new_inherited() -> AbortSignal {
55 AbortSignal {
56 eventtarget: EventTarget::new_inherited(),
57 abort_reason: Default::default(),
58 abort_algorithms: Default::default(),
59 }
60 }
61
62 pub(crate) fn new_with_proto(
63 global: &GlobalScope,
64 proto: Option<HandleObject>,
65 can_gc: CanGc,
66 ) -> DomRoot<AbortSignal> {
67 reflect_dom_object_with_proto(
68 Box::new(AbortSignal::new_inherited()),
69 global,
70 proto,
71 can_gc,
72 )
73 }
74
75 pub(crate) fn signal_abort(
77 &self,
78 cx: SafeJSContext,
79 reason: HandleValue,
80 realm: InRealm,
81 can_gc: CanGc,
82 ) {
83 let global = self.global();
84
85 if self.Aborted() {
87 return;
88 }
89
90 let abort_reason = reason.get();
91
92 if !abort_reason.is_undefined() {
94 self.abort_reason.set(abort_reason);
95 } else {
96 rooted!(in(*cx) let mut rooted_error = UndefinedValue());
98 Error::Abort.to_jsval(cx, &global, rooted_error.handle_mut(), can_gc);
99 self.abort_reason.set(rooted_error.get())
100 }
101
102 self.run_the_abort_steps(cx, &global, realm, can_gc);
108
109 }
112
113 pub(crate) fn add(&self, algorithm: &AbortAlgorithm) {
115 if self.aborted() {
117 return;
118 }
119
120 self.abort_algorithms.borrow_mut().push(algorithm.clone());
122 }
123
124 pub(crate) fn run_abort_algorithm(
126 &self,
127 cx: SafeJSContext,
128 global: &GlobalScope,
129 algorithm: &AbortAlgorithm,
130 realm: InRealm,
131 can_gc: CanGc,
132 ) {
133 match algorithm {
134 AbortAlgorithm::StreamPiping(pipe) => {
135 rooted!(in(*cx) let mut reason = UndefinedValue());
136 reason.set(self.abort_reason.get());
137 pipe.abort_with_reason(cx, global, reason.handle(), realm, can_gc);
138 },
139 _ => {
140 },
143 }
144 }
145
146 fn run_the_abort_steps(
148 &self,
149 cx: SafeJSContext,
150 global: &GlobalScope,
151 realm: InRealm,
152 can_gc: CanGc,
153 ) {
154 for algo in self.abort_algorithms.borrow().iter() {
156 self.run_abort_algorithm(cx, global, algo, realm, can_gc);
157 }
158
159 self.abort_algorithms.borrow_mut().clear();
161
162 self.upcast::<EventTarget>()
164 .fire_event(atom!("abort"), can_gc);
165 }
166
167 pub(crate) fn aborted(&self) -> bool {
169 !self.abort_reason.get().is_undefined()
171 }
172}
173
174impl AbortSignalMethods<crate::DomTypeHolder> for AbortSignal {
175 fn Aborted(&self) -> bool {
177 self.aborted()
179 }
180
181 fn Abort(
183 cx: SafeJSContext,
184 global: &GlobalScope,
185 reason: HandleValue,
186 can_gc: CanGc,
187 ) -> DomRoot<AbortSignal> {
188 let signal = AbortSignal::new_with_proto(global, None, can_gc);
190
191 let abort_reason = reason.get();
192 if !abort_reason.is_undefined() {
194 signal.abort_reason.set(abort_reason);
195 } else {
196 rooted!(in(*cx) let mut rooted_error = UndefinedValue());
198 Error::Abort.to_jsval(cx, global, rooted_error.handle_mut(), can_gc);
199 signal.abort_reason.set(rooted_error.get())
200 }
201
202 signal
204 }
205
206 fn Reason(&self, _cx: SafeJSContext, mut rval: MutableHandleValue) {
208 rval.set(self.abort_reason.get());
210 }
211
212 #[allow(unsafe_code)]
214 fn ThrowIfAborted(&self) {
215 if self.aborted() {
217 let cx = GlobalScope::get_cx();
218 unsafe {
219 JS_SetPendingException(
220 *cx,
221 self.abort_reason.handle(),
222 ExceptionStackBehavior::Capture,
223 )
224 };
225 }
226 }
227
228 event_handler!(abort, GetOnabort, SetOnabort);
230}