你应该能够使用 substack 的 burrito 构建类似的东西,它使用来自Uglify-JS的解析器,我认为,你需要的一切。快速示例:
src.js:
var jobs, found, list;
jobs = mylibrary.getJobs();
found = jobs.find("Python");
list = found.convert("html");
ast.js:
var fs = require('fs'),
burrito = require('burrito');
var src = fs.readFileSync('./src.js', 'utf8');
burrito(src, function (node) {
console.log(node.name, node.value);
});
您将如何构建您请求的结构,我不太确定(我自己对 AST 解析不是很精通),但我相信这需要您付出一些努力。也许你不需要中间的结构,可以这么说,但可以只验证墨西哥卷饼中的每个节点,每个call
节点都将根据它的值(函数名、对象名等)进行验证,如果没有,则会发出警告'不验证。
这是burrito
上面调用的输出(注意:每个[Object]
或这样都已被 node.js' 截断console.log
。值实际上是 burrito 解析树中的节点,因此每个值都有它的关联状态等)。
var [ [ [ 'jobs' ], [ 'found' ], [ 'list' ] ] ]
stat [ [ { name: 'assign', start: [Object], end: [Object] },
true,
[ [Object], 'jobs' ],
[ [Object], [Object], [] ] ] ]
assign [ true,
[ { name: 'name', start: [Object], end: [Object] }, 'jobs' ],
[ { name: 'call', start: [Object], end: [Object] },
[ 'dot', [Object], 'getJobs' ],
[] ] ]
name [ 'jobs' ]
call [ [ 'dot', [ 'name', 'mylibrary' ], 'getJobs' ], [] ]
stat [ [ { name: 'assign', start: [Object], end: [Object] },
true,
[ [Object], 'found' ],
[ [Object], [Object], [Object] ] ] ]
assign [ true,
[ { name: 'name', start: [Object], end: [Object] }, 'found' ],
[ { name: 'call', start: [Object], end: [Object] },
[ 'dot', [Object], 'find' ],
[ [Object] ] ] ]
name [ 'found' ]
call [ [ 'dot', [ 'name', 'jobs' ], 'find' ],
[ [ [Object], 'Python' ] ] ]
string [ 'Python' ]
stat [ [ { name: 'assign', start: [Object], end: [Object] },
true,
[ [Object], 'list' ],
[ [Object], [Object], [Object] ] ] ]
assign [ true,
[ { name: 'name', start: [Object], end: [Object] }, 'list' ],
[ { name: 'call', start: [Object], end: [Object] },
[ 'dot', [Object], 'convert' ],
[ [Object] ] ] ]
name [ 'list' ]
call [ [ 'dot', [ 'name', 'found' ], 'convert' ],
[ [ [Object], 'html' ] ] ]
string [ 'html' ]
更新:
另一种选择是更新的(?) ES 解析器Esprima,它似乎更积极地开发和更好地记录。据报道,它也比 Uglify 更快。您可以在Parsing Demo 页面上尝试例如解析。我认为,您应该能够使用它构建一个好的解决方案。