2015-06-08 20:21:19 +02:00
|
|
|
"use strict";
|
|
|
|
|
2018-01-14 21:44:53 +01:00
|
|
|
const util = require( "../util" );
|
|
|
|
const __slice = Array.prototype.slice;
|
|
|
|
|
2018-01-16 03:54:14 +01:00
|
|
|
// Abstract syntax tree visitor for PEG.js
|
|
|
|
class ASTVisitor {
|
2014-06-04 07:23:34 +02:00
|
|
|
|
2018-01-16 03:54:14 +01:00
|
|
|
// Will traverse the node, strictly assuming the visitor can handle the node type.
|
|
|
|
visit( node ) {
|
2014-06-04 07:23:34 +02:00
|
|
|
|
2018-01-24 19:12:13 +01:00
|
|
|
// istanbul ignore next
|
|
|
|
if ( ! node ) throw new Error( "Visitor function called with no or `null` node" );
|
|
|
|
|
|
|
|
const func = this[ node.type ];
|
|
|
|
|
|
|
|
// istanbul ignore next
|
|
|
|
if ( ! func ) throw new Error( "Visitor function for node type '" + node.type + " not defined" );
|
|
|
|
|
2018-01-16 03:54:14 +01:00
|
|
|
const args = __slice.call( arguments, 0 );
|
2014-06-04 07:23:34 +02:00
|
|
|
|
2018-01-24 19:12:13 +01:00
|
|
|
return func.apply( this, args );
|
2018-01-16 03:54:14 +01:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
// Simple AST node visitor builder for PEG.js
|
|
|
|
static build( functions ) {
|
|
|
|
|
2018-01-16 03:57:12 +01:00
|
|
|
let visitor = new ASTVisitor();
|
2018-01-16 03:54:14 +01:00
|
|
|
util.extend( visitor, functions );
|
2018-01-16 03:57:12 +01:00
|
|
|
visitor = util.enforceFastProperties( visitor );
|
2018-01-16 03:54:14 +01:00
|
|
|
return visitor.visit.bind( visitor );
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
2018-01-16 05:28:27 +01:00
|
|
|
ASTVisitor.ASTVisitor = ASTVisitor;
|
2018-01-16 03:54:14 +01:00
|
|
|
module.exports = ASTVisitor;
|
2014-06-04 07:23:34 +02:00
|
|
|
|
2018-01-16 03:54:14 +01:00
|
|
|
// Helper's to create visitor's for use with the ASTVisitor class
|
|
|
|
const on = ASTVisitor.for = {
|
2014-06-04 07:23:34 +02:00
|
|
|
|
2018-01-16 03:54:14 +01:00
|
|
|
// Visit a node that is defined as a property on another node
|
|
|
|
property( name ) {
|
|
|
|
|
|
|
|
return function visitProperty( node ) {
|
2017-10-25 20:19:42 +02:00
|
|
|
|
2018-01-14 21:44:53 +01:00
|
|
|
const extraArgs = __slice.call( arguments, 1 );
|
2018-01-16 03:54:14 +01:00
|
|
|
const value = node[ name ];
|
2014-06-04 07:23:34 +02:00
|
|
|
|
2018-01-16 03:54:14 +01:00
|
|
|
if ( extraArgs.length )
|
2014-06-04 07:23:34 +02:00
|
|
|
|
2018-01-16 03:54:14 +01:00
|
|
|
this.visit.apply( this, [ value ].concat( extraArgs ) );
|
2014-06-04 07:23:34 +02:00
|
|
|
|
2018-01-16 03:54:14 +01:00
|
|
|
else
|
2017-10-25 20:19:42 +02:00
|
|
|
|
2018-01-16 03:54:14 +01:00
|
|
|
this.visit( value );
|
2017-10-25 20:19:42 +02:00
|
|
|
|
2018-01-16 03:54:14 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
},
|
|
|
|
|
|
|
|
// Visit an array of nodes that are defined as a property on another node
|
|
|
|
children( name ) {
|
|
|
|
|
|
|
|
return function visitProperty( node ) {
|
|
|
|
|
|
|
|
const args = __slice.call( arguments, 0 );
|
|
|
|
const children = node[ name ];
|
|
|
|
const visitor = this;
|
|
|
|
|
|
|
|
const cb = args.length < 2
|
|
|
|
? function withoutArgs( child ) {
|
|
|
|
|
|
|
|
visitor.visit( child );
|
2017-10-25 20:19:42 +02:00
|
|
|
|
2018-01-14 21:44:53 +01:00
|
|
|
}
|
2018-01-16 03:54:14 +01:00
|
|
|
: function withArgs( child ) {
|
2017-10-25 20:19:42 +02:00
|
|
|
|
2018-01-16 03:54:14 +01:00
|
|
|
args[ 0 ] = child;
|
|
|
|
visitor.visit.apply( visitor, args );
|
2017-10-25 20:19:42 +02:00
|
|
|
|
2018-01-14 21:44:53 +01:00
|
|
|
};
|
2017-10-25 20:19:42 +02:00
|
|
|
|
2018-01-14 21:44:53 +01:00
|
|
|
children.forEach( cb );
|
2017-10-25 20:19:42 +02:00
|
|
|
|
2018-01-16 03:54:14 +01:00
|
|
|
};
|
2017-10-25 20:19:42 +02:00
|
|
|
|
2018-01-16 03:54:14 +01:00
|
|
|
},
|
2017-10-25 20:19:42 +02:00
|
|
|
|
2018-01-16 03:54:14 +01:00
|
|
|
};
|
2017-10-25 20:19:42 +02:00
|
|
|
|
2018-01-16 03:54:14 +01:00
|
|
|
// Build the default ast visitor functions
|
2017-10-25 20:19:42 +02:00
|
|
|
|
2018-01-16 03:54:14 +01:00
|
|
|
const visitNop = util.noop;
|
|
|
|
const visitExpression = on.property( "expression" );
|
2017-10-25 20:19:42 +02:00
|
|
|
|
2018-01-16 03:54:14 +01:00
|
|
|
const DEFAULT_FUNCTIONS = {
|
2017-10-25 20:19:42 +02:00
|
|
|
|
2018-01-16 03:54:14 +01:00
|
|
|
grammar( node ) {
|
2017-10-25 20:19:42 +02:00
|
|
|
|
2018-01-16 03:54:14 +01:00
|
|
|
const extraArgs = __slice.call( arguments, 1 );
|
2017-10-25 20:19:42 +02:00
|
|
|
|
2018-01-16 03:54:14 +01:00
|
|
|
if ( node.initializer ) {
|
|
|
|
|
|
|
|
this.visit.apply( this, [ node.initializer ].concat( extraArgs ) );
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
node.rules.forEach( rule => {
|
|
|
|
|
|
|
|
this.visit.apply( this, [ rule ].concat( 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,
|
2017-10-25 20:19:42 +02:00
|
|
|
|
2014-05-08 11:48:56 +02:00
|
|
|
};
|
|
|
|
|
2018-01-16 03:54:14 +01:00
|
|
|
util.each( DEFAULT_FUNCTIONS, ( fn, name ) => {
|
|
|
|
|
|
|
|
ASTVisitor.prototype[ name ] = fn;
|
|
|
|
|
|
|
|
} );
|