mirror of
https://github.com/fluencelabs/jsonpath
synced 2025-04-03 15:31:04 +00:00
fix wildcard filter
This commit is contained in:
parent
d384079842
commit
053be432f2
@ -470,12 +470,12 @@ fn walk_all<'a>(vec: &Vec<&'a Value>, tmp: &mut Vec<&'a Value>) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fn walk<'a, F>(vec: &Vec<&'a Value>, tmp: &mut Vec<&'a Value>, fun: &F)
|
fn walk<'a, F>(vec: &Vec<&'a Value>, tmp: &mut Vec<&'a Value>, fun: &F)
|
||||||
where
|
where
|
||||||
F: Fn(&Value) -> Option<Vec<&Value>>,
|
F: Fn(&Value) -> Option<Vec<&Value>>,
|
||||||
{
|
{
|
||||||
fn _walk<'a, F>(v: &'a Value, tmp: &mut Vec<&'a Value>, fun: &F)
|
fn _walk<'a, F>(v: &'a Value, tmp: &mut Vec<&'a Value>, fun: &F)
|
||||||
where
|
where
|
||||||
F: Fn(&Value) -> Option<Vec<&Value>>,
|
F: Fn(&Value) -> Option<Vec<&Value>>,
|
||||||
{
|
{
|
||||||
if let Some(mut ret) = fun(v) {
|
if let Some(mut ret) = fun(v) {
|
||||||
tmp.append(&mut ret);
|
tmp.append(&mut ret);
|
||||||
@ -758,38 +758,19 @@ impl<'a, 'b> Selector<'a, 'b> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fn next_from_current_with_str(&mut self, keys: &Vec<String>) {
|
fn next_from_current_with_str(&mut self, keys: &Vec<String>) {
|
||||||
fn _collect<'a>(
|
if let Some(current) = self.current.take() {
|
||||||
v: &'a Value,
|
let mut tmp = Vec::new();
|
||||||
tmp: &mut Vec<&'a Value>,
|
for c in current {
|
||||||
keys: &Vec<String>,
|
match c {
|
||||||
visited: &mut HashSet<*const Value>,
|
Value::Object(map) => {
|
||||||
) {
|
for key in keys {
|
||||||
match v {
|
if let Some(v) = map.get(key) {
|
||||||
Value::Object(map) => {
|
|
||||||
for key in keys {
|
|
||||||
if let Some(v) = map.get(key) {
|
|
||||||
let ptr = v as *const Value;
|
|
||||||
if !visited.contains(&ptr) {
|
|
||||||
visited.insert(ptr);
|
|
||||||
tmp.push(v)
|
tmp.push(v)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
_ => {}
|
||||||
}
|
}
|
||||||
Value::Array(vec) => {
|
|
||||||
for v in vec {
|
|
||||||
_collect(v, tmp, keys, visited);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
_ => {}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if let Some(current) = self.current.take() {
|
|
||||||
let mut tmp = Vec::new();
|
|
||||||
let mut visited = HashSet::new();
|
|
||||||
for c in current {
|
|
||||||
_collect(c, &mut tmp, keys, &mut visited);
|
|
||||||
}
|
}
|
||||||
self.current = Some(tmp);
|
self.current = Some(tmp);
|
||||||
}
|
}
|
||||||
@ -801,26 +782,22 @@ impl<'a, 'b> Selector<'a, 'b> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fn next_all_from_current(&mut self) {
|
fn next_all_from_current(&mut self) {
|
||||||
fn _collect<'a>(v: &'a Value, tmp: &mut Vec<&'a Value>) {
|
|
||||||
match v {
|
|
||||||
Value::Object(map) => {
|
|
||||||
for (_, v) in map {
|
|
||||||
tmp.push(v)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Value::Array(vec) => {
|
|
||||||
for v in vec {
|
|
||||||
_collect(v, tmp);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
_ => {}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if let Some(current) = self.current.take() {
|
if let Some(current) = self.current.take() {
|
||||||
let mut tmp = Vec::new();
|
let mut tmp = Vec::new();
|
||||||
for c in current {
|
for c in current {
|
||||||
_collect(c, &mut tmp);
|
match c {
|
||||||
|
Value::Object(map) => {
|
||||||
|
for (_, v) in map {
|
||||||
|
tmp.push(v)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Value::Array(vec) => {
|
||||||
|
for v in vec {
|
||||||
|
tmp.push(v);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_ => {}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
self.current = Some(tmp);
|
self.current = Some(tmp);
|
||||||
}
|
}
|
||||||
@ -934,17 +911,28 @@ impl<'a, 'b> NodeVisitor for Selector<'a, 'b> {
|
|||||||
|
|
||||||
self.tokens.pop();
|
self.tokens.pop();
|
||||||
}
|
}
|
||||||
ParseToken::All => match self.tokens.last() {
|
ParseToken::All => {
|
||||||
Some(ParseToken::Leaves) => {
|
match self.tokens.last() {
|
||||||
self.tokens.pop();
|
Some(ParseToken::Array) => {
|
||||||
self.all_from_current();
|
self.tokens.pop();
|
||||||
|
}
|
||||||
|
_ => {}
|
||||||
}
|
}
|
||||||
Some(ParseToken::In) => {
|
|
||||||
self.tokens.pop();
|
match self.tokens.last() {
|
||||||
self.next_all_from_current();
|
Some(ParseToken::Leaves) => {
|
||||||
|
self.tokens.pop();
|
||||||
|
self.all_from_current();
|
||||||
|
}
|
||||||
|
Some(ParseToken::In) => {
|
||||||
|
self.tokens.pop();
|
||||||
|
self.next_all_from_current();
|
||||||
|
}
|
||||||
|
_ => {
|
||||||
|
self.next_all_from_current();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
_ => {}
|
}
|
||||||
},
|
|
||||||
ParseToken::Bool(b) => {
|
ParseToken::Bool(b) => {
|
||||||
self.terms.push(Some(ExprTerm::Bool(*b)));
|
self.terms.push(Some(ExprTerm::Bool(*b)));
|
||||||
}
|
}
|
||||||
|
@ -619,3 +619,47 @@ fn quote() {
|
|||||||
json!(["value"]),
|
json!(["value"]),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn all_filter() {
|
||||||
|
setup();
|
||||||
|
|
||||||
|
for path in vec![r#"$.*"#, r#"$[*]"#] {
|
||||||
|
select_and_then_compare(
|
||||||
|
path,
|
||||||
|
json!(["string", 42, { "key": "value" }, [0, 1]]),
|
||||||
|
json!(["string", 42, { "key": "value" }, [0, 1]]),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
for path in vec![r#"$..*"#, r#"$..[*]"#] {
|
||||||
|
select_and_then_compare(
|
||||||
|
path,
|
||||||
|
json!(["string", 42, { "key": "value" }, [0, 1]]),
|
||||||
|
json!([ "string", 42, { "key" : "value" }, [ 0, 1 ], "value", 0, 1 ]),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
for path in vec![r#"$.*.*"#, r#"$[*].*"#, r#"$.*[*]"#, r#"$[*][*]"#] {
|
||||||
|
select_and_then_compare(
|
||||||
|
path,
|
||||||
|
json!(["string", 42, { "key": "value" }, [0, 1]]),
|
||||||
|
json!(["value", 0, 1]),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
for path in vec![r#"$..friends.*"#, r#"$[*].friends.*"#] {
|
||||||
|
select_and_then_compare(
|
||||||
|
path,
|
||||||
|
read_json("./benches/data_array.json"),
|
||||||
|
json!([
|
||||||
|
{ "id" : 0, "name" : "Millicent Norman" },
|
||||||
|
{ "id" : 1, "name" : "Vincent Cannon" },
|
||||||
|
{ "id" : 2, "name" : "Gray Berry" },
|
||||||
|
{ "id" : 0, "name" : "Tillman Mckay" },
|
||||||
|
{ "id" : 1, "name" : "Rivera Berg" },
|
||||||
|
{ "id" : 2, "name" : "Rosetta Erickson" }
|
||||||
|
]),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user