mirror of
https://github.com/fluencelabs/jsonpath
synced 2025-04-30 11:52:12 +00:00
269 lines
7.9 KiB
Rust
269 lines
7.9 KiB
Rust
extern crate jsonpath_lib as jsonpath;
|
|
#[macro_use]
|
|
extern crate neon;
|
|
extern crate serde_json;
|
|
|
|
use jsonpath::{JsonPathError, Node, Parser, Selector};
|
|
use neon::prelude::*;
|
|
use serde_json::Value;
|
|
|
|
fn select_str(mut ctx: FunctionContext) -> JsResult<JsValue> {
|
|
let json_val = ctx.argument::<JsString>(0)?.value();
|
|
let path = ctx.argument::<JsString>(1)?.value();
|
|
match jsonpath::select_as_str(&json_val, path.as_str()) {
|
|
Ok(value) => Ok(JsString::new(&mut ctx, &value).upcast()),
|
|
Err(e) => panic!("{:?}", e),
|
|
}
|
|
}
|
|
|
|
fn delete(mut ctx: FunctionContext) -> JsResult<JsValue> {
|
|
let json_val = ctx.argument::<JsString>(0)?.value();
|
|
let json: Value = match serde_json::from_str(&json_val) {
|
|
Ok(value) => value,
|
|
Err(e) => panic!("{:?}", JsonPathError::Serde(e.to_string())),
|
|
};
|
|
let path = ctx.argument::<JsString>(1)?.value();
|
|
match jsonpath::delete(json, &path) {
|
|
Ok(value) => Ok(JsString::new(
|
|
&mut ctx,
|
|
match serde_json::to_string(&value) {
|
|
Ok(value) => value,
|
|
Err(e) => panic!("{:?}", JsonPathError::Serde(e.to_string())),
|
|
},
|
|
)
|
|
.upcast()),
|
|
Err(e) => panic!("{:?}", e),
|
|
}
|
|
}
|
|
|
|
fn replace_with(mut ctx: FunctionContext) -> JsResult<JsValue> {
|
|
let json_val = ctx.argument::<JsString>(0)?.value();
|
|
let json: Value = match serde_json::from_str(&json_val) {
|
|
Ok(value) => value,
|
|
Err(e) => panic!("{:?}", JsonPathError::Serde(e.to_string())),
|
|
};
|
|
let path = ctx.argument::<JsString>(1)?.value();
|
|
let fun = ctx.argument::<JsFunction>(2)?;
|
|
match jsonpath::replace_with(json, &path, &mut |v| {
|
|
let json_str = JsString::new(
|
|
&mut ctx,
|
|
match serde_json::to_string(v) {
|
|
Ok(value) => value,
|
|
Err(e) => panic!("{:?}", JsonPathError::Serde(e.to_string())),
|
|
},
|
|
);
|
|
|
|
let null = ctx.null();
|
|
let args = vec![ctx.string(json_str.value())];
|
|
let result = match fun.call(&mut ctx, null, args) {
|
|
Ok(result) => result,
|
|
Err(e) => panic!("{:?}", e),
|
|
};
|
|
let json_str = match result.downcast::<JsString>() {
|
|
Ok(v) => v.value(),
|
|
Err(e) => panic!("{:?}", JsonPathError::Serde(e.to_string())),
|
|
};
|
|
match serde_json::from_str(&json_str) {
|
|
Ok(v) => v,
|
|
Err(e) => panic!("{:?}", JsonPathError::Serde(e.to_string())),
|
|
}
|
|
}) {
|
|
Ok(value) => Ok(JsString::new(
|
|
&mut ctx,
|
|
match serde_json::to_string(&value) {
|
|
Ok(value) => value,
|
|
Err(e) => panic!("{:?}", JsonPathError::Serde(e.to_string())),
|
|
},
|
|
)
|
|
.upcast()),
|
|
Err(e) => panic!("{:?}", e),
|
|
}
|
|
}
|
|
|
|
pub struct SelectorCls {
|
|
node: Option<Node>,
|
|
value: Option<Value>,
|
|
}
|
|
|
|
impl SelectorCls {
|
|
fn path(&mut self, path: &str) {
|
|
let node = match Parser::compile(path) {
|
|
Ok(node) => node,
|
|
Err(e) => panic!("{:?}", e),
|
|
};
|
|
|
|
self.node = Some(node);
|
|
}
|
|
|
|
fn value(&mut self, json_str: &str) {
|
|
let value: Value = match serde_json::from_str(&json_str) {
|
|
Ok(value) => value,
|
|
Err(e) => panic!("{:?}", JsonPathError::Serde(e.to_string())),
|
|
};
|
|
|
|
self.value = Some(value);
|
|
}
|
|
|
|
fn select(&self) -> String {
|
|
let node = match &self.node {
|
|
Some(node) => node,
|
|
None => panic!("{:?}", JsonPathError::EmptyPath),
|
|
};
|
|
|
|
let value = match &self.value {
|
|
Some(value) => value,
|
|
None => panic!("{:?}", JsonPathError::EmptyValue),
|
|
};
|
|
|
|
let mut selector = Selector::new();
|
|
selector.compiled_path(node);
|
|
selector.value(&value);
|
|
match selector.select_as_str() {
|
|
Ok(ret) => ret,
|
|
Err(e) => panic!("{:?}", e),
|
|
}
|
|
}
|
|
}
|
|
|
|
pub struct SelectorMutCls {}
|
|
|
|
declare_types! {
|
|
pub class JsCompileFn for SelectorCls {
|
|
init(mut ctx) {
|
|
let path = ctx.argument::<JsString>(0)?.value();
|
|
let node = match Parser::compile(path.as_str()) {
|
|
Ok(node) => node,
|
|
Err(e) => panic!("{:?}", e)
|
|
};
|
|
|
|
Ok(SelectorCls { node: Some(node), value: None })
|
|
}
|
|
|
|
method template(mut ctx) {
|
|
let mut this = ctx.this();
|
|
|
|
let json_str = ctx.argument::<JsString>(0)?.value();
|
|
{
|
|
let guard = ctx.lock();
|
|
let mut this = this.borrow_mut(&guard);
|
|
let value: Value = match serde_json::from_str(&json_str) {
|
|
Ok(value) => value,
|
|
Err(e) => panic!("{:?}", JsonPathError::Serde(e.to_string()))
|
|
};
|
|
this.value = Some(value);
|
|
};
|
|
|
|
let result_str = {
|
|
let guard = ctx.lock();
|
|
let this = this.borrow(&guard);
|
|
this.select()
|
|
};
|
|
|
|
Ok(JsString::new(&mut ctx, &result_str).upcast())
|
|
}
|
|
}
|
|
|
|
pub class JsSelectorFn for SelectorCls {
|
|
init(mut ctx) {
|
|
let json_str = ctx.argument::<JsString>(0)?.value();
|
|
let value: Value = match serde_json::from_str(&json_str) {
|
|
Ok(value) => value,
|
|
Err(e) => panic!("{:?}", JsonPathError::Serde(e.to_string()))
|
|
};
|
|
|
|
Ok(SelectorCls { node: None, value: Some(value) })
|
|
}
|
|
|
|
method select(mut ctx) {
|
|
let mut this = ctx.this();
|
|
|
|
let path = ctx.argument::<JsString>(0)?.value();
|
|
{
|
|
let guard = ctx.lock();
|
|
let mut this = this.borrow_mut(&guard);
|
|
this.path(&path);
|
|
}
|
|
|
|
let result_str = {
|
|
let guard = ctx.lock();
|
|
let this = this.borrow(&guard);
|
|
this.select()
|
|
};
|
|
|
|
Ok(JsString::new(&mut ctx, &result_str).upcast())
|
|
}
|
|
}
|
|
|
|
pub class JsSelector for SelectorCls {
|
|
init(mut _ctx) {
|
|
Ok(SelectorCls { node: None, value: None })
|
|
}
|
|
|
|
method path(mut ctx) {
|
|
let mut this = ctx.this();
|
|
|
|
let path = ctx.argument::<JsString>(0)?.value();
|
|
{
|
|
let guard = ctx.lock();
|
|
let mut this = this.borrow_mut(&guard);
|
|
this.path(&path);
|
|
}
|
|
|
|
Ok(JsUndefined::new().upcast())
|
|
}
|
|
|
|
method value(mut ctx) {
|
|
let mut this = ctx.this();
|
|
|
|
let json_str = ctx.argument::<JsString>(0)?.value();
|
|
{
|
|
let guard = ctx.lock();
|
|
let mut this = this.borrow_mut(&guard);
|
|
this.value(&json_str);
|
|
}
|
|
|
|
Ok(JsUndefined::new().upcast())
|
|
}
|
|
|
|
method select(mut ctx) {
|
|
let this = ctx.this();
|
|
|
|
let result_str = {
|
|
let guard = ctx.lock();
|
|
let this = this.borrow(&guard);
|
|
this.select()
|
|
};
|
|
|
|
Ok(JsString::new(&mut ctx, &result_str).upcast())
|
|
}
|
|
}
|
|
|
|
pub class JsSelectorMut for SelectorMutCls {
|
|
init(mut _ctx) {
|
|
Ok(SelectorMutCls {})
|
|
}
|
|
|
|
method emptyPathError(mut _ctx) {
|
|
panic!("{:?}", JsonPathError::EmptyPath);
|
|
}
|
|
|
|
method emptyValueError(mut _ctx) {
|
|
panic!("{:?}", JsonPathError::EmptyValue);
|
|
}
|
|
}
|
|
}
|
|
register_module!(mut m, {
|
|
m.export_class::<JsCompileFn>("CompileFn")
|
|
.expect("CompileFn class error");
|
|
m.export_class::<JsSelectorFn>("SelectorFn")
|
|
.expect("SelectorFn class error");
|
|
m.export_class::<JsSelector>("Selector")
|
|
.expect("Selector class error");
|
|
m.export_class::<JsSelectorMut>("SelectorMut")
|
|
.expect("SelectorMut class error");
|
|
m.export_function("deleteValue", delete)?;
|
|
m.export_function("replaceWith", replace_with)?;
|
|
m.export_function("selectStr", select_str)?;
|
|
Ok(())
|
|
});
|