2016-06-27 13:42:28 +02:00
|
|
|
"use strict";
|
|
|
|
|
2017-10-25 20:19:42 +02:00
|
|
|
const GrammarError = require( "../../grammar-error" );
|
|
|
|
const visitor = require( "../visitor" );
|
2016-06-27 13:42:28 +02:00
|
|
|
|
2016-09-17 16:28:28 +02:00
|
|
|
// Checks that each label is defined only once within each scope.
|
2017-10-25 20:19:42 +02:00
|
|
|
function reportDuplicateLabels( ast ) {
|
|
|
|
|
|
|
|
let check;
|
|
|
|
|
|
|
|
function cloneEnv( env ) {
|
|
|
|
|
|
|
|
const clone = {};
|
|
|
|
|
|
|
|
Object.keys( env ).forEach( name => {
|
|
|
|
|
|
|
|
clone[ name ] = env[ name ];
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
return clone;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
function checkExpressionWithClonedEnv( node, env ) {
|
|
|
|
|
|
|
|
check( node.expression, cloneEnv( env ) );
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
check = visitor.build( {
|
|
|
|
rule( node ) {
|
|
|
|
|
|
|
|
check( node.expression, {} );
|
|
|
|
|
|
|
|
},
|
|
|
|
|
|
|
|
choice( node, env ) {
|
|
|
|
|
|
|
|
node.alternatives.forEach( alternative => {
|
|
|
|
|
|
|
|
check( alternative, cloneEnv( env ) );
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
},
|
|
|
|
|
|
|
|
action: checkExpressionWithClonedEnv,
|
|
|
|
|
|
|
|
labeled( node, env ) {
|
|
|
|
|
|
|
|
const label = node.label;
|
|
|
|
|
|
|
|
if ( Object.prototype.hasOwnProperty.call( env, label ) ) {
|
|
|
|
|
|
|
|
const start = env[ label ].start;
|
|
|
|
|
|
|
|
throw new GrammarError(
|
|
|
|
`Label "${ label }" is already defined at line ${ start.line }, column ${ start.column }.`,
|
|
|
|
node.location
|
|
|
|
);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
check( node.expression, env );
|
|
|
|
env[ label ] = node.location;
|
|
|
|
|
|
|
|
},
|
|
|
|
|
|
|
|
text: checkExpressionWithClonedEnv,
|
|
|
|
simple_and: checkExpressionWithClonedEnv,
|
|
|
|
simple_not: checkExpressionWithClonedEnv,
|
|
|
|
optional: checkExpressionWithClonedEnv,
|
|
|
|
zero_or_more: checkExpressionWithClonedEnv,
|
|
|
|
one_or_more: checkExpressionWithClonedEnv,
|
|
|
|
group: checkExpressionWithClonedEnv
|
|
|
|
} );
|
|
|
|
|
|
|
|
check( ast );
|
|
|
|
|
2016-06-27 13:42:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = reportDuplicateLabels;
|