flecks/packages/dox/src/parser.js

261 lines
7.3 KiB
JavaScript
Raw Normal View History

2022-03-07 00:21:16 -06:00
import {readFile} from 'fs/promises';
import {dirname, join} from 'path';
import {transformAsync} from '@babel/core';
import traverse from '@babel/traverse';
import {
isIdentifier,
isLiteral,
isMemberExpression,
isObjectExpression,
isStringLiteral,
isThisExpression,
} from '@babel/types';
import {require as R} from '@flecks/core/server';
import {parse as parseComment} from 'comment-parser';
import glob from 'glob';
const flecksCorePath = dirname(__non_webpack_require__.resolve('@flecks/core/package.json'));
class ParserState {
constructor() {
this.hooks = {};
this.todos = [];
}
addImplementation(hook, filename, loc) {
this.hooks[hook] = this.hooks[hook] || {};
this.hooks[hook].implementations = this.hooks[hook].implementations || [];
this.hooks[hook].implementations.push({filename, loc});
}
addInvocation(hook, type, filename, loc) {
this.hooks[hook] = this.hooks[hook] || {};
this.hooks[hook].invocations = this.hooks[hook].invocations || [];
this.hooks[hook].invocations.push({filename, loc, type});
}
addSpecification(hook, specification) {
this.hooks[hook] = this.hooks[hook] || {};
this.hooks[hook].specification = specification;
}
addTodo(comment, filename) {
this.todos.push({filename, loc: comment.loc, text: comment.value.trim()});
}
}
const FlecksInvocations = (state, filename) => ({
CallExpression(path) {
if (isMemberExpression(path.node.callee)) {
if (
(isIdentifier(path.node.callee.object) && 'flecks' === path.node.callee.object.name)
|| (
(
isThisExpression(path.node.callee.object)
&& (filename === join(flecksCorePath, 'src', 'flecks.js'))
)
)
) {
if (isIdentifier(path.node.callee.property)) {
if (path.node.callee.property.name.match(/^invoke.*/)) {
if (path.node.arguments.length > 0) {
if (isStringLiteral(path.node.arguments[0])) {
state.addInvocation(
path.node.arguments[0].value,
path.node.callee.property.name,
filename,
path.node.loc,
);
}
}
}
if ('up' === path.node.callee.property.name) {
if (path.node.arguments.length > 0) {
if (isStringLiteral(path.node.arguments[0])) {
state.addInvocation(
path.node.arguments[0].value,
'invokeSequentialAsync',
filename,
path.node.loc,
);
state.addInvocation(
'@flecks/core/starting',
'invokeFlat',
filename,
path.node.loc,
);
}
}
}
if ('gather' === path.node.callee.property.name) {
if (path.node.arguments.length > 0) {
if (isStringLiteral(path.node.arguments[0])) {
state.addInvocation(
path.node.arguments[0].value,
'invokeReduce',
filename,
path.node.loc,
);
state.addInvocation(
`${path.node.arguments[0].value}.decorate`,
'invokeComposed',
filename,
path.node.loc,
);
}
}
}
}
}
}
},
});
const implementationVisitor = (fn) => {
let hooksSymbol;
return {
ExportDefaultDeclaration(path) {
const {declaration} = path.node;
if (isObjectExpression(declaration)) {
const {properties} = declaration;
properties.forEach((property) => {
const {key, value} = property;
if (isIdentifier(key) && key.name === hooksSymbol) {
if (isObjectExpression(value)) {
const {properties} = value;
properties.forEach((property) => {
const {key} = property;
if (isLiteral(key)) {
2022-03-07 02:08:24 -06:00
fn(property);
2022-03-07 00:21:16 -06:00
}
});
}
}
});
}
},
ImportDeclaration(path) {
const {source, specifiers} = path.node;
if ('@flecks/core' === source.value) {
specifiers.forEach((specifier) => {
const {imported, local} = specifier;
if ('Hooks' === imported.name) {
hooksSymbol = local.name;
}
});
}
},
};
};
const FlecksImplementations = (state, filename) => (
2022-03-07 02:08:24 -06:00
implementationVisitor(({key}) => {
2022-03-07 00:21:16 -06:00
state.addImplementation(
key.value,
filename,
2022-03-07 02:08:24 -06:00
key.loc,
2022-03-07 00:21:16 -06:00
);
})
);
const FlecksSpecifications = (state, source) => (
2022-03-07 02:08:24 -06:00
implementationVisitor((property) => {
2022-03-07 00:21:16 -06:00
if (property.leadingComments) {
const {key, value: example} = property;
const [{value}] = property.leadingComments;
const [{description, tags}] = parseComment(`/**\n${value}\n*/`, {spacing: 'preserve'});
const params = tags
.filter(({tag}) => 'param' === tag)
.map(({description, name, type}) => ({description, name, type}));
state.addSpecification(
key.value,
{
description,
example: source.slice(example.start, example.end),
params,
},
);
}
})
);
const FlecksTodos = (state, filename) => ({
enter(path) {
if (path.node.leadingComments) {
path.node.leadingComments.forEach((comment) => {
if (comment.value.toLowerCase().match('@todo')) {
state.addTodo(comment, filename);
}
});
}
},
});
export const parseCode = async (code) => {
const config = {
ast: true,
code: false,
};
const {ast} = await transformAsync(code, config);
return ast;
};
export const parseFile = async (filename, resolved, state) => {
const buffer = await readFile(filename);
const ast = await parseCode(buffer.toString('utf8'));
traverse(ast, FlecksInvocations(state, resolved));
traverse(ast, FlecksImplementations(state, resolved));
traverse(ast, FlecksTodos(state, resolved));
};
const fleckSources = async (path) => (
new Promise((r, e) => (
glob(
join(path, 'src', '**', '*.js'),
(error, result) => (error ? e(error) : r(result)),
)
))
);
export const parseFleckRoot = async (root, state) => {
const resolved = dirname(R.resolve(join(root, 'package.json')));
const sources = await fleckSources(resolved);
await Promise.all(
sources.map(async (source) => {
2022-03-07 02:44:48 -06:00
// @todo Aliased fleck paths are gonna be bad.
2022-03-07 00:21:16 -06:00
await parseFile(source, join(root, source.slice(resolved.length)), state);
}),
);
try {
const buffer = await readFile(join(resolved, 'build', 'dox', 'hooks.js'));
const source = buffer.toString('utf8');
const ast = await parseCode(source);
traverse(ast, FlecksSpecifications(state, source));
}
catch (error) {
if ('ENOENT' !== error.code) {
throw error;
}
}
};
export const parseFlecks = async (flecks) => {
const state = new ParserState();
const paths = Object.keys(flecks.resolver);
const roots = Array.from(new Set(
paths
.map((path) => flecks.root(path))
.filter((e) => !!e),
));
await Promise.all(
roots
.map(async (root) => {
await parseFleckRoot(root, state);
}),
);
return state;
};