/**
* @namespace kql
* @memberof module:kdljs.parser
*/
import { Lexer } from 'chevrotain'
import { BaseParser } from './base.js'
import * as Tokens from './tokens.js'
const tokens = {
defaultMode: 'main',
modes: {
main: [
Tokens.LeftBracket,
Tokens.RightBracket,
Tokens.Descendant,
Tokens.GreaterOrEqualThan,
Tokens.LessOrEqualThan,
Tokens.GreaterThan,
Tokens.LessThan,
Tokens.Or,
Tokens.Sibling,
Tokens.AdjacentSibling,
Tokens.NotEquals,
Tokens.StartsWith,
Tokens.EndsWith,
Tokens.Contains,
Tokens.Map,
Tokens.TopAccessor,
Tokens.ValAccessor,
Tokens.PropAccessor,
Tokens.Accessor,
Tokens.Comma,
Tokens.WhiteSpace,
Tokens.NewLine,
Tokens.BlockComment,
Tokens.LineComment,
Tokens.OpenMultiLineComment,
Tokens.Boolean,
Tokens.Null,
Tokens.FloatKeyword,
Tokens.MultiLineRawString,
Tokens.RawString,
Tokens.Integer,
Tokens.Float,
Tokens.Equals,
Tokens.LeftParenthesis,
Tokens.RightParenthesis,
Tokens.EscLine,
Tokens.MultiLineOpenQuote,
Tokens.OpenQuote,
Tokens.Identifier
],
multilineComment: [
Tokens.OpenMultiLineComment,
Tokens.CloseMultiLineComment,
Tokens.MultiLineCommentContent
],
string: [
Tokens.Unicode,
Tokens.Escape,
Tokens.UnicodeEscape,
Tokens.CloseQuote
],
multilineString: [
Tokens.MultiLineCloseQuote,
Tokens.MultiLineSingleQuote,
Tokens.NewLine,
Tokens.WhiteSpace,
Tokens.Unicode,
Tokens.Escape,
Tokens.UnicodeEscape,
Tokens.WhiteSpaceEscape
]
}
}
/**
* @class
* @extends module:kdljs.parser.base.BaseParser
* @memberof module:kdljs.parser.kql
*/
class KqlParser extends BaseParser {
constructor () {
super(tokens)
/**
* Consume a query
* @method #query
* @memberof module:kdljs.parser.kql.KqlParser
* @return {module:kdljs.query.Query}
*/
this.RULE('query', () => {
const alternatives = [this.SUBRULE(this.selector)]
this.MANY(() => {
this.CONSUME(Tokens.Or)
this.CONSUME1(Tokens.WhiteSpace)
const selector = this.SUBRULE1(this.selector)
alternatives.push(selector)
})
const mapping = this.OPTION(() => {
this.CONSUME(Tokens.Map)
this.CONSUME2(Tokens.WhiteSpace)
return this.SUBRULE(this.mapTuple)
})
this.CONSUME(Tokens.EOF)
return {
alternatives,
mapping
}
})
/**
* Consume a selector
* @method #selector
* @memberof module:kdljs.parser.kql.KqlParser
* @return {module:kdljs.query.Selector}
*/
this.RULE('selector', () => {
const selector = [this.SUBRULE(this.nodeFilter)]
this.OR([
{ ALT: () => this.CONSUME(Tokens.WhiteSpace) },
{ ALT: () => this.CONSUME(Tokens.EOF) }
])
this.MANY(() => {
const operator = this.SUBRULE(this.selectionOperator)
this.CONSUME1(Tokens.WhiteSpace)
const nodeFilter = this.SUBRULE1(this.nodeFilter)
selector.push({ ...nodeFilter, operator })
this.OR1([
{ ALT: () => this.CONSUME2(Tokens.WhiteSpace) },
{ ALT: () => this.CONSUME1(Tokens.EOF) }
])
})
return selector
})
/**
* Consume a nodeFilter
* @method #nodeFilter
* @memberof module:kdljs.parser.kql.KqlParser
* @return {module:kdljs.query.NodeFilter}
*/
this.RULE('nodeFilter', () => {
const matchers = []
this.AT_LEAST_ONE(() => {
matchers.push(this.SUBRULE(this.matcher))
})
return { matchers }
})
/**
* Consume a matcher
* @method #matcher
* @memberof module:kdljs.parser.kql.KqlParser
* @return {module:kdljs.query.Matcher}
*/
this.RULE('matcher', () => {
return this.OR([
{
ALT: () => ({
accessor: { type: 'name' },
operator: '=',
value: this.SUBRULE(this.string)
})
},
{
ALT: () => {
this.CONSUME(Tokens.TopAccessor)
this.CONSUME(Tokens.RightParenthesis)
return { accessor: { type: 'top' } }
}
},
{
ALT: () => {
this.CONSUME1(Tokens.LeftParenthesis)
this.CONSUME1(Tokens.RightParenthesis)
return { accessor: { type: 'tag' } }
}
},
{
ALT: () => ({
accessor: { type: 'name' },
operator: '=',
tag: this.SUBRULE(this.tag)
})
},
{ ALT: () => this.SUBRULE(this.accessorMatcher) }
])
})
/**
* Consume a [accessor]-type matcher
* @method #accessorMatcher
* @memberof module:kdljs.parser.kql.KqlParser
* @return {module:kdljs.query.Matcher}
*/
this.RULE('accessorMatcher', () => {
this.CONSUME(Tokens.LeftBracket)
const matcher = this.OPTION(() => {
const accessor = this.SUBRULE(this.accessor)
const matcher = this.OPTION1(() => {
this.CONSUME(Tokens.WhiteSpace)
const operator = this.SUBRULE(this.matcherOperator)
this.CONSUME1(Tokens.WhiteSpace)
const matcher = { accessor, operator }
this.OR([
{ ALT: () => { matcher.value = this.SUBRULE(this.value) } },
{ ALT: () => { matcher.tag = this.SUBRULE(this.tag) } }
])
return matcher
})
return matcher || { accessor }
})
this.CONSUME(Tokens.RightBracket)
return matcher || {}
})
/**
* Consume an accessor
* @method #accessor
* @memberof module:kdljs.parser.kql.KqlParser
* @return {module:kdljs.query.Accessor}
*/
this.RULE('accessor', () => {
return this.OR([
{
ALT: () => {
this.CONSUME(Tokens.ValAccessor)
const parameter = this.OPTION(() => this.SUBRULE(this.value))
this.CONSUME(Tokens.RightParenthesis)
return { type: 'val', parameter }
}
},
{
ALT: () => {
this.CONSUME(Tokens.PropAccessor)
const parameter = this.SUBRULE(this.string)
this.CONSUME1(Tokens.RightParenthesis)
return { type: 'prop', parameter }
}
},
{
ALT: () => {
const type = this.CONSUME(Tokens.Accessor).image.slice(0, -1)
this.CONSUME2(Tokens.RightParenthesis)
return { type }
}
},
{
ALT: () => {
const parameter = this.SUBRULE1(this.string)
return { type: 'prop', parameter }
}
}
])
})
/**
* Consume an operator within selectors
* @method #selectionOperator
* @memberof module:kdljs.parser.kql.KqlParser
* @return {string}
*/
this.RULE('selectionOperator', () => {
return this.OR([
{ ALT: () => this.CONSUME(Tokens.Descendant) },
{ ALT: () => this.CONSUME(Tokens.GreaterThan) },
{ ALT: () => this.CONSUME(Tokens.AdjacentSibling) },
{ ALT: () => this.CONSUME(Tokens.Sibling) }
]).image
})
/**
* Consume an operator within matchers
* @method #matcherOperator
* @memberof module:kdljs.parser.kql.KqlParser
* @return {string}
*/
this.RULE('matcherOperator', () => {
return this.OR([
{ ALT: () => this.CONSUME(Tokens.GreaterThan) },
{ ALT: () => this.CONSUME(Tokens.LessThan) },
{ ALT: () => this.CONSUME(Tokens.GreaterOrEqualThan) },
{ ALT: () => this.CONSUME(Tokens.LessOrEqualThan) },
{ ALT: () => this.CONSUME(Tokens.Equals) },
{ ALT: () => this.CONSUME(Tokens.NotEquals) },
{ ALT: () => this.CONSUME(Tokens.StartsWith) },
{ ALT: () => this.CONSUME(Tokens.EndsWith) },
{ ALT: () => this.CONSUME(Tokens.Contains) }
]).image
})
/**
* Consume a mapping
* @method #mapTuple
* @memberof module:kdljs.parser.kql.KqlParser
* @return {module:kdljs.query.Mapping}
*/
this.RULE('mapTuple', () => {
return this.OR([
{ ALT: () => this.SUBRULE(this.accessor) },
{
ALT: () => {
this.CONSUME(Tokens.LeftParenthesis)
const tuple = [this.SUBRULE1(this.accessor)]
this.MANY(() => {
this.CONSUME(Tokens.Comma)
this.CONSUME(Tokens.WhiteSpace)
tuple.push(this.SUBRULE2(this.accessor))
})
this.CONSUME(Tokens.RightParenthesis)
return tuple
}
}
])
})
this.performSelfAnalysis()
}
}
const lexer = new Lexer(tokens)
/**
* @constant {module:kdljs.parser.kql.KqlParser}
* @memberof module:kdljs.parser.kql
*/
const parser = new KqlParser()
/**
* @typedef ParseResult
* @memberof module:kdljs.parser.kql
* @type {Object}
* @property {Array} errors - Parsing errors
* @property {module:kdljs.queryEngine.Query} output - KQL query
*/
/**
* @function parse
* @memberof module:kdljs.parser.kql
* @param {string} text - Input KQL file (or fragment)
* @return {module:kdljs.parser.kql.ParseResult} Output
*/
export function parse (text) {
parser.input = lexer.tokenize(text).tokens
const output = parser.query()
return {
output,
errors: parser.errors
}
}
export {
lexer,
parser,
KqlParser
}