Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: add support for json path syntax #54

Merged
merged 29 commits into from
Jun 4, 2024
Merged
Show file tree
Hide file tree
Changes from 4 commits
Commits
Show all changes
29 commits
Select commit Hold shift + click to select a range
63f4953
feat: add support for json path syntax
koladilip May 17, 2024
4b2c463
chore: update read me for json paths
koladilip May 17, 2024
ffe4f14
fix: array all filters syntax tree
koladilip May 20, 2024
3b90cc9
chore: add json path tests
koladilip May 22, 2024
a2af49f
refactor: use path type stack to handle child paths
koladilip May 22, 2024
e6309c1
feat: add flat to object mapping converter
koladilip May 24, 2024
312cad1
refactor: static utils to modules
koladilip May 24, 2024
f864b2f
feat: add support for regexp
koladilip May 25, 2024
7789cae
feat: add anyof noneof operators
koladilip May 25, 2024
17241c3
feat: add support for json path functions
koladilip May 26, 2024
89a7388
refactor: update mappings test case
koladilip May 27, 2024
078895a
refactor: address pr comments
koladilip May 27, 2024
f6b2fab
fix: typos
koladilip May 27, 2024
b1cc70d
fix: parse mappings paths
koladilip May 28, 2024
dab28b3
fix: multiple indexes in json paths
koladilip May 28, 2024
7bc0733
fix: unused import
koladilip May 28, 2024
2ff4c0a
refactor: convertToObjectMapping
koladilip May 28, 2024
aeb5912
fix: imports
koladilip May 28, 2024
e7b37fa
refactor: comparisons tests
koladilip May 29, 2024
a9e591a
feat: add reverse translator
koladilip May 30, 2024
71f3b06
fix: formatting issue reverse translator
koladilip May 30, 2024
6b5269e
refactor: add error handling in flat mapping convertor
koladilip Jun 1, 2024
9fa7b50
fix: error handling in convertor
koladilip Jun 3, 2024
644f7aa
fix: typo in file name
koladilip Jun 3, 2024
0625be4
refactor: address pr comments on template parsing
koladilip Jun 3, 2024
9ca6100
fix: sonar lint issues
koladilip Jun 3, 2024
e386ebb
fix: eslint issue
koladilip Jun 3, 2024
71e11c0
refactor: mappings convertor
koladilip Jun 3, 2024
84aaabf
feat: add util function convert mappings to template
koladilip Jun 4, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
8 changes: 8 additions & 0 deletions readme.md
Original file line number Diff line number Diff line change
Expand Up @@ -216,6 +216,12 @@ If we use this rich path`~r a.b.c` then it automatically handles following varia
- `{"a": [{ "b": [{"c": 2}]}]}`
Refer this [example](test/scenarios/paths/rich_path.jt) for more details.

#### Json Paths
We support some features of [JSON Path](https://goessner.net/articles/JsonPath/index.html#) syntax using path option (`~j`).
Note: This is an experimental feature and may not support all the features of JSON Paths.

Refer this [example](test/scenarios/paths/json_path.jt) for more details.

#### Simple selectors

```js
Expand Down Expand Up @@ -333,6 +339,8 @@ We can override the default path option using tags.
~s a.b.c
// Use ~r to treat a.b.c as rich path
~r a.b.c
// Use ~j for using json paths
~j items[?(@.a>1)]
```

**Note:** Rich paths are slower compare to the simple paths.
Expand Down
38 changes: 29 additions & 9 deletions src/lexer.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
import { BINDINGS_PARAM_KEY, VARS_PREFIX } from './constants';
import { VARS_PREFIX } from './constants';
import { JsonTemplateLexerError } from './errors';
import { Keyword, Token, TokenType } from './types';

Expand Down Expand Up @@ -81,12 +81,16 @@ export class JsonTemplateLexer {
return this.match('~r');
}

matchJsonPath(): boolean {
return this.match('~j');
}

matchPathType(): boolean {
return this.matchRichPath() || this.matchSimplePath();
return this.matchRichPath() || this.matchJsonPath() || this.matchSimplePath();
}

matchPath(): boolean {
return this.matchPathSelector() || this.matchID();
return this.matchPathType() || this.matchPathSelector() || this.matchID();
}

matchSpread(): boolean {
Expand All @@ -113,7 +117,7 @@ export class JsonTemplateLexer {
const token = this.lookahead();
if (token.type === TokenType.PUNCT) {
const { value } = token;
return value === '.' || value === '..' || value === '^';
return value === '.' || value === '..' || value === '^' || value === '$' || value === '@';
}

return false;
Expand Down Expand Up @@ -145,8 +149,24 @@ export class JsonTemplateLexer {
return token.type === TokenType.KEYWORD && token.value === val;
}

matchContains(): boolean {
return this.matchKeywordValue(Keyword.CONTAINS);
}

matchEmpty(): boolean {
return this.matchKeywordValue(Keyword.EMPTY);
}

matchSize(): boolean {
return this.matchKeywordValue(Keyword.SIZE);
}

matchSubsetOf(): boolean {
return this.matchKeywordValue(Keyword.SUBSETOF);
}

matchIN(): boolean {
return this.matchKeywordValue(Keyword.IN);
return this.matchKeywordValue(Keyword.IN) || this.matchKeywordValue(Keyword.NOT_IN);
}

matchFunction(): boolean {
Expand Down Expand Up @@ -317,7 +337,7 @@ export class JsonTemplateLexer {
}

private static isIdStart(ch: string) {
return ch === '$' || ch === '_' || (ch >= 'a' && ch <= 'z') || (ch >= 'A' && ch <= 'Z');
return ch === '_' || (ch >= 'a' && ch <= 'z') || (ch >= 'A' && ch <= 'Z');
koladilip marked this conversation as resolved.
Show resolved Hide resolved
}

private static isIdPart(ch: string) {
Expand Down Expand Up @@ -383,7 +403,7 @@ export class JsonTemplateLexer {
JsonTemplateLexer.validateID(id);
return {
type: TokenType.ID,
value: id.replace(/^\$/, `${BINDINGS_PARAM_KEY}`),
value: id,
range: [start, this.idx],
};
}
Expand Down Expand Up @@ -565,7 +585,7 @@ export class JsonTemplateLexer {
const start = this.idx;
const ch1 = this.codeChars[this.idx];

if (',;:{}()[]^+-*/%!><|=@~#?\n'.includes(ch1)) {
if (',;:{}()[]^+-*/%!><|=@~$#?\n'.includes(ch1)) {
return {
type: TokenType.PUNCT,
value: ch1,
Expand Down Expand Up @@ -594,7 +614,7 @@ export class JsonTemplateLexer {
const ch1 = this.codeChars[this.idx];
const ch2 = this.codeChars[this.idx + 1];

if (ch1 === '~' && 'rs'.includes(ch2)) {
if (ch1 === '~' && 'rsj'.includes(ch2)) {
this.idx += 2;
return {
type: TokenType.PUNCT,
Expand Down
10 changes: 9 additions & 1 deletion src/operators.ts
Original file line number Diff line number Diff line change
Expand Up @@ -23,7 +23,7 @@ function endsWith(val1, val2): string {
}

function containsStrict(val1, val2): string {
return `(typeof ${val1} === 'string' && ${val1}.includes(${val2}))`;
return `((typeof ${val1} === 'string' || Array.isArray(${val1})) && ${val1}.includes(${val2}))`;
}

function contains(val1, val2): string {
Expand Down Expand Up @@ -74,10 +74,18 @@ export const binaryOperators = {

'=$': (val1, val2): string => endsWith(val2, val1),

contains: containsStrict,

'==*': (val1, val2): string => containsStrict(val2, val1),

'=*': (val1, val2): string => contains(val2, val1),

size: (val1, val2): string => `${val1}.length === ${val2}`,

empty: (val1, val2): string => `(${val1}.length === 0) === ${val2}`,

subsetof: (val1, val2): string => `${val1}.every((el) => {return ${val2}.includes(el);})`,

'+': (val1, val2): string => `${val1}+${val2}`,

'-': (val1, val2): string => `${val1}-${val2}`,
Expand Down
68 changes: 44 additions & 24 deletions src/parser.ts
Original file line number Diff line number Diff line change
Expand Up @@ -211,10 +211,10 @@ export class JsonTemplateParser {
return this.parseSelector();
} else if (this.lexer.matchToArray()) {
return this.parsePathOptions();
} else if (this.lexer.match('[')) {
return this.parseArrayFilterExpr();
} else if (this.lexer.match('{')) {
return this.parseObjectFiltersExpr();
} else if (this.lexer.match('[')) {
return this.parseArrayFilterExpr();
koladilip marked this conversation as resolved.
Show resolved Hide resolved
} else if (this.lexer.match('@') || this.lexer.match('#')) {
return this.parsePathOptions();
}
Expand Down Expand Up @@ -265,14 +265,25 @@ export class JsonTemplateParser {
};
}

private parsePathRoot(root?: Expression): Expression | string | undefined {
private parsePathRoot(pathType: PathType, root?: Expression): Expression | string | undefined {
if (root) {
return root;
}
if (this.lexer.match('^')) {
this.lexer.ignoreTokens(1);
return DATA_PARAM_KEY;
const nextToken = this.lexer.lookahead();
switch (nextToken.value) {
case '^':
this.lexer.ignoreTokens(1);
return DATA_PARAM_KEY;
case '$':
this.lexer.ignoreTokens(1);
return pathType === PathType.JSON ? DATA_PARAM_KEY : BINDINGS_PARAM_KEY;
case '@':
this.lexer.ignoreTokens(1);
return undefined;
default:
break;
koladilip marked this conversation as resolved.
Show resolved Hide resolved
}

if (this.lexer.matchID()) {
return this.lexer.value();
}
Expand All @@ -287,23 +298,18 @@ export class JsonTemplateParser {
this.lexer.ignoreTokens(1);
return PathType.RICH;
}
return this.options?.defaultPathType ?? PathType.RICH;
}

private parsePathTypeExpr(): Expression {
const pathType = this.parsePathType();
const expr = this.parseBaseExpr();
if (expr.type === SyntaxType.PATH) {
expr.pathType = pathType;
if (this.lexer.matchJsonPath()) {
this.lexer.ignoreTokens(1);
return PathType.JSON;
}
return expr;
return this.options?.defaultPathType ?? PathType.RICH;
}

private parsePath(options?: { root?: Expression }): PathExpression | Expression {
const pathType = this.parsePathType();
const expr: PathExpression = {
type: SyntaxType.PATH,
root: this.parsePathRoot(options?.root),
root: this.parsePathRoot(pathType, options?.root),
parts: this.parsePathParts(),
pathType,
};
Expand Down Expand Up @@ -548,13 +554,27 @@ export class JsonTemplateParser {
};
}

private parseArrayFilterExpr(): ArrayFilterExpression {
private parseArrayFilterExpr(): ArrayFilterExpression | ObjectFilterExpression {
let exprType = SyntaxType.ARRAY_FILTER_EXPR;
let filter: Expression | undefined;
this.lexer.expect('[');
const filter = this.parseArrayFilter();
if (this.lexer.match('?')) {
this.lexer.ignoreTokens(1);
exprType = SyntaxType.OBJECT_FILTER_EXPR;
filter = this.parseBaseExpr();
} else if (this.lexer.match('*')) {
// this selects all the items so no filter
this.lexer.ignoreTokens(1);
filter = {
type: SyntaxType.ALL_FILTER_EXPR,
};
} else {
filter = this.parseArrayFilter();
}
this.lexer.expect(']');

return {
type: SyntaxType.ARRAY_FILTER_EXPR,
type: exprType,
koladilip marked this conversation as resolved.
Show resolved Hide resolved
filter,
};
}
Expand Down Expand Up @@ -666,7 +686,11 @@ export class JsonTemplateParser {
this.lexer.match('>') ||
this.lexer.match('<=') ||
this.lexer.match('>=') ||
this.lexer.matchIN()
this.lexer.matchIN() ||
this.lexer.matchContains() ||
this.lexer.matchSize() ||
this.lexer.matchEmpty() ||
this.lexer.matchSubsetOf()
) {
return {
type: this.lexer.matchIN() ? SyntaxType.IN_EXPR : SyntaxType.COMPARISON_EXPR,
Expand Down Expand Up @@ -1247,10 +1271,6 @@ export class JsonTemplateParser {
return this.parseArrayExpr();
}

if (this.lexer.matchPathType()) {
return this.parsePathTypeExpr();
}

if (this.lexer.matchPath()) {
return this.parsePath();
}
Expand Down
12 changes: 9 additions & 3 deletions src/translator.ts
Original file line number Diff line number Diff line change
Expand Up @@ -39,6 +39,7 @@ import {
LoopExpression,
IncrementExpression,
LoopControlExpression,
Keyword,
koladilip marked this conversation as resolved.
Show resolved Hide resolved
} from './types';
import { CommonUtils } from './utils';

Expand Down Expand Up @@ -686,8 +687,13 @@ export class JsonTemplateTranslator {
code.push(this.translateExpr(expr.args[0], val1, ctx));
code.push(this.translateExpr(expr.args[1], val2, ctx));
code.push(`if(typeof ${val2} === 'object'){`);
const inCode = `(Array.isArray(${val2}) ? ${val2}.includes(${val1}) : ${val1} in ${val2})`;
code.push(JsonTemplateTranslator.generateAssignmentCode(resultVar, inCode));
if (expr.op === Keyword.IN) {
const inCode = `(Array.isArray(${val2}) ? ${val2}.includes(${val1}) : ${val1} in ${val2})`;
code.push(JsonTemplateTranslator.generateAssignmentCode(resultVar, inCode));
} else {
const notInCode = `(Array.isArray(${val2}) ? !${val2}.includes(${val1}) : !(${val1} in ${val2}))`;
code.push(JsonTemplateTranslator.generateAssignmentCode(resultVar, notInCode));
}
koladilip marked this conversation as resolved.
Show resolved Hide resolved
code.push('} else {');
code.push(JsonTemplateTranslator.generateAssignmentCode(resultVar, 'false'));
code.push('}');
Expand Down Expand Up @@ -715,7 +721,7 @@ export class JsonTemplateTranslator {
const code: string[] = [];
if (expr.filter.type === SyntaxType.ARRAY_INDEX_FILTER_EXPR) {
code.push(this.translateIndexFilterExpr(expr.filter as IndexFilterExpression, dest, ctx));
} else {
} else if (expr.filter.type === SyntaxType.RANGE_FILTER_EXPR) {
koladilip marked this conversation as resolved.
Show resolved Hide resolved
code.push(this.translateRangeFilterExpr(expr.filter as RangeFilterExpression, dest, ctx));
}
return code.join('');
Expand Down
11 changes: 10 additions & 1 deletion src/types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,12 @@ export enum Keyword {
AWAIT = 'await',
ASYNC = 'async',
IN = 'in',
NOT_IN = 'nin',
NOT = 'not',
CONTAINS = 'contains',
SUBSETOF = 'subsetof',
EMPTY = 'empty',
SIZE = 'size',
RETURN = 'return',
THROW = 'throw',
CONTINUE = 'continue',
Expand Down Expand Up @@ -69,6 +74,7 @@ export enum SyntaxType {
SPREAD_EXPR = 'spread_expr',
CONDITIONAL_EXPR = 'conditional_expr',
ARRAY_INDEX_FILTER_EXPR = 'array_index_filter_expr',
ALL_FILTER_EXPR = 'all_filter_expr',
OBJECT_INDEX_FILTER_EXPR = 'object_index_filter_expr',
RANGE_FILTER_EXPR = 'range_filter_expr',
OBJECT_FILTER_EXPR = 'object_filter_expr',
Expand All @@ -92,6 +98,7 @@ export enum SyntaxType {
export enum PathType {
SIMPLE = 'simple',
RICH = 'rich',
JSON = 'json',
}

export interface EngineOptions {
Expand Down Expand Up @@ -186,12 +193,14 @@ export interface IndexFilterExpression extends Expression {
exclude?: boolean;
}

export interface AllFilterExpression extends Expression {}

export interface ObjectFilterExpression extends Expression {
filter: Expression;
}

export interface ArrayFilterExpression extends Expression {
filter: RangeFilterExpression | IndexFilterExpression;
filter: RangeFilterExpression | IndexFilterExpression | AllFilterExpression;
}
export interface LiteralExpression extends Expression {
value: string | number | boolean | null | undefined;
Expand Down
12 changes: 12 additions & 0 deletions test/scenarios/comparisons/data.ts
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,18 @@ export const data: Scenario[] = [
true,
true,
true,
true,
true,
true,
true,
true,
true,
true,
true,
true,
true,
true,
true,
],
},
];
14 changes: 13 additions & 1 deletion test/scenarios/comparisons/template.jt
Original file line number Diff line number Diff line change
Expand Up @@ -18,5 +18,17 @@
'I start with' ^== 'I',
'I' ==^ 'I start with',
"a" in ["a", "b"],
"a" in {a: 1, b: 2}
"a" in {a: 1, b: 2},
koladilip marked this conversation as resolved.
Show resolved Hide resolved
"a" nin ["b"],
"a" nin {b: 1},
["a", "b"] contains "a",
"abc" contains "a",
["a", "b"] size 2,
"abc" size 3,
[] empty true,
["a"] empty false,
"" empty true,
"abc" empty false,
["c", "a"] subsetof ["a", "b", "c"],
[] subsetof ["a", "b", "c"]
]
Loading
Loading