commit
6a59c3549c
6 changed files with 129 additions and 10 deletions
|
@ -320,20 +320,19 @@ impl Env {
|
||||||
/// Provided `finalize_callback` will be called when `Buffer` got dropped.
|
/// Provided `finalize_callback` will be called when `Buffer` got dropped.
|
||||||
pub unsafe fn create_buffer_with_borrowed_data<Hint, Finalize>(
|
pub unsafe fn create_buffer_with_borrowed_data<Hint, Finalize>(
|
||||||
&self,
|
&self,
|
||||||
data: &[u8],
|
data: *const u8,
|
||||||
|
length: usize,
|
||||||
hint: Hint,
|
hint: Hint,
|
||||||
finalize_callback: Option<Finalize>,
|
finalize_callback: Option<Finalize>,
|
||||||
) -> Result<JsBufferValue>
|
) -> Result<JsBufferValue>
|
||||||
where
|
where
|
||||||
Finalize: FnOnce(Env, Hint),
|
Finalize: FnOnce(Hint, Env),
|
||||||
{
|
{
|
||||||
let length = data.len();
|
|
||||||
let mut raw_value = ptr::null_mut();
|
let mut raw_value = ptr::null_mut();
|
||||||
let data_ptr = data.as_ptr();
|
|
||||||
check_status!(sys::napi_create_external_buffer(
|
check_status!(sys::napi_create_external_buffer(
|
||||||
self.0,
|
self.0,
|
||||||
length,
|
length,
|
||||||
data_ptr as *mut c_void,
|
data as *mut c_void,
|
||||||
Some(
|
Some(
|
||||||
raw_finalize_with_custom_callback::<Hint, Finalize>
|
raw_finalize_with_custom_callback::<Hint, Finalize>
|
||||||
as unsafe extern "C" fn(
|
as unsafe extern "C" fn(
|
||||||
|
@ -351,7 +350,7 @@ impl Env {
|
||||||
value: raw_value,
|
value: raw_value,
|
||||||
value_type: ValueType::Object,
|
value_type: ValueType::Object,
|
||||||
}),
|
}),
|
||||||
mem::ManuallyDrop::new(Vec::from_raw_parts(data_ptr as *mut u8, length, length)),
|
mem::ManuallyDrop::new(Vec::from_raw_parts(data as *mut u8, length, length)),
|
||||||
))
|
))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1065,6 +1064,9 @@ impl Env {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// This function could be used for `create_buffer_with_borrowed_data` and want do noting when Buffer finalized.
|
||||||
|
pub fn noop_finalize<Hint>(_hint: Hint, _env: Env) {}
|
||||||
|
|
||||||
unsafe extern "C" fn drop_buffer(
|
unsafe extern "C" fn drop_buffer(
|
||||||
_env: sys::napi_env,
|
_env: sys::napi_env,
|
||||||
finalize_data: *mut c_void,
|
finalize_data: *mut c_void,
|
||||||
|
@ -1125,10 +1127,10 @@ unsafe extern "C" fn raw_finalize_with_custom_callback<Hint, Finalize>(
|
||||||
_finalize_data: *mut c_void,
|
_finalize_data: *mut c_void,
|
||||||
finalize_hint: *mut c_void,
|
finalize_hint: *mut c_void,
|
||||||
) where
|
) where
|
||||||
Finalize: FnOnce(Env, Hint),
|
Finalize: FnOnce(Hint, Env),
|
||||||
{
|
{
|
||||||
let (hint, maybe_callback) = *Box::from_raw(finalize_hint as *mut (Hint, Option<Finalize>));
|
let (hint, maybe_callback) = *Box::from_raw(finalize_hint as *mut (Hint, Option<Finalize>));
|
||||||
if let Some(callback) = maybe_callback {
|
if let Some(callback) = maybe_callback {
|
||||||
callback(Env::from_raw(env), hint);
|
callback(hint, Env::from_raw(env));
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
|
@ -8,6 +8,8 @@ use std::os::raw::{c_char, c_void};
|
||||||
|
|
||||||
#[cfg(feature = "serde-json")]
|
#[cfg(feature = "serde-json")]
|
||||||
use serde::{de, ser};
|
use serde::{de, ser};
|
||||||
|
#[cfg(feature = "serde-json")]
|
||||||
|
use serde_json::Error as SerdeJSONError;
|
||||||
|
|
||||||
use crate::{sys, Status};
|
use crate::{sys, Status};
|
||||||
|
|
||||||
|
@ -38,6 +40,13 @@ impl de::Error for Error {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[cfg(feature = "serde-json")]
|
||||||
|
impl From<SerdeJSONError> for Error {
|
||||||
|
fn from(value: SerdeJSONError) -> Self {
|
||||||
|
Error::new(Status::InvalidArg, format!("{}", value))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl fmt::Display for Error {
|
impl fmt::Display for Error {
|
||||||
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
||||||
if !self.reason.is_empty() {
|
if !self.reason.is_empty() {
|
||||||
|
|
|
@ -18,3 +18,17 @@ test('should copy', (t) => {
|
||||||
t.deepEqual(copyBuffer, fixture)
|
t.deepEqual(copyBuffer, fixture)
|
||||||
t.not(fixture, copyBuffer)
|
t.not(fixture, copyBuffer)
|
||||||
})
|
})
|
||||||
|
|
||||||
|
test('should create borrowed buffer with noop finalize', (t) => {
|
||||||
|
t.deepEqual(
|
||||||
|
bindings.createBorrowedBufferWithNoopFinalize(),
|
||||||
|
Buffer.from([1, 2, 3]),
|
||||||
|
)
|
||||||
|
})
|
||||||
|
|
||||||
|
test('should create borrowed buffer with finalize', (t) => {
|
||||||
|
t.deepEqual(
|
||||||
|
bindings.createBorrowedBufferWithFinalize(),
|
||||||
|
Buffer.from([1, 2, 3]),
|
||||||
|
)
|
||||||
|
})
|
||||||
|
|
28
test_module/__test__/serde/serde-json.spec.ts
Normal file
28
test_module/__test__/serde/serde-json.spec.ts
Normal file
|
@ -0,0 +1,28 @@
|
||||||
|
import test from 'ava'
|
||||||
|
|
||||||
|
const bindings = require('../../index.node')
|
||||||
|
|
||||||
|
const ValidObject = {
|
||||||
|
a: 1,
|
||||||
|
b: [-1.2, 1.1, 2.2, 3.3],
|
||||||
|
c: 'Hi',
|
||||||
|
}
|
||||||
|
|
||||||
|
const InValidObject = {
|
||||||
|
a: -1,
|
||||||
|
b: [-1, 1.1, 2.2, 3.3],
|
||||||
|
c: 'Hello',
|
||||||
|
}
|
||||||
|
|
||||||
|
test('should from json string', (t) => {
|
||||||
|
t.throws(() => bindings.from_json_string(JSON.stringify(InValidObject)))
|
||||||
|
t.deepEqual(
|
||||||
|
ValidObject,
|
||||||
|
bindings.from_json_string(JSON.stringify(ValidObject)),
|
||||||
|
)
|
||||||
|
})
|
||||||
|
|
||||||
|
test('should convert to json string', (t) => {
|
||||||
|
t.throws(() => bindings.json_to_string(InValidObject))
|
||||||
|
t.deepEqual(JSON.stringify(ValidObject), bindings.json_to_string(ValidObject))
|
||||||
|
})
|
|
@ -1,6 +1,10 @@
|
||||||
|
use std::mem::ManuallyDrop;
|
||||||
use std::str;
|
use std::str;
|
||||||
|
|
||||||
use napi::{CallContext, Error, JsBuffer, JsNumber, JsObject, JsString, Result, Status};
|
use napi::{
|
||||||
|
noop_finalize, CallContext, ContextlessResult, Env, Error, JsBuffer, JsNumber, JsObject,
|
||||||
|
JsString, Result, Status,
|
||||||
|
};
|
||||||
|
|
||||||
#[js_function(1)]
|
#[js_function(1)]
|
||||||
pub fn get_buffer_length(ctx: CallContext) -> Result<JsNumber> {
|
pub fn get_buffer_length(ctx: CallContext) -> Result<JsNumber> {
|
||||||
|
@ -22,9 +26,50 @@ pub fn copy_buffer(ctx: CallContext) -> Result<JsBuffer> {
|
||||||
ctx.env.create_buffer_copy(buffer).map(|b| b.into_raw())
|
ctx.env.create_buffer_copy(buffer).map(|b| b.into_raw())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[contextless_function]
|
||||||
|
pub fn create_borrowed_buffer_with_noop_finalize(env: Env) -> ContextlessResult<JsBuffer> {
|
||||||
|
let data = vec![1, 2, 3];
|
||||||
|
let data_ptr = data.as_ptr();
|
||||||
|
let length = data.len();
|
||||||
|
let manually_drop = ManuallyDrop::new(data);
|
||||||
|
|
||||||
|
unsafe {
|
||||||
|
env.create_buffer_with_borrowed_data(data_ptr, length, manually_drop, Some(noop_finalize))
|
||||||
|
}
|
||||||
|
.map(|b| Some(b.into_raw()))
|
||||||
|
}
|
||||||
|
|
||||||
|
#[contextless_function]
|
||||||
|
pub fn create_borrowed_buffer_with_finalize(env: Env) -> ContextlessResult<JsBuffer> {
|
||||||
|
let data = vec![1, 2, 3];
|
||||||
|
let data_ptr = data.as_ptr();
|
||||||
|
let length = data.len();
|
||||||
|
let manually_drop = ManuallyDrop::new(data);
|
||||||
|
|
||||||
|
unsafe {
|
||||||
|
env.create_buffer_with_borrowed_data(
|
||||||
|
data_ptr,
|
||||||
|
length,
|
||||||
|
manually_drop,
|
||||||
|
Some(|mut hint: ManuallyDrop<Vec<u8>>, _| {
|
||||||
|
ManuallyDrop::drop(&mut hint);
|
||||||
|
}),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
.map(|b| Some(b.into_raw()))
|
||||||
|
}
|
||||||
|
|
||||||
pub fn register_js(exports: &mut JsObject) -> Result<()> {
|
pub fn register_js(exports: &mut JsObject) -> Result<()> {
|
||||||
exports.create_named_method("getBufferLength", get_buffer_length)?;
|
exports.create_named_method("getBufferLength", get_buffer_length)?;
|
||||||
exports.create_named_method("bufferToString", buffer_to_string)?;
|
exports.create_named_method("bufferToString", buffer_to_string)?;
|
||||||
exports.create_named_method("copyBuffer", copy_buffer)?;
|
exports.create_named_method("copyBuffer", copy_buffer)?;
|
||||||
|
exports.create_named_method(
|
||||||
|
"createBorrowedBufferWithNoopFinalize",
|
||||||
|
create_borrowed_buffer_with_noop_finalize,
|
||||||
|
)?;
|
||||||
|
exports.create_named_method(
|
||||||
|
"createBorrowedBufferWithFinalize",
|
||||||
|
create_borrowed_buffer_with_finalize,
|
||||||
|
)?;
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,4 +1,6 @@
|
||||||
use napi::{CallContext, JsObject, JsUndefined, JsUnknown, Result};
|
use napi::{CallContext, JsObject, JsString, JsUndefined, JsUnknown, Result};
|
||||||
|
|
||||||
|
use serde_json::{from_str, to_string};
|
||||||
|
|
||||||
#[derive(Serialize, Debug, Deserialize)]
|
#[derive(Serialize, Debug, Deserialize)]
|
||||||
struct AnObject {
|
struct AnObject {
|
||||||
|
@ -162,6 +164,23 @@ fn roundtrip_object(ctx: CallContext) -> Result<JsUnknown> {
|
||||||
ctx.env.to_js_value(&de_serialized)
|
ctx.env.to_js_value(&de_serialized)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[js_function(1)]
|
||||||
|
fn from_json_string(ctx: CallContext) -> Result<JsUnknown> {
|
||||||
|
let arg0 = ctx.get::<JsString>(0)?.into_utf8()?;
|
||||||
|
|
||||||
|
let de_serialized: AnObject = from_str(arg0.as_str()?)?;
|
||||||
|
ctx.env.to_js_value(&de_serialized)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[js_function(1)]
|
||||||
|
fn json_to_string(ctx: CallContext) -> Result<JsString> {
|
||||||
|
let arg0 = ctx.get::<JsObject>(0)?;
|
||||||
|
|
||||||
|
let de_serialized: AnObject = ctx.env.from_js_value(arg0)?;
|
||||||
|
let json_string = to_string(&de_serialized)?;
|
||||||
|
ctx.env.create_string_from_std(json_string)
|
||||||
|
}
|
||||||
|
|
||||||
pub fn register_js(exports: &mut JsObject) -> Result<()> {
|
pub fn register_js(exports: &mut JsObject) -> Result<()> {
|
||||||
exports.create_named_method("make_num_77", make_num_77)?;
|
exports.create_named_method("make_num_77", make_num_77)?;
|
||||||
exports.create_named_method("make_num_32", make_num_32)?;
|
exports.create_named_method("make_num_32", make_num_32)?;
|
||||||
|
@ -178,5 +197,7 @@ pub fn register_js(exports: &mut JsObject) -> Result<()> {
|
||||||
exports.create_named_method("expect_buffer", expect_buffer)?;
|
exports.create_named_method("expect_buffer", expect_buffer)?;
|
||||||
|
|
||||||
exports.create_named_method("roundtrip_object", roundtrip_object)?;
|
exports.create_named_method("roundtrip_object", roundtrip_object)?;
|
||||||
|
exports.create_named_method("from_json_string", from_json_string)?;
|
||||||
|
exports.create_named_method("json_to_string", json_to_string)?;
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue