/* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #![allow(dead_code)] use std::collections::VecDeque; use std::mem; use std::rc::Rc; use dom_struct::dom_struct; use js::gc::CustomAutoRooterGuard; use js::jsapi::Heap; use js::jsval::{JSVal, UndefinedValue}; use js::rust::{HandleObject as SafeHandleObject, HandleValue as SafeHandleValue}; use js::typedarray::ArrayBufferView; use super::bindings::reflector::reflect_dom_object; use super::readablestreamgenericreader::ReadableStreamGenericReader; use crate::dom::bindings::cell::DomRefCell; use crate::dom::bindings::codegen::Bindings::ReadableStreamBYOBReaderBinding::ReadableStreamBYOBReaderMethods; use crate::dom::bindings::error::Error; use crate::dom::bindings::import::module::Fallible; use crate::dom::bindings::reflector::{reflect_dom_object_with_proto, DomObject, Reflector}; use crate::dom::bindings::root::{DomRoot, MutNullableDom}; use crate::dom::bindings::trace::RootedTraceableBox; use crate::dom::globalscope::GlobalScope; use crate::dom::promise::Promise; use crate::dom::readablestream::ReadableStream; use crate::script_runtime::{CanGc, JSContext as SafeJSContext}; /// #[derive(JSTraceable, MallocSizeOf)] pub enum ReadIntoRequest { /// Read(#[ignore_malloc_size_of = "Rc is hard"] Rc), } impl ReadIntoRequest { /// pub fn chunk_steps(&self, _chunk: RootedTraceableBox>) { todo!() } /// pub fn close_steps(&self, _chunk: Option>>) { todo!() } /// pub(crate) fn error_steps(&self, _e: SafeHandleValue) { todo!() } } /// #[dom_struct] pub(crate) struct ReadableStreamBYOBReader { reflector_: Reflector, /// stream: MutNullableDom, read_into_requests: DomRefCell>, /// #[ignore_malloc_size_of = "Rc is hard"] closed_promise: DomRefCell>, } impl ReadableStreamBYOBReader { fn new_with_proto( global: &GlobalScope, proto: Option, can_gc: CanGc, ) -> DomRoot { reflect_dom_object_with_proto( Box::new(ReadableStreamBYOBReader::new_inherited(global, can_gc)), global, proto, can_gc, ) } fn new_inherited(global: &GlobalScope, can_gc: CanGc) -> ReadableStreamBYOBReader { ReadableStreamBYOBReader { reflector_: Reflector::new(), stream: MutNullableDom::new(None), read_into_requests: DomRefCell::new(Default::default()), closed_promise: DomRefCell::new(Promise::new(global, can_gc)), } } pub(crate) fn new(global: &GlobalScope, can_gc: CanGc) -> DomRoot { reflect_dom_object( Box::new(Self::new_inherited(global, can_gc)), global, can_gc, ) } /// pub(crate) fn set_up( &self, stream: &ReadableStream, global: &GlobalScope, can_gc: CanGc, ) -> Fallible<()> { // If ! IsReadableStreamLocked(stream) is true, throw a TypeError exception. if stream.is_locked() { return Err(Error::Type("stream is locked".to_owned())); } // If stream.[[controller]] does not implement ReadableByteStreamController, throw a TypeError exception. if !stream.has_byte_controller() { return Err(Error::Type( "stream controller is not a byte stream controller".to_owned(), )); } // Perform ! ReadableStreamReaderGenericInitialize(reader, stream). self.generic_initialize(global, stream, can_gc)?; // Set reader.[[readIntoRequests]] to a new empty list. self.read_into_requests.borrow_mut().clear(); Ok(()) } /// #[allow(unsafe_code)] pub(crate) fn release(&self) -> Fallible<()> { // Perform ! ReadableStreamReaderGenericRelease(reader). self.generic_release()?; // Let e be a new TypeError exception. let cx = GlobalScope::get_cx(); rooted!(in(*cx) let mut error = UndefinedValue()); unsafe { Error::Type("Reader is released".to_owned()).to_jsval( *cx, &self.global(), error.handle_mut(), ) }; // Perform ! ReadableStreamBYOBReaderErrorReadIntoRequests(reader, e). self.error_read_into_requests(error.handle()); Ok(()) } /// fn error_read_into_requests(&self, rval: SafeHandleValue) { // Let readRequests be reader.[[readRequests]]. let mut read_into_requests = self.take_read_into_requests(); // Set reader.[[readIntoRequests]] to a new empty list. for request in read_into_requests.drain(0..) { // Perform readIntoRequest’s error steps, given e. request.error_steps(rval); } } fn take_read_into_requests(&self) -> VecDeque { mem::take(&mut *self.read_into_requests.borrow_mut()) } /// pub(crate) fn close(&self) { // If reader is not undefined and reader implements ReadableStreamBYOBReader, // Let readIntoRequests be reader.[[readIntoRequests]]. let mut read_into_requests = self.take_read_into_requests(); // Set reader.[[readIntoRequests]] to an empty list. // Perform readIntoRequest’s close steps, given undefined. for request in read_into_requests.drain(0..) { // Perform readIntoRequest’s close steps, given undefined. request.close_steps(None); } } } impl ReadableStreamBYOBReaderMethods for ReadableStreamBYOBReader { /// fn Constructor( global: &GlobalScope, proto: Option, can_gc: CanGc, stream: &ReadableStream, ) -> Fallible> { let reader = Self::new_with_proto(global, proto, can_gc); // Perform ? SetUpReadableStreamBYOBReader(this, stream). Self::set_up(&reader, stream, global, can_gc)?; Ok(reader) } /// fn Read(&self, _view: CustomAutoRooterGuard, can_gc: CanGc) -> Rc { // TODO Promise::new(&self.reflector_.global(), can_gc) } /// fn ReleaseLock(&self) -> Fallible<()> { if self.stream.get().is_none() { // If this.[[stream]] is undefined, return. return Ok(()); } // Perform !ReadableStreamBYOBReaderRelease(this). self.release() } /// fn Closed(&self) -> Rc { self.closed() } /// fn Cancel(&self, _cx: SafeJSContext, reason: SafeHandleValue, can_gc: CanGc) -> Rc { self.cancel(&self.global(), reason, can_gc) } } impl ReadableStreamGenericReader for ReadableStreamBYOBReader { fn get_closed_promise(&self) -> Rc { self.closed_promise.borrow().clone() } fn set_closed_promise(&self, promise: Rc) { *self.closed_promise.borrow_mut() = promise; } fn set_stream(&self, stream: Option<&ReadableStream>) { self.stream.set(stream); } fn get_stream(&self) -> Option> { self.stream.get() } fn as_byob_reader(&self) -> Option<&ReadableStreamBYOBReader> { Some(self) } }