mirror of
https://github.com/philomena-dev/philomena.git
synced 2025-01-07 16:58:01 +01:00
53 lines
1.1 KiB
TypeScript
53 lines
1.1 KiB
TypeScript
|
import { matchAll, matchAny, matchNone, matchNot } from './boolean';
|
||
|
import { AstMatcher, ParseError, TokenList } from './types';
|
||
|
|
||
|
export function parseTokens(lexicalArray: TokenList): AstMatcher {
|
||
|
const operandStack: AstMatcher[] = [];
|
||
|
|
||
|
lexicalArray.forEach((token, i) => {
|
||
|
if (token === 'not_op') {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
let intermediate: AstMatcher;
|
||
|
|
||
|
if (typeof token === 'string') {
|
||
|
const op2 = operandStack.pop();
|
||
|
const op1 = operandStack.pop();
|
||
|
|
||
|
if (typeof op1 === 'undefined' || typeof op2 === 'undefined') {
|
||
|
throw new ParseError('Missing operand.');
|
||
|
}
|
||
|
|
||
|
if (token === 'and_op') {
|
||
|
intermediate = matchAll(op1, op2);
|
||
|
}
|
||
|
else {
|
||
|
intermediate = matchAny(op1, op2);
|
||
|
}
|
||
|
}
|
||
|
else {
|
||
|
intermediate = token;
|
||
|
}
|
||
|
|
||
|
if (lexicalArray[i + 1] === 'not_op') {
|
||
|
operandStack.push(matchNot(intermediate));
|
||
|
}
|
||
|
else {
|
||
|
operandStack.push(intermediate);
|
||
|
}
|
||
|
});
|
||
|
|
||
|
if (operandStack.length > 1) {
|
||
|
throw new ParseError('Missing operator.');
|
||
|
}
|
||
|
|
||
|
const op1 = operandStack.pop();
|
||
|
|
||
|
if (typeof op1 === 'undefined') {
|
||
|
return matchNone();
|
||
|
}
|
||
|
|
||
|
return op1;
|
||
|
}
|