2016-09-22 16:23:24 +02:00
|
|
|
"use strict";
|
|
|
|
|
2016-09-08 13:29:06 +02:00
|
|
|
/* global console */
|
2015-06-08 20:21:19 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
let chai = require("chai");
|
2016-09-08 16:04:36 +02:00
|
|
|
let peg = require("../../lib/peg");
|
2016-12-08 08:59:04 +01:00
|
|
|
let sinon = require("sinon");
|
|
|
|
|
|
|
|
let expect = chai.expect;
|
2016-09-08 13:29:06 +02:00
|
|
|
|
2014-05-23 15:07:22 +02:00
|
|
|
describe("generated parser behavior", function() {
|
2014-05-16 17:55:30 +02:00
|
|
|
function varyOptimizationOptions(block) {
|
|
|
|
function clone(object) {
|
2016-09-09 11:43:06 +02:00
|
|
|
let result = {};
|
2012-04-22 10:52:41 +02:00
|
|
|
|
2016-09-14 16:08:14 +02:00
|
|
|
Object.keys(object).forEach(key => {
|
|
|
|
result[key] = object[key];
|
|
|
|
});
|
2012-04-22 10:52:41 +02:00
|
|
|
|
2014-05-16 17:55:30 +02:00
|
|
|
return result;
|
|
|
|
}
|
2012-04-22 10:52:41 +02:00
|
|
|
|
2016-09-08 16:04:36 +02:00
|
|
|
let optionsVariants = [
|
2016-10-04 11:00:25 +02:00
|
|
|
{ cache: false, optimize: "speed", trace: false },
|
|
|
|
{ cache: false, optimize: "speed", trace: true },
|
|
|
|
{ cache: false, optimize: "size", trace: false },
|
|
|
|
{ cache: false, optimize: "size", trace: true },
|
|
|
|
{ cache: true, optimize: "speed", trace: false },
|
|
|
|
{ cache: true, optimize: "speed", trace: true },
|
|
|
|
{ cache: true, optimize: "size", trace: false },
|
|
|
|
{ cache: true, optimize: "size", trace: true }
|
|
|
|
];
|
2014-05-16 17:55:30 +02:00
|
|
|
|
2016-09-09 13:27:24 +02:00
|
|
|
optionsVariants.forEach(variant => {
|
2014-05-16 17:55:30 +02:00
|
|
|
describe(
|
2016-12-08 08:59:04 +01:00
|
|
|
"with options " + chai.util.inspect(variant),
|
2016-09-01 13:03:51 +02:00
|
|
|
function() { block(clone(variant)); }
|
2014-05-16 17:55:30 +02:00
|
|
|
);
|
2016-09-01 13:03:51 +02:00
|
|
|
});
|
2012-04-22 10:52:41 +02:00
|
|
|
}
|
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
function withConsoleStub(block) {
|
|
|
|
if (typeof console === "object") {
|
|
|
|
sinon.stub(console, "log");
|
|
|
|
}
|
2012-04-22 10:52:41 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
try {
|
|
|
|
return block();
|
|
|
|
} finally {
|
|
|
|
if (typeof console === "object") {
|
|
|
|
console.log.restore();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-08-01 11:37:23 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
function helpers(chai, utils) {
|
|
|
|
let Assertion = chai.Assertion;
|
2016-08-01 11:37:23 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
Assertion.addMethod("parse", function(input, expected, options) {
|
|
|
|
options = options !== undefined ? options : {};
|
|
|
|
|
|
|
|
let result = withConsoleStub(() =>
|
|
|
|
utils.flag(this, "object").parse(input, options)
|
|
|
|
);
|
|
|
|
|
|
|
|
if (expected !== undefined) {
|
|
|
|
this.assert(
|
|
|
|
utils.eql(result, expected),
|
|
|
|
"expected #{this} to parse input as #{exp} but got #{act}",
|
|
|
|
"expected #{this} to not parse input as #{exp}",
|
|
|
|
expected,
|
|
|
|
result,
|
|
|
|
!utils.flag(this, "negate")
|
|
|
|
);
|
2012-04-22 10:52:41 +02:00
|
|
|
}
|
|
|
|
});
|
2016-08-01 11:50:00 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
Assertion.addMethod("failToParse", function(input, props, options) {
|
|
|
|
options = options !== undefined ? options : {};
|
|
|
|
|
|
|
|
let passed, result;
|
|
|
|
|
|
|
|
try {
|
|
|
|
result = withConsoleStub(() =>
|
|
|
|
utils.flag(this, "object").parse(input, options)
|
|
|
|
);
|
|
|
|
passed = true;
|
|
|
|
} catch (e) {
|
|
|
|
result = e;
|
|
|
|
passed = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.assert(
|
|
|
|
!passed,
|
|
|
|
"expected #{this} to fail to parse input but got #{act}",
|
|
|
|
"expected #{this} to not fail to parse input but #{act} was thrown",
|
|
|
|
null,
|
|
|
|
result
|
|
|
|
);
|
|
|
|
|
|
|
|
if (!passed && props !== undefined) {
|
|
|
|
Object.keys(props).forEach(key => {
|
|
|
|
new Assertion(result).to.have.property(key)
|
|
|
|
.that.is.deep.equal(props[key]);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
// Helper activation needs to put inside a |beforeEach| block because the
|
2016-12-08 10:05:17 +01:00
|
|
|
// helpers conflict with the ones in test/unit/parser.spec.js.
|
2016-12-08 08:59:04 +01:00
|
|
|
beforeEach(function() {
|
|
|
|
chai.use(helpers);
|
2012-04-22 10:52:41 +02:00
|
|
|
});
|
|
|
|
|
2014-05-16 17:55:30 +02:00
|
|
|
varyOptimizationOptions(function(options) {
|
2015-01-12 14:48:29 +01:00
|
|
|
describe("initializer", function() {
|
2015-01-26 07:44:51 +01:00
|
|
|
it("executes the code before parsing starts", function() {
|
2016-09-08 16:04:36 +02:00
|
|
|
let parser = peg.generate([
|
2016-10-04 11:00:25 +02:00
|
|
|
"{ var result = 42; }",
|
|
|
|
"start = 'a' { return result; }"
|
|
|
|
].join("\n"), options);
|
2012-04-22 16:45:28 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("a", 42);
|
2012-04-22 16:45:28 +02:00
|
|
|
});
|
2012-09-19 08:32:21 +02:00
|
|
|
|
2015-01-26 07:44:51 +01:00
|
|
|
describe("available variables and functions", function() {
|
|
|
|
it("|options| contains options", function() {
|
2016-09-08 16:04:36 +02:00
|
|
|
let parser = peg.generate([
|
2016-10-04 11:00:25 +02:00
|
|
|
"{ var result = options; }",
|
|
|
|
"start = 'a' { return result; }"
|
|
|
|
].join("\n"), options);
|
2012-09-19 08:32:21 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("a", { a: 42 }, { a: 42 });
|
2015-01-26 07:44:51 +01:00
|
|
|
});
|
2012-09-19 08:32:21 +02:00
|
|
|
});
|
2012-04-22 16:45:28 +02:00
|
|
|
});
|
|
|
|
|
2015-01-12 14:48:29 +01:00
|
|
|
describe("rule", function() {
|
2012-04-22 17:01:02 +02:00
|
|
|
if (options.cache) {
|
|
|
|
it("caches rule match results", function() {
|
2016-09-08 16:04:36 +02:00
|
|
|
let parser = peg.generate([
|
2016-10-04 11:00:25 +02:00
|
|
|
"{ var n = 0; }",
|
|
|
|
"start = (a 'b') / (a 'c') { return n; }",
|
|
|
|
"a = 'a' { n++; }"
|
|
|
|
].join("\n"), options);
|
2012-04-22 17:01:02 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("ac", 1);
|
2012-04-22 17:01:02 +02:00
|
|
|
});
|
|
|
|
} else {
|
2015-01-24 20:17:52 +01:00
|
|
|
it("doesn't cache rule match results", function() {
|
2016-09-08 16:04:36 +02:00
|
|
|
let parser = peg.generate([
|
2016-10-04 11:00:25 +02:00
|
|
|
"{ var n = 0; }",
|
|
|
|
"start = (a 'b') / (a 'c') { return n; }",
|
|
|
|
"a = 'a' { n++; }"
|
|
|
|
].join("\n"), options);
|
2012-04-22 17:01:02 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("ac", 2);
|
2012-04-22 17:01:02 +02:00
|
|
|
});
|
|
|
|
}
|
2015-01-12 15:28:31 +01:00
|
|
|
|
2015-01-24 20:17:52 +01:00
|
|
|
describe("when the expression matches", function() {
|
|
|
|
it("returns its match result", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = 'a'");
|
2015-01-12 15:28:31 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("a", "a");
|
2015-01-12 15:28:31 +01:00
|
|
|
});
|
2015-01-24 20:17:52 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
describe("when the expression doesn't match", function() {
|
|
|
|
describe("without display name", function() {
|
|
|
|
it("reports match failure and doesn't record any expectation", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = 'a'");
|
2015-01-24 20:17:52 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("b", {
|
2016-06-17 15:14:47 +02:00
|
|
|
expected: [{ type: "literal", text: "a", ignoreCase: false }]
|
2015-01-24 20:17:52 +01:00
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("with display name", function() {
|
|
|
|
it("reports match failure and records an expectation of type \"other\"", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start 'start' = 'a'");
|
2015-01-24 20:17:52 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("b", {
|
2015-01-24 20:17:52 +01:00
|
|
|
expected: [{ type: "other", description: "start" }]
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it("discards any expectations recorded when matching the expression", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start 'start' = 'a'");
|
2015-01-12 15:28:31 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("b", {
|
2015-01-24 20:17:52 +01:00
|
|
|
expected: [{ type: "other", description: "start" }]
|
|
|
|
});
|
2015-01-12 15:28:31 +01:00
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
2012-06-24 16:55:30 +02:00
|
|
|
});
|
2012-04-30 10:28:01 +02:00
|
|
|
|
2015-01-12 14:48:29 +01:00
|
|
|
describe("literal", function() {
|
2015-01-12 16:09:38 +01:00
|
|
|
describe("matching", function() {
|
|
|
|
it("matches empty literals", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = ''", options);
|
2012-04-30 10:28:01 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("");
|
2015-01-12 16:09:38 +01:00
|
|
|
});
|
2012-04-22 17:01:02 +02:00
|
|
|
|
2015-01-12 16:09:38 +01:00
|
|
|
it("matches one-character literals", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = 'a'", options);
|
2012-04-22 16:33:12 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("a");
|
|
|
|
expect(parser).to.failToParse("b");
|
2015-01-12 16:09:38 +01:00
|
|
|
});
|
2012-06-26 20:28:06 +02:00
|
|
|
|
2015-01-12 16:09:38 +01:00
|
|
|
it("matches multi-character literals", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = 'abcd'", options);
|
2012-06-26 20:28:06 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("abcd");
|
|
|
|
expect(parser).to.failToParse("efgh");
|
2015-01-12 16:09:38 +01:00
|
|
|
});
|
2012-06-26 20:28:06 +02:00
|
|
|
|
2015-01-12 16:09:38 +01:00
|
|
|
it("is case sensitive without the \"i\" flag", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = 'a'", options);
|
2012-06-26 20:28:06 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("a");
|
|
|
|
expect(parser).to.failToParse("A");
|
2015-01-12 16:09:38 +01:00
|
|
|
});
|
2012-06-26 20:28:06 +02:00
|
|
|
|
2015-01-12 16:09:38 +01:00
|
|
|
it("is case insensitive with the \"i\" flag", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = 'a'i", options);
|
2012-12-01 13:10:50 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("a");
|
|
|
|
expect(parser).to.parse("A");
|
2015-01-12 16:09:38 +01:00
|
|
|
});
|
2012-12-01 13:10:50 +01:00
|
|
|
});
|
2012-06-26 20:28:06 +02:00
|
|
|
|
2015-01-12 16:09:38 +01:00
|
|
|
describe("when it matches", function() {
|
|
|
|
it("returns the matched text", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = 'a'", options);
|
2015-01-12 16:09:38 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("a", "a");
|
2015-01-12 16:09:38 +01:00
|
|
|
});
|
|
|
|
|
2015-09-04 17:23:42 +02:00
|
|
|
it("consumes the matched text", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = 'a' .", options);
|
2012-06-26 20:28:06 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("ab");
|
2015-01-12 16:09:38 +01:00
|
|
|
});
|
2012-06-26 20:28:06 +02:00
|
|
|
});
|
|
|
|
|
2015-01-12 16:09:38 +01:00
|
|
|
describe("when it doesn't match", function() {
|
|
|
|
it("reports match failure and records an expectation of type \"literal\"", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = 'a'", options);
|
2013-11-29 16:34:09 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("b", {
|
2016-06-17 15:14:47 +02:00
|
|
|
expected: [{ type: "literal", text: "a", ignoreCase: false }]
|
2015-01-12 16:09:38 +01:00
|
|
|
});
|
2013-11-29 16:34:09 +01:00
|
|
|
});
|
|
|
|
});
|
2015-01-12 14:48:29 +01:00
|
|
|
});
|
2013-11-29 16:34:09 +01:00
|
|
|
|
2015-01-12 14:48:29 +01:00
|
|
|
describe("character class", function() {
|
2015-01-12 16:23:08 +01:00
|
|
|
describe("matching", function() {
|
|
|
|
it("matches empty classes", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = []", options);
|
2013-11-30 17:26:06 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("a");
|
2015-01-12 16:23:08 +01:00
|
|
|
});
|
2013-11-30 17:26:06 +01:00
|
|
|
|
2015-01-12 16:23:08 +01:00
|
|
|
it("matches classes with a character list", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = [abc]", options);
|
2012-06-26 20:28:06 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("a");
|
|
|
|
expect(parser).to.parse("b");
|
|
|
|
expect(parser).to.parse("c");
|
|
|
|
expect(parser).to.failToParse("d");
|
2015-01-12 16:23:08 +01:00
|
|
|
});
|
2012-06-26 20:28:06 +02:00
|
|
|
|
2015-01-12 16:23:08 +01:00
|
|
|
it("matches classes with a character range", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = [a-c]", options);
|
2014-04-19 18:14:12 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("a");
|
|
|
|
expect(parser).to.parse("b");
|
|
|
|
expect(parser).to.parse("c");
|
|
|
|
expect(parser).to.failToParse("d");
|
2015-01-12 16:23:08 +01:00
|
|
|
});
|
2014-04-19 18:14:12 +02:00
|
|
|
|
2015-01-12 16:23:08 +01:00
|
|
|
it("matches inverted classes", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = [^a]", options);
|
2012-09-19 08:32:21 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("a");
|
|
|
|
expect(parser).to.parse("b");
|
2015-01-12 16:23:08 +01:00
|
|
|
});
|
2012-06-26 20:28:06 +02:00
|
|
|
|
2015-01-12 16:23:08 +01:00
|
|
|
it("is case sensitive without the \"i\" flag", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = [a]", options);
|
2012-04-22 16:29:22 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("a");
|
|
|
|
expect(parser).to.failToParse("A");
|
2015-01-12 16:23:08 +01:00
|
|
|
});
|
2012-04-22 16:29:22 +02:00
|
|
|
|
2015-01-12 16:23:08 +01:00
|
|
|
it("is case insensitive with the \"i\" flag", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = [a]i", options);
|
2012-04-22 16:29:22 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("a");
|
|
|
|
expect(parser).to.parse("A");
|
2015-01-12 16:23:08 +01:00
|
|
|
});
|
2012-04-22 16:29:22 +02:00
|
|
|
});
|
|
|
|
|
2015-01-12 16:23:08 +01:00
|
|
|
describe("when it matches", function() {
|
|
|
|
it("returns the matched character", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = [a]", options);
|
2012-04-22 16:03:43 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("a", "a");
|
2015-01-12 16:23:08 +01:00
|
|
|
});
|
|
|
|
|
2015-09-04 17:23:42 +02:00
|
|
|
it("consumes the matched character", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = [a] .", options);
|
2015-01-12 16:23:08 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("ab");
|
2015-01-12 16:23:08 +01:00
|
|
|
});
|
2012-04-22 16:03:43 +02:00
|
|
|
});
|
|
|
|
|
2015-01-12 16:23:08 +01:00
|
|
|
describe("when it doesn't match", function() {
|
|
|
|
it("reports match failure and records an expectation of type \"class\"", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = [a]", options);
|
2012-12-01 15:46:14 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("b", {
|
2016-06-17 15:14:47 +02:00
|
|
|
expected: [{ type: "class", parts: ["a"], inverted: false, ignoreCase: false }]
|
2015-01-12 16:23:08 +01:00
|
|
|
});
|
2015-01-12 14:48:29 +01:00
|
|
|
});
|
2012-12-01 15:46:14 +01:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2015-01-12 14:48:29 +01:00
|
|
|
describe("dot", function() {
|
2015-01-12 16:38:29 +01:00
|
|
|
describe("matching", function() {
|
|
|
|
it("matches any character", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = .", options);
|
2012-04-22 16:00:54 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("a");
|
|
|
|
expect(parser).to.parse("b");
|
|
|
|
expect(parser).to.parse("c");
|
2015-01-12 16:38:29 +01:00
|
|
|
});
|
2012-04-22 16:00:54 +02:00
|
|
|
});
|
|
|
|
|
2015-01-12 16:38:29 +01:00
|
|
|
describe("when it matches", function() {
|
|
|
|
it("returns the matched character", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = .", options);
|
2015-01-12 16:38:29 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("a", "a");
|
2015-01-12 16:38:29 +01:00
|
|
|
});
|
|
|
|
|
2015-09-04 17:23:42 +02:00
|
|
|
it("consumes the matched character", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = . .", options);
|
2012-04-22 16:00:54 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("ab");
|
2015-01-12 16:38:29 +01:00
|
|
|
});
|
2012-04-22 16:00:54 +02:00
|
|
|
});
|
2012-04-30 10:28:01 +02:00
|
|
|
|
2015-01-12 16:38:29 +01:00
|
|
|
describe("when it doesn't match", function() {
|
|
|
|
it("reports match failure and records an expectation of type \"any\"", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = .", options);
|
2012-04-30 10:28:01 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("", {
|
2016-06-17 15:14:47 +02:00
|
|
|
expected: [{ type: "any" }]
|
2015-01-12 16:38:29 +01:00
|
|
|
});
|
2015-01-12 14:48:29 +01:00
|
|
|
});
|
2012-04-30 10:28:01 +02:00
|
|
|
});
|
2012-04-22 16:00:54 +02:00
|
|
|
});
|
|
|
|
|
2015-01-12 14:48:29 +01:00
|
|
|
describe("rule reference", function() {
|
2015-01-12 16:53:59 +01:00
|
|
|
describe("when referenced rule's expression matches", function() {
|
|
|
|
it("returns its result", function() {
|
2016-09-08 16:04:36 +02:00
|
|
|
let parser = peg.generate([
|
2016-10-04 11:00:25 +02:00
|
|
|
"start = a",
|
|
|
|
"a = 'a'"
|
|
|
|
].join("\n"), options);
|
2012-04-30 10:28:01 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("a", "a");
|
2015-01-12 16:53:59 +01:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("when referenced rule's expression doesn't match", function() {
|
|
|
|
it("reports match failure", function() {
|
2016-09-08 16:04:36 +02:00
|
|
|
let parser = peg.generate([
|
2016-10-04 11:00:25 +02:00
|
|
|
"start = a",
|
|
|
|
"a = 'a'"
|
|
|
|
].join("\n"), options);
|
2015-01-12 16:53:59 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("b");
|
2015-01-12 16:53:59 +01:00
|
|
|
});
|
2012-04-30 10:28:01 +02:00
|
|
|
});
|
2012-04-22 15:56:00 +02:00
|
|
|
});
|
|
|
|
|
2015-01-12 14:48:29 +01:00
|
|
|
describe("positive semantic predicate", function() {
|
2015-01-26 09:37:20 +01:00
|
|
|
describe("when the code returns a truthy value", function() {
|
|
|
|
it("returns |undefined|", function() {
|
2016-09-17 16:28:28 +02:00
|
|
|
// The |""| is needed so that the parser doesn't return just
|
|
|
|
// |undefined| which we can't compare against in |toParse| due to the
|
|
|
|
// way optional parameters work.
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = &{ return true; } ''", options);
|
2012-04-22 15:45:13 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("", [undefined, ""]);
|
2015-01-26 09:37:20 +01:00
|
|
|
});
|
2012-04-22 15:45:13 +02:00
|
|
|
});
|
|
|
|
|
2015-01-26 09:37:20 +01:00
|
|
|
describe("when the code returns a falsey value", function() {
|
|
|
|
it("reports match failure", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = &{ return false; }", options);
|
2012-12-10 21:01:30 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("");
|
2015-01-26 09:37:20 +01:00
|
|
|
});
|
2012-12-10 21:01:30 +01:00
|
|
|
});
|
|
|
|
|
2015-01-26 09:37:20 +01:00
|
|
|
describe("label variables", function() {
|
2016-03-04 17:04:24 +01:00
|
|
|
describe("in containing sequence", function() {
|
2015-07-17 14:16:46 +02:00
|
|
|
it("can access variables defined by preceding labeled elements", function() {
|
2016-09-08 16:04:36 +02:00
|
|
|
let parser = peg.generate(
|
2016-10-04 11:00:25 +02:00
|
|
|
"start = a:'a' &{ return a === 'a'; }",
|
|
|
|
options
|
|
|
|
);
|
2012-04-22 15:45:13 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("a");
|
2015-07-17 14:16:46 +02:00
|
|
|
});
|
2016-03-06 17:12:28 +01:00
|
|
|
|
|
|
|
it("cannot access variable defined by labeled predicate element", function() {
|
2016-09-08 16:04:36 +02:00
|
|
|
let parser = peg.generate(
|
2016-10-04 11:00:25 +02:00
|
|
|
"start = 'a' b:&{ return b === undefined; } 'c'",
|
|
|
|
options
|
|
|
|
);
|
2016-03-06 17:12:28 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("ac");
|
2016-03-06 17:12:28 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
it("cannot access variables defined by following labeled elements", function() {
|
2016-09-08 16:04:36 +02:00
|
|
|
let parser = peg.generate(
|
2016-10-04 11:00:25 +02:00
|
|
|
"start = &{ return a === 'a'; } a:'a'",
|
|
|
|
options
|
|
|
|
);
|
2016-03-06 17:12:28 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("a");
|
2016-03-06 17:12:28 +01:00
|
|
|
});
|
2016-03-06 17:20:19 +01:00
|
|
|
|
|
|
|
it("cannot access variables defined by subexpressions", function() {
|
2016-09-08 16:04:36 +02:00
|
|
|
let testcases = [
|
2016-10-04 11:00:25 +02:00
|
|
|
{
|
|
|
|
grammar: "start = (a:'a') &{ return a === 'a'; }",
|
|
|
|
input: "a"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
grammar: "start = (a:'a')? &{ return a === 'a'; }",
|
|
|
|
input: "a"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
grammar: "start = (a:'a')* &{ return a === 'a'; }",
|
|
|
|
input: "a"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
grammar: "start = (a:'a')+ &{ return a === 'a'; }",
|
|
|
|
input: "a"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
grammar: "start = $(a:'a') &{ return a === 'a'; }",
|
|
|
|
input: "a"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
grammar: "start = &(a:'a') 'a' &{ return a === 'a'; }",
|
|
|
|
input: "a"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
grammar: "start = !(a:'a') 'b' &{ return a === 'a'; }",
|
|
|
|
input: "b"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
grammar: "start = b:(a:'a') &{ return a === 'a'; }",
|
|
|
|
input: "a"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
grammar: "start = ('a' b:'b' 'c') &{ return b === 'b'; }",
|
|
|
|
input: "abc"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
grammar: "start = (a:'a' { return a; }) &{ return a === 'a'; }",
|
|
|
|
input: "a"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
grammar: "start = ('a' / b:'b' / 'c') &{ return b === 'b'; }",
|
|
|
|
input: "b"
|
|
|
|
}
|
|
|
|
];
|
2016-03-06 17:20:19 +01:00
|
|
|
|
2016-09-09 13:27:24 +02:00
|
|
|
testcases.forEach(testcase => {
|
2016-09-23 05:56:52 +02:00
|
|
|
let parser = peg.generate(testcase.grammar, options);
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse(testcase.input);
|
2016-09-01 13:03:51 +02:00
|
|
|
});
|
2016-03-06 17:20:19 +01:00
|
|
|
});
|
2015-01-26 09:37:20 +01:00
|
|
|
});
|
2015-02-13 14:10:32 +01:00
|
|
|
|
2016-03-04 17:04:24 +01:00
|
|
|
describe("in outer sequence", function() {
|
2016-03-04 16:42:42 +01:00
|
|
|
it("can access variables defined by preceding labeled elements", function() {
|
2016-09-08 16:04:36 +02:00
|
|
|
let parser = peg.generate(
|
2016-10-04 11:00:25 +02:00
|
|
|
"start = a:'a' ('b' &{ return a === 'a'; })",
|
|
|
|
options
|
|
|
|
);
|
2015-02-13 14:10:32 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("ab");
|
2015-07-17 14:16:46 +02:00
|
|
|
});
|
2016-03-06 17:12:28 +01:00
|
|
|
|
|
|
|
it("cannot access variable defined by labeled predicate element", function() {
|
2016-09-08 16:04:36 +02:00
|
|
|
let parser = peg.generate(
|
2016-10-04 11:00:25 +02:00
|
|
|
"start = 'a' b:('b' &{ return b === undefined; }) 'c'",
|
|
|
|
options
|
|
|
|
);
|
2016-03-06 17:12:28 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("abc");
|
2016-03-06 17:12:28 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
it("cannot access variables defined by following labeled elements", function() {
|
2016-09-08 16:04:36 +02:00
|
|
|
let parser = peg.generate(
|
2016-10-04 11:00:25 +02:00
|
|
|
"start = ('a' &{ return b === 'b'; }) b:'b'",
|
|
|
|
options
|
|
|
|
);
|
2016-03-06 17:12:28 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("ab");
|
2016-03-06 17:12:28 +01:00
|
|
|
});
|
2015-02-13 14:10:32 +01:00
|
|
|
});
|
2012-04-22 15:45:13 +02:00
|
|
|
});
|
|
|
|
|
2015-01-26 09:37:20 +01:00
|
|
|
describe("initializer variables & functions", function() {
|
|
|
|
it("can access variables defined in the initializer", function() {
|
2016-09-08 16:04:36 +02:00
|
|
|
let parser = peg.generate([
|
2016-10-04 11:00:25 +02:00
|
|
|
"{ var v = 42 }",
|
|
|
|
"start = &{ return v === 42; }"
|
|
|
|
].join("\n"), options);
|
2012-12-01 13:10:50 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("");
|
2015-01-26 09:37:20 +01:00
|
|
|
});
|
2012-12-01 13:10:50 +01:00
|
|
|
|
2015-01-26 09:37:20 +01:00
|
|
|
it("can access functions defined in the initializer", function() {
|
2016-09-08 16:04:36 +02:00
|
|
|
let parser = peg.generate([
|
2016-10-04 11:00:25 +02:00
|
|
|
"{ function f() { return 42; } }",
|
|
|
|
"start = &{ return f() === 42; }"
|
|
|
|
].join("\n"), options);
|
2012-12-01 13:10:50 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("");
|
2015-01-26 09:37:20 +01:00
|
|
|
});
|
2012-12-01 13:10:50 +01:00
|
|
|
});
|
2012-04-22 16:45:28 +02:00
|
|
|
|
2015-01-26 09:37:20 +01:00
|
|
|
describe("available variables & functions", function() {
|
|
|
|
it("|options| contains options", function() {
|
2016-09-08 16:04:36 +02:00
|
|
|
let parser = peg.generate([
|
2016-10-04 11:00:25 +02:00
|
|
|
"{ var result; }",
|
|
|
|
"start = &{ result = options; return true; } { return result; }"
|
|
|
|
].join("\n"), options);
|
2012-04-22 16:45:28 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("", { a: 42 }, { a: 42 });
|
2015-01-26 09:37:20 +01:00
|
|
|
});
|
2012-09-19 08:32:21 +02:00
|
|
|
|
2015-04-02 16:28:08 +02:00
|
|
|
it("|location| returns current location info", function() {
|
2016-09-08 16:04:36 +02:00
|
|
|
let parser = peg.generate([
|
2016-10-04 11:00:25 +02:00
|
|
|
"{ var result; }",
|
|
|
|
"start = line (nl+ line)* { return result; }",
|
|
|
|
"line = thing (' '+ thing)*",
|
|
|
|
"thing = digit / mark",
|
|
|
|
"digit = [0-9]",
|
|
|
|
"mark = &{ result = location(); return true; } 'x'",
|
|
|
|
"nl = '\\r'? '\\n'"
|
|
|
|
].join("\n"), options);
|
2014-04-19 18:14:12 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("1\n2\n\n3\n\n\n4 5 x", {
|
2015-04-02 16:28:08 +02:00
|
|
|
start: { offset: 13, line: 7, column: 5 },
|
2016-09-22 05:25:09 +02:00
|
|
|
end: { offset: 13, line: 7, column: 5 }
|
2015-04-02 16:28:08 +02:00
|
|
|
});
|
2014-04-19 18:14:12 +02:00
|
|
|
|
2016-09-17 16:28:28 +02:00
|
|
|
// Newline representations
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("1\nx", { // Unix
|
2015-04-02 16:28:08 +02:00
|
|
|
start: { offset: 2, line: 2, column: 1 },
|
2016-09-22 05:25:09 +02:00
|
|
|
end: { offset: 2, line: 2, column: 1 }
|
2015-04-02 16:28:08 +02:00
|
|
|
});
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("1\r\nx", { // Windows
|
2015-04-02 16:28:08 +02:00
|
|
|
start: { offset: 3, line: 2, column: 1 },
|
2016-09-22 05:25:09 +02:00
|
|
|
end: { offset: 3, line: 2, column: 1 }
|
2015-04-02 16:28:08 +02:00
|
|
|
});
|
2015-01-26 09:37:20 +01:00
|
|
|
});
|
2012-09-19 08:32:21 +02:00
|
|
|
});
|
2012-04-22 15:45:13 +02:00
|
|
|
});
|
|
|
|
|
2015-01-12 14:48:29 +01:00
|
|
|
describe("negative semantic predicate", function() {
|
2015-01-26 09:37:20 +01:00
|
|
|
describe("when the code returns a falsey value", function() {
|
|
|
|
it("returns |undefined|", function() {
|
2016-09-17 16:28:28 +02:00
|
|
|
// The |""| is needed so that the parser doesn't return just
|
|
|
|
// |undefined| which we can't compare against in |toParse| due to the
|
|
|
|
// way optional parameters work.
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = !{ return false; } ''", options);
|
2012-04-22 15:39:23 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("", [undefined, ""]);
|
2015-01-26 09:37:20 +01:00
|
|
|
});
|
2012-04-22 15:39:23 +02:00
|
|
|
});
|
|
|
|
|
2015-01-26 09:37:20 +01:00
|
|
|
describe("when the code returns a truthy value", function() {
|
|
|
|
it("reports match failure", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = !{ return true; }", options);
|
2012-12-10 21:01:30 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("");
|
2015-01-26 09:37:20 +01:00
|
|
|
});
|
2012-12-10 21:01:30 +01:00
|
|
|
});
|
|
|
|
|
2015-01-26 09:37:20 +01:00
|
|
|
describe("label variables", function() {
|
2016-03-04 17:04:24 +01:00
|
|
|
describe("in containing sequence", function() {
|
2015-07-17 14:16:46 +02:00
|
|
|
it("can access variables defined by preceding labeled elements", function() {
|
2016-09-08 16:04:36 +02:00
|
|
|
let parser = peg.generate(
|
2016-10-04 11:00:25 +02:00
|
|
|
"start = a:'a' !{ return a !== 'a'; }",
|
|
|
|
options
|
|
|
|
);
|
2012-04-22 15:39:23 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("a");
|
2015-07-17 14:16:46 +02:00
|
|
|
});
|
2016-03-06 17:12:28 +01:00
|
|
|
|
|
|
|
it("cannot access variable defined by labeled predicate element", function() {
|
2016-09-08 16:04:36 +02:00
|
|
|
let parser = peg.generate(
|
2016-10-04 11:00:25 +02:00
|
|
|
"start = 'a' b:!{ return b !== undefined; } 'c'",
|
|
|
|
options
|
|
|
|
);
|
2016-03-06 17:12:28 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("ac");
|
2016-03-06 17:12:28 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
it("cannot access variables defined by following labeled elements", function() {
|
2016-09-08 16:04:36 +02:00
|
|
|
let parser = peg.generate(
|
2016-10-04 11:00:25 +02:00
|
|
|
"start = !{ return a !== 'a'; } a:'a'",
|
|
|
|
options
|
|
|
|
);
|
2016-03-06 17:12:28 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("a");
|
2016-03-06 17:12:28 +01:00
|
|
|
});
|
2016-03-06 17:20:19 +01:00
|
|
|
|
|
|
|
it("cannot access variables defined by subexpressions", function() {
|
2016-09-08 16:04:36 +02:00
|
|
|
let testcases = [
|
2016-10-04 11:00:25 +02:00
|
|
|
{
|
|
|
|
grammar: "start = (a:'a') !{ return a !== 'a'; }",
|
|
|
|
input: "a"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
grammar: "start = (a:'a')? !{ return a !== 'a'; }",
|
|
|
|
input: "a"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
grammar: "start = (a:'a')* !{ return a !== 'a'; }",
|
|
|
|
input: "a"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
grammar: "start = (a:'a')+ !{ return a !== 'a'; }",
|
|
|
|
input: "a"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
grammar: "start = $(a:'a') !{ return a !== 'a'; }",
|
|
|
|
input: "a"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
grammar: "start = &(a:'a') 'a' !{ return a !== 'a'; }",
|
|
|
|
input: "a"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
grammar: "start = !(a:'a') 'b' !{ return a !== 'a'; }",
|
|
|
|
input: "b"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
grammar: "start = b:(a:'a') !{ return a !== 'a'; }",
|
|
|
|
input: "a"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
grammar: "start = ('a' b:'b' 'c') !{ return b !== 'b'; }",
|
|
|
|
input: "abc"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
grammar: "start = (a:'a' { return a; }) !{ return a !== 'a'; }",
|
|
|
|
input: "a"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
grammar: "start = ('a' / b:'b' / 'c') !{ return b !== 'b'; }",
|
|
|
|
input: "b"
|
|
|
|
}
|
|
|
|
];
|
2016-03-06 17:20:19 +01:00
|
|
|
|
2016-09-09 13:27:24 +02:00
|
|
|
testcases.forEach(testcase => {
|
2016-09-23 05:56:52 +02:00
|
|
|
let parser = peg.generate(testcase.grammar, options);
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse(testcase.input);
|
2016-09-01 13:03:51 +02:00
|
|
|
});
|
2016-03-06 17:20:19 +01:00
|
|
|
});
|
2015-01-26 09:37:20 +01:00
|
|
|
});
|
2015-02-13 14:10:32 +01:00
|
|
|
|
2016-03-04 17:04:24 +01:00
|
|
|
describe("in outer sequence", function() {
|
2016-03-04 16:42:42 +01:00
|
|
|
it("can access variables defined by preceding labeled elements", function() {
|
2016-09-08 16:04:36 +02:00
|
|
|
let parser = peg.generate(
|
2016-10-04 11:00:25 +02:00
|
|
|
"start = a:'a' ('b' !{ return a !== 'a'; })",
|
|
|
|
options
|
|
|
|
);
|
2015-02-13 14:10:32 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("ab");
|
2015-07-17 14:16:46 +02:00
|
|
|
});
|
2016-03-06 17:12:28 +01:00
|
|
|
|
|
|
|
it("cannot access variable defined by labeled predicate element", function() {
|
2016-09-08 16:04:36 +02:00
|
|
|
let parser = peg.generate(
|
2016-10-04 11:00:25 +02:00
|
|
|
"start = 'a' b:('b' !{ return b !== undefined; }) 'c'",
|
|
|
|
options
|
|
|
|
);
|
2016-03-06 17:12:28 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("abc");
|
2016-03-06 17:12:28 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
it("cannot access variables defined by following labeled elements", function() {
|
2016-09-08 16:04:36 +02:00
|
|
|
let parser = peg.generate(
|
2016-10-04 11:00:25 +02:00
|
|
|
"start = ('a' !{ return b !== 'b'; }) b:'b'",
|
|
|
|
options
|
|
|
|
);
|
2016-03-06 17:12:28 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("ab");
|
2016-03-06 17:12:28 +01:00
|
|
|
});
|
2015-02-13 14:10:32 +01:00
|
|
|
});
|
2012-04-22 15:39:23 +02:00
|
|
|
});
|
|
|
|
|
2015-01-26 09:37:20 +01:00
|
|
|
describe("initializer variables & functions", function() {
|
|
|
|
it("can access variables defined in the initializer", function() {
|
2016-09-08 16:04:36 +02:00
|
|
|
let parser = peg.generate([
|
2016-10-04 11:00:25 +02:00
|
|
|
"{ var v = 42 }",
|
|
|
|
"start = !{ return v !== 42; }"
|
|
|
|
].join("\n"), options);
|
2012-12-01 13:10:50 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("");
|
2015-01-26 09:37:20 +01:00
|
|
|
});
|
2012-12-01 13:10:50 +01:00
|
|
|
|
2015-01-26 09:37:20 +01:00
|
|
|
it("can access functions defined in the initializer", function() {
|
2016-09-08 16:04:36 +02:00
|
|
|
let parser = peg.generate([
|
2016-10-04 11:00:25 +02:00
|
|
|
"{ function f() { return 42; } }",
|
|
|
|
"start = !{ return f() !== 42; }"
|
|
|
|
].join("\n"), options);
|
2012-12-01 13:10:50 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("");
|
2015-01-26 09:37:20 +01:00
|
|
|
});
|
2012-12-01 13:10:50 +01:00
|
|
|
});
|
2012-04-22 16:45:28 +02:00
|
|
|
|
2015-01-26 09:37:20 +01:00
|
|
|
describe("available variables & functions", function() {
|
|
|
|
it("|options| contains options", function() {
|
2016-09-08 16:04:36 +02:00
|
|
|
let parser = peg.generate([
|
2016-10-04 11:00:25 +02:00
|
|
|
"{ var result; }",
|
|
|
|
"start = !{ result = options; return false; } { return result; }"
|
|
|
|
].join("\n"), options);
|
2012-04-22 16:45:28 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("", { a: 42 }, { a: 42 });
|
2015-01-26 09:37:20 +01:00
|
|
|
});
|
2012-09-19 08:32:21 +02:00
|
|
|
|
2015-04-02 16:28:08 +02:00
|
|
|
it("|location| returns current location info", function() {
|
2016-09-08 16:04:36 +02:00
|
|
|
let parser = peg.generate([
|
2016-10-04 11:00:25 +02:00
|
|
|
"{ var result; }",
|
|
|
|
"start = line (nl+ line)* { return result; }",
|
|
|
|
"line = thing (' '+ thing)*",
|
|
|
|
"thing = digit / mark",
|
|
|
|
"digit = [0-9]",
|
|
|
|
"mark = !{ result = location(); return false; } 'x'",
|
|
|
|
"nl = '\\r'? '\\n'"
|
|
|
|
].join("\n"), options);
|
2014-04-19 18:14:12 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("1\n2\n\n3\n\n\n4 5 x", {
|
2015-04-02 16:28:08 +02:00
|
|
|
start: { offset: 13, line: 7, column: 5 },
|
2016-09-22 05:25:09 +02:00
|
|
|
end: { offset: 13, line: 7, column: 5 }
|
2015-04-02 16:28:08 +02:00
|
|
|
});
|
2014-04-19 18:14:12 +02:00
|
|
|
|
2016-09-17 16:28:28 +02:00
|
|
|
// Newline representations
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("1\nx", { // Unix
|
2015-04-02 16:28:08 +02:00
|
|
|
start: { offset: 2, line: 2, column: 1 },
|
2016-09-22 05:25:09 +02:00
|
|
|
end: { offset: 2, line: 2, column: 1 }
|
2015-04-02 16:28:08 +02:00
|
|
|
});
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("1\r\nx", { // Windows
|
2015-04-02 16:28:08 +02:00
|
|
|
start: { offset: 3, line: 2, column: 1 },
|
2016-09-22 05:25:09 +02:00
|
|
|
end: { offset: 3, line: 2, column: 1 }
|
2015-04-02 16:28:08 +02:00
|
|
|
});
|
2015-01-26 09:37:20 +01:00
|
|
|
});
|
2012-09-19 08:32:21 +02:00
|
|
|
});
|
2012-04-22 15:39:23 +02:00
|
|
|
});
|
|
|
|
|
2015-01-26 09:44:42 +01:00
|
|
|
describe("group", function() {
|
|
|
|
describe("when the expression matches", function() {
|
|
|
|
it("returns its match result", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = ('a')", options);
|
2015-01-26 09:44:42 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("a", "a");
|
2015-01-26 09:44:42 +01:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("when the expression doesn't match", function() {
|
|
|
|
it("reports match failure", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = ('a')", options);
|
2015-01-26 09:44:42 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("b");
|
2015-01-26 09:44:42 +01:00
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2015-01-12 14:48:29 +01:00
|
|
|
describe("optional", function() {
|
2015-01-16 15:08:11 +01:00
|
|
|
describe("when the expression matches", function() {
|
|
|
|
it("returns its match result", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = 'a'?", options);
|
2012-04-22 11:31:15 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("a", "a");
|
2015-01-16 15:08:11 +01:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("when the expression doesn't match", function() {
|
|
|
|
it("returns |null|", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = 'a'?", options);
|
2015-01-16 15:08:11 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("", null);
|
2015-01-16 15:08:11 +01:00
|
|
|
});
|
2012-04-22 11:31:15 +02:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2015-01-12 14:48:29 +01:00
|
|
|
describe("zero or more", function() {
|
2015-01-16 15:20:33 +01:00
|
|
|
describe("when the expression matches zero or more times", function() {
|
|
|
|
it("returns an array of its match results", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = 'a'*", options);
|
2012-04-22 11:27:17 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("", []);
|
|
|
|
expect(parser).to.parse("a", ["a"]);
|
|
|
|
expect(parser).to.parse("aaa", ["a", "a", "a"]);
|
2015-01-16 15:20:33 +01:00
|
|
|
});
|
2012-04-22 11:27:17 +02:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2015-01-12 14:48:29 +01:00
|
|
|
describe("one or more", function() {
|
2015-01-16 15:24:48 +01:00
|
|
|
describe("when the expression matches one or more times", function() {
|
|
|
|
it("returns an array of its match results", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = 'a'+", options);
|
2012-04-22 11:24:16 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("a", ["a"]);
|
|
|
|
expect(parser).to.parse("aaa", ["a", "a", "a"]);
|
2015-01-16 15:24:48 +01:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("when the expression doesn't match", function() {
|
|
|
|
it("reports match failure", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = 'a'+", options);
|
2015-01-16 15:24:48 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("");
|
2015-01-16 15:24:48 +01:00
|
|
|
});
|
2012-04-22 11:24:16 +02:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2015-01-12 14:48:29 +01:00
|
|
|
describe("text", function() {
|
2015-01-16 15:42:32 +01:00
|
|
|
describe("when the expression matches", function() {
|
|
|
|
it("returns the matched text", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = $('a' 'b' 'c')", options);
|
2015-01-16 15:42:32 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("abc", "abc");
|
2015-01-16 15:42:32 +01:00
|
|
|
});
|
|
|
|
});
|
2012-04-22 11:12:30 +02:00
|
|
|
|
2015-01-16 15:42:32 +01:00
|
|
|
describe("when the expression doesn't match", function() {
|
|
|
|
it("reports match failure", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = $('a')", options);
|
2015-01-16 15:42:32 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("b");
|
2015-01-16 15:42:32 +01:00
|
|
|
});
|
2012-04-22 11:12:30 +02:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2015-01-12 14:48:29 +01:00
|
|
|
describe("positive simple predicate", function() {
|
2015-01-16 16:24:42 +01:00
|
|
|
describe("when the expression matches", function() {
|
|
|
|
it("returns |undefined|", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = &'a' 'a'", options);
|
2012-04-22 10:52:41 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("a", [undefined, "a"]);
|
2015-01-16 16:24:42 +01:00
|
|
|
});
|
2012-04-22 10:52:41 +02:00
|
|
|
|
2015-01-16 16:24:42 +01:00
|
|
|
it("resets parse position", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = &'a' 'a'", options);
|
2012-04-22 10:52:41 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("a");
|
2015-01-16 16:24:42 +01:00
|
|
|
});
|
2012-04-22 10:52:41 +02:00
|
|
|
});
|
|
|
|
|
2015-01-16 16:24:42 +01:00
|
|
|
describe("when the expression doesn't match", function() {
|
|
|
|
it("reports match failure", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = &'a'", options);
|
2015-01-16 16:24:42 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("b");
|
2015-01-16 16:24:42 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
it("discards any expectations recorded when matching the expression", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = 'a' / &'b' / 'c'", options);
|
2012-04-22 10:52:41 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("d", {
|
2015-01-16 16:24:42 +01:00
|
|
|
expected: [
|
2016-06-17 15:14:47 +02:00
|
|
|
{ type: "literal", text: "a", ignoreCase: false },
|
|
|
|
{ type: "literal", text: "c", ignoreCase: false }
|
2015-01-16 16:24:42 +01:00
|
|
|
]
|
|
|
|
});
|
|
|
|
});
|
2012-04-22 10:52:41 +02:00
|
|
|
});
|
2015-01-12 14:48:29 +01:00
|
|
|
});
|
2012-04-22 10:52:41 +02:00
|
|
|
|
2015-01-12 14:48:29 +01:00
|
|
|
describe("negative simple predicate", function() {
|
2015-01-16 16:24:42 +01:00
|
|
|
describe("when the expression matches", function() {
|
|
|
|
it("reports match failure", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = !'a'", options);
|
2012-04-22 10:52:41 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("a");
|
2015-01-16 16:24:42 +01:00
|
|
|
});
|
2012-04-22 10:52:41 +02:00
|
|
|
});
|
|
|
|
|
2015-01-16 16:24:42 +01:00
|
|
|
describe("when the expression doesn't match", function() {
|
|
|
|
it("returns |undefined|", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = !'a' 'b'", options);
|
2012-04-22 10:52:41 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("b", [undefined, "b"]);
|
2015-01-16 16:24:42 +01:00
|
|
|
});
|
2012-04-22 10:52:41 +02:00
|
|
|
|
2015-01-16 16:24:42 +01:00
|
|
|
it("resets parse position", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = !'a' 'b'", options);
|
2012-04-22 10:52:41 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("b");
|
2015-01-16 16:24:42 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
it("discards any expectations recorded when matching the expression", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = 'a' / !'b' / 'c'", options);
|
2015-01-16 16:24:42 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("b", {
|
2015-01-16 16:24:42 +01:00
|
|
|
expected: [
|
2016-06-17 15:14:47 +02:00
|
|
|
{ type: "literal", text: "a", ignoreCase: false },
|
|
|
|
{ type: "literal", text: "c", ignoreCase: false }
|
2015-01-16 16:24:42 +01:00
|
|
|
]
|
|
|
|
});
|
|
|
|
});
|
2012-04-22 10:52:41 +02:00
|
|
|
});
|
2015-01-12 14:48:29 +01:00
|
|
|
});
|
2012-04-30 10:28:01 +02:00
|
|
|
|
2015-01-12 14:48:29 +01:00
|
|
|
describe("label", function() {
|
2015-01-16 15:45:54 +01:00
|
|
|
describe("when the expression matches", function() {
|
|
|
|
it("returns its match result", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = a:'a'", options);
|
2012-04-30 10:28:01 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("a", "a");
|
2015-01-16 15:45:54 +01:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("when the expression doesn't match", function() {
|
|
|
|
it("reports match failure", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = a:'a'", options);
|
2015-01-16 15:45:54 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("b");
|
2015-01-16 15:45:54 +01:00
|
|
|
});
|
2012-04-30 10:28:01 +02:00
|
|
|
});
|
2012-04-22 10:52:41 +02:00
|
|
|
});
|
2012-04-22 10:59:14 +02:00
|
|
|
|
2015-01-12 14:48:29 +01:00
|
|
|
describe("sequence", function() {
|
2015-01-24 20:02:06 +01:00
|
|
|
describe("when all expressions match", function() {
|
|
|
|
it("returns an array of their match results", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = 'a' 'b' 'c'", options);
|
2012-04-22 11:07:00 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("abc", ["a", "b", "c"]);
|
2015-01-24 20:02:06 +01:00
|
|
|
});
|
2012-04-22 11:07:00 +02:00
|
|
|
});
|
|
|
|
|
2015-01-24 20:02:06 +01:00
|
|
|
describe("when any expression doesn't match", function() {
|
|
|
|
it("reports match failure", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = 'a' 'b' 'c'", options);
|
2015-01-24 20:02:06 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("dbc");
|
|
|
|
expect(parser).to.failToParse("adc");
|
|
|
|
expect(parser).to.failToParse("abd");
|
2015-01-24 20:02:06 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
it("resets parse position", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = 'a' 'b' / 'a'", options);
|
2012-04-22 11:07:00 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("a", "a");
|
2015-01-24 20:02:06 +01:00
|
|
|
});
|
2012-04-22 11:07:00 +02:00
|
|
|
});
|
2015-01-12 14:48:29 +01:00
|
|
|
});
|
2012-04-22 11:07:00 +02:00
|
|
|
|
2015-01-12 14:48:29 +01:00
|
|
|
describe("action", function() {
|
2015-01-26 09:10:52 +01:00
|
|
|
describe("when the expression matches", function() {
|
|
|
|
it("returns the value returned by the code", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = 'a' { return 42; }", options);
|
2012-04-22 11:07:00 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("a", 42);
|
2015-01-26 09:10:52 +01:00
|
|
|
});
|
2012-04-22 11:07:00 +02:00
|
|
|
|
2015-01-26 09:10:52 +01:00
|
|
|
describe("label variables", function() {
|
2015-07-17 14:16:46 +02:00
|
|
|
describe("in the expression", function() {
|
2016-03-04 17:13:14 +01:00
|
|
|
it("can access variable defined by labeled expression", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = a:'a' { return a; }", options);
|
2012-04-22 11:07:00 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("a", "a");
|
2015-07-17 14:16:46 +02:00
|
|
|
});
|
2012-04-22 11:07:00 +02:00
|
|
|
|
2016-03-04 17:13:14 +01:00
|
|
|
it("can access variables defined by labeled sequence elements", function() {
|
2016-09-08 16:04:36 +02:00
|
|
|
let parser = peg.generate(
|
2016-10-04 11:00:25 +02:00
|
|
|
"start = a:'a' b:'b' c:'c' { return [a, b, c]; }",
|
|
|
|
options
|
|
|
|
);
|
2012-04-22 11:07:00 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("abc", ["a", "b", "c"]);
|
2015-07-17 14:16:46 +02:00
|
|
|
});
|
2016-03-06 17:20:19 +01:00
|
|
|
|
|
|
|
it("cannot access variables defined by subexpressions", function() {
|
2016-09-08 16:04:36 +02:00
|
|
|
let testcases = [
|
2016-10-04 11:00:25 +02:00
|
|
|
{
|
|
|
|
grammar: "start = (a:'a') { return a; }",
|
|
|
|
input: "a"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
grammar: "start = (a:'a')? { return a; }",
|
|
|
|
input: "a"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
grammar: "start = (a:'a')* { return a; }",
|
|
|
|
input: "a"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
grammar: "start = (a:'a')+ { return a; }",
|
|
|
|
input: "a"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
grammar: "start = $(a:'a') { return a; }",
|
|
|
|
input: "a"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
grammar: "start = &(a:'a') 'a' { return a; }",
|
|
|
|
input: "a"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
grammar: "start = !(a:'a') 'b' { return a; }",
|
|
|
|
input: "b"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
grammar: "start = b:(a:'a') { return a; }",
|
|
|
|
input: "a"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
grammar: "start = ('a' b:'b' 'c') { return b; }",
|
|
|
|
input: "abc"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
grammar: "start = (a:'a' { return a; }) { return a; }",
|
|
|
|
input: "a"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
grammar: "start = ('a' / b:'b' / 'c') { return b; }",
|
|
|
|
input: "b"
|
|
|
|
}
|
|
|
|
];
|
2016-03-06 17:20:19 +01:00
|
|
|
|
2016-09-09 13:27:24 +02:00
|
|
|
testcases.forEach(testcase => {
|
2016-09-23 05:56:52 +02:00
|
|
|
let parser = peg.generate(testcase.grammar, options);
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse(testcase.input);
|
2016-09-01 13:03:51 +02:00
|
|
|
});
|
2016-03-06 17:20:19 +01:00
|
|
|
});
|
2015-01-26 09:10:52 +01:00
|
|
|
});
|
2015-02-13 14:10:32 +01:00
|
|
|
|
2016-03-04 17:04:24 +01:00
|
|
|
describe("in outer sequence", function() {
|
2016-03-04 16:42:42 +01:00
|
|
|
it("can access variables defined by preceding labeled elements", function() {
|
2016-09-08 16:04:36 +02:00
|
|
|
let parser = peg.generate(
|
2016-10-04 11:00:25 +02:00
|
|
|
"start = a:'a' ('b' { return a; })",
|
|
|
|
options
|
|
|
|
);
|
2016-03-06 16:41:37 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("ab", ["a", "a"]);
|
2015-07-17 14:16:46 +02:00
|
|
|
});
|
2016-03-06 17:12:28 +01:00
|
|
|
|
|
|
|
it("cannot access variable defined by labeled action element", function() {
|
2016-09-08 16:04:36 +02:00
|
|
|
let parser = peg.generate(
|
2016-10-04 11:00:25 +02:00
|
|
|
"start = 'a' b:('b' { return b; }) c:'c'",
|
|
|
|
options
|
|
|
|
);
|
2016-03-06 17:12:28 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("abc");
|
2016-03-06 17:12:28 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
it("cannot access variables defined by following labeled elements", function() {
|
2016-09-08 16:04:36 +02:00
|
|
|
let parser = peg.generate(
|
2016-10-04 11:00:25 +02:00
|
|
|
"start = ('a' { return b; }) b:'b'",
|
|
|
|
options
|
|
|
|
);
|
2016-03-06 17:12:28 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("ab");
|
2016-03-06 17:12:28 +01:00
|
|
|
});
|
2015-02-13 14:10:32 +01:00
|
|
|
});
|
2015-01-26 09:10:52 +01:00
|
|
|
});
|
2012-04-22 11:07:00 +02:00
|
|
|
|
2015-01-26 09:10:52 +01:00
|
|
|
describe("initializer variables & functions", function() {
|
|
|
|
it("can access variables defined in the initializer", function() {
|
2016-09-08 16:04:36 +02:00
|
|
|
let parser = peg.generate([
|
2016-10-04 11:00:25 +02:00
|
|
|
"{ var v = 42 }",
|
|
|
|
"start = 'a' { return v; }"
|
|
|
|
].join("\n"), options);
|
2012-04-22 11:07:00 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("a", 42);
|
2015-01-26 09:10:52 +01:00
|
|
|
});
|
2012-04-22 11:07:00 +02:00
|
|
|
|
2015-01-26 09:10:52 +01:00
|
|
|
it("can access functions defined in the initializer", function() {
|
2016-09-08 16:04:36 +02:00
|
|
|
let parser = peg.generate([
|
2016-10-04 11:00:25 +02:00
|
|
|
"{ function f() { return 42; } }",
|
|
|
|
"start = 'a' { return f(); }"
|
|
|
|
].join("\n"), options);
|
2012-04-30 10:28:01 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("a", 42);
|
2015-01-26 09:10:52 +01:00
|
|
|
});
|
|
|
|
});
|
2012-04-30 10:28:01 +02:00
|
|
|
|
2015-01-26 09:10:52 +01:00
|
|
|
describe("available variables & functions", function() {
|
|
|
|
it("|options| contains options", function() {
|
2016-09-08 16:04:36 +02:00
|
|
|
let parser = peg.generate(
|
2016-10-04 11:00:25 +02:00
|
|
|
"start = 'a' { return options; }",
|
|
|
|
options
|
|
|
|
);
|
2012-04-22 17:16:26 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("a", { a: 42 }, { a: 42 });
|
2015-01-26 09:10:52 +01:00
|
|
|
});
|
2012-06-25 21:46:47 +02:00
|
|
|
|
2015-01-26 09:10:52 +01:00
|
|
|
it("|text| returns text matched by the expression", function() {
|
2016-09-08 16:04:36 +02:00
|
|
|
let parser = peg.generate(
|
2016-10-04 11:00:25 +02:00
|
|
|
"start = 'a' { return text(); }",
|
|
|
|
options
|
|
|
|
);
|
2012-06-25 21:46:47 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("a", "a");
|
2015-01-26 09:10:52 +01:00
|
|
|
});
|
2012-06-25 21:46:47 +02:00
|
|
|
|
2015-04-02 16:28:08 +02:00
|
|
|
it("|location| returns location info of the expression", function() {
|
2016-09-08 16:04:36 +02:00
|
|
|
let parser = peg.generate([
|
2016-10-04 11:00:25 +02:00
|
|
|
"{ var result; }",
|
|
|
|
"start = line (nl+ line)* { return result; }",
|
|
|
|
"line = thing (' '+ thing)*",
|
|
|
|
"thing = digit / mark",
|
|
|
|
"digit = [0-9]",
|
|
|
|
"mark = 'x' { result = location(); }",
|
|
|
|
"nl = '\\r'? '\\n'"
|
|
|
|
].join("\n"), options);
|
2015-01-26 09:10:52 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("1\n2\n\n3\n\n\n4 5 x", {
|
2015-04-02 16:28:08 +02:00
|
|
|
start: { offset: 13, line: 7, column: 5 },
|
2016-09-22 05:25:09 +02:00
|
|
|
end: { offset: 14, line: 7, column: 6 }
|
2015-04-02 16:28:08 +02:00
|
|
|
});
|
2015-01-26 09:10:52 +01:00
|
|
|
|
2016-09-17 16:28:28 +02:00
|
|
|
// Newline representations
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("1\nx", { // Unix
|
2015-04-02 16:28:08 +02:00
|
|
|
start: { offset: 2, line: 2, column: 1 },
|
2016-09-22 05:25:09 +02:00
|
|
|
end: { offset: 3, line: 2, column: 2 }
|
2015-04-02 16:28:08 +02:00
|
|
|
});
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("1\r\nx", { // Windows
|
2015-04-02 16:28:08 +02:00
|
|
|
start: { offset: 3, line: 2, column: 1 },
|
2016-09-22 05:25:09 +02:00
|
|
|
end: { offset: 4, line: 2, column: 2 }
|
2015-04-02 16:28:08 +02:00
|
|
|
});
|
2015-01-26 09:10:52 +01:00
|
|
|
});
|
2012-06-25 21:46:47 +02:00
|
|
|
|
2016-04-27 14:11:32 +02:00
|
|
|
describe("|expected|", function() {
|
|
|
|
it("terminates parsing and throws an exception", function() {
|
2016-09-08 16:04:36 +02:00
|
|
|
let parser = peg.generate(
|
2016-10-04 11:00:25 +02:00
|
|
|
"start = 'a' { expected('a'); }",
|
|
|
|
options
|
|
|
|
);
|
2015-01-26 09:10:52 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("a", {
|
2016-09-22 05:25:09 +02:00
|
|
|
message: "Expected a but \"a\" found.",
|
2016-04-27 14:11:32 +02:00
|
|
|
expected: [{ type: "other", description: "a" }],
|
2016-09-22 05:25:09 +02:00
|
|
|
found: "a",
|
2016-04-27 14:11:32 +02:00
|
|
|
location: {
|
|
|
|
start: { offset: 0, line: 1, column: 1 },
|
2016-09-22 05:25:09 +02:00
|
|
|
end: { offset: 1, line: 1, column: 2 }
|
2016-04-27 14:11:32 +02:00
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it("allows to set custom location info", function() {
|
2016-09-08 16:04:36 +02:00
|
|
|
let parser = peg.generate([
|
2016-10-04 11:00:25 +02:00
|
|
|
"start = 'a' {",
|
|
|
|
" expected('a', {",
|
|
|
|
" start: { offset: 1, line: 1, column: 2 },",
|
|
|
|
" end: { offset: 2, line: 1, column: 3 }",
|
|
|
|
" });",
|
|
|
|
"}"
|
|
|
|
].join("\n"), options);
|
2016-04-27 14:11:32 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("a", {
|
2016-09-22 05:25:09 +02:00
|
|
|
message: "Expected a but \"a\" found.",
|
2016-04-27 14:11:32 +02:00
|
|
|
expected: [{ type: "other", description: "a" }],
|
2016-09-22 05:25:09 +02:00
|
|
|
found: "a",
|
2016-04-27 14:11:32 +02:00
|
|
|
location: {
|
|
|
|
start: { offset: 1, line: 1, column: 2 },
|
2016-09-22 05:25:09 +02:00
|
|
|
end: { offset: 2, line: 1, column: 3 }
|
2016-04-27 14:11:32 +02:00
|
|
|
}
|
|
|
|
});
|
2015-01-26 09:10:52 +01:00
|
|
|
});
|
|
|
|
});
|
2012-06-25 21:46:47 +02:00
|
|
|
|
2016-04-27 14:11:32 +02:00
|
|
|
describe("|error|", function() {
|
|
|
|
it("terminates parsing and throws an exception", function() {
|
2016-09-08 16:04:36 +02:00
|
|
|
let parser = peg.generate(
|
2016-10-04 11:00:25 +02:00
|
|
|
"start = 'a' { error('a'); }",
|
|
|
|
options
|
|
|
|
);
|
2015-01-26 09:10:52 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("a", {
|
2016-09-22 05:25:09 +02:00
|
|
|
message: "a",
|
|
|
|
found: null,
|
2016-04-27 14:11:32 +02:00
|
|
|
expected: null,
|
|
|
|
location: {
|
|
|
|
start: { offset: 0, line: 1, column: 1 },
|
2016-09-22 05:25:09 +02:00
|
|
|
end: { offset: 1, line: 1, column: 2 }
|
2016-04-27 14:11:32 +02:00
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it("allows to set custom location info", function() {
|
2016-09-08 16:04:36 +02:00
|
|
|
let parser = peg.generate([
|
2016-10-04 11:00:25 +02:00
|
|
|
"start = 'a' {",
|
|
|
|
" error('a', {",
|
|
|
|
" start: { offset: 1, line: 1, column: 2 },",
|
|
|
|
" end: { offset: 2, line: 1, column: 3 }",
|
|
|
|
" });",
|
|
|
|
"}"
|
|
|
|
].join("\n"), options);
|
2016-04-27 14:11:32 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("a", {
|
2016-09-22 05:25:09 +02:00
|
|
|
message: "a",
|
2016-04-27 14:11:32 +02:00
|
|
|
expected: null,
|
2016-09-22 05:25:09 +02:00
|
|
|
found: null,
|
2016-04-27 14:11:32 +02:00
|
|
|
location: {
|
|
|
|
start: { offset: 1, line: 1, column: 2 },
|
2016-09-22 05:25:09 +02:00
|
|
|
end: { offset: 2, line: 1, column: 3 }
|
2016-04-27 14:11:32 +02:00
|
|
|
}
|
|
|
|
});
|
2015-01-26 09:10:52 +01:00
|
|
|
});
|
|
|
|
});
|
2013-08-28 19:33:21 +02:00
|
|
|
});
|
2012-06-25 21:46:47 +02:00
|
|
|
});
|
2015-01-12 14:48:29 +01:00
|
|
|
|
2015-01-26 09:10:52 +01:00
|
|
|
describe("when the expression doesn't match", function() {
|
|
|
|
it("reports match failure", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = 'a' { return 42; }", options);
|
2015-01-12 14:48:29 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("b");
|
2015-01-26 09:10:52 +01:00
|
|
|
});
|
2015-01-12 14:48:29 +01:00
|
|
|
|
2015-01-26 09:10:52 +01:00
|
|
|
it("doesn't execute the code", function() {
|
2016-09-08 16:04:36 +02:00
|
|
|
let parser = peg.generate(
|
2016-10-04 11:00:25 +02:00
|
|
|
"start = 'a' { throw 'Boom!'; } / 'b'",
|
|
|
|
options
|
|
|
|
);
|
2015-01-12 14:48:29 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("b");
|
2015-01-26 09:10:52 +01:00
|
|
|
});
|
2015-01-12 14:48:29 +01:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("choice", function() {
|
2015-01-24 20:04:49 +01:00
|
|
|
describe("when any expression matches", function() {
|
|
|
|
it("returns its match result", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = 'a' / 'b' / 'c'", options);
|
2015-01-12 14:48:29 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("a", "a");
|
|
|
|
expect(parser).to.parse("b", "b");
|
|
|
|
expect(parser).to.parse("c", "c");
|
2015-01-24 20:04:49 +01:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("when all expressions don't match", function() {
|
|
|
|
it("reports match failure", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = 'a' / 'b' / 'c'", options);
|
2015-01-24 20:04:49 +01:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("d");
|
2015-01-24 20:04:49 +01:00
|
|
|
});
|
2015-01-12 14:48:29 +01:00
|
|
|
});
|
2012-06-25 21:46:47 +02:00
|
|
|
});
|
|
|
|
|
2012-04-30 10:42:20 +02:00
|
|
|
describe("error reporting", function() {
|
2012-04-30 10:28:01 +02:00
|
|
|
describe("behavior", function() {
|
|
|
|
it("reports only the rightmost error", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = 'a' 'b' / 'a' 'c' 'd'", options);
|
2012-04-30 10:28:01 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("ace", {
|
2016-06-17 15:14:47 +02:00
|
|
|
expected: [{ type: "literal", text: "d", ignoreCase: false }]
|
2013-08-28 19:33:21 +02:00
|
|
|
});
|
2012-04-30 10:28:01 +02:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2013-08-28 19:33:21 +02:00
|
|
|
describe("expectations reporting", function() {
|
|
|
|
it("reports expectations correctly with no alternative", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = 'a'", options);
|
2012-04-30 10:28:01 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("ab", {
|
2016-06-17 15:14:47 +02:00
|
|
|
expected: [{ type: "end" }]
|
2013-12-08 14:48:30 +01:00
|
|
|
});
|
2012-04-30 10:28:01 +02:00
|
|
|
});
|
|
|
|
|
2013-08-28 19:33:21 +02:00
|
|
|
it("reports expectations correctly with one alternative", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = 'a'", options);
|
2012-04-30 10:28:01 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("b", {
|
2016-06-17 15:14:47 +02:00
|
|
|
expected: [{ type: "literal", text: "a", ignoreCase: false }]
|
2013-08-28 19:33:21 +02:00
|
|
|
});
|
2012-04-30 10:28:01 +02:00
|
|
|
});
|
|
|
|
|
2013-08-28 19:33:21 +02:00
|
|
|
it("reports expectations correctly with multiple alternatives", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = 'a' / 'b' / 'c'", options);
|
2012-04-30 10:28:01 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("d", {
|
2013-08-28 19:33:21 +02:00
|
|
|
expected: [
|
2016-06-17 15:14:47 +02:00
|
|
|
{ type: "literal", text: "a", ignoreCase: false },
|
|
|
|
{ type: "literal", text: "b", ignoreCase: false },
|
|
|
|
{ type: "literal", text: "c", ignoreCase: false }
|
2013-08-28 19:33:21 +02:00
|
|
|
]
|
2012-04-30 10:28:01 +02:00
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2016-06-10 15:18:25 +02:00
|
|
|
describe("found string reporting", function() {
|
|
|
|
it("reports found string correctly at the end of input", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = 'a'", options);
|
2016-06-10 15:18:25 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("", { found: null });
|
2016-06-10 15:18:25 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
it("reports found string correctly in the middle of input", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = 'a'", options);
|
2016-06-10 15:18:25 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("b", { found: "b" });
|
2016-06-10 15:18:25 +02:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2012-04-30 10:28:01 +02:00
|
|
|
describe("message building", function() {
|
|
|
|
it("builds message correctly with no alternative", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = 'a'", options);
|
2012-04-30 10:28:01 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("ab", {
|
2016-09-21 15:06:56 +02:00
|
|
|
message: "Expected end of input but \"b\" found."
|
2012-04-30 10:28:01 +02:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it("builds message correctly with one alternative", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = 'a'", options);
|
2012-04-30 10:28:01 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("b", {
|
2016-09-21 15:06:56 +02:00
|
|
|
message: "Expected \"a\" but \"b\" found."
|
2012-04-30 10:28:01 +02:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it("builds message correctly with multiple alternatives", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = 'a' / 'b' / 'c'", options);
|
2012-04-30 10:28:01 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("d", {
|
2016-09-21 15:06:56 +02:00
|
|
|
message: "Expected \"a\", \"b\", or \"c\" but \"d\" found."
|
2016-06-10 15:18:25 +02:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it("builds message correctly at the end of input", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = 'a'", options);
|
2016-06-10 15:18:25 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("", {
|
2016-09-21 15:06:56 +02:00
|
|
|
message: "Expected \"a\" but end of input found."
|
2016-06-10 15:18:25 +02:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it("builds message correctly in the middle of input", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = 'a'", options);
|
2016-06-10 15:18:25 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("b", {
|
2016-09-21 15:06:56 +02:00
|
|
|
message: "Expected \"a\" but \"b\" found."
|
2012-04-30 10:28:01 +02:00
|
|
|
});
|
|
|
|
});
|
2016-06-13 13:39:13 +02:00
|
|
|
|
|
|
|
it("removes duplicates from expectations", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = 'a' / 'a'", options);
|
2016-06-13 13:39:13 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("b", {
|
2016-09-21 15:06:56 +02:00
|
|
|
message: "Expected \"a\" but \"b\" found."
|
2016-06-13 13:39:13 +02:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it("sorts expectations", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = 'c' / 'b' / 'a'", options);
|
2016-06-13 13:39:13 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("d", {
|
2016-09-21 15:06:56 +02:00
|
|
|
message: "Expected \"a\", \"b\", or \"c\" but \"d\" found."
|
2016-06-13 13:39:13 +02:00
|
|
|
});
|
|
|
|
});
|
2012-04-30 10:28:01 +02:00
|
|
|
});
|
|
|
|
|
2012-04-22 19:45:34 +02:00
|
|
|
describe("position reporting", function() {
|
2016-06-10 15:18:25 +02:00
|
|
|
it("reports position correctly at the end of input", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = 'a'", options);
|
2015-04-03 15:59:46 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("", {
|
2015-04-03 15:59:46 +02:00
|
|
|
location: {
|
|
|
|
start: { offset: 0, line: 1, column: 1 },
|
2016-09-22 05:25:09 +02:00
|
|
|
end: { offset: 0, line: 1, column: 1 }
|
2015-04-03 15:59:46 +02:00
|
|
|
}
|
|
|
|
});
|
2012-04-22 19:45:34 +02:00
|
|
|
});
|
|
|
|
|
2016-06-10 15:18:25 +02:00
|
|
|
it("reports position correctly in the middle of input", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = 'a'", options);
|
2016-06-10 15:18:25 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("b", {
|
2016-06-10 15:18:25 +02:00
|
|
|
location: {
|
|
|
|
start: { offset: 0, line: 1, column: 1 },
|
2016-09-22 05:25:09 +02:00
|
|
|
end: { offset: 1, line: 1, column: 2 }
|
2016-06-10 15:18:25 +02:00
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2012-04-22 19:45:34 +02:00
|
|
|
it("reports position correctly with trailing input", function() {
|
2016-09-21 15:06:56 +02:00
|
|
|
let parser = peg.generate("start = 'a'", options);
|
2012-04-22 19:45:34 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("aa", {
|
2015-04-03 15:59:46 +02:00
|
|
|
location: {
|
|
|
|
start: { offset: 1, line: 1, column: 2 },
|
2016-09-22 05:25:09 +02:00
|
|
|
end: { offset: 2, line: 1, column: 3 }
|
2015-04-03 15:59:46 +02:00
|
|
|
}
|
|
|
|
});
|
2012-04-22 19:45:34 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
it("reports position correctly in complex cases", function() {
|
2016-09-08 16:04:36 +02:00
|
|
|
let parser = peg.generate([
|
2016-10-04 11:00:25 +02:00
|
|
|
"start = line (nl+ line)*",
|
|
|
|
"line = digit (' '+ digit)*",
|
|
|
|
"digit = [0-9]",
|
|
|
|
"nl = '\\r'? '\\n'"
|
|
|
|
].join("\n"), options);
|
2012-04-22 19:45:34 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("1\n2\n\n3\n\n\n4 5 x", {
|
2015-04-03 15:59:46 +02:00
|
|
|
location: {
|
|
|
|
start: { offset: 13, line: 7, column: 5 },
|
2016-09-22 05:25:09 +02:00
|
|
|
end: { offset: 14, line: 7, column: 6 }
|
2015-04-03 15:59:46 +02:00
|
|
|
}
|
2012-04-22 19:45:34 +02:00
|
|
|
});
|
|
|
|
|
2016-09-17 16:28:28 +02:00
|
|
|
// Newline representations
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("1\nx", { // Old Mac
|
2015-04-03 15:59:46 +02:00
|
|
|
location: {
|
|
|
|
start: { offset: 2, line: 2, column: 1 },
|
2016-09-22 05:25:09 +02:00
|
|
|
end: { offset: 3, line: 2, column: 2 }
|
2015-04-03 15:59:46 +02:00
|
|
|
}
|
2012-04-22 19:45:34 +02:00
|
|
|
});
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.failToParse("1\r\nx", { // Windows
|
2015-04-03 15:59:46 +02:00
|
|
|
location: {
|
|
|
|
start: { offset: 3, line: 2, column: 1 },
|
2016-09-22 05:25:09 +02:00
|
|
|
end: { offset: 4, line: 2, column: 2 }
|
2015-04-03 15:59:46 +02:00
|
|
|
}
|
2012-04-22 19:45:34 +02:00
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2016-09-17 16:28:28 +02:00
|
|
|
// Following examples are from Wikipedia, see
|
|
|
|
// http://en.wikipedia.org/w/index.php?title=Parsing_expression_grammar&oldid=335106938.
|
2012-04-30 10:42:20 +02:00
|
|
|
describe("complex examples", function() {
|
2012-04-22 17:16:26 +02:00
|
|
|
it("handles arithmetics example correctly", function() {
|
2016-09-17 16:28:28 +02:00
|
|
|
// Value ← [0-9]+ / '(' Expr ')'
|
|
|
|
// Product ← Value (('*' / '/') Value)*
|
|
|
|
// Sum ← Product (('+' / '-') Product)*
|
|
|
|
// Expr ← Sum
|
2016-09-08 16:04:36 +02:00
|
|
|
let parser = peg.generate([
|
2016-10-04 11:00:25 +02:00
|
|
|
"Expr = Sum",
|
|
|
|
"Sum = head:Product tail:(('+' / '-') Product)* {",
|
|
|
|
" return tail.reduce(function(result, element) {",
|
|
|
|
" if (element[0] === '+') { return result + element[1]; }",
|
|
|
|
" if (element[0] === '-') { return result - element[1]; }",
|
|
|
|
" }, head);",
|
|
|
|
" }",
|
|
|
|
"Product = head:Value tail:(('*' / '/') Value)* {",
|
|
|
|
" return tail.reduce(function(result, element) {",
|
|
|
|
" if (element[0] === '*') { return result * element[1]; }",
|
|
|
|
" if (element[0] === '/') { return result / element[1]; }",
|
|
|
|
" }, head);",
|
|
|
|
" }",
|
|
|
|
"Value = digits:[0-9]+ { return parseInt(digits.join(''), 10); }",
|
|
|
|
" / '(' expr:Expr ')' { return expr; }"
|
|
|
|
].join("\n"), options);
|
2012-04-22 17:16:26 +02:00
|
|
|
|
2016-09-17 16:28:28 +02:00
|
|
|
// The "value" rule
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("0", 0);
|
|
|
|
expect(parser).to.parse("123", 123);
|
|
|
|
expect(parser).to.parse("(42+43)", 42 + 43);
|
2012-04-22 17:16:26 +02:00
|
|
|
|
2016-09-17 16:28:28 +02:00
|
|
|
// The "product" rule
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("42", 42);
|
|
|
|
expect(parser).to.parse("42*43", 42 * 43);
|
|
|
|
expect(parser).to.parse("42*43*44*45", 42 * 43 * 44 * 45);
|
|
|
|
expect(parser).to.parse("42/43", 42 / 43);
|
|
|
|
expect(parser).to.parse("42/43/44/45", 42 / 43 / 44 / 45);
|
2012-04-22 17:16:26 +02:00
|
|
|
|
2016-09-17 16:28:28 +02:00
|
|
|
// The "sum" rule
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("42*43", 42 * 43);
|
|
|
|
expect(parser).to.parse("42*43+44*45", 42 * 43 + 44 * 45);
|
|
|
|
expect(parser).to.parse("42*43+44*45+46*47+48*49", 42 * 43 + 44 * 45 + 46 * 47 + 48 * 49);
|
|
|
|
expect(parser).to.parse("42*43-44*45", 42 * 43 - 44 * 45);
|
|
|
|
expect(parser).to.parse("42*43-44*45-46*47-48*49", 42 * 43 - 44 * 45 - 46 * 47 - 48 * 49);
|
2012-04-22 17:16:26 +02:00
|
|
|
|
2016-09-17 16:28:28 +02:00
|
|
|
// The "expr" rule
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("42+43", 42 + 43);
|
2012-04-22 17:16:26 +02:00
|
|
|
|
2016-09-17 16:28:28 +02:00
|
|
|
// Complex test
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("(1+2)*(3+4)", (1 + 2) * (3 + 4));
|
2012-04-22 17:16:26 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
it("handles non-context-free language correctly", function() {
|
2016-09-17 16:28:28 +02:00
|
|
|
// The following parsing expression grammar describes the classic
|
|
|
|
// non-context-free language { a^n b^n c^n : n >= 1 }:
|
|
|
|
//
|
|
|
|
// S ← &(A c) a+ B !(a/b/c)
|
|
|
|
// A ← a A? b
|
|
|
|
// B ← b B? c
|
2016-09-08 16:04:36 +02:00
|
|
|
let parser = peg.generate([
|
2016-10-04 11:00:25 +02:00
|
|
|
"S = &(A 'c') a:'a'+ B:B !('a' / 'b' / 'c') { return a.join('') + B; }",
|
|
|
|
"A = a:'a' A:A? b:'b' { return [a, A, b].join(''); }",
|
|
|
|
"B = b:'b' B:B? c:'c' { return [b, B, c].join(''); }"
|
|
|
|
].join("\n"), options);
|
2012-04-22 17:16:26 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("abc", "abc");
|
|
|
|
expect(parser).to.parse("aaabbbccc", "aaabbbccc");
|
|
|
|
expect(parser).to.failToParse("aabbbccc");
|
|
|
|
expect(parser).to.failToParse("aaaabbbccc");
|
|
|
|
expect(parser).to.failToParse("aaabbccc");
|
|
|
|
expect(parser).to.failToParse("aaabbbbccc");
|
|
|
|
expect(parser).to.failToParse("aaabbbcc");
|
|
|
|
expect(parser).to.failToParse("aaabbbcccc");
|
2012-04-22 17:16:26 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
it("handles nested comments example correctly", function() {
|
2016-09-17 16:28:28 +02:00
|
|
|
// Begin ← "(*"
|
|
|
|
// End ← "*)"
|
|
|
|
// C ← Begin N* End
|
|
|
|
// N ← C / (!Begin !End Z)
|
|
|
|
// Z ← any single character
|
2016-09-08 16:04:36 +02:00
|
|
|
let parser = peg.generate([
|
2016-10-04 11:00:25 +02:00
|
|
|
"C = begin:Begin ns:N* end:End { return begin + ns.join('') + end; }",
|
|
|
|
"N = C",
|
|
|
|
" / !Begin !End z:Z { return z; }",
|
|
|
|
"Z = .",
|
|
|
|
"Begin = '(*'",
|
|
|
|
"End = '*)'"
|
|
|
|
].join("\n"), options);
|
2012-04-22 17:16:26 +02:00
|
|
|
|
2016-12-08 08:59:04 +01:00
|
|
|
expect(parser).to.parse("(**)", "(**)");
|
|
|
|
expect(parser).to.parse("(*abc*)", "(*abc*)");
|
|
|
|
expect(parser).to.parse("(*(**)*)", "(*(**)*)");
|
|
|
|
expect(parser).to.parse(
|
2012-04-22 17:16:26 +02:00
|
|
|
"(*abc(*def*)ghi(*(*(*jkl*)*)*)mno*)",
|
|
|
|
"(*abc(*def*)ghi(*(*(*jkl*)*)*)mno*)"
|
|
|
|
);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
2012-04-22 10:52:41 +02:00
|
|
|
});
|