2016-05-04 12:37:13 +02:00
|
|
|
/* global peg */
|
2015-06-08 20:21:19 +02:00
|
|
|
|
|
|
|
"use strict";
|
|
|
|
|
2012-04-30 13:42:13 +02:00
|
|
|
describe("compiler pass |reportLeftRecursion|", function() {
|
2016-05-04 12:37:13 +02:00
|
|
|
var pass = peg.compiler.passes.check.reportLeftRecursion;
|
2012-04-30 13:42:13 +02:00
|
|
|
|
2014-06-07 09:11:00 +02:00
|
|
|
it("reports direct left recursion", function() {
|
|
|
|
expect(pass).toReportError('start = start', {
|
2015-09-11 15:06:57 +02:00
|
|
|
message: 'Possible left recursion detected (start -> start).',
|
2015-04-06 10:34:07 +02:00
|
|
|
location: {
|
|
|
|
start: { offset: 8, line: 1, column: 9 },
|
|
|
|
end: { offset: 13, line: 1, column: 14 }
|
|
|
|
}
|
2012-04-30 13:42:13 +02:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it("reports indirect left recursion", function() {
|
2014-06-07 09:11:00 +02:00
|
|
|
expect(pass).toReportError([
|
2012-04-30 13:42:13 +02:00
|
|
|
'start = stop',
|
|
|
|
'stop = start'
|
2014-06-07 09:11:00 +02:00
|
|
|
].join("\n"), {
|
2015-09-11 15:06:57 +02:00
|
|
|
message: 'Possible left recursion detected (start -> stop -> start).',
|
2015-04-06 10:34:07 +02:00
|
|
|
location: {
|
|
|
|
start: { offset: 21, line: 2, column: 9 },
|
|
|
|
end: { offset: 26, line: 2, column: 14 }
|
|
|
|
}
|
2014-06-07 09:11:00 +02:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("in sequences", function() {
|
2015-03-31 19:02:59 +02:00
|
|
|
it("reports left recursion if all preceding elements match empty string", function() {
|
|
|
|
expect(pass).toReportError('start = "" "" "" start');
|
|
|
|
});
|
|
|
|
|
|
|
|
it("doesn't report left recursion if some preceding element doesn't match empty string", function() {
|
|
|
|
expect(pass).not.toReportError('start = "a" "" "" start');
|
|
|
|
expect(pass).not.toReportError('start = "" "a" "" start');
|
|
|
|
expect(pass).not.toReportError('start = "" "" "a" start');
|
|
|
|
});
|
|
|
|
|
2015-08-17 10:58:37 +02:00
|
|
|
/* Regression test for #359. */
|
|
|
|
it("reports left recursion when rule reference is wrapped in an expression", function() {
|
|
|
|
expect(pass).toReportError('start = "" start?');
|
|
|
|
});
|
|
|
|
|
2015-09-04 17:35:37 +02:00
|
|
|
it("computes expressions that always consume input on success correctly", function() {
|
2015-08-03 16:56:57 +02:00
|
|
|
expect(pass).toReportError([
|
|
|
|
'start = a start',
|
|
|
|
'a "a" = ""'
|
|
|
|
].join('\n'));
|
|
|
|
expect(pass).not.toReportError([
|
|
|
|
'start = a start',
|
|
|
|
'a "a" = "a"'
|
|
|
|
].join('\n'));
|
|
|
|
|
2015-03-31 19:02:59 +02:00
|
|
|
expect(pass).toReportError('start = ("" / "a" / "b") start');
|
|
|
|
expect(pass).toReportError('start = ("a" / "" / "b") start');
|
|
|
|
expect(pass).toReportError('start = ("a" / "b" / "") start');
|
|
|
|
expect(pass).not.toReportError('start = ("a" / "b" / "c") start');
|
|
|
|
|
|
|
|
expect(pass).toReportError('start = ("" { }) start');
|
|
|
|
expect(pass).not.toReportError('start = ("a" { }) start');
|
|
|
|
|
|
|
|
expect(pass).toReportError('start = ("" "" "") start');
|
|
|
|
expect(pass).not.toReportError('start = ("a" "" "") start');
|
|
|
|
expect(pass).not.toReportError('start = ("" "a" "") start');
|
|
|
|
expect(pass).not.toReportError('start = ("" "" "a") start');
|
|
|
|
|
|
|
|
expect(pass).toReportError('start = a:"" start');
|
|
|
|
expect(pass).not.toReportError('start = a:"a" start');
|
|
|
|
|
|
|
|
expect(pass).toReportError('start = $"" start');
|
|
|
|
expect(pass).not.toReportError('start = $"a" start');
|
|
|
|
|
|
|
|
expect(pass).toReportError('start = &"" start');
|
|
|
|
expect(pass).toReportError('start = &"a" start');
|
|
|
|
|
|
|
|
expect(pass).toReportError('start = !"" start');
|
|
|
|
expect(pass).toReportError('start = !"a" start');
|
|
|
|
|
|
|
|
expect(pass).toReportError('start = ""? start');
|
|
|
|
expect(pass).toReportError('start = "a"? start');
|
|
|
|
|
|
|
|
expect(pass).toReportError('start = ""* start');
|
|
|
|
expect(pass).toReportError('start = "a"* start');
|
|
|
|
|
|
|
|
expect(pass).toReportError('start = ""+ start');
|
|
|
|
expect(pass).not.toReportError('start = "a"+ start');
|
|
|
|
|
2016-03-11 16:18:29 +01:00
|
|
|
expect(pass).toReportError('start = ("") start');
|
|
|
|
expect(pass).not.toReportError('start = ("a") start');
|
|
|
|
|
2015-03-31 19:02:59 +02:00
|
|
|
expect(pass).toReportError('start = &{ } start');
|
|
|
|
|
|
|
|
expect(pass).toReportError('start = !{ } start');
|
|
|
|
|
|
|
|
expect(pass).toReportError([
|
|
|
|
'start = a start',
|
|
|
|
'a = ""'
|
|
|
|
].join('\n'));
|
|
|
|
expect(pass).not.toReportError([
|
|
|
|
'start = a start',
|
|
|
|
'a = "a"'
|
|
|
|
].join('\n'));
|
|
|
|
|
|
|
|
expect(pass).toReportError('start = "" start');
|
|
|
|
expect(pass).not.toReportError('start = "a" start');
|
|
|
|
|
|
|
|
expect(pass).not.toReportError('start = [a-d] start');
|
2014-06-07 09:11:00 +02:00
|
|
|
|
2015-07-17 15:31:46 +02:00
|
|
|
expect(pass).not.toReportError('start = . start');
|
2014-06-07 09:11:00 +02:00
|
|
|
});
|
2012-04-30 13:42:13 +02:00
|
|
|
});
|
|
|
|
});
|