2015-06-08 20:21:19 +02:00
|
|
|
"use strict";
|
|
|
|
|
2016-09-17 16:28:28 +02:00
|
|
|
// Simple AST node visitor builder.
|
2017-10-25 20:19:42 +02:00
|
|
|
const visitor = {
|
|
|
|
build( functions ) {
|
2014-06-04 07:23:34 +02:00
|
|
|
|
2017-10-25 20:19:42 +02:00
|
|
|
function visit( node ) {
|
2014-06-04 07:23:34 +02:00
|
|
|
|
2017-10-25 20:19:42 +02:00
|
|
|
return functions[ node.type ].apply( null, arguments );
|
2014-06-04 07:23:34 +02:00
|
|
|
|
2017-10-25 20:19:42 +02:00
|
|
|
}
|
2014-06-04 07:23:34 +02:00
|
|
|
|
2017-10-25 20:19:42 +02:00
|
|
|
function visitNop() {
|
|
|
|
// Do nothing.
|
|
|
|
}
|
2014-06-04 07:23:34 +02:00
|
|
|
|
2017-10-25 20:19:42 +02:00
|
|
|
function visitExpression( node ) {
|
|
|
|
|
|
|
|
const extraArgs = Array.prototype.slice.call( arguments, 1 );
|
2014-06-04 07:23:34 +02:00
|
|
|
|
2017-10-26 11:20:29 +02:00
|
|
|
visit.apply( null, [ node.expression ].concat( extraArgs ) );
|
2014-06-04 07:23:34 +02:00
|
|
|
|
2016-10-04 11:00:25 +02:00
|
|
|
}
|
2014-06-04 07:23:34 +02:00
|
|
|
|
2017-10-25 20:19:42 +02:00
|
|
|
function visitChildren( property ) {
|
|
|
|
|
|
|
|
return function visitProperty( node ) {
|
|
|
|
|
|
|
|
const extraArgs = Array.prototype.slice.call( arguments, 1 );
|
|
|
|
|
|
|
|
node[ property ].forEach( child => {
|
|
|
|
|
2017-10-26 11:20:29 +02:00
|
|
|
visit.apply( null, [ child ].concat( extraArgs ) );
|
2017-10-25 20:19:42 +02:00
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
const DEFAULT_FUNCTIONS = {
|
|
|
|
grammar( node ) {
|
|
|
|
|
|
|
|
const extraArgs = Array.prototype.slice.call( arguments, 1 );
|
|
|
|
|
|
|
|
if ( node.initializer ) {
|
|
|
|
|
2017-10-26 11:20:29 +02:00
|
|
|
visit.apply( null, [ node.initializer ].concat( extraArgs ) );
|
2017-10-25 20:19:42 +02:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
node.rules.forEach( rule => {
|
|
|
|
|
2017-10-26 11:20:29 +02:00
|
|
|
visit.apply( null, [ rule ].concat( extraArgs ) );
|
2017-10-25 20:19:42 +02:00
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
},
|
|
|
|
|
|
|
|
initializer: visitNop,
|
|
|
|
rule: visitExpression,
|
|
|
|
named: visitExpression,
|
|
|
|
choice: visitChildren( "alternatives" ),
|
|
|
|
action: visitExpression,
|
|
|
|
sequence: visitChildren( "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
|
|
|
|
};
|
|
|
|
|
|
|
|
Object.keys( DEFAULT_FUNCTIONS ).forEach( type => {
|
|
|
|
|
|
|
|
if ( ! Object.prototype.hasOwnProperty.call( functions, type ) ) {
|
|
|
|
|
|
|
|
functions[ type ] = DEFAULT_FUNCTIONS[ type ];
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
return visit;
|
|
|
|
|
|
|
|
}
|
2014-05-08 11:48:56 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
module.exports = visitor;
|