/* 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 https://mozilla.org/MPL/2.0/. */ use dom_struct::dom_struct; use js::jsapi::{JSObject, Type}; use js::rust::CustomAutoRooterGuard; use js::typedarray::{ArrayBufferView, ArrayBufferViewU8, TypedArray}; use servo_rand::{RngCore, ServoRng}; use uuid::Uuid; use crate::dom::bindings::cell::DomRefCell; use crate::dom::bindings::codegen::Bindings::CryptoBinding::CryptoMethods; use crate::dom::bindings::error::{Error, Fallible}; use crate::dom::bindings::reflector::{DomGlobal, Reflector, reflect_dom_object}; use crate::dom::bindings::root::{DomRoot, MutNullableDom}; use crate::dom::bindings::str::DOMString; use crate::dom::globalscope::GlobalScope; use crate::dom::subtlecrypto::SubtleCrypto; use crate::script_runtime::{CanGc, JSContext}; // https://developer.mozilla.org/en-US/docs/Web/API/Crypto #[dom_struct] pub(crate) struct Crypto { reflector_: Reflector, #[no_trace] rng: DomRefCell, subtle: MutNullableDom, } impl Crypto { fn new_inherited() -> Crypto { Crypto { reflector_: Reflector::new(), rng: DomRefCell::new(ServoRng::default()), subtle: MutNullableDom::default(), } } pub(crate) fn new(global: &GlobalScope, can_gc: CanGc) -> DomRoot { reflect_dom_object(Box::new(Crypto::new_inherited()), global, can_gc) } } impl CryptoMethods for Crypto { /// fn Subtle(&self, can_gc: CanGc) -> DomRoot { self.subtle .or_init(|| SubtleCrypto::new(&self.global(), can_gc)) } #[allow(unsafe_code)] // https://w3c.github.io/webcrypto/#Crypto-method-getRandomValues fn GetRandomValues( &self, _cx: JSContext, mut input: CustomAutoRooterGuard, ) -> Fallible { let array_type = input.get_array_type(); if !is_integer_buffer(array_type) { Err(Error::TypeMismatch) } else { let data = unsafe { input.as_mut_slice() }; if data.len() > 65536 { return Err(Error::QuotaExceeded); } self.rng.borrow_mut().fill_bytes(data); let underlying_object = unsafe { input.underlying_object() }; TypedArray::::from(*underlying_object) .map_err(|_| Error::JSFailed) } } // https://w3c.github.io/webcrypto/#Crypto-method-randomUUID fn RandomUUID(&self) -> DOMString { let uuid = Uuid::new_v4(); uuid.hyphenated() .encode_lower(&mut Uuid::encode_buffer()) .to_owned() .into() } } fn is_integer_buffer(array_type: Type) -> bool { matches!( array_type, Type::Uint8 | Type::Uint8Clamped | Type::Int8 | Type::Uint16 | Type::Int16 | Type::Uint32 | Type::Int32 | Type::BigInt64 | Type::BigUint64 ) }