Package Exports
- parse-statements
Readme
parse-statements ✂️
Fast and easy parser of statements in source code in any language.
parse-statements ✂️ allows you to parse statements consisting of a sequence of tokens
with arbitrary text between them. Statements cannot overlap.
In addition to statements, language comments can be described, which can also be located inside statements (between its neighboring tokens).
Strings are used to describe (find) tokens, from which regexps with gmu flags are generated
(therefore, the backslash in these lines must be escaped, that is, it must be doubled).
For each parsed statement, the optional onParse callback is called with the context,
source text, and an array of tokens of statement
(and an array of comments between this token and the next one, if any).
If the sequence of tokens of statement has not completed, instead of the onParse callback,
an onError callback with the same signature is called, receiving an incomplete sequence
of parsed tokens of statement.
Similar optional callbacks can be set for comments.
Basic example
import {createParseFunction} from 'parse-statements';
import type {OnCommentError, OnCommentParse, OnParse} from 'parse-statements';
const throwError = (message: string): void => {
throw new Error(message);
};
type Context = Readonly<{
errors: unknown[];
exports: [exports: string, ...comments: string[]][];
imports: [import: string, ...comments: string[]][];
multilineComments: string[];
singlelineComments: string[];
}>;
const getCommentSource = (
source: string,
pair: readonly [{end: number}, {start: number}],
): string => source.slice(pair[0].end, pair[1].start);
const onCommentError: OnCommentError<Context> = (_context, source, {start}) => {
throwError(source.slice(start));
};
const onCommentParse: OnCommentParse<Context> = ({singlelineComments}, source, {end}, {start}) => {
singlelineComments.push(source.slice(end, start));
};
const onError: OnParse<Context> = ({errors}, source, ...tokens) => {
errors.push(source.slice(tokens[0]!.start, tokens[tokens.length - 1]!.end + 30));
};
const onExportParse: OnParse<Context, 3> = (
{exports},
source,
exportStart,
exportListEnd,
exportEnd,
) => {
const exportStartComments = exportStart.comments?.map((pair) => getCommentSource(source, pair));
const exportListComments = exportListEnd.comments?.map((pair) => getCommentSource(source, pair));
exports.push([
source.slice(exportStart.end, exportEnd.start),
...(exportStartComments || []),
...(exportListComments || []),
]);
};
const onImportParse: OnParse<Context, 3> = (
{imports},
source,
importStart,
importFrom,
importEnd,
) => {
const importStartComments = importStart.comments?.map((pair) => getCommentSource(source, pair));
const importFromComments = importFrom.comments?.map((pair) => getCommentSource(source, pair));
imports.push([
source.slice(importStart.end, importEnd.start),
...(importStartComments || []),
...(importFromComments || []),
]);
};
const parseImportsExports = createParseFunction<Context>({
comments: [
{
onError: onCommentError,
onParse: onCommentParse,
tokens: ['\\/\\/', '$\\n?'],
},
{
onError: onCommentError,
onParse: ({multilineComments}, source, {end}, {start}) => {
multilineComments.push(source.slice(end, start));
},
tokens: ['\\/\\*', '\\*\\/'],
},
],
onError: (_context, _source, message) => throwError(message),
statements: [
{
onError,
onParse: onImportParse as OnParse,
tokens: ['^import\\b', '\\bfrom\\b', '$\\n?'],
},
{
onError,
onParse: onExportParse as OnParse,
tokens: ['^export\\b', '\\}', '$\\n?'],
},
],
});
const importsExports: Context = {
errors: [],
exports: [],
imports: [],
multilineComments: [],
singlelineComments: [],
};
parseImportsExports(
importsExports,
`
import {foo} from './foo';
import bar from './bar'
// This is a comment
import /* some comment */ bar from bar;
'also import from bar;'
import bar from './baz'
import with error;
import // comment in import without from;
export {foo} /* also comment} */;
export /* comment in export} */ {bar}
`,
);
console.log(importsExports);Install
Requires node version 10 or higher:
npm install parse-statementsparse-statements ✂️ works in any environment that supports ES2018
(because package uses RegExp Named Capture Groups).