function transform(src, walker) { try { var ast = acorn.parse(src, { ecmaVersion: 6, allowReturnOutsideFunction: true, allowImportExportEverywhere: true, allowHashBang: true }); } catch (ex) { if (typeof ex.loc === 'object' && typeof ex.loc.line === 'number' && typeof ex.loc.column === 'number') { var lines = src.split(/\n/g); ex.message += '\n\n | ' + (lines[ex.loc.line - 2] || '') + '\n> | ' + (lines[ex.loc.line - 1] || '') + '\n | ' + (lines[ex.loc.line] || ''); } throw ex; } src = src.split(''); function getSource(node) { return src.slice(node.start, node.end).join(''); } function setSource(node, str) { for (var i = node.start; i < node.end; i++) { src[i] = ''; } src[node.start] = str; } module.exports.getSource = getSource; module.exports.setSource = setSource; walk.ancestor(ast, walker); return src.join(''); }
function findGlobals(source, options) { options = options || {}; var globals = []; var ast; // istanbul ignore else if (typeof source === 'string') { ast = reallyParse(source, options); } else { ast = source; } // istanbul ignore if if (!(ast && typeof ast === 'object' && ast.type === 'Program')) { throw new TypeError('Source must be either a string of JavaScript or an acorn AST'); } var declareFunction = function (node) { var fn = node; fn.locals = fn.locals || {}; node.params.forEach(function (node) { declarePattern(node, fn); }); if (node.id) { fn.locals[node.id.name] = true; } } var declarePattern = function (node, parent) { switch (node.type) { case 'Identifier': parent.locals[node.name] = true; break; case 'ObjectPattern': node.properties.forEach(function (node) { declarePattern(node.value, parent); }); break; case 'ArrayPattern': node.elements.forEach(function (node) { if (node) declarePattern(node, parent); }); break; case 'RestElement': declarePattern(node.argument, parent); break; case 'AssignmentPattern': declarePattern(node.left, parent); break; // istanbul ignore next default: throw new Error('Unrecognized pattern type: ' + node.type); } } var declareModuleSpecifier = function (node, parents) { ast.locals = ast.locals || {}; ast.locals[node.local.name] = true; } walk.ancestor(ast, { 'VariableDeclaration': function (node, parents) { var parent = null; for (var i = parents.length - 1; i >= 0 && parent === null; i--) { if (node.kind === 'var' ? isScope(parents[i]) : isBlockScope(parents[i])) { parent = parents[i]; } } parent.locals = parent.locals || {}; node.declarations.forEach(function (declaration) { declarePattern(declaration.id, parent); }); }, 'FunctionDeclaration': function (node, parents) { var parent = null; for (var i = parents.length - 2; i >= 0 && parent === null; i--) { if (isScope(parents[i])) { parent = parents[i]; } } parent.locals = parent.locals || {}; parent.locals[node.id.name] = true; declareFunction(node); }, 'Function': declareFunction, 'ClassDeclaration': function (node, parents) { var parent = null; for (var i = parents.length - 2; i >= 0 && parent === null; i--) { if (isScope(parents[i])) { parent = parents[i]; } } parent.locals = parent.locals || {}; parent.locals[node.id.name] = true; }, 'TryStatement': function (node) { if (node.handler === null) return; node.handler.locals = node.handler.locals || {}; node.handler.locals[node.handler.param.name] = true; }, 'ImportDefaultSpecifier': declareModuleSpecifier, 'ImportSpecifier': declareModuleSpecifier, 'ImportNamespaceSpecifier': declareModuleSpecifier }); function identifier(node, parents) { var name = node.name; if (name === 'undefined') return; for (var i = 0; i < parents.length; i++) { if (name === 'arguments' && declaresArguments(parents[i])) { return; } if (parents[i].locals && name in parents[i].locals) { return; } } node.parents = parents; globals.push(node); } walk.ancestor(ast, { 'VariablePattern': identifier, 'Identifier': identifier, 'ThisExpression': function (node, parents) { for (var i = 0; i < parents.length; i++) { if (declaresThis(parents[i])) { return; } } node.parents = parents; globals.push(node); } }); var groupedGlobals = {}; globals.forEach(function (node) { var name = node.type === 'ThisExpression' ? 'this' : node.name; groupedGlobals[name] = (groupedGlobals[name] || []); groupedGlobals[name].push(node); }); return Object.keys(groupedGlobals).sort().map(function (name) { return {name: name, nodes: groupedGlobals[name]}; }); }
function findGlobals(source) { var globals = []; var ast = typeof source === 'string' ? acorn.parse(source, { ecmaVersion: 6, allowReturnOutsideFunction: true, sourceType: 'module' }) : source; if (!(ast && typeof ast === 'object' && ast.type === 'Program')) { throw new TypeError('Source must be either a string of JavaScript or an acorn AST'); } var declareFunction = function (node) { var fn = node; fn.locals = fn.locals || {}; node.params.forEach(function (node) { fn.locals[node.name] = true; }); if (node.id) { fn.locals[node.id.name] = true; } } walk.ancestor(ast, { 'VariableDeclaration': function (node, parents) { var parent = null; for (var i = parents.length - 1; i >= 0 && parent === null; i--) { if (node.kind === 'var' ? isScope(parents[i]) : isBlockScope(parents[i])) { parent = parents[i]; } } parent.locals = parent.locals || {}; node.declarations.forEach(function (declaration) { parent.locals[declaration.id.name] = true; }); }, 'FunctionDeclaration': function (node, parents) { var parent = null; for (var i = parents.length - 2; i >= 0 && parent === null; i--) { if (isScope(parents[i])) { parent = parents[i]; } } parent.locals = parent.locals || {}; parent.locals[node.id.name] = true; declareFunction(node); }, 'Function': declareFunction, 'TryStatement': function (node) { node.handler.body.locals = node.handler.body.locals || {}; node.handler.body.locals[node.handler.param.name] = true; }, 'ImportDefaultSpecifier': function (node) { if (node.local.type === 'Identifier') { ast.locals = ast.locals || {}; ast.locals[node.local.name] = true; } }, 'ImportSpecifier': function (node) { var id = node.local ? node.local : node.imported; if (id.type === 'Identifier') { ast.locals = ast.locals || {}; ast.locals[id.name] = true; } }, 'ImportNamespaceSpecifier': function (node) { if (node.local.type === 'Identifier') { ast.locals = ast.locals || {}; ast.locals[node.local.name] = true; } } }); walk.ancestor(ast, { 'Identifier': function (node, parents) { var name = node.name; if (name === 'undefined') return; for (var i = 0; i < parents.length; i++) { if (name === 'arguments' && declaresArguments(parents[i])) { return; } if (parents[i].locals && name in parents[i].locals) { return; } } node.parents = parents; globals.push(node); }, ThisExpression: function (node, parents) { for (var i = 0; i < parents.length; i++) { if (declaresThis(parents[i])) { return; } } node.parents = parents; globals.push(node); } }); var groupedGlobals = {}; globals.forEach(function (node) { groupedGlobals[node.name] = (groupedGlobals[node.name] || []); groupedGlobals[node.name].push(node); }); return Object.keys(groupedGlobals).sort().map(function (name) { return {name: name, nodes: groupedGlobals[name]}; }); }