This project is part of the @thi.ng/umbrella monorepo.
Purely functional parser combinators & AST generation for generic inputs.
- Parser generation/compilation via built-in grammar DSL
- Small API surface, easy-to-grok syntax
- All parsers implemented as composable, higher-order functions
- All state centrally kept/managed by a parser context given as arg
- Support for custom readers (currently only string & array-like numeric inputs (incl. typed arrays) supported)
- Automatic AST generation & ability to transform/prune nodes during parsing
- Node transforms are composable too
- Each AST node (optionally) retains reader information (position, line num, column) - disabled by default to save memory
- Common, re-usable preset parsers & node transforms included
ALPHA - bleeding edge / work-in-progress
- @thi.ng/fsm - Composable primitives for building declarative, transducer based Finite-State Machines & matchers for arbitrary data streams
- @thi.ng/transducers-fsm - Transducer-based Finite State Machine transformer
yarn add @thi.ng/parse
// ES module
<script type="module" src="https://unpkg.com/@thi.ng/parse?module" crossorigin></script>
// UMD
<script src="https://unpkg.com/@thi.ng/parse/lib/index.umd.js" crossorigin></script>
Package sizes (gzipped, pre-treeshake): ESM: 4.66 KB / CJS: 5.02 KB / UMD: 4.71 KB
TODO
Source: /readers
defArrayReader
defStringReader
Source: /presets
WS
/WS0
/WS1
/NL
/DNL
/SPACE
/SPACES
/SPACES0
ALPHA
/LOWER_CASE
/UPPER_CASE
/ALPHA_NUM
ESC
/UNICODE
DIGIT
/DIGITS
/DIGITS0
HEX_DIGIT
/HEX_DIGITS
/HEX_UINT
BIT
/BINARY_UINT
INT
/UINT
/REAL
/FLOAT
/SIGN
Source: /prims
The D
suffix is used for discarding behavior, i.e. parsers
which do not retain their result in the AST after successful matching.
These should be preferred whenever possible, for lower memory usage and
(possibly) better performance...
The P
suffix is used to indicate predicate based parsers, i.e.
using user provided functions applied to individual inputs (characters)
in order to decide about successful matching.
always
fail
lit
/litD
/litP
noneOf
/noneOfD
/noneOfP
oneOf
/oneOfD
/oneOfP
pass
/passD
range
/rangeD
/rangeP
satisfy
/satisfyD
skipWhile
string
/stringD
stringOf
anchor
inputStart
/inputEnd
lineStart
/lineEnd
wordBoundary
startsWith
/endsWith
entireLine
entirely
Source: /combinators
alt
/altD
check
expect
lookahead
maybe
not
oneOrMore
/zeroOrMore
/oneOrMoreD
/zeroOrMoreD
repeat
/repeatD
seq
/seqD
xform
Source: /xform
Syntax sugars for xform(parser, fn)
:
collect
- collect child results into arraycount
- count number of childrendiscard
- discard resulthoist
/hoistResult
- hoist first child / child resultjoin
- join child results into stringprint
- print AST
Actual transforms:
comp
- scope transform compositionxfCollect
xfCount
xfDiscard
xfFloat
xfHoist
/xfHoistResult
xfInt(radix)
xfJoin
xfPrint
Complex parsers can be constructed via
defGrammar()
,
which accepts a string of rule definitions in the built-in (and still
WIP) grammar rule definition language, similar to PEGs and regular
expressions:
Example grammar
ws: ' '+ => discard ;
sym: [a-z] [a-z0-9_]* => join ;
num: [0-9]+ => int ;
program: ( <num> | <sym> | <ws> )* ;
Here, each line is a single parse rule definition, with each rule consisting of a sequence of one or more:
'x'
- single char literal"abc"
- multi-char string[a-z0-9_@]
- regex style char set (incl. char range support, inversion via^
)<rule_id>
- rule references (order independent)( term | ... | )
- choice of sub-terms
Literals, strings and char sets can include \uXXXX
unicode escapes (if
given within a JS source string, double escaping must be used, i.e.
\\uXXXX
).
All of these terms can be immediately followed by one of these regexp style repetition specs:
?
- zero or one occurrence*
- zero or more+
- one or more{min,max}
- min-max repetitions
All terms can be suffixed with the !
modifier to discard their results
and help produce a cleaner result AST.
link: '['! <linktitle> "]("! <linkurl> ')'! => collect ;
linktitle: [^\\u005d]+ => join ;
linkurl: [^\\u0029]+ => join ;
Given a valid input like [abc](def)
, the link
parser's result will
just be the array ["abc", "def"]
with other terms discarded from AST.
Note: A parse rule must produce a result for at least one of its
successfully matched terms. Use the discard
rule transform to discard
the entire result (instead of !
modifier)...
Furthermore, each rule can specify an optional rule transform function
which will only be applied after the rule's parser has successfully
completed. The transform is given at the end of a rule, separated by
=>
.
Custom transforms can be supplied via an additional arg to
defGrammar()
. The following default transforms are available by
default (can be overwritten) and correspond to the above mentioned
transforms:
collect
- collect sub terms into arraydiscard
- discard resulthoist
- replace AST node with its 1st childhoistR
- use result of 1st child term onlyjoin
- join sub terms into single stringfloat
- parse as floating point numberint
- parse as integerhex
- parse as hex int
For convenience, the following built-in parser presets are available as rule references in the grammar definition as well:
ALPHA
ALPHA_NUM
DIGIT
END
- input endESC
FLOAT
HEX_DIGIT
INT
LEND
- line endLSTART
- line startSTART
- input startSTRING
UNICODE
WB
- word boundaryWS
WS0
WS1
(Also see @thi.ng/defmulti as a useful tool for processing/interpreting/compiling the result AST)
// define language via grammar DSL
// the upper-cased rule names are built-ins
const lang = defGrammar(`
list: '('! <expr> ')'! ;
sym: ( <ALPHA_NUM> | [?!$+\\u002d*/.~#^=<>] )+ => join ;
expr: ( <FLOAT> | <STRING> | <sym> | <list> | <WS1> )* ;
`);
// define input & parser context
const ctx = defContext(`
(def hello (x) (str "hello, " x))
(print (hello 42))
`);
// parse & print AST
print(lang.rules.expr)(ctx)
// expr: null
// list: null
// expr: null
// sym: "def"
// sym: "hello"
// list: null
// expr: null
// sym: "x"
// list: null
// expr: null
// sym: "str"
// string: "hello, "
// sym: "x"
// list: null
// expr: null
// sym: "print"
// list: null
// expr: null
// sym: "hello"
// real: 42
// parse result
// true
// the two top-level s-expressions...
ctx.children
// [
// { id: 'list', state: null, children: [ [Object] ], result: null },
// { id: 'list', state: null, children: [ [Object] ], result: null }
// ]
import {
INT, WS0,
alt, oneOf, seq, zeroOrMore,
collect, discard, xform,
defContext
} from "@thi.ng/parse";
// whitespace parser
// discard() removes results from AST
const wsc = discard(zeroOrMore(oneOf(" ,")));
// svg path parser rules
// collect() collects child results in array, then removes children
// INT & WS0 are preset parsers (see section above)
const point = collect(seq([INT, wsc, INT]));
const move = collect(seq([oneOf("Mm"), WS0, point, WS0]));
const line = collect(seq([oneOf("Ll"), WS0, point, WS0]));
const curve = collect(seq([oneOf("Cc"), WS0, point, wsc, point, wsc, point, WS0]));
// xform used here to wrap result in array
// (to produce same result format as parsers above)
const close = xform(oneOf("Zz"), ($) => ($.result = [$.result], $));
// main path parser
const path = collect(zeroOrMore(alt([move, line, curve, close])));
// prepare parse context & reader
const ctx = defContext("M0,1L2 3c4,5-6,7 8 9z");
// parse input into AST
path(ctx);
// true
// transformed result of AST root node
ctx.result
// [["M", [0, 1]], ["L", [2, 3]], ["c", [4, 5], [-6, 7], [8, 9]], ["z"]]
import {
INT, WS0,
alt, oneOf, xform, zeroOrMore
defContext
} from "@thi.ng/parse";
// data stack for execution
const stack = [];
// operator implementations
const ops = {
"+": (a, b) => a + b,
"-": (a, b) => a - b,
"*": (a, b) => a * b,
"/": (a, b) => a / b,
};
// signed integer parser (using INT preset) with transform fn
// user fn here only used for pushing values on data stack
// also, if a transform returns null, the parse scope will
// be removed from the result AST
const value = xform(INT, (scope) => {
stack.push(scope.result);
return null;
});
// parser for any of the registered operators (again w/ transform)
// the transform here applies the op in RPN fashion to the data stack
const op = xform(oneOf(Object.keys(ops)), (scope) => {
const b = stack.pop();
const a = stack.pop();
stack.push(ops[scope.result](a, b));
return null;
});
// parser for complete RPN program, combines above two parsers
// and the whitespace preset as alternatives
const program = zeroOrMore(alt([value, op, WS0]))
// prepare parser context (incl. reader) and execute
program(defContext("10 5 3 * + -2 * 10 /"));
// print result data stack, i.e. result of:
// 3 * 5 + 10 * -2 / 10 (in infix notation)
console.log(stack);
// [-5]
Karsten Schmidt
© 2020 Karsten Schmidt // Apache Software License 2.0