2020-01-02 15:13:47 -05:00
|
|
|
// Copyright 2018-2020 the Deno authors. All rights reserved. MIT license.
|
2020-02-23 14:51:29 -05:00
|
|
|
use crate::op_error::OpError;
|
2020-01-05 11:56:18 -05:00
|
|
|
use deno_core::*;
|
2019-11-17 01:17:47 +01:00
|
|
|
use futures::future::FutureExt;
|
2019-08-22 22:30:14 -07:00
|
|
|
pub use serde_derive::Deserialize;
|
|
|
|
use serde_json::json;
|
|
|
|
pub use serde_json::Value;
|
2019-11-17 01:17:47 +01:00
|
|
|
use std::future::Future;
|
|
|
|
use std::pin::Pin;
|
2019-08-22 22:30:14 -07:00
|
|
|
|
2020-02-23 14:51:29 -05:00
|
|
|
pub type JsonResult = Result<Value, OpError>;
|
2020-02-03 18:08:44 -05:00
|
|
|
|
|
|
|
pub type AsyncJsonOp = Pin<Box<dyn Future<Output = JsonResult>>>;
|
2019-08-22 22:30:14 -07:00
|
|
|
|
|
|
|
pub enum JsonOp {
|
|
|
|
Sync(Value),
|
|
|
|
Async(AsyncJsonOp),
|
2020-01-22 02:01:10 +09:00
|
|
|
/// AsyncUnref is the variation of Async, which doesn't block the program
|
|
|
|
/// exiting.
|
|
|
|
AsyncUnref(AsyncJsonOp),
|
2019-08-22 22:30:14 -07:00
|
|
|
}
|
|
|
|
|
2020-02-23 14:51:29 -05:00
|
|
|
fn json_err(err: OpError) -> Value {
|
2019-08-22 22:30:14 -07:00
|
|
|
json!({
|
2020-02-23 14:51:29 -05:00
|
|
|
"message": err.msg,
|
|
|
|
"kind": err.kind as u32,
|
2019-08-22 22:30:14 -07:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2020-02-03 18:08:44 -05:00
|
|
|
fn serialize_result(promise_id: Option<u64>, result: JsonResult) -> Buf {
|
2019-08-22 22:30:14 -07:00
|
|
|
let value = match result {
|
|
|
|
Ok(v) => json!({ "ok": v, "promiseId": promise_id }),
|
|
|
|
Err(err) => json!({ "err": json_err(err), "promiseId": promise_id }),
|
|
|
|
};
|
2020-02-09 13:54:16 -05:00
|
|
|
serde_json::to_vec(&value).unwrap().into_boxed_slice()
|
2019-08-22 22:30:14 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Deserialize)]
|
|
|
|
#[serde(rename_all = "camelCase")]
|
|
|
|
struct AsyncArgs {
|
|
|
|
promise_id: Option<u64>,
|
|
|
|
}
|
|
|
|
|
2020-01-24 15:10:49 -05:00
|
|
|
pub fn json_op<D>(d: D) -> impl Fn(&[u8], Option<ZeroCopyBuf>) -> CoreOp
|
2019-10-02 00:51:05 +02:00
|
|
|
where
|
2020-02-23 14:51:29 -05:00
|
|
|
D: Fn(Value, Option<ZeroCopyBuf>) -> Result<JsonOp, OpError>,
|
2019-10-02 00:51:05 +02:00
|
|
|
{
|
2020-01-24 15:10:49 -05:00
|
|
|
move |control: &[u8], zero_copy: Option<ZeroCopyBuf>| {
|
2019-10-25 19:23:16 +02:00
|
|
|
let async_args: AsyncArgs = match serde_json::from_slice(control) {
|
|
|
|
Ok(args) => args,
|
|
|
|
Err(e) => {
|
2020-02-23 14:51:29 -05:00
|
|
|
let buf = serialize_result(None, Err(OpError::from(e)));
|
2019-10-25 19:23:16 +02:00
|
|
|
return CoreOp::Sync(buf);
|
|
|
|
}
|
|
|
|
};
|
2019-10-02 00:51:05 +02:00
|
|
|
let promise_id = async_args.promise_id;
|
|
|
|
let is_sync = promise_id.is_none();
|
2019-08-22 22:30:14 -07:00
|
|
|
|
2019-10-02 00:51:05 +02:00
|
|
|
let result = serde_json::from_slice(control)
|
2020-02-23 14:51:29 -05:00
|
|
|
.map_err(OpError::from)
|
2019-10-02 00:51:05 +02:00
|
|
|
.and_then(|args| d(args, zero_copy));
|
|
|
|
|
|
|
|
// Convert to CoreOp
|
|
|
|
match result {
|
|
|
|
Ok(JsonOp::Sync(sync_value)) => {
|
|
|
|
assert!(promise_id.is_none());
|
|
|
|
CoreOp::Sync(serialize_result(promise_id, Ok(sync_value)))
|
|
|
|
}
|
|
|
|
Ok(JsonOp::Async(fut)) => {
|
|
|
|
assert!(promise_id.is_some());
|
2019-11-17 01:17:47 +01:00
|
|
|
let fut2 = fut.then(move |result| {
|
|
|
|
futures::future::ok(serialize_result(promise_id, result))
|
|
|
|
});
|
2020-02-03 18:08:44 -05:00
|
|
|
CoreOp::Async(fut2.boxed_local())
|
2019-10-02 00:51:05 +02:00
|
|
|
}
|
2020-01-22 02:01:10 +09:00
|
|
|
Ok(JsonOp::AsyncUnref(fut)) => {
|
|
|
|
assert!(promise_id.is_some());
|
|
|
|
let fut2 = fut.then(move |result| {
|
|
|
|
futures::future::ok(serialize_result(promise_id, result))
|
|
|
|
});
|
2020-02-03 18:08:44 -05:00
|
|
|
CoreOp::AsyncUnref(fut2.boxed_local())
|
2020-01-22 02:01:10 +09:00
|
|
|
}
|
2019-10-02 00:51:05 +02:00
|
|
|
Err(sync_err) => {
|
|
|
|
let buf = serialize_result(promise_id, Err(sync_err));
|
|
|
|
if is_sync {
|
|
|
|
CoreOp::Sync(buf)
|
|
|
|
} else {
|
2020-02-03 18:08:44 -05:00
|
|
|
CoreOp::Async(futures::future::ok(buf).boxed_local())
|
2019-10-02 00:51:05 +02:00
|
|
|
}
|
2019-08-22 22:30:14 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-23 14:51:29 -05:00
|
|
|
pub fn blocking_json<F>(is_sync: bool, f: F) -> Result<JsonOp, OpError>
|
2019-08-22 22:30:14 -07:00
|
|
|
where
|
2020-02-03 18:08:44 -05:00
|
|
|
F: 'static + Send + FnOnce() -> JsonResult,
|
2019-08-22 22:30:14 -07:00
|
|
|
{
|
|
|
|
if is_sync {
|
|
|
|
Ok(JsonOp::Sync(f()?))
|
|
|
|
} else {
|
2020-02-07 17:54:44 -05:00
|
|
|
let fut = async move { tokio::task::spawn_blocking(f).await.unwrap() };
|
2020-02-03 18:08:44 -05:00
|
|
|
Ok(JsonOp::Async(fut.boxed_local()))
|
2019-08-22 22:30:14 -07:00
|
|
|
}
|
|
|
|
}
|