From 37f3247adddc07f625423dbdd261078d53040631 Mon Sep 17 00:00:00 2001 From: Xavier Montillet <git@xavier.montillet.pro> Date: Tue, 5 Dec 2023 11:30:37 +0100 Subject: [PATCH 1/6] Added JsFn --- crates/jstz_core/src/js_fn.rs | 136 +++++++++++++++++++++++++++++++++ crates/jstz_core/src/lib.rs | 1 + crates/jstz_core/src/native.rs | 2 +- 3 files changed, 138 insertions(+), 1 deletion(-) create mode 100644 crates/jstz_core/src/js_fn.rs diff --git a/crates/jstz_core/src/js_fn.rs b/crates/jstz_core/src/js_fn.rs new file mode 100644 index 000000000..7f7c4247d --- /dev/null +++ b/crates/jstz_core/src/js_fn.rs @@ -0,0 +1,136 @@ +use std::{marker::PhantomData, ops::Deref}; + +use boa_engine::{ + object::builtins::JsFunction, value::TryFromJs, Context, JsResult, JsValue, +}; +use boa_gc::{custom_trace, Finalize, Trace}; + +use crate::value::IntoJs; + +pub trait IntoJsArgs<const N: usize> { + fn into_js_args(self, context: &mut Context<'_>) -> [JsValue; N]; +} + +impl IntoJsArgs<0> for () { + fn into_js_args(self, _context: &mut Context<'_>) -> [JsValue; 0] { + [] + } +} + +impl<T0: IntoJs> IntoJsArgs<1> for (T0,) { + fn into_js_args(self, context: &mut Context<'_>) -> [JsValue; 1] { + [self.0.into_js(context)] + } +} + +impl<T0: IntoJs, T1: IntoJs> IntoJsArgs<2> for (T0, T1) { + fn into_js_args(self, context: &mut Context<'_>) -> [JsValue; 2] { + [self.0.into_js(context), self.1.into_js(context)] + } +} + +impl<T0: IntoJs, T1: IntoJs, T2: IntoJs> IntoJsArgs<3> for (T0, T1, T2) { + fn into_js_args(self, context: &mut Context<'_>) -> [JsValue; 3] { + [ + self.0.into_js(context), + self.1.into_js(context), + self.2.into_js(context), + ] + } +} + +/// A `JsFn<T, N, I, O>` is a `JsFunction` tagged with some Rust types used to handle the `TryFromJs` and `IntoJs` conversions automatically: +/// - `T` is the type of the `this` parameter; +/// - `N` is the arity; +/// - `I` is a tuple `(I1, ..., IN)` that contains the types of the parameters; +/// - `O` is the type of the output. +#[derive(Debug)] +pub struct JsFn<T: IntoJs, const N: usize, I: IntoJsArgs<N>, O: TryFromJs> { + function: JsFunction, + _this_type: PhantomData<T>, + _inputs_type: PhantomData<I>, + _output_type: PhantomData<O>, +} + +impl<T: IntoJs, const N: usize, I: IntoJsArgs<N>, O: TryFromJs> Finalize + for JsFn<T, N, I, O> +{ + fn finalize(&self) {} +} + +unsafe impl<T: IntoJs, const N: usize, I: IntoJsArgs<N>, O: TryFromJs> Trace + for JsFn<T, N, I, O> +{ + custom_trace!(this, { + mark(&this.function); + }); +} + +impl<T: IntoJs, const N: usize, I: IntoJsArgs<N>, O: TryFromJs> Deref + for JsFn<T, N, I, O> +{ + type Target = JsFunction; + + fn deref(&self) -> &Self::Target { + &self.function + } +} + +impl<T: IntoJs, const N: usize, I: IntoJsArgs<N>, O: TryFromJs> Into<JsFunction> + for JsFn<T, N, I, O> +{ + fn into(self) -> JsFunction { + self.function + } +} + +impl<T: IntoJs, const N: usize, I: IntoJsArgs<N>, O: TryFromJs> From<JsFunction> + for JsFn<T, N, I, O> +{ + fn from(value: JsFunction) -> Self { + JsFn { + function: value, + _this_type: PhantomData, + _inputs_type: PhantomData, + _output_type: PhantomData, + } + } +} + +impl<T: IntoJs, const N: usize, I: IntoJsArgs<N>, O: TryFromJs> Into<JsValue> + for JsFn<T, N, I, O> +{ + fn into(self) -> JsValue { + self.function.into() + } +} + +// impl<T: IntoJs, const N: usize, I: IntoJsArgs<N>, O: TryFromJs> TryFrom<JsValue> for JsFn<T, N, I, O> +// This is implementable, but the right way to implement it would be to lift the implementation of `TryFromJs` for `JsFunction` (that does not use the context) to an implementation of `TryFrom<JsFunction>` in boa +// (If it is eventually implemented, then the implementation of TryFromJs below should use it) + +impl<T: IntoJs, const N: usize, I: IntoJsArgs<N>, O: TryFromJs> IntoJs + for JsFn<T, N, I, O> +{ + fn into_js(self, _context: &mut Context<'_>) -> JsValue { + self.function.into() + } +} + +impl<T: IntoJs, const N: usize, I: IntoJsArgs<N>, O: TryFromJs> TryFromJs + for JsFn<T, N, I, O> +{ + fn try_from_js(value: &JsValue, context: &mut Context<'_>) -> JsResult<Self> { + JsFunction::try_from_js(value, context).map(JsFn::from) + } +} + +impl<T: IntoJs, const N: usize, I: IntoJsArgs<N>, O: TryFromJs> JsFn<T, N, I, O> { + pub fn call(&self, this: T, inputs: I, context: &mut Context<'_>) -> JsResult<O> { + let js_this = this.into_js(context); + let js_args = inputs.into_js_args(context); + self.deref() + .call(&js_this, &js_args, context) + .and_then(|output| O::try_from_js(&output, context)) + } +} diff --git a/crates/jstz_core/src/lib.rs b/crates/jstz_core/src/lib.rs index 99dcd70c4..bac9b0803 100644 --- a/crates/jstz_core/src/lib.rs +++ b/crates/jstz_core/src/lib.rs @@ -6,6 +6,7 @@ pub use error::{Error, Result}; pub mod future; pub mod host; pub mod iterators; +pub mod js_fn; pub mod kv; pub mod native; pub mod realm; diff --git a/crates/jstz_core/src/native.rs b/crates/jstz_core/src/native.rs index 0f9624159..37399194c 100644 --- a/crates/jstz_core/src/native.rs +++ b/crates/jstz_core/src/native.rs @@ -18,7 +18,7 @@ pub use boa_engine::{object::NativeObject, NativeFunction}; use crate::value::IntoJs; /// This struct permits Rust types to be passed around as JavaScript objects. -#[derive(Trace, Finalize)] +#[derive(Trace, Finalize, Debug)] pub struct JsNativeObject<T: NativeObject> { inner: JsValue, _phantom: PhantomData<T>, From a7c62b01414aac490b7d16b2de2b51bc471c4cd1 Mon Sep 17 00:00:00 2001 From: Xavier Montillet <git@xavier.montillet.pro> Date: Tue, 5 Dec 2023 11:31:56 +0100 Subject: [PATCH 2/6] Added IDL type aliases --- crates/jstz_api/src/idl.rs | 17 +++++++++++++++++ 1 file changed, 17 insertions(+) diff --git a/crates/jstz_api/src/idl.rs b/crates/jstz_api/src/idl.rs index 9f8a65bbf..c366de120 100644 --- a/crates/jstz_api/src/idl.rs +++ b/crates/jstz_api/src/idl.rs @@ -182,3 +182,20 @@ impl ArrayBufferLike for JsBufferSource { } } } + +// https://webidl.spec.whatwg.org/#idl-types + +pub type Any = JsValue; +pub type Bytes = i8; +pub type Octet = u8; +pub type Short = i16; +pub type UnsignedShort = u16; +pub type Long = i32; +pub type UnsignedLong = u32; +pub type LongLong = i64; +pub type UnsignedLongLong = u64; +pub type UnrestrictedFloat = f32; +pub type UnrestrictedDouble = f64; + +pub type PositiveInteger = UnsignedLongLong; +pub type Number = f64; From e4fd7e5480548e0edd541c1dbc8452ea994fcc19 Mon Sep 17 00:00:00 2001 From: Xavier Montillet <git@xavier.montillet.pro> Date: Tue, 5 Dec 2023 11:36:42 +0100 Subject: [PATCH 3/6] Register trivial `ReadableStream` class --- crates/jstz_api/src/lib.rs | 1 + crates/jstz_api/src/stream/mod.rs | 13 ++++++ crates/jstz_api/src/stream/readable/mod.rs | 50 ++++++++++++++++++++++ crates/jstz_cli/src/repl.rs | 3 +- 4 files changed, 66 insertions(+), 1 deletion(-) create mode 100644 crates/jstz_api/src/stream/mod.rs create mode 100644 crates/jstz_api/src/stream/readable/mod.rs diff --git a/crates/jstz_api/src/lib.rs b/crates/jstz_api/src/lib.rs index 85f84806d..0dea6f293 100644 --- a/crates/jstz_api/src/lib.rs +++ b/crates/jstz_api/src/lib.rs @@ -4,6 +4,7 @@ mod kv; pub mod encoding; pub mod http; pub mod idl; +pub mod stream; pub mod url; pub mod urlpattern; pub use console::{ConsoleApi, LogRecord, LOG_PREFIX}; diff --git a/crates/jstz_api/src/stream/mod.rs b/crates/jstz_api/src/stream/mod.rs new file mode 100644 index 000000000..ad54dec54 --- /dev/null +++ b/crates/jstz_api/src/stream/mod.rs @@ -0,0 +1,13 @@ +use boa_engine::Context; + +use self::readable::ReadableStreamApi; + +pub mod readable; + +pub struct StreamApi; + +impl jstz_core::Api for StreamApi { + fn init(self, context: &mut Context<'_>) { + ReadableStreamApi.init(context); + } +} diff --git a/crates/jstz_api/src/stream/readable/mod.rs b/crates/jstz_api/src/stream/readable/mod.rs new file mode 100644 index 000000000..0b801c17f --- /dev/null +++ b/crates/jstz_api/src/stream/readable/mod.rs @@ -0,0 +1,50 @@ +use boa_engine::{Context, JsResult}; +use boa_gc::{custom_trace, Finalize, Trace}; +use jstz_core::native::{ + register_global_class, ClassBuilder, JsNativeObject, NativeClass, +}; + +pub struct ReadableStream { + // TODO +} + +impl Finalize for ReadableStream { + fn finalize(&self) { + todo!() + } +} + +unsafe impl Trace for ReadableStream { + custom_trace!(this, todo!()); +} + +pub struct ReadableStreamClass; + +impl NativeClass for ReadableStreamClass { + type Instance = ReadableStream; + + const NAME: &'static str = "ReadableStream"; + + fn constructor( + _this: &JsNativeObject<Self::Instance>, + args: &[boa_engine::JsValue], + context: &mut Context<'_>, + ) -> JsResult<Self::Instance> { + todo!() + } + + fn init(class: &mut ClassBuilder<'_, '_>) -> JsResult<()> { + // TODO + Ok(()) + } +} + +pub struct ReadableStreamApi; + +impl jstz_core::Api for ReadableStreamApi { + fn init(self, context: &mut Context<'_>) { + register_global_class::<ReadableStreamClass>(context) + .expect("The `ReadableStream` class shouldn't exist yet") + // TODO + } +} diff --git a/crates/jstz_cli/src/repl.rs b/crates/jstz_cli/src/repl.rs index 132108d69..cca46b629 100644 --- a/crates/jstz_cli/src/repl.rs +++ b/crates/jstz_cli/src/repl.rs @@ -2,7 +2,7 @@ use anyhow::Result; use boa_engine::{js_string, JsResult, JsValue, Source}; use jstz_api::{ encoding::EncodingApi, http::HttpApi, url::UrlApi, urlpattern::UrlPatternApi, - ConsoleApi, KvApi, + ConsoleApi, KvApi, stream::StreamApi }; use jstz_core::host::HostRuntime; use jstz_core::{ @@ -48,6 +48,7 @@ pub fn exec(self_address: Option<String>, cfg: &Config) -> Result<()> { rt.context(), ); realm_clone.register_api(EncodingApi, rt.context()); + realm_clone.register_api(StreamApi, rt.context()); realm_clone.register_api(UrlApi, rt.context()); realm_clone.register_api(UrlPatternApi, rt.context()); realm_clone.register_api(HttpApi, rt.context()); From dc1f1f0a59635230f8c8e055723bb54aa5c15b01 Mon Sep 17 00:00:00 2001 From: Xavier Montillet <git@xavier.montillet.pro> Date: Tue, 5 Dec 2023 11:37:38 +0100 Subject: [PATCH 4/6] Export the `impl_into_js_from_into` macro --- crates/jstz_core/src/value.rs | 1 + 1 file changed, 1 insertion(+) diff --git a/crates/jstz_core/src/value.rs b/crates/jstz_core/src/value.rs index b25069057..f9f9bb99f 100644 --- a/crates/jstz_core/src/value.rs +++ b/crates/jstz_core/src/value.rs @@ -15,6 +15,7 @@ pub trait IntoJs { fn into_js(self, context: &mut Context<'_>) -> JsValue; } +#[macro_export] macro_rules! impl_into_js_from_into { ($($T: ty), *) => { $( From d4a78b5f00f8742b3b83a9532ce5dda8181a1870 Mon Sep 17 00:00:00 2001 From: Xavier Montillet <git@xavier.montillet.pro> Date: Tue, 5 Dec 2023 11:38:21 +0100 Subject: [PATCH 5/6] Added `Todo` placeholder type --- crates/jstz_api/src/lib.rs | 1 + crates/jstz_api/src/todo.rs | 40 +++++++++++++++++++++++++++++++++++++ 2 files changed, 41 insertions(+) create mode 100644 crates/jstz_api/src/todo.rs diff --git a/crates/jstz_api/src/lib.rs b/crates/jstz_api/src/lib.rs index 0dea6f293..bed936b10 100644 --- a/crates/jstz_api/src/lib.rs +++ b/crates/jstz_api/src/lib.rs @@ -5,6 +5,7 @@ pub mod encoding; pub mod http; pub mod idl; pub mod stream; +pub mod todo; pub mod url; pub mod urlpattern; pub use console::{ConsoleApi, LogRecord, LOG_PREFIX}; diff --git a/crates/jstz_api/src/todo.rs b/crates/jstz_api/src/todo.rs new file mode 100644 index 000000000..37d487585 --- /dev/null +++ b/crates/jstz_api/src/todo.rs @@ -0,0 +1,40 @@ +use boa_engine::value::TryFromJs; +use boa_gc::{custom_trace, Finalize, Trace}; +use jstz_core::value::IntoJs; + +/// A placeholder for types that have yet to be defined +#[derive(Debug)] +pub enum Todo { + Todo, +} + +impl Finalize for Todo { + fn finalize(&self) { + todo!() + } +} + +#[allow(unused_variables)] +unsafe impl Trace for Todo { + custom_trace!(this, todo!()); +} + +#[allow(unused_variables)] +impl IntoJs for Todo { + fn into_js( + self, + context: &mut boa_engine::prelude::Context<'_>, + ) -> boa_engine::prelude::JsValue { + todo!() + } +} + +#[allow(unused_variables)] +impl TryFromJs for Todo { + fn try_from_js( + value: &boa_engine::prelude::JsValue, + context: &mut boa_engine::prelude::Context<'_>, + ) -> boa_engine::prelude::JsResult<Self> { + todo!() + } +} From 85387e77ccccdf593ef1dac62887a66ae98ab7f0 Mon Sep 17 00:00:00 2001 From: Xavier Montillet <git@xavier.montillet.pro> Date: Tue, 5 Dec 2023 11:50:35 +0100 Subject: [PATCH 6/6] Implemented `UnderlyingSource` --- crates/jstz_api/src/stream/mod.rs | 1 + crates/jstz_api/src/stream/readable/mod.rs | 10 +- .../src/stream/readable/underlying_source.rs | 411 ++++++++++++++++++ crates/jstz_api/src/stream/tmp.rs | 6 + 4 files changed, 427 insertions(+), 1 deletion(-) create mode 100644 crates/jstz_api/src/stream/readable/underlying_source.rs create mode 100644 crates/jstz_api/src/stream/tmp.rs diff --git a/crates/jstz_api/src/stream/mod.rs b/crates/jstz_api/src/stream/mod.rs index ad54dec54..85b88cab1 100644 --- a/crates/jstz_api/src/stream/mod.rs +++ b/crates/jstz_api/src/stream/mod.rs @@ -3,6 +3,7 @@ use boa_engine::Context; use self::readable::ReadableStreamApi; pub mod readable; +mod tmp; pub struct StreamApi; diff --git a/crates/jstz_api/src/stream/readable/mod.rs b/crates/jstz_api/src/stream/readable/mod.rs index 0b801c17f..a569adc41 100644 --- a/crates/jstz_api/src/stream/readable/mod.rs +++ b/crates/jstz_api/src/stream/readable/mod.rs @@ -1,9 +1,15 @@ -use boa_engine::{Context, JsResult}; +use boa_engine::{value::TryFromJs, Context, JsArgs, JsResult}; use boa_gc::{custom_trace, Finalize, Trace}; use jstz_core::native::{ register_global_class, ClassBuilder, JsNativeObject, NativeClass, }; +use crate::stream::readable::underlying_source::{ + UnderlyingSource, UnderlyingSourceTrait, +}; + +pub mod underlying_source; + pub struct ReadableStream { // TODO } @@ -30,6 +36,8 @@ impl NativeClass for ReadableStreamClass { args: &[boa_engine::JsValue], context: &mut Context<'_>, ) -> JsResult<Self::Instance> { + let underlying_source = + Option::<UnderlyingSource>::try_from_js(args.get_or_undefined(0), context)?; todo!() } diff --git a/crates/jstz_api/src/stream/readable/underlying_source.rs b/crates/jstz_api/src/stream/readable/underlying_source.rs new file mode 100644 index 000000000..8202ac982 --- /dev/null +++ b/crates/jstz_api/src/stream/readable/underlying_source.rs @@ -0,0 +1,411 @@ +//! https://streams.spec.whatwg.org/#underlying-source-api + +use boa_engine::{ + js_string, object::builtins::JsPromise, property::PropertyKey, value::TryFromJs, + Context, JsNativeError, JsObject, JsResult, JsValue, +}; +use boa_gc::{custom_trace, Finalize, Trace}; +use jstz_core::{ + impl_into_js_from_into, js_fn::JsFn, native::JsNativeObject, value::IntoJs, +}; +use std::str::FromStr; + +use crate::idl; + +use crate::stream::tmp::*; + +/// dictionary [UnderlyingSource][spec] { +/// UnderlyingSourceStartCallback start; +/// UnderlyingSourcePullCallback pull; +/// UnderlyingSourceCancelCallback cancel; +/// ReadableStreamType type; +/// \[EnforceRange\] unsigned long long autoAllocateChunkSize; +/// }; +/// +/// [Note][spec2]: We cannot declare the underlyingSource argument as having the UnderlyingSource type directly, because doing so would lose the reference to the original object. We need to retain the object so we can invoke the various methods on it. +/// +/// [spec]: https://streams.spec.whatwg.org/#dictdef-underlyingsource +/// [spec2]: https://streams.spec.whatwg.org/#rs-constructor +#[derive(Debug)] +pub struct UnderlyingSource { + /// TODO + pub this: JsObject, + + /// **[start][spec](controller), of type UnderlyingSourceStartCallback** + /// + /// A function that is called immediately during creation of the ReadableStream. + /// + /// Typically this is used to adapt a push source by setting up relevant event listeners, as in the example of § 10.1 A readable stream with an underlying push source (no backpressure support), or to acquire access to a pull source, as in § 10.4 A readable stream with an underlying pull source. + /// + /// If this setup process is asynchronous, it can return a promise to signal success or failure; a rejected promise will error the stream. Any thrown exceptions will be re-thrown by the ReadableStream() constructor. + /// + /// [spec]: https://streams.spec.whatwg.org/#dom-underlyingsource-start + pub start: Option<UnderlyingSourceStartCallback>, + + /// **[pull][spec](controller), of type UnderlyingSourcePullCallback** + /// + /// A function that is called whenever the stream’s internal queue of chunks becomes not full, i.e. whenever the queue’s desired size becomes positive. Generally, it will be called repeatedly until the queue reaches its high water mark (i.e. until the desired size becomes non-positive). + /// + /// For push sources, this can be used to resume a paused flow, as in § 10.2 A readable stream with an underlying push source and backpressure support. For pull sources, it is used to acquire new chunks to enqueue into the stream, as in § 10.4 A readable stream with an underlying pull source. + /// + /// This function will not be called until start() successfully completes. Additionally, it will only be called repeatedly if it enqueues at least one chunk or fulfills a BYOB request; a no-op pull() implementation will not be continually called. + /// + /// If the function returns a promise, then it will not be called again until that promise fulfills. (If the promise rejects, the stream will become errored.) This is mainly used in the case of pull sources, where the promise returned represents the process of acquiring a new chunk. Throwing an exception is treated the same as returning a rejected promise. + /// + /// [spec]: https://streams.spec.whatwg.org/#dom-underlyingsource-pull + pub pull: Option<UnderlyingSourcePullCallback>, + + /// **cancel(reason), of type UnderlyingSourceCancelCallback** + /// A function that is called whenever the consumer cancels the stream, via stream.cancel() or reader.cancel(). It takes as its argument the same value as was passed to those methods by the consumer. + /// + /// Readable streams can additionally be canceled under certain conditions during piping; see the definition of the pipeTo() method for more details. + /// + // For all streams, this is generally used to release access to the underlying resource; see for example § 10.1 A readable stream with an underlying push source (no backpressure support). + /// + /// If the shutdown process is asynchronous, it can return a promise to signal success or failure; the result will be communicated via the return value of the cancel() method that was called. Throwing an exception is treated the same as returning a rejected promise. + /// + /// [spec]: https://streams.spec.whatwg.org/#dom-underlyingsource-cancel + /// + /// *Even if the cancelation process fails, the stream will still close; it will not be put into an errored state. This is because a failure in the cancelation process doesn’t matter to the consumer’s view of the stream, once they’ve expressed disinterest in it by canceling. The failure is only communicated to the immediate caller of the corresponding method.* + /// + /// *This is different from the behavior of the close and abort options of a WritableStream's underlying sink, which upon failure put the corresponding WritableStream into an errored state. Those correspond to specific actions the producer is requesting and, if those actions fail, they indicate something more persistently wrong.* + pub cancel: Option<UnderlyingSourceCancelCallback>, + + /// **[type][spec] (byte streams only), of type ReadableStreamType** + /// + /// Can be set to "bytes" to signal that the constructed ReadableStream is a readable byte stream. This ensures that the resulting ReadableStream will successfully be able to vend BYOB readers via its getReader() method. It also affects the controller argument passed to the start() and pull() methods; see below. + /// + /// For an example of how to set up a readable byte stream, including using the different controller interface, see § 10.3 A readable byte stream with an underlying push source (no backpressure support). + /// + /// Setting any value other than "bytes" or undefined will cause the ReadableStream() constructor to throw an exception. + /// + /// [spec]: https://streams.spec.whatwg.org/#dom-underlyingsource-type + pub r#type: Option<ReadableStreamType>, + + /// **[autoAllocateChunkSize][spec] (byte streams only), of type unsigned long long** + /// + /// Can be set to a positive integer to cause the implementation to automatically allocate buffers for the underlying source code to write into. In this case, when a consumer is using a default reader, the stream implementation will automatically allocate an ArrayBuffer of the given size, so that controller.byobRequest is always present, as if the consumer was using a BYOB reader. + /// + /// This is generally used to cut down on the amount of code needed to handle consumers that use default readers, as can be seen by comparing § 10.3 A readable byte stream with an underlying push source (no backpressure support) without auto-allocation to § 10.5 A readable byte stream with an underlying pull source with auto-allocation. + /// + /// [spec]: https://streams.spec.whatwg.org/#dom-underlyingsource-autoallocatechunksize + pub auto_allocate_chunk_size: Option<idl::UnsignedLongLong>, // TODO [EnforceRange] +} + +impl Finalize for UnderlyingSource { + fn finalize(&self) {} +} + +unsafe impl Trace for UnderlyingSource { + custom_trace!(this, { + mark(&this.this); + mark(&this.start); + mark(&this.pull); + mark(&this.cancel); + }); +} + +// TODO derive this implementation with a macro? +impl TryFromJs for UnderlyingSource { + fn try_from_js(value: &JsValue, context: &mut Context<'_>) -> JsResult<Self> { + // TODO check that this function works as intended in all cases, + // and move it either to a new derive macro for TryFromJs, or to JsObject + #[allow(non_snake_case)] + pub fn get_JsObject_property( + obj: &JsObject, + name: &str, + context: &mut Context<'_>, + ) -> JsResult<JsValue> { + let key = PropertyKey::from(js_string!(name)); + let key2 = key.clone(); + let has_prop = obj.has_property(key, context)?; + if has_prop { + obj.get(key2, context) + } else { + Ok(JsValue::Undefined) + } + } + + let this = value.to_object(context)?; + let start: Option<UnderlyingSourceStartCallback> = + get_JsObject_property(&this, "start", context)?.try_js_into(context)?; + let pull: Option<UnderlyingSourcePullCallback> = + get_JsObject_property(&this, "pull", context)?.try_js_into(context)?; + let cancel: Option<UnderlyingSourceCancelCallback> = + get_JsObject_property(&this, "cancel", context)?.try_js_into(context)?; + let r#type = + get_JsObject_property(&this, "type", context)?.try_js_into(context)?; + let auto_allocate_chunk_size = + get_JsObject_property(&this, "autoAllocateChunkSize", context)? + .try_js_into(context)?; + Ok(UnderlyingSource { + this, + start, + pull, + cancel, + r#type, + auto_allocate_chunk_size, + }) + } +} + +impl Into<JsValue> for UnderlyingSource { + fn into(self) -> JsValue { + self.this.into() + } +} + +impl_into_js_from_into!(UnderlyingSource); + +pub trait UnderlyingSourceTrait { + /// **[start][spec](controller), of type UnderlyingSourceStartCallback** + /// + /// A function that is called immediately during creation of the ReadableStream. + /// + /// Typically this is used to adapt a push source by setting up relevant event listeners, as in the example of § 10.1 A readable stream with an underlying push source (no backpressure support), or to acquire access to a pull source, as in § 10.4 A readable stream with an underlying pull source. + /// + /// If this setup process is asynchronous, it can return a promise to signal success or failure; a rejected promise will error the stream. Any thrown exceptions will be re-thrown by the ReadableStream() constructor. + /// + /// [spec]: https://streams.spec.whatwg.org/#dom-underlyingsource-start + fn start( + &self, + controller: JsNativeObject<ReadableStreamController>, + context: &mut Context, + ) -> JsResult<JsValue>; + + /// **[pull][spec](controller), of type UnderlyingSourcePullCallback** + /// + /// A function that is called whenever the stream’s internal queue of chunks becomes not full, i.e. whenever the queue’s desired size becomes positive. Generally, it will be called repeatedly until the queue reaches its high water mark (i.e. until the desired size becomes non-positive). + /// + /// For push sources, this can be used to resume a paused flow, as in § 10.2 A readable stream with an underlying push source and backpressure support. For pull sources, it is used to acquire new chunks to enqueue into the stream, as in § 10.4 A readable stream with an underlying pull source. + /// + /// This function will not be called until start() successfully completes. Additionally, it will only be called repeatedly if it enqueues at least one chunk or fulfills a BYOB request; a no-op pull() implementation will not be continually called. + /// + /// If the function returns a promise, then it will not be called again until that promise fulfills. (If the promise rejects, the stream will become errored.) This is mainly used in the case of pull sources, where the promise returned represents the process of acquiring a new chunk. Throwing an exception is treated the same as returning a rejected promise. + /// + /// [spec]: https://streams.spec.whatwg.org/#dom-underlyingsource-pull + fn pull( + &self, + controller: JsNativeObject<ReadableStreamController>, + context: &mut Context, + ) -> JsResult<Option<JsPromise>>; + + /// **cancel(reason), of type UnderlyingSourceCancelCallback** + /// A function that is called whenever the consumer cancels the stream, via stream.cancel() or reader.cancel(). It takes as its argument the same value as was passed to those methods by the consumer. + /// + /// Readable streams can additionally be canceled under certain conditions during piping; see the definition of the pipeTo() method for more details. + /// + // For all streams, this is generally used to release access to the underlying resource; see for example § 10.1 A readable stream with an underlying push source (no backpressure support). + /// + /// If the shutdown process is asynchronous, it can return a promise to signal success or failure; the result will be communicated via the return value of the cancel() method that was called. Throwing an exception is treated the same as returning a rejected promise. + /// + /// [spec]: https://streams.spec.whatwg.org/#dom-underlyingsource-cancel + /// + /// *Even if the cancelation process fails, the stream will still close; it will not be put into an errored state. This is because a failure in the cancelation process doesn’t matter to the consumer’s view of the stream, once they’ve expressed disinterest in it by canceling. The failure is only communicated to the immediate caller of the corresponding method.* + /// + /// *This is different from the behavior of the close and abort options of a WritableStream's underlying sink, which upon failure put the corresponding WritableStream into an errored state. Those correspond to specific actions the producer is requesting and, if those actions fail, they indicate something more persistently wrong.* + fn cancel( + &self, + reason: Option<JsValue>, + context: &mut Context, + ) -> JsResult<Option<JsPromise>>; +} + +#[derive(Default)] +pub struct UndefinedUnderlyingSource {} + +impl UnderlyingSourceTrait for UndefinedUnderlyingSource { + fn start( + &self, + _controller: JsNativeObject<ReadableStreamController>, + _context: &mut Context, + ) -> JsResult<JsValue> { + Ok(JsValue::Undefined) // TODO spec link + } + + fn pull( + &self, + _controller: JsNativeObject<ReadableStreamController>, + context: &mut Context, + ) -> JsResult<Option<JsPromise>> { + JsPromise::resolve(JsValue::Undefined, context).map(Option::Some) // TODO spec link + } + + fn cancel( + &self, + _reason: Option<JsValue>, + context: &mut Context, + ) -> JsResult<Option<JsPromise>> { + JsPromise::resolve(JsValue::Undefined, context).map(Option::Some) // TODO spec link + } +} + +impl UnderlyingSourceTrait for UnderlyingSource { + fn start( + &self, + controller: JsNativeObject<ReadableStreamController>, + context: &mut Context, + ) -> JsResult<JsValue> { + if let Some(ref start) = self.start { + start.call( + self.this.clone(), // TODO remove clone? https://tezos-dev.slack.com/archives/C061SSDBN69/p1701192316869399 + (controller,), + context, + ) + } else { + UndefinedUnderlyingSource::default().start(controller, context) + } + } + + fn pull( + &self, + controller: JsNativeObject<ReadableStreamController>, + context: &mut Context, + ) -> JsResult<Option<JsPromise>> { + if let Some(ref pull) = self.pull { + pull.call( + self.this.clone(), // TODO remove clone? https://tezos-dev.slack.com/archives/C061SSDBN69/p1701192316869399 + (controller,), + context, + ) + } else { + UndefinedUnderlyingSource::default().pull(controller, context) + } + } + + fn cancel( + &self, + reason: Option<JsValue>, + context: &mut Context, + ) -> JsResult<Option<JsPromise>> { + if let Some(ref cancel) = self.cancel { + cancel.call( + self.this.clone(), // TODO remove clone? https://tezos-dev.slack.com/archives/C061SSDBN69/p1701192316869399 + (reason.unwrap_or(JsValue::Undefined),), + context, + ) + } else { + UndefinedUnderlyingSource::default().cancel(reason, context) + } + } +} + +impl UnderlyingSourceTrait for Option<UnderlyingSource> { + fn start( + &self, + controller: JsNativeObject<ReadableStreamController>, + context: &mut Context, + ) -> JsResult<JsValue> { + match self { + Some(underlying_source) => underlying_source.start(controller, context), + None => UndefinedUnderlyingSource::default().start(controller, context), + } + } + + fn pull( + &self, + controller: JsNativeObject<ReadableStreamController>, + context: &mut Context, + ) -> JsResult<Option<JsPromise>> { + match self { + Some(underlying_source) => underlying_source.pull(controller, context), + None => UndefinedUnderlyingSource::default().pull(controller, context), + } + } + + fn cancel( + &self, + reason: Option<JsValue>, + context: &mut Context, + ) -> JsResult<Option<JsPromise>> { + match self { + Some(underlying_source) => underlying_source.cancel(reason, context), + None => UndefinedUnderlyingSource::default().cancel(reason, context), + } + } +} + +/// typedef (ReadableStreamDefaultController or ReadableByteStreamController) [ReadableStreamController][spec]; +/// +/// [spec]: https://streams.spec.whatwg.org/#typedefdef-readablestreamcontroller +#[derive(Debug)] +pub enum ReadableStreamController { + DefaultController(ReadableStreamDefaultController), + ByteController(ReadableByteStreamController), +} + +impl Finalize for ReadableStreamController { + fn finalize(&self) {} +} + +unsafe impl Trace for ReadableStreamController { + custom_trace!(this, { + match this { + ReadableStreamController::DefaultController(value) => mark(value), + ReadableStreamController::ByteController(value) => mark(value), + } + }); +} + +/// callback [UnderlyingSourceStartCallback][spec] = any (ReadableStreamController controller); +/// +/// [spec]: https://streams.spec.whatwg.org/#callbackdef-underlyingsourcestartcallback +pub type UnderlyingSourceStartCallback = + JsFn<JsObject, 1, (JsNativeObject<ReadableStreamController>,), idl::Any>; + +/// callback [UnderlyingSourcePullCallback][spec] = Promise<undefined> (ReadableStreamController controller); +/// +/// [spec]: https://streams.spec.whatwg.org/#callbackdef-underlyingsourcepullcallback +pub type UnderlyingSourcePullCallback = + JsFn<JsObject, 1, (JsNativeObject<ReadableStreamController>,), Option<JsPromise>>; + +/// callback [UnderlyingSourceCancelCallback][spec] = Promise<undefined> (optional any reason); +/// +/// [spec]: https://streams.spec.whatwg.org/#callbackdef-underlyingsourcecancelcallback +pub type UnderlyingSourceCancelCallback = + JsFn<JsObject, 1, (idl::Any,), Option<JsPromise>>; + +#[derive(Debug, PartialEq)] +pub enum ReadableStreamType { + Bytes, +} + +impl Into<&str> for ReadableStreamType { + fn into(self) -> &'static str { + match self { + ReadableStreamType::Bytes => "bytes", + } + } +} + +impl FromStr for ReadableStreamType { + type Err = (); + + fn from_str(s: &str) -> Result<Self, Self::Err> { + match s { + "bytes" => Ok(ReadableStreamType::Bytes), + _ => Err(()), + } + } +} + +impl IntoJs for ReadableStreamType { + fn into_js(self, context: &mut Context<'_>) -> JsValue { + let str: &str = self.into(); + String::from(str).into_js(context) + } +} + +impl TryFromJs for ReadableStreamType { + fn try_from_js(value: &JsValue, context: &mut Context<'_>) -> JsResult<Self> { + let str = String::try_from_js(value, context)?; + ReadableStreamType::from_str(&str).map_err(|()| { + JsNativeError::typ() + .with_message(format!( + "{} is not a valid value for enumeration ReadableStreamType.", + str + )) + .into() + }) + } +} diff --git a/crates/jstz_api/src/stream/tmp.rs b/crates/jstz_api/src/stream/tmp.rs new file mode 100644 index 000000000..4b8e0d772 --- /dev/null +++ b/crates/jstz_api/src/stream/tmp.rs @@ -0,0 +1,6 @@ +//! Temporary definitions to allow compiling before defining all types + +use crate::todo::Todo; + +pub type ReadableStreamDefaultController = Todo; +pub type ReadableByteStreamController = Todo;