2015-06-08 20:21:19 +02:00
|
|
|
"use strict";
|
|
|
|
|
2017-10-25 20:19:42 +02:00
|
|
|
const chai = require( "chai" );
|
2018-09-08 03:39:12 +02:00
|
|
|
const peg = require( "pegjs" );
|
2017-10-25 20:19:42 +02:00
|
|
|
const sinon = require( "sinon" );
|
2016-05-02 16:05:09 +02:00
|
|
|
|
2017-10-25 20:19:42 +02:00
|
|
|
const expect = chai.expect;
|
2014-05-17 07:52:49 +02:00
|
|
|
|
2017-10-25 20:19:42 +02:00
|
|
|
describe( "PEG.js API", function () {
|
|
|
|
|
|
|
|
describe( "generate", function () {
|
|
|
|
|
|
|
|
it( "generates a parser", function () {
|
|
|
|
|
|
|
|
const parser = peg.generate( "start = 'a'" );
|
|
|
|
|
|
|
|
expect( parser ).to.be.an( "object" );
|
|
|
|
expect( parser.parse( "a" ) ).to.equal( "a" );
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
it( "throws an exception on syntax error", function () {
|
|
|
|
|
|
|
|
expect( () => {
|
|
|
|
|
|
|
|
peg.generate( "start = @" );
|
|
|
|
|
|
|
|
} ).to.throw();
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
it( "throws an exception on semantic error", function () {
|
|
|
|
|
|
|
|
expect( () => {
|
|
|
|
|
|
|
|
peg.generate( "start = undefined" );
|
|
|
|
|
|
|
|
} ).to.throw();
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
describe( "allowed start rules", function () {
|
|
|
|
|
|
|
|
const grammar = `
|
|
|
|
|
|
|
|
a = 'x'
|
|
|
|
b = 'x'
|
|
|
|
c = 'x'
|
|
|
|
|
|
|
|
`;
|
|
|
|
|
|
|
|
it( "throws an error on missing rule", function () {
|
|
|
|
|
|
|
|
expect( () => {
|
|
|
|
|
|
|
|
peg.generate( grammar, { allowedStartRules: [ "missing" ] } );
|
|
|
|
|
|
|
|
} ).to.throw();
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
// The |allowedStartRules| option is implemented separately for each
|
|
|
|
// optimization mode, so we need to test it in both.
|
|
|
|
|
|
|
|
describe( "when optimizing for parsing speed", function () {
|
|
|
|
|
|
|
|
describe( "when |allowedStartRules| is not set", function () {
|
|
|
|
|
|
|
|
it( "generated parser can start only from the first rule", function () {
|
|
|
|
|
|
|
|
const parser = peg.generate( grammar, { optimize: "speed" } );
|
|
|
|
|
|
|
|
expect( parser.parse( "x", { startRule: "a" } ) ).to.equal( "x" );
|
|
|
|
expect( () => {
|
|
|
|
|
|
|
|
parser.parse( "x", { startRule: "b" } );
|
|
|
|
|
|
|
|
} ).to.throw();
|
|
|
|
expect( () => {
|
|
|
|
|
|
|
|
parser.parse( "x", { startRule: "c" } );
|
|
|
|
|
|
|
|
} ).to.throw();
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
describe( "when |allowedStartRules| is set", function () {
|
|
|
|
|
|
|
|
it( "generated parser can start only from specified rules", function () {
|
|
|
|
|
|
|
|
const parser = peg.generate( grammar, {
|
|
|
|
optimize: "speed",
|
2019-09-29 04:24:42 +02:00
|
|
|
allowedStartRules: [ "b", "c" ],
|
2017-10-25 20:19:42 +02:00
|
|
|
} );
|
|
|
|
|
|
|
|
expect( () => {
|
|
|
|
|
|
|
|
parser.parse( "x", { startRule: "a" } );
|
|
|
|
|
|
|
|
} ).to.throw();
|
|
|
|
expect( parser.parse( "x", { startRule: "b" } ) ).to.equal( "x" );
|
|
|
|
expect( parser.parse( "x", { startRule: "c" } ) ).to.equal( "x" );
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
describe( "when optimizing for code size", function () {
|
|
|
|
|
|
|
|
describe( "when |allowedStartRules| is not set", function () {
|
|
|
|
|
|
|
|
it( "generated parser can start only from the first rule", function () {
|
|
|
|
|
|
|
|
const parser = peg.generate( grammar, { optimize: "size" } );
|
|
|
|
|
|
|
|
expect( parser.parse( "x", { startRule: "a" } ) ).to.equal( "x" );
|
|
|
|
expect( () => {
|
|
|
|
|
|
|
|
parser.parse( "x", { startRule: "b" } );
|
|
|
|
|
|
|
|
} ).to.throw();
|
|
|
|
expect( () => {
|
|
|
|
|
|
|
|
parser.parse( "x", { startRule: "c" } );
|
|
|
|
|
|
|
|
} ).to.throw();
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
describe( "when |allowedStartRules| is set", function () {
|
|
|
|
|
|
|
|
it( "generated parser can start only from specified rules", function () {
|
|
|
|
|
|
|
|
const parser = peg.generate( grammar, {
|
|
|
|
optimize: "size",
|
2019-09-29 04:24:42 +02:00
|
|
|
allowedStartRules: [ "b", "c" ],
|
2017-10-25 20:19:42 +02:00
|
|
|
} );
|
|
|
|
|
|
|
|
expect( () => {
|
|
|
|
|
|
|
|
parser.parse( "x", { startRule: "a" } );
|
|
|
|
|
|
|
|
} ).to.throw();
|
|
|
|
expect( parser.parse( "x", { startRule: "b" } ) ).to.equal( "x" );
|
|
|
|
expect( parser.parse( "x", { startRule: "c" } ) ).to.equal( "x" );
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
describe( "intermediate results caching", function () {
|
|
|
|
|
|
|
|
const grammar = `
|
|
|
|
|
|
|
|
{ var n = 0; }
|
|
|
|
start = (a 'b') / (a 'c') { return n; }
|
|
|
|
a = 'a' { n++; }
|
|
|
|
|
|
|
|
`;
|
|
|
|
|
|
|
|
describe( "when |cache| is not set", function () {
|
|
|
|
|
|
|
|
it( "generated parser doesn't cache intermediate parse results", function () {
|
|
|
|
|
|
|
|
const parser = peg.generate( grammar );
|
|
|
|
expect( parser.parse( "ac" ) ).to.equal( 2 );
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
describe( "when |cache| is set to |false|", function () {
|
|
|
|
|
|
|
|
it( "generated parser doesn't cache intermediate parse results", function () {
|
|
|
|
|
|
|
|
const parser = peg.generate( grammar, { cache: false } );
|
|
|
|
expect( parser.parse( "ac" ) ).to.equal( 2 );
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
describe( "when |cache| is set to |true|", function () {
|
|
|
|
|
|
|
|
it( "generated parser caches intermediate parse results", function () {
|
|
|
|
|
|
|
|
const parser = peg.generate( grammar, { cache: true } );
|
|
|
|
expect( parser.parse( "ac" ) ).to.equal( 1 );
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
describe( "tracing", function () {
|
|
|
|
|
|
|
|
const grammar = "start = 'a'";
|
|
|
|
|
|
|
|
describe( "when |trace| is not set", function () {
|
|
|
|
|
|
|
|
it( "generated parser doesn't trace", function () {
|
|
|
|
|
|
|
|
const parser = peg.generate( grammar );
|
|
|
|
const tracer = { trace: sinon.spy() };
|
|
|
|
|
|
|
|
parser.parse( "a", { tracer: tracer } );
|
|
|
|
|
|
|
|
expect( tracer.trace.called ).to.equal( false );
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
describe( "when |trace| is set to |false|", function () {
|
|
|
|
|
|
|
|
it( "generated parser doesn't trace", function () {
|
|
|
|
|
|
|
|
const parser = peg.generate( grammar, { trace: false } );
|
|
|
|
const tracer = { trace: sinon.spy() };
|
|
|
|
|
|
|
|
parser.parse( "a", { tracer: tracer } );
|
|
|
|
|
|
|
|
expect( tracer.trace.called ).to.equal( false );
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
describe( "when |trace| is set to |true|", function () {
|
|
|
|
|
|
|
|
it( "generated parser traces", function () {
|
|
|
|
|
|
|
|
const parser = peg.generate( grammar, { trace: true } );
|
|
|
|
const tracer = { trace: sinon.spy() };
|
|
|
|
|
|
|
|
parser.parse( "a", { tracer: tracer } );
|
|
|
|
|
|
|
|
expect( tracer.trace.called ).to.equal( true );
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
// The |optimize| option isn't tested because there is no meaningful way to
|
|
|
|
// write the tests without turning this into a performance test.
|
|
|
|
|
|
|
|
describe( "output", function () {
|
|
|
|
|
|
|
|
const grammar = "start = 'a'";
|
|
|
|
|
|
|
|
describe( "when |output| is not set", function () {
|
|
|
|
|
|
|
|
it( "returns generated parser object", function () {
|
|
|
|
|
|
|
|
const parser = peg.generate( grammar );
|
|
|
|
|
|
|
|
expect( parser ).to.be.an( "object" );
|
|
|
|
expect( parser.parse( "a" ) ).to.equal( "a" );
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
describe( "when |output| is set to |\"parser\"|", function () {
|
|
|
|
|
|
|
|
it( "returns generated parser object", function () {
|
|
|
|
|
|
|
|
const parser = peg.generate( grammar, { output: "parser" } );
|
|
|
|
|
|
|
|
expect( parser ).to.be.an( "object" );
|
|
|
|
expect( parser.parse( "a" ) ).to.equal( "a" );
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
describe( "when |output| is set to |\"source\"|", function () {
|
|
|
|
|
|
|
|
it( "returns generated parser source code", function () {
|
|
|
|
|
|
|
|
const source = peg.generate( grammar, { output: "source" } );
|
|
|
|
|
|
|
|
expect( source ).to.be.a( "string" );
|
|
|
|
expect( eval( source ).parse( "a" ) ).to.equal( "a" );
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
// The |format|, |exportVars|, and |dependencies| options are not tested
|
|
|
|
// becasue there is no meaningful way to thest their effects without turning
|
|
|
|
// this into an integration test.
|
|
|
|
|
|
|
|
// The |plugins| option is tested in plugin API tests.
|
|
|
|
|
Improve error when reserved word used as label (#552)
Before this commit error looks like (for input `start = break:'a'`)
> Expected "!", "$", "&", "(", "*", "+", ".", "/", "/*", "//", ";", "?", character class, code block, comment, end of line, identifier, literal, or whitespace but ":" found.
After this error looks like
> Label can't be a reserved word "break".
2018-01-07 15:51:06 +01:00
|
|
|
describe( "reserved words", function () {
|
|
|
|
|
2018-04-03 04:16:55 +02:00
|
|
|
const RESERVED_WORDS = peg.util.reservedWords;
|
Improve error when reserved word used as label (#552)
Before this commit error looks like (for input `start = break:'a'`)
> Expected "!", "$", "&", "(", "*", "+", ".", "/", "/*", "//", ";", "?", character class, code block, comment, end of line, identifier, literal, or whitespace but ":" found.
After this error looks like
> Label can't be a reserved word "break".
2018-01-07 15:51:06 +01:00
|
|
|
|
|
|
|
describe( "throws an exception on reserved JS words used as labels", function () {
|
|
|
|
|
|
|
|
for ( const label of RESERVED_WORDS ) {
|
|
|
|
|
|
|
|
it( label, function () {
|
|
|
|
|
|
|
|
expect( () => {
|
|
|
|
|
|
|
|
peg.generate( [
|
|
|
|
"start = " + label + ":end",
|
2019-09-29 04:24:42 +02:00
|
|
|
"end = 'a'",
|
Improve error when reserved word used as label (#552)
Before this commit error looks like (for input `start = break:'a'`)
> Expected "!", "$", "&", "(", "*", "+", ".", "/", "/*", "//", ";", "?", character class, code block, comment, end of line, identifier, literal, or whitespace but ":" found.
After this error looks like
> Label can't be a reserved word "break".
2018-01-07 15:51:06 +01:00
|
|
|
].join( "\n" ), { output: "source" } );
|
|
|
|
|
|
|
|
} ).to.throw( peg.parser.SyntaxError );
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
describe( "not throws an exception on reserved JS words used as rule name", function () {
|
|
|
|
|
|
|
|
for ( const rule of RESERVED_WORDS ) {
|
|
|
|
|
|
|
|
it( rule, function () {
|
|
|
|
|
|
|
|
expect( () => {
|
|
|
|
|
|
|
|
peg.generate( [
|
|
|
|
"start = " + rule,
|
2019-09-29 04:24:42 +02:00
|
|
|
rule + " = 'a'",
|
Improve error when reserved word used as label (#552)
Before this commit error looks like (for input `start = break:'a'`)
> Expected "!", "$", "&", "(", "*", "+", ".", "/", "/*", "//", ";", "?", character class, code block, comment, end of line, identifier, literal, or whitespace but ":" found.
After this error looks like
> Label can't be a reserved word "break".
2018-01-07 15:51:06 +01:00
|
|
|
].join( "\n" ), { output: "source" } );
|
|
|
|
|
|
|
|
} ).to.not.throw( peg.parser.SyntaxError );
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
2017-10-25 20:19:42 +02:00
|
|
|
it( "accepts custom options", function () {
|
|
|
|
|
|
|
|
peg.generate( "start = 'a'", { foo: 42 } );
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
} );
|
|
|
|
|
|
|
|
} );
|