use std::mem; use std::ops::Deref; use std::ptr; use super::Value; #[cfg(feature = "serde-json")] use super::ValueType; use crate::check_status; use crate::{sys, JsUnknown, NapiValue, Ref, Result}; pub struct JsBuffer(pub(crate) Value); pub struct JsBufferValue { pub(crate) value: JsBuffer, data: mem::ManuallyDrop>, } impl JsBuffer { #[inline] pub fn into_value(self) -> Result { let mut data = ptr::null_mut(); let mut len: usize = 0; check_status!(unsafe { sys::napi_get_buffer_info( self.0.env, self.0.value, &mut data, &mut len as *mut usize as *mut _, ) })?; Ok(JsBufferValue { data: mem::ManuallyDrop::new(unsafe { Vec::from_raw_parts(data as *mut _, len, len) }), value: self, }) } #[inline] pub fn into_ref(self) -> Result> { Ref::new(self.0, 1, self.into_value()?) } } impl JsBufferValue { #[cfg(feature = "serde-json")] #[inline] pub(crate) fn from_raw(env: sys::napi_env, value: sys::napi_value) -> Result { let mut data = ptr::null_mut(); let mut len = 0usize; check_status!(unsafe { sys::napi_get_buffer_info(env, value, &mut data, &mut len as *mut usize as *mut _) })?; Ok(Self { value: JsBuffer(Value { env, value, value_type: ValueType::Object, }), data: mem::ManuallyDrop::new(unsafe { Vec::from_raw_parts(data as *mut _, len, len) }), }) } #[inline] pub fn new(value: JsBuffer, data: Vec) -> Self { JsBufferValue { value, data: mem::ManuallyDrop::new(data), } } #[inline] pub fn into_raw(self) -> JsBuffer { self.value } #[inline] pub fn into_unknown(self) -> JsUnknown { unsafe { JsUnknown::from_raw_unchecked(self.value.0.env, self.value.0.value) } } } impl AsRef<[u8]> for JsBufferValue { fn as_ref(&self) -> &[u8] { self.data.as_slice() } } impl Deref for JsBufferValue { type Target = [u8]; fn deref(&self) -> &[u8] { self.data.as_slice() } }