2020-09-11 17:39:10 +08:00
|
|
|
import { Token, Lexer } from 'antlr4';
|
|
|
|
import { ParseTreeWalker } from 'antlr4/tree';
|
|
|
|
|
|
|
|
import ParserErrorListener, {
|
|
|
|
ParserError,
|
|
|
|
ErrorHandler,
|
|
|
|
ParserErrorCollector,
|
|
|
|
} from './parserErrorListener';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Custom Parser class, subclass needs extends it.
|
|
|
|
*/
|
|
|
|
export default abstract class BasicParser<C = any> {
|
|
|
|
private _parser;
|
|
|
|
|
|
|
|
public parse(
|
|
|
|
input: string,
|
|
|
|
errorListener?: ErrorHandler,
|
|
|
|
) {
|
|
|
|
const parser = this.createParser(input);
|
|
|
|
this._parser = parser;
|
|
|
|
|
|
|
|
parser.removeErrorListeners();
|
|
|
|
parser.addErrorListener(new ParserErrorListener(errorListener));
|
|
|
|
|
2022-12-16 02:58:20 +08:00
|
|
|
const parserTree = parser.program();
|
2020-09-11 17:39:10 +08:00
|
|
|
|
|
|
|
return parserTree;
|
|
|
|
}
|
|
|
|
|
|
|
|
public validate(input: string): ParserError[] {
|
|
|
|
const lexerError = []; const syntaxErrors = [];
|
|
|
|
|
|
|
|
const parser = this.createParser(input);
|
|
|
|
this._parser = parser;
|
|
|
|
|
|
|
|
parser.removeErrorListeners();
|
|
|
|
parser.addErrorListener(new ParserErrorCollector(syntaxErrors));
|
|
|
|
|
2022-12-16 02:58:20 +08:00
|
|
|
parser.program();
|
2020-09-11 17:39:10 +08:00
|
|
|
|
|
|
|
return lexerError.concat(syntaxErrors);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Create antrl4 Lexer object
|
|
|
|
* @param input source string
|
|
|
|
*/
|
|
|
|
public abstract createLexer(input: string): Lexer;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Create Parser by lexer
|
|
|
|
* @param lexer Lexer
|
|
|
|
*/
|
|
|
|
public abstract createParserFromLexer(lexer: Lexer);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Visit parser tree
|
|
|
|
* @param parserTree
|
|
|
|
*/
|
|
|
|
// public abstract visit(visitor: any, parserTree: any);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The source string
|
|
|
|
* @param input string
|
|
|
|
*/
|
|
|
|
public getAllTokens(input: string): Token[] {
|
|
|
|
return this.createLexer(input).getAllTokens();
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get Parser instance by input string
|
|
|
|
* @param input
|
|
|
|
*/
|
|
|
|
public createParser(input: string) {
|
|
|
|
const lexer = this.createLexer(input);
|
|
|
|
const parser: any = this.createParserFromLexer(lexer);
|
|
|
|
parser.buildParseTrees = true;
|
|
|
|
this._parser = parser;
|
|
|
|
|
|
|
|
return parser;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* It convert tree to string, it's convenient to use in unit test.
|
|
|
|
* @param string input
|
|
|
|
*/
|
|
|
|
public parserTreeToString(input: string): string {
|
|
|
|
const parser = this.createParser(input);
|
|
|
|
this._parser = parser;
|
|
|
|
|
2022-12-16 02:58:20 +08:00
|
|
|
const tree = parser.program();
|
2020-09-11 17:39:10 +08:00
|
|
|
return tree.toStringTree(parser.ruleNames);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get List-like style tree string
|
|
|
|
* @param parserTree
|
|
|
|
*/
|
|
|
|
public toString(parserTree: any): string {
|
|
|
|
return parserTree.toStringTree(this._parser.ruleNames);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param listener Listener instance extends ParserListener
|
|
|
|
* @param parserTree parser Tree
|
|
|
|
*/
|
|
|
|
public listen(listener: any, parserTree: any) {
|
|
|
|
ParseTreeWalker.DEFAULT.walk(listener, parserTree);
|
|
|
|
}
|
|
|
|
}
|