PHP WebShell
Текущая директория: /opt/BitGoJS/node_modules/jsdoc/lib/jsdoc/src
Просмотр файла: walker.js
/**
* Traversal utilities for ASTs that are compatible with the ESTree API.
*
* @module jsdoc/src/walker
*/
const astnode = require('jsdoc/src/astnode');
const logger = require('jsdoc/util/logger');
const Syntax = require('jsdoc/src/syntax').Syntax;
// TODO: docs
function getCurrentScope(scopes) {
return scopes[scopes.length - 1] || null;
}
// TODO: docs
function moveLeadingComments(source, target, count) {
if (source.leadingComments) {
if (count === undefined) {
count = source.leadingComments.length;
}
target.leadingComments = source.leadingComments.slice(0, count);
source.leadingComments = source.leadingComments.slice(count);
}
}
// TODO: docs
function moveTrailingComments(source, target, count) {
if (source.trailingComments) {
if (count === undefined) {
count = source.trailingComments.length;
}
target.trailingComments = source.trailingComments.slice(
source.trailingComments.length - count, count
);
source.trailingComments = source.trailingComments.slice(0);
}
}
/* eslint-disable no-empty-function, no-unused-vars */
function leafNode(node, parent, state, cb) {}
/* eslint-enable no-empty-function, no-unused-vars */
// TODO: docs
const walkers = exports.walkers = {};
walkers[Syntax.ArrayExpression] = (node, parent, state, cb) => {
for (let element of node.elements) {
if (element) {
cb(element, node, state);
}
}
};
// TODO: verify correctness
walkers[Syntax.ArrayPattern] = (node, parent, state, cb) => {
for (let element of node.elements) {
// must be an identifier or an expression
if (element && element.type !== Syntax.Identifier) {
cb(element, node, state);
}
}
};
walkers[Syntax.ArrowFunctionExpression] = (node, parent, state, cb) => {
if (node.id) {
cb(node.id, node, state);
}
for (let param of node.params) {
cb(param, node, state);
}
cb(node.body, node, state);
};
walkers[Syntax.AssignmentExpression] = (node, parent, state, cb) => {
cb(node.left, node, state);
cb(node.right, node, state);
};
walkers[Syntax.AssignmentPattern] = walkers[Syntax.AssignmentExpression];
walkers[Syntax.AwaitExpression] = (node, parent, state, cb) => {
cb(node.argument, node, state);
};
walkers[Syntax.BigIntLiteral] = leafNode;
walkers[Syntax.BinaryExpression] = (node, parent, state, cb) => {
cb(node.left, node, state);
cb(node.right, node, state);
};
walkers[Syntax.BindExpression] = (node, parent, state, cb) => {
if (node.object) {
cb(node.object, node, state);
}
cb(node.callee, node, state);
};
walkers[Syntax.BlockStatement] = (node, parent, state, cb) => {
for (let bodyItem of node.body) {
cb(bodyItem, node, state);
}
};
walkers[Syntax.BreakStatement] = leafNode;
walkers[Syntax.CallExpression] = function(node, parent, state, cb) {
cb(node.callee, node, state);
if (node.arguments) {
for (let arg of node.arguments) {
cb(arg, node, state);
}
}
};
walkers[Syntax.CatchClause] = leafNode;
walkers[Syntax.ClassBody] = walkers[Syntax.BlockStatement];
walkers[Syntax.ClassDeclaration] = (node, parent, state, cb) => {
if (node.id) {
cb(node.id, node, state);
}
if (node.superClass) {
cb(node.superClass, node, state);
}
if (node.body) {
cb(node.body, node, state);
}
if (node.decorators) {
for (let decorator of node.decorators) {
cb(decorator, node, state);
}
}
};
walkers[Syntax.ClassExpression] = walkers[Syntax.ClassDeclaration];
// walkers[Syntax.ClassPrivateProperty] is defined later
// walkers[Syntax.ClassProperty] is defined later
// TODO: verify correctness
walkers[Syntax.ComprehensionBlock] = walkers[Syntax.AssignmentExpression];
// TODO: verify correctness
walkers[Syntax.ComprehensionExpression] = (node, parent, state, cb) => {
cb(node.body, node, state);
if (node.filter) {
cb(node.filter, node, state);
}
for (let block of node.blocks) {
cb(block, node, state);
}
};
walkers[Syntax.ConditionalExpression] = (node, parent, state, cb) => {
cb(node.test, node, state);
cb(node.consequent, node, state);
cb(node.alternate, node, state);
};
walkers[Syntax.ContinueStatement] = leafNode;
walkers[Syntax.DebuggerStatement] = leafNode;
walkers[Syntax.Decorator] = (node, parent, state, cb) => {
cb(node.expression, node, state);
};
walkers[Syntax.DoExpression] = (node, parent, state, cb) => {
cb(node.body, node, state);
};
walkers[Syntax.DoWhileStatement] = (node, parent, state, cb) => {
cb(node.test, node, state);
cb(node.body, node, state);
};
walkers[Syntax.EmptyStatement] = leafNode;
walkers[Syntax.ExperimentalRestProperty] = (node, parent, state, cb) => {
cb(node.argument, node, state);
};
walkers[Syntax.ExperimentalSpreadProperty] = walkers[Syntax.ExperimentalRestProperty];
walkers[Syntax.ExportAllDeclaration] = (node, parent, state, cb) => {
if (node.source) {
cb(node.source, node, state);
}
};
walkers[Syntax.ExportDefaultDeclaration] = (node, parent, state, cb) => {
// if the declaration target is a class, move leading comments to the declaration target
if (node.declaration && node.declaration.type === Syntax.ClassDeclaration) {
moveLeadingComments(node, node.declaration);
}
if (node.declaration) {
cb(node.declaration, node, state);
}
};
walkers[Syntax.ExportDefaultSpecifier] = (node, parent, state, cb) => {
cb(node.exported, node, state);
};
walkers[Syntax.ExportNamedDeclaration] = (node, parent, state, cb) => {
if (node.declaration) {
cb(node.declaration, node, state);
}
for (let specifier of node.specifiers) {
cb(specifier, node, state);
}
if (node.source) {
cb(node.source, node, state);
}
};
walkers[Syntax.ExportNamespaceSpecifier] = (node, parent, state, cb) => {
cb(node.exported, node, state);
};
walkers[Syntax.ExportSpecifier] = (node, parent, state, cb) => {
if (node.exported) {
cb(node.exported, node, state);
}
if (node.local) {
cb(node.local, node, state);
}
};
walkers[Syntax.ExpressionStatement] = (node, parent, state, cb) => {
moveLeadingComments(node, node.expression);
cb(node.expression, node, state);
};
walkers[Syntax.File] = (node, parent, state, cb) => {
cb(node.program, node, state);
};
walkers[Syntax.ForInStatement] = (node, parent, state, cb) => {
cb(node.left, node, state);
cb(node.right, node, state);
cb(node.body, node, state);
};
walkers[Syntax.ForOfStatement] = walkers[Syntax.ForInStatement];
walkers[Syntax.ForStatement] = (node, parent, state, cb) => {
if (node.init) {
cb(node.init, node, state);
}
if (node.test) {
cb(node.test, node, state);
}
if (node.update) {
cb(node.update, node, state);
}
cb(node.body, node, state);
};
walkers[Syntax.FunctionDeclaration] = walkers[Syntax.ArrowFunctionExpression];
walkers[Syntax.FunctionExpression] = walkers[Syntax.ArrowFunctionExpression];
walkers[Syntax.Identifier] = leafNode;
walkers[Syntax.IfStatement] = (node, parent, state, cb) => {
cb(node.test, node, state);
cb(node.consequent, node, state);
if (node.alternate) {
cb(node.alternate, node, state);
}
};
walkers[Syntax.Import] = leafNode;
walkers[Syntax.ImportDeclaration] = (node, parent, state, cb) => {
if (node.specifiers) {
for (let specifier of node.specifiers) {
cb(specifier, node, state);
}
}
if (node.source) {
cb(node.source, node, state);
}
};
walkers[Syntax.ImportDefaultSpecifier] = (node, parent, state, cb) => {
if (node.local) {
cb(node.local, node, state);
}
};
walkers[Syntax.ImportNamespaceSpecifier] = walkers[Syntax.ImportDefaultSpecifier];
walkers[Syntax.ImportSpecifier] = walkers[Syntax.ExportSpecifier];
walkers[Syntax.JSXAttribute] = (node, parent, state, cb) => {
cb(node.name, node, state);
if (node.value) {
cb(node.value, node, state);
}
};
walkers[Syntax.JSXClosingElement] = (node, parent, state, cb) => {
cb(node.name, node, state);
};
walkers[Syntax.JSXElement] = (node, parent, state, cb) => {
cb(node.openingElement, node, state);
if (node.closingElement) {
cb(node.closingElement, node, state);
}
for (let child of node.children) {
cb(child, node, state);
}
};
walkers[Syntax.JSXEmptyExpression] = leafNode;
walkers[Syntax.JSXExpressionContainer] = (node, parent, state, cb) => {
cb(node.expression, node, state);
};
walkers[Syntax.JSXIdentifier] = leafNode;
walkers[Syntax.JSXMemberExpression] = (node, parent, state, cb) => {
cb(node.object, node, state);
cb(node.property, node, state);
};
walkers[Syntax.JSXNamespacedName] = (node, parent, state, cb) => {
cb(node.namespace, node, state);
cb(node.name, node, state);
};
walkers[Syntax.JSXOpeningElement] = (node, parent, state, cb) => {
cb(node.name, node, state);
for (let attribute of node.attributes) {
cb(attribute, node, state);
}
};
walkers[Syntax.JSXSpreadAttribute] = (node, parent, state, cb) => {
cb(node.argument, node, state);
};
walkers[Syntax.JSXText] = leafNode;
walkers[Syntax.LabeledStatement] = (node, parent, state, cb) => {
cb(node.body, node, state);
};
// TODO: add scope info??
walkers[Syntax.LetStatement] = (node, parent, state, cb) => {
for (let headItem of node.head) {
cb(headItem.id, node, state);
if (headItem.init) {
cb(headItem.init, node, state);
}
}
cb(node.body, node, state);
};
walkers[Syntax.Literal] = leafNode;
walkers[Syntax.LogicalExpression] = walkers[Syntax.BinaryExpression];
walkers[Syntax.MemberExpression] = (node, parent, state, cb) => {
cb(node.object, node, state);
if (node.property) {
cb(node.property, node, state);
}
};
walkers[Syntax.MetaProperty] = leafNode;
walkers[Syntax.MethodDefinition] = (node, parent, state, cb) => {
if (node.key) {
cb(node.key, node, state);
}
if (node.value) {
cb(node.value, node, state);
}
if (node.decorators) {
for (let decorator of node.decorators) {
cb(decorator, node, state);
}
}
};
walkers[Syntax.ModuleDeclaration] = (node, parent, state, cb) => {
if (node.id) {
cb(node.id, node, state);
}
if (node.source) {
cb(node.source, node, state);
}
if (node.body) {
cb(node.body, node, state);
}
};
walkers[Syntax.NewExpression] = walkers[Syntax.CallExpression];
walkers[Syntax.ObjectExpression] = (node, parent, state, cb) => {
for (let property of node.properties) {
cb(property, node, state);
}
};
walkers[Syntax.ObjectPattern] = walkers[Syntax.ObjectExpression];
walkers[Syntax.PrivateName] = (node, parent, state, cb) => {
cb(node.id, node, state);
};
walkers[Syntax.Program] = (node, parent, state, cb) => {
// if the first item in the body has multiple leading comments, move all but the last one to
// this node. this happens, for example, when a file has a /** @module */ standalone comment
// followed by one or more other comments.
if (node.body[0] && node.body[0].leadingComments && node.body[0].leadingComments.length > 1) {
moveLeadingComments(node.body[0], node, node.body[0].leadingComments.length - 1);
}
// if the last item in the body has trailing comments, move them to this node
if (node.body.length && node.body[node.body.length - 1].trailingComments) {
moveTrailingComments(node.body[node.body.length - 1], node);
}
for (let bodyItem of node.body) {
cb(bodyItem, node, state);
}
};
walkers[Syntax.Property] = (node, parent, state, cb) => {
// move leading comments from key to property node
moveLeadingComments(node.key, node);
if (node.value) {
cb(node.value, node, state);
}
if (node.decorators) {
for (let decorator of node.decorators) {
cb(decorator, node, state);
}
}
};
walkers[Syntax.ClassPrivateProperty] = (node, parent, state, cb) => {
// move leading comments from key to property node
moveLeadingComments(node.key, node);
// add `name` property to key, so we don't have to give this type of node special treatment
// when we resolve its name
node.key.name = node.key.id.name;
if (node.value) {
cb(node.value, node, state);
}
if (node.decorators) {
for (let decorator of node.decorators) {
cb(decorator, node, state);
}
}
};
walkers[Syntax.ClassProperty] = walkers[Syntax.Property];
walkers[Syntax.RestElement] = (node, parent, state, cb) => {
if (node.argument) {
cb(node.argument, node, state);
}
};
walkers[Syntax.ReturnStatement] = (node, parent, state, cb) => {
if (node.argument) {
cb(node.argument, node, state);
}
};
walkers[Syntax.SequenceExpression] = (node, parent, state, cb) => {
for (let expression of node.expressions) {
cb(expression, node, state);
}
};
walkers[Syntax.SpreadElement] = (node, parent, state, cb) => {
if (node.argument) {
cb(node.argument, node, state);
}
};
walkers[Syntax.Super] = leafNode;
walkers[Syntax.SwitchCase] = (node, parent, state, cb) => {
if (node.test) {
cb(node.test, node, state);
}
for (let consequentItem of node.consequent) {
cb(consequentItem, node, state);
}
};
walkers[Syntax.SwitchStatement] = (node, parent, state, cb) => {
cb(node.discriminant, node, state);
for (let caseItem of node.cases) {
cb(caseItem, node, state);
}
};
walkers[Syntax.TaggedTemplateExpression] = (node, parent, state, cb) => {
if (node.tag) {
cb(node.tag, node, state);
}
if (node.quasi) {
cb(node.quasi, node, state);
}
};
walkers[Syntax.TemplateElement] = leafNode;
walkers[Syntax.TemplateLiteral] = (node, parent, state, cb) => {
if (node.quasis && node.quasis.length) {
for (let quasi of node.quasis) {
cb(quasi, node, state);
}
}
if (node.expressions && node.expressions.length) {
for (let expression of node.expressions) {
cb(expression, node, state);
}
}
};
walkers[Syntax.ThisExpression] = leafNode;
walkers[Syntax.ThrowStatement] = (node, parent, state, cb) => {
cb(node.argument, node, state);
};
walkers[Syntax.TryStatement] = (node, parent, state, cb) => {
cb(node.block, node, state);
if (node.handler) {
cb(node.handler.body, node, state);
}
if (node.finalizer) {
cb(node.finalizer, node, state);
}
};
walkers[Syntax.UnaryExpression] = (node, parent, state, cb) => {
cb(node.argument, node, state);
};
walkers[Syntax.UpdateExpression] = walkers[Syntax.UnaryExpression];
walkers[Syntax.VariableDeclaration] = (node, parent, state, cb) => {
// move leading comments to first declarator
moveLeadingComments(node, node.declarations[0]);
for (let declaration of node.declarations) {
cb(declaration, node, state);
}
};
walkers[Syntax.VariableDeclarator] = (node, parent, state, cb) => {
cb(node.id, node, state);
if (node.init) {
cb(node.init, node, state);
}
};
walkers[Syntax.WhileStatement] = walkers[Syntax.DoWhileStatement];
walkers[Syntax.WithStatement] = (node, parent, state, cb) => {
cb(node.object, node, state);
cb(node.body, node, state);
};
walkers[Syntax.YieldExpression] = (node, parent, state, cb) => {
if (node.argument) {
cb(node.argument, node, state);
}
};
/**
* Create a walker that can traverse an AST that is consistent with the Mozilla Parser API.
*
* @memberof module:jsdoc/src/walker
*/
class Walker {
// TODO: docs
constructor(walkerFuncs = walkers) {
this._walkers = walkerFuncs;
}
// TODO: docs
_recurse(filename, ast) {
const self = this;
const state = {
filename: filename,
nodes: [],
scopes: []
};
function logUnknownNodeType({type}) {
logger.debug('Found a node with unrecognized type %s. Ignoring the node and its ' +
'descendants.', type);
}
function cb(node, parent, cbState) {
let currentScope;
const isScope = astnode.isScope(node);
astnode.addNodeProperties(node);
node.parent = parent || null;
currentScope = getCurrentScope(cbState.scopes);
if (currentScope) {
node.enclosingScope = currentScope;
}
if (isScope) {
cbState.scopes.push(node);
}
cbState.nodes.push(node);
if (!self._walkers[node.type]) {
logUnknownNodeType(node);
} else {
self._walkers[node.type](node, parent, cbState, cb);
}
if (isScope) {
cbState.scopes.pop();
}
}
cb(ast, null, state);
return state;
}
// TODO: docs
recurse(ast, visitor, filename) {
let shouldContinue;
const state = this._recurse(filename, ast);
if (visitor) {
for (let node of state.nodes) {
shouldContinue = visitor.visit(node, filename);
if (!shouldContinue) {
break;
}
}
}
return ast;
}
}
exports.Walker = Walker;
Выполнить команду
Для локальной разработки. Не используйте в интернете!