use std::cell::{Cell, RefCell};
use std::os::raw::c_void;
use std::ptr::{self, NonNull};
use std::rc::Rc;
use std::slice;
use dom_struct::dom_struct;
use js::glue::{
CreateReadableStreamUnderlyingSource, DeleteReadableStreamUnderlyingSource,
ReadableStreamUnderlyingSourceTraps,
};
use js::jsapi::{
AutoRequireNoGC, HandleObject, HandleValue, Heap, IsReadableStream, JSContext, JSObject,
JS_GetArrayBufferViewData, NewReadableExternalSourceStreamObject, ReadableStreamClose,
ReadableStreamDefaultReaderRead, ReadableStreamError, ReadableStreamGetReader,
ReadableStreamIsDisturbed, ReadableStreamIsLocked, ReadableStreamIsReadable,
ReadableStreamReaderMode, ReadableStreamReaderReleaseLock, ReadableStreamUnderlyingSource,
ReadableStreamUpdateDataAvailableFromSource, UnwrapReadableStream,
};
use js::jsval::{JSVal, UndefinedValue};
use js::rust::{HandleValue as SafeHandleValue, IntoHandle};
use crate::dom::bindings::conversions::{ConversionBehavior, ConversionResult};
use crate::dom::bindings::error::Error;
use crate::dom::bindings::reflector::{reflect_dom_object, DomObject, Reflector};
use crate::dom::bindings::root::DomRoot;
use crate::dom::bindings::settings_stack::{AutoEntryScript, AutoIncumbentScript};
use crate::dom::bindings::utils::get_dictionary_property;
use crate::dom::globalscope::GlobalScope;
use crate::dom::promise::Promise;
use crate::js::conversions::FromJSValConvertible;
use crate::realms::{enter_realm, InRealm};
use crate::script_runtime::JSContext as SafeJSContext;
static UNDERLYING_SOURCE_TRAPS: ReadableStreamUnderlyingSourceTraps =
ReadableStreamUnderlyingSourceTraps {
requestData: Some(request_data),
writeIntoReadRequestBuffer: Some(write_into_read_request_buffer),
cancel: Some(cancel),
onClosed: Some(close),
onErrored: Some(error),
finalize: Some(finalize),
};
#[dom_struct]
pub struct ReadableStream {
reflector_: Reflector,
#[ignore_malloc_size_of = "SM handles JS values"]
js_stream: Heap<*mut JSObject>,
#[ignore_malloc_size_of = "SM handles JS values"]
js_reader: Heap<*mut JSObject>,
has_reader: Cell<bool>,
#[ignore_malloc_size_of = "Rc is hard"]
external_underlying_source: Option<Rc<ExternalUnderlyingSourceController>>,
}
impl ReadableStream {
fn new_inherited(
external_underlying_source: Option<Rc<ExternalUnderlyingSourceController>>,
) -> ReadableStream {
ReadableStream {
reflector_: Reflector::new(),
js_stream: Heap::default(),
js_reader: Heap::default(),
has_reader: Default::default(),
external_underlying_source,
}
}
fn new(
global: &GlobalScope,
external_underlying_source: Option<Rc<ExternalUnderlyingSourceController>>,
) -> DomRoot<ReadableStream> {
reflect_dom_object(
Box::new(ReadableStream::new_inherited(external_underlying_source)),
global,
)
}
#[allow(unsafe_code)]
pub unsafe fn from_js(
cx: SafeJSContext,
obj: *mut JSObject,
realm: InRealm,
) -> Result<DomRoot<ReadableStream>, ()> {
if !IsReadableStream(obj) {
return Err(());
}
let global = GlobalScope::from_safe_context(cx, realm);
let stream = ReadableStream::new(&global, None);
stream.js_stream.set(UnwrapReadableStream(obj));
Ok(stream)
}
pub fn new_from_bytes(global: &GlobalScope, bytes: Vec<u8>) -> DomRoot<ReadableStream> {
let stream = ReadableStream::new_with_external_underlying_source(
global,
ExternalUnderlyingSource::Memory(bytes.len()),
);
stream.enqueue_native(bytes);
stream.close_native();
stream
}
#[allow(unsafe_code)]
pub fn new_with_external_underlying_source(
global: &GlobalScope,
source: ExternalUnderlyingSource,
) -> DomRoot<ReadableStream> {
let _ar = enter_realm(global);
let _ais = AutoIncumbentScript::new(global);
let cx = GlobalScope::get_cx();
let source = Rc::new(ExternalUnderlyingSourceController::new(source));
let stream = ReadableStream::new(global, Some(source.clone()));
unsafe {
let js_wrapper = CreateReadableStreamUnderlyingSource(
&UNDERLYING_SOURCE_TRAPS,
&*source as *const _ as *const c_void,
);
rooted!(in(*cx)
let js_stream = NewReadableExternalSourceStreamObject(
*cx,
js_wrapper,
ptr::null_mut(),
HandleObject::null(),
)
);
stream.js_stream.set(UnwrapReadableStream(js_stream.get()));
}
stream
}
pub fn get_js_stream(&self) -> NonNull<JSObject> {
NonNull::new(self.js_stream.get())
.expect("Couldn't get a non-null pointer to JS stream object.")
}
#[allow(unsafe_code)]
pub fn enqueue_native(&self, bytes: Vec<u8>) {
let global = self.global();
let _ar = enter_realm(&*global);
let cx = GlobalScope::get_cx();
let handle = unsafe { self.js_stream.handle() };
self.external_underlying_source
.as_ref()
.expect("No external source to enqueue bytes.")
.enqueue_chunk(cx, handle, bytes);
}
#[allow(unsafe_code)]
pub fn error_native(&self, error: Error) {
let global = self.global();
let _ar = enter_realm(&*global);
let cx = GlobalScope::get_cx();
unsafe {
rooted!(in(*cx) let mut js_error = UndefinedValue());
error.to_jsval(*cx, &global, js_error.handle_mut());
ReadableStreamError(
*cx,
self.js_stream.handle(),
js_error.handle().into_handle(),
);
}
}
#[allow(unsafe_code)]
pub fn close_native(&self) {
let global = self.global();
let _ar = enter_realm(&*global);
let cx = GlobalScope::get_cx();
let handle = unsafe { self.js_stream.handle() };
self.external_underlying_source
.as_ref()
.expect("No external source to close.")
.close(cx, handle);
}
pub fn in_memory(&self) -> bool {
self.external_underlying_source
.as_ref()
.map(|source| source.in_memory())
.unwrap_or(false)
}
pub fn get_in_memory_bytes(&self) -> Option<Vec<u8>> {
self.external_underlying_source
.as_ref()
.and_then(|source| source.get_in_memory_bytes())
}
#[allow(unsafe_code)]
pub fn start_reading(&self) -> Result<(), ()> {
if self.is_locked() || self.is_disturbed() {
return Err(());
}
let global = self.global();
let _ar = enter_realm(&*global);
let cx = GlobalScope::get_cx();
unsafe {
rooted!(in(*cx) let reader = ReadableStreamGetReader(
*cx,
self.js_stream.handle(),
ReadableStreamReaderMode::Default,
));
self.js_reader.set(reader.get());
}
self.has_reader.set(true);
Ok(())
}
#[allow(unsafe_code)]
pub fn read_a_chunk(&self) -> Rc<Promise> {
if !self.has_reader.get() {
panic!("Attempt to read stream chunk without having acquired a reader.");
}
let global = self.global();
let _ar = enter_realm(&*global);
let _aes = AutoEntryScript::new(&global);
let cx = GlobalScope::get_cx();
unsafe {
rooted!(in(*cx) let promise_obj = ReadableStreamDefaultReaderRead(
*cx,
self.js_reader.handle(),
));
Promise::new_with_js_promise(promise_obj.handle(), cx)
}
}
#[allow(unsafe_code)]
pub fn stop_reading(&self) {
if !self.has_reader.get() {
panic!("ReadableStream::stop_reading called on a readerless stream.");
}
self.has_reader.set(false);
let global = self.global();
let _ar = enter_realm(&*global);
let cx = GlobalScope::get_cx();
unsafe {
ReadableStreamReaderReleaseLock(*cx, self.js_reader.handle());
self.js_reader.set(ptr::null_mut());
}
}
#[allow(unsafe_code)]
pub fn is_locked(&self) -> bool {
if self.has_reader.get() {
return true;
}
let cx = GlobalScope::get_cx();
let mut locked_or_disturbed = false;
unsafe {
ReadableStreamIsLocked(*cx, self.js_stream.handle(), &mut locked_or_disturbed);
}
locked_or_disturbed
}
#[allow(unsafe_code)]
pub fn is_disturbed(&self) -> bool {
let cx = GlobalScope::get_cx();
let mut locked_or_disturbed = false;
unsafe {
ReadableStreamIsDisturbed(*cx, self.js_stream.handle(), &mut locked_or_disturbed);
}
locked_or_disturbed
}
}
#[allow(unsafe_code)]
unsafe extern "C" fn request_data(
source: *const c_void,
cx: *mut JSContext,
stream: HandleObject,
desired_size: usize,
) {
let source = &*(source as *const ExternalUnderlyingSourceController);
source.pull(SafeJSContext::from_ptr(cx), stream, desired_size);
}
#[allow(unsafe_code)]
unsafe extern "C" fn write_into_read_request_buffer(
source: *const c_void,
_cx: *mut JSContext,
_stream: HandleObject,
chunk: HandleObject,
length: usize,
bytes_written: *mut usize,
) {
let source = &*(source as *const ExternalUnderlyingSourceController);
let mut is_shared_memory = false;
let buffer = JS_GetArrayBufferViewData(
*chunk,
&mut is_shared_memory,
&AutoRequireNoGC { _address: 0 },
);
assert!(!is_shared_memory);
let slice = slice::from_raw_parts_mut(buffer as *mut u8, length);
source.write_into_buffer(slice);
*bytes_written = length;
}
#[allow(unsafe_code)]
unsafe extern "C" fn cancel(
_source: *const c_void,
_cx: *mut JSContext,
_stream: HandleObject,
_reason: HandleValue,
_resolve_to: *mut JSVal,
) {
}
#[allow(unsafe_code)]
unsafe extern "C" fn close(_source: *const c_void, _cx: *mut JSContext, _stream: HandleObject) {}
#[allow(unsafe_code)]
unsafe extern "C" fn error(
_source: *const c_void,
_cx: *mut JSContext,
_stream: HandleObject,
_reason: HandleValue,
) {
}
#[allow(unsafe_code)]
unsafe extern "C" fn finalize(source: *mut ReadableStreamUnderlyingSource) {
DeleteReadableStreamUnderlyingSource(source);
}
pub enum ExternalUnderlyingSource {
Memory(usize),
Blob(usize),
FetchResponse,
}
#[derive(JSTraceable, MallocSizeOf)]
struct ExternalUnderlyingSourceController {
buffer: RefCell<Vec<u8>>,
closed: Cell<bool>,
in_memory: Cell<bool>,
}
impl ExternalUnderlyingSourceController {
fn new(source: ExternalUnderlyingSource) -> ExternalUnderlyingSourceController {
let (buffer, in_mem) = match source {
ExternalUnderlyingSource::Blob(size) => (Vec::with_capacity(size), false),
ExternalUnderlyingSource::Memory(size) => (Vec::with_capacity(size), true),
ExternalUnderlyingSource::FetchResponse => (vec![], false),
};
ExternalUnderlyingSourceController {
buffer: RefCell::new(buffer),
closed: Cell::new(false),
in_memory: Cell::new(in_mem),
}
}
pub fn in_memory(&self) -> bool {
self.in_memory.get()
}
pub fn get_in_memory_bytes(&self) -> Option<Vec<u8>> {
if self.in_memory.get() {
return Some(self.buffer.borrow().clone());
}
None
}
#[allow(unsafe_code)]
fn maybe_signal_available_bytes(
&self,
cx: SafeJSContext,
stream: HandleObject,
available: usize,
) {
if available == 0 {
return;
}
unsafe {
let mut readable = false;
if !ReadableStreamIsReadable(*cx, stream, &mut readable) {
return;
}
if readable {
ReadableStreamUpdateDataAvailableFromSource(*cx, stream, available as u32);
}
}
}
#[allow(unsafe_code)]
fn maybe_close_js_stream(&self, cx: SafeJSContext, stream: HandleObject) {
unsafe {
let mut readable = false;
if !ReadableStreamIsReadable(*cx, stream, &mut readable) {
return;
}
if readable {
ReadableStreamClose(*cx, stream);
}
}
}
fn close(&self, cx: SafeJSContext, stream: HandleObject) {
self.closed.set(true);
self.maybe_close_js_stream(cx, stream);
}
fn enqueue_chunk(&self, cx: SafeJSContext, stream: HandleObject, mut chunk: Vec<u8>) {
let available = {
let mut buffer = self.buffer.borrow_mut();
chunk.append(&mut buffer);
*buffer = chunk;
buffer.len()
};
self.maybe_signal_available_bytes(cx, stream, available);
}
#[allow(unsafe_code)]
fn pull(&self, cx: SafeJSContext, stream: HandleObject, _desired_size: usize) {
if self.closed.get() {
return self.maybe_close_js_stream(cx, stream);
}
let available = {
let buffer = self.buffer.borrow();
buffer.len()
};
self.maybe_signal_available_bytes(cx, stream, available);
}
fn get_chunk_with_length(&self, length: usize) -> Vec<u8> {
let mut buffer = self.buffer.borrow_mut();
let buffer_len = buffer.len();
assert!(buffer_len >= length);
buffer.split_off(buffer_len - length)
}
fn write_into_buffer(&self, dest: &mut [u8]) {
let length = dest.len();
let chunk = self.get_chunk_with_length(length);
dest.copy_from_slice(chunk.as_slice());
}
}
#[allow(unsafe_code)]
pub fn get_read_promise_done(cx: SafeJSContext, v: &SafeHandleValue) -> Result<bool, Error> {
unsafe {
rooted!(in(*cx) let object = v.to_object());
rooted!(in(*cx) let mut done = UndefinedValue());
match get_dictionary_property(*cx, object.handle(), "done", done.handle_mut()) {
Ok(true) => match bool::from_jsval(*cx, done.handle(), ()) {
Ok(ConversionResult::Success(val)) => Ok(val),
Ok(ConversionResult::Failure(error)) => Err(Error::Type(error.to_string())),
_ => Err(Error::Type("Unknown format for done property.".to_string())),
},
Ok(false) => Err(Error::Type("Promise has no done property.".to_string())),
Err(()) => Err(Error::JSFailed),
}
}
}
#[allow(unsafe_code)]
pub fn get_read_promise_bytes(cx: SafeJSContext, v: &SafeHandleValue) -> Result<Vec<u8>, Error> {
unsafe {
rooted!(in(*cx) let object = v.to_object());
rooted!(in(*cx) let mut bytes = UndefinedValue());
match get_dictionary_property(*cx, object.handle(), "value", bytes.handle_mut()) {
Ok(true) => {
match Vec::<u8>::from_jsval(*cx, bytes.handle(), ConversionBehavior::EnforceRange) {
Ok(ConversionResult::Success(val)) => Ok(val),
Ok(ConversionResult::Failure(error)) => Err(Error::Type(error.to_string())),
_ => Err(Error::Type("Unknown format for bytes read.".to_string())),
}
},
Ok(false) => Err(Error::Type("Promise has no value property.".to_string())),
Err(()) => Err(Error::JSFailed),
}
}
}