pegjs/lib/ast/visitor.js
2018-09-05 10:16:34 +01:00

143 lines
3 KiB
JavaScript

"use strict";
const util = require( "../util" );
const __slice = Array.prototype.slice;
// Abstract syntax tree visitor for PEG.js
class ASTVisitor {
// Will traverse the node, strictly assuming the visitor can handle the node type.
visit( node ) {
// istanbul ignore next
if ( ! node ) throw new Error( "Visitor function called with no arguments or a `falsy` node" );
const fn = this[ node.type ];
// istanbul ignore next
if ( ! fn ) throw new Error( `Visitor function for node type "${ node.type }" not defined` );
return fn.apply( this, arguments ); // eslint-disable-line prefer-rest-params
}
}
module.exports = {
ASTVisitor,
// Simple AST node visitor builder for PEG.js
build( functions ) {
let visitor = new ASTVisitor();
util.extend( visitor, functions );
visitor = util.enforceFastProperties( visitor );
return visitor.visit.bind( visitor );
},
};
// Helper's to create visitor's for use with the ASTVisitor class
const on = ASTVisitor.on = {
// Visit a node that is defined as a property on another node
property( name ) {
return function visitProperty( node, ...extraArgs ) {
const value = node[ name ];
if ( extraArgs.length )
this.visit( value, ...extraArgs );
else
this.visit( value );
};
},
// Visit an array of nodes that are defined as a property on another node
children( name ) {
return function visitProperty( node, ...extraArgs ) {
const children = node[ name ];
const visitor = this;
const cb = extraArgs.length < 1
? function withoutArgs( child ) {
visitor.visit( child );
}
: function withArgs( child ) {
visitor.visit( child, ...extraArgs );
};
children.forEach( cb );
};
},
};
// Build the default ast visitor functions
const visitNop = util.noop;
const visitExpression = on.property( "expression" );
const DEFAULT_FUNCTIONS = {
grammar( node, ...extraArgs ) {
if ( node.initializer ) {
this.visit( node.initializer, ...extraArgs );
}
node.rules.forEach( rule => {
this.visit( rule, ...extraArgs );
} );
},
initializer: visitNop,
rule: visitExpression,
named: visitExpression,
choice: on.children( "alternatives" ),
action: visitExpression,
sequence: on.children( "elements" ),
labeled: visitExpression,
text: visitExpression,
simple_and: visitExpression,
simple_not: visitExpression,
optional: visitExpression,
zero_or_more: visitExpression,
one_or_more: visitExpression,
group: visitExpression,
semantic_and: visitNop,
semantic_not: visitNop,
rule_ref: visitNop,
literal: visitNop,
class: visitNop,
any: visitNop,
};
util.each( DEFAULT_FUNCTIONS, ( fn, name ) => {
ASTVisitor.prototype[ name ] = fn;
} );