use std::ptr;
use std::rc::Rc;
use dom_struct::dom_struct;
use encoding_rs::UTF_8;
use js::jsapi::JSObject;
use js::realm::CurrentRealm;
use js::rust::HandleObject;
use js::typedarray::{ArrayBufferU8, Uint8};
use net_traits::filemanager_thread::RelativePos;
use rustc_hash::FxHashMap;
use servo_base::id::{BlobId, BlobIndex};
use servo_constellation_traits::{BlobData, BlobImpl};
use uuid::Uuid;
use crate::dom::bindings::buffer_source::create_buffer_source;
use crate::dom::bindings::codegen::Bindings::BlobBinding;
use crate::dom::bindings::codegen::Bindings::BlobBinding::BlobMethods;
use crate::dom::bindings::codegen::UnionTypes::ArrayBufferOrArrayBufferViewOrBlobOrString;
use crate::dom::bindings::error::{Error, Fallible};
use crate::dom::bindings::reflector::{
DomGlobal, Reflector, reflect_dom_object_with_proto, reflect_dom_object_with_proto_and_cx,
};
use crate::dom::bindings::root::DomRoot;
use crate::dom::bindings::serializable::Serializable;
use crate::dom::bindings::str::DOMString;
use crate::dom::bindings::structuredclone::StructuredData;
use crate::dom::globalscope::GlobalScope;
use crate::dom::promise::Promise;
use crate::dom::stream::readablestream::ReadableStream;
use crate::realms::InRealm;
use crate::script_runtime::CanGc;
#[dom_struct]
pub(crate) struct Blob {
reflector_: Reflector,
#[no_trace]
blob_id: BlobId,
}
impl Blob {
pub(crate) fn new(global: &GlobalScope, blob_impl: BlobImpl, can_gc: CanGc) -> DomRoot<Blob> {
Self::new_with_proto(global, None, blob_impl, can_gc)
}
fn new_with_proto(
global: &GlobalScope,
proto: Option<HandleObject>,
blob_impl: BlobImpl,
can_gc: CanGc,
) -> DomRoot<Blob> {
let dom_blob = reflect_dom_object_with_proto(
Box::new(Blob::new_inherited(&blob_impl)),
global,
proto,
can_gc,
);
global.track_blob(&dom_blob, blob_impl);
dom_blob
}
fn new_with_proto_and_cx(
global: &GlobalScope,
proto: Option<HandleObject>,
blob_impl: BlobImpl,
cx: &mut js::context::JSContext,
) -> DomRoot<Blob> {
let dom_blob = reflect_dom_object_with_proto_and_cx(
Box::new(Blob::new_inherited(&blob_impl)),
global,
proto,
cx,
);
global.track_blob(&dom_blob, blob_impl);
dom_blob
}
pub(crate) fn new_inherited(blob_impl: &BlobImpl) -> Blob {
Blob {
reflector_: Reflector::new(),
blob_id: blob_impl.blob_id(),
}
}
pub(crate) fn get_bytes(&self) -> Result<Vec<u8>, ()> {
self.global().get_blob_bytes(&self.blob_id)
}
pub(crate) fn type_string(&self) -> String {
self.global().get_blob_type_string(&self.blob_id)
}
pub(crate) fn get_blob_url_id(&self) -> Uuid {
self.global().get_blob_url_id(&self.blob_id)
}
pub(crate) fn get_stream(&self, can_gc: CanGc) -> Fallible<DomRoot<ReadableStream>> {
self.global().get_blob_stream(&self.blob_id, can_gc)
}
}
impl Serializable for Blob {
type Index = BlobIndex;
type Data = BlobImpl;
fn serialize(&self) -> Result<(BlobId, BlobImpl), ()> {
let blob_id = self.blob_id;
let blob_impl = self.global().serialize_blob(&blob_id);
let new_blob_id = blob_impl.blob_id();
Ok((new_blob_id, blob_impl))
}
fn deserialize(
owner: &GlobalScope,
serialized: BlobImpl,
can_gc: CanGc,
) -> Result<DomRoot<Self>, ()> {
let deserialized_blob = Blob::new(owner, serialized, can_gc);
Ok(deserialized_blob)
}
fn serialized_storage<'a>(
reader: StructuredData<'a, '_>,
) -> &'a mut Option<FxHashMap<BlobId, Self::Data>> {
match reader {
StructuredData::Reader(r) => &mut r.blob_impls,
StructuredData::Writer(w) => &mut w.blobs,
}
}
}
#[expect(unsafe_code)]
pub(crate) fn blob_parts_to_bytes(
mut blobparts: Vec<ArrayBufferOrArrayBufferViewOrBlobOrString>,
) -> Result<Vec<u8>, ()> {
let mut ret = vec![];
for blobpart in &mut blobparts {
match blobpart {
ArrayBufferOrArrayBufferViewOrBlobOrString::String(s) => {
ret.extend_from_slice(&s.as_bytes());
},
ArrayBufferOrArrayBufferViewOrBlobOrString::Blob(b) => {
let bytes = b.get_bytes().unwrap_or(vec![]);
ret.extend(bytes);
},
ArrayBufferOrArrayBufferViewOrBlobOrString::ArrayBuffer(a) => unsafe {
let bytes = a.as_slice();
ret.extend(bytes);
},
ArrayBufferOrArrayBufferViewOrBlobOrString::ArrayBufferView(a) => unsafe {
let bytes = a.as_slice();
ret.extend(bytes);
},
}
}
Ok(ret)
}
impl BlobMethods<crate::DomTypeHolder> for Blob {
#[expect(non_snake_case)]
fn Constructor(
cx: &mut js::context::JSContext,
global: &GlobalScope,
proto: Option<HandleObject>,
blobParts: Option<Vec<ArrayBufferOrArrayBufferViewOrBlobOrString>>,
blobPropertyBag: &BlobBinding::BlobPropertyBag,
) -> Fallible<DomRoot<Blob>> {
let bytes: Vec<u8> = match blobParts {
None => Vec::new(),
Some(blobparts) => match blob_parts_to_bytes(blobparts) {
Ok(bytes) => bytes,
Err(_) => return Err(Error::InvalidCharacter(None)),
},
};
let type_string = normalize_type_string(&blobPropertyBag.type_.str());
let blob_impl = BlobImpl::new_from_bytes(bytes, type_string);
Ok(Blob::new_with_proto_and_cx(global, proto, blob_impl, cx))
}
fn Size(&self) -> u64 {
self.global().get_blob_size(&self.blob_id)
}
fn Type(&self) -> DOMString {
DOMString::from(self.type_string())
}
fn Stream(&self, cx: &mut js::context::JSContext) -> Fallible<DomRoot<ReadableStream>> {
self.get_stream(CanGc::from_cx(cx))
}
fn Slice(
&self,
cx: &mut js::context::JSContext,
start: Option<i64>,
end: Option<i64>,
content_type: Option<DOMString>,
) -> DomRoot<Blob> {
let global = self.global();
let type_string = normalize_type_string(&content_type.unwrap_or_default().str());
let (parent, range) = match *global.get_blob_data(&self.blob_id) {
BlobData::Sliced(grandparent, parent_range) => {
let range = RelativePos {
start: parent_range.start + start.unwrap_or_default(),
end: end.map(|end| end + parent_range.start).or(parent_range.end),
};
(grandparent, range)
},
_ => (self.blob_id, RelativePos::from_opts(start, end)),
};
let blob_impl = BlobImpl::new_sliced(range, parent, type_string);
Blob::new(&global, blob_impl, CanGc::from_cx(cx))
}
fn Text(&self, cx: &mut CurrentRealm) -> Rc<Promise> {
let global = self.global();
let p = Promise::new_in_realm(cx);
let id = self.get_blob_url_id();
global.read_file_async(
id,
p.clone(),
Box::new(|cx, promise, bytes| match bytes {
Ok(b) => {
let (text, _) = UTF_8.decode_with_bom_removal(&b);
let text = DOMString::from(text);
promise.resolve_native(&text, CanGc::from_cx(cx));
},
Err(e) => {
promise.reject_error(e, CanGc::from_cx(cx));
},
}),
);
p
}
fn ArrayBuffer(&self, in_realm: InRealm, can_gc: CanGc) -> Rc<Promise> {
let cx = GlobalScope::get_cx();
let promise = Promise::new_in_current_realm(in_realm, can_gc);
let stream = self.get_stream(can_gc);
let reader = match stream.and_then(|s| s.acquire_default_reader(can_gc)) {
Ok(reader) => reader,
Err(error) => {
promise.reject_error(error, can_gc);
return promise;
},
};
let success_promise = promise.clone();
let failure_promise = promise.clone();
reader.read_all_bytes(
cx,
Rc::new(move |bytes| {
rooted!(in(*cx) let mut js_object = ptr::null_mut::<JSObject>());
let array_buffer = create_buffer_source::<ArrayBufferU8>(
cx,
bytes,
js_object.handle_mut(),
can_gc,
)
.expect("Converting input to ArrayBufferU8 should never fail");
success_promise.resolve_native(&array_buffer, can_gc);
}),
Rc::new(move |cx, value| {
failure_promise.reject(cx, value, can_gc);
}),
can_gc,
);
promise
}
fn Bytes(&self, in_realm: InRealm, can_gc: CanGc) -> Rc<Promise> {
let cx = GlobalScope::get_cx();
let p = Promise::new_in_current_realm(in_realm, can_gc);
let stream = self.get_stream(can_gc);
let reader = match stream.and_then(|s| s.acquire_default_reader(can_gc)) {
Ok(r) => r,
Err(e) => {
p.reject_error(e, can_gc);
return p;
},
};
let p_success = p.clone();
let p_failure = p.clone();
reader.read_all_bytes(
cx,
Rc::new(move |bytes| {
rooted!(in(*cx) let mut js_object = ptr::null_mut::<JSObject>());
let arr = create_buffer_source::<Uint8>(cx, bytes, js_object.handle_mut(), can_gc)
.expect("Converting input to uint8 array should never fail");
p_success.resolve_native(&arr, can_gc);
}),
Rc::new(move |cx, v| {
p_failure.reject(cx, v, can_gc);
}),
can_gc,
);
p
}
}
pub(crate) fn normalize_type_string(s: &str) -> String {
if is_ascii_printable(s) {
s.to_ascii_lowercase()
} else {
"".to_string()
}
}
fn is_ascii_printable(string: &str) -> bool {
string.chars().all(|c| ('\x20'..='\x7E').contains(&c))
}