2016-05-04 12:37:13 +02:00
|
|
|
/* global peg */
|
2015-06-08 20:21:19 +02:00
|
|
|
|
|
|
|
"use strict";
|
|
|
|
|
2014-05-17 07:52:49 +02:00
|
|
|
describe("PEG.js API", function() {
|
2016-05-04 14:01:14 +02:00
|
|
|
describe("generate", function() {
|
2016-05-04 14:08:43 +02:00
|
|
|
it("generates a parser", function() {
|
2016-05-04 14:01:14 +02:00
|
|
|
var parser = peg.generate('start = "a"');
|
2014-05-17 07:52:49 +02:00
|
|
|
|
|
|
|
expect(typeof parser).toBe("object");
|
|
|
|
expect(parser.parse("a")).toBe("a");
|
|
|
|
});
|
|
|
|
|
|
|
|
it("throws an exception on syntax error", function() {
|
2016-05-04 14:01:14 +02:00
|
|
|
expect(function() { peg.generate('start = @'); }).toThrow();
|
2014-05-17 07:52:49 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
it("throws an exception on semantic error", function() {
|
2016-05-04 14:01:14 +02:00
|
|
|
expect(function() { peg.generate('start = missing'); }).toThrow();
|
2014-05-17 07:52:49 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
describe("allowed start rules", function() {
|
|
|
|
var grammar = [
|
|
|
|
'a = "x"',
|
|
|
|
'b = "x"',
|
|
|
|
'c = "x"'
|
|
|
|
].join("\n");
|
|
|
|
|
|
|
|
/*
|
|
|
|
* 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() {
|
2014-12-05 15:33:29 +01:00
|
|
|
it("generated parser can start only from the first rule", function() {
|
2016-05-04 14:01:14 +02:00
|
|
|
var parser = peg.generate(grammar, { optimize: "speed" });
|
2014-05-17 07:52:49 +02:00
|
|
|
|
|
|
|
expect(parser.parse("x", { startRule: "a" })).toBe("x");
|
|
|
|
expect(
|
|
|
|
function() { parser.parse("x", { startRule: "b" }); }
|
|
|
|
).toThrow();
|
|
|
|
expect(
|
|
|
|
function() { parser.parse("x", { startRule: "c" }); }
|
|
|
|
).toThrow();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("when |allowedStartRules| is set", function() {
|
2014-12-05 15:33:29 +01:00
|
|
|
it("generated parser can start only from specified rules", function() {
|
2016-05-04 14:01:14 +02:00
|
|
|
var parser = peg.generate(grammar, {
|
2014-05-17 07:52:49 +02:00
|
|
|
optimize: "speed",
|
|
|
|
allowedStartRules: ["b", "c"]
|
|
|
|
});
|
|
|
|
|
|
|
|
expect(
|
|
|
|
function() { parser.parse("x", { startRule: "a" }); }
|
|
|
|
).toThrow();
|
|
|
|
expect(parser.parse("x", { startRule: "b" })).toBe("x");
|
|
|
|
expect(parser.parse("x", { startRule: "c" })).toBe("x");
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("when optimizing for code size", function() {
|
|
|
|
describe("when |allowedStartRules| is not set", function() {
|
2014-12-05 15:33:29 +01:00
|
|
|
it("generated parser can start only from the first rule", function() {
|
2016-05-04 14:01:14 +02:00
|
|
|
var parser = peg.generate(grammar, { optimize: "size" });
|
2014-05-17 07:52:49 +02:00
|
|
|
|
|
|
|
expect(parser.parse("x", { startRule: "a" })).toBe("x");
|
|
|
|
expect(
|
|
|
|
function() { parser.parse("x", { startRule: "b" }); }
|
|
|
|
).toThrow();
|
|
|
|
expect(
|
|
|
|
function() { parser.parse("x", { startRule: "c" }); }
|
|
|
|
).toThrow();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("when |allowedStartRules| is set", function() {
|
2014-12-05 15:33:29 +01:00
|
|
|
it("generated parser can start only from specified rules", function() {
|
2016-05-04 14:01:14 +02:00
|
|
|
var parser = peg.generate(grammar, {
|
2014-05-17 07:52:49 +02:00
|
|
|
optimize: "size",
|
|
|
|
allowedStartRules: ["b", "c"]
|
|
|
|
});
|
|
|
|
|
|
|
|
expect(
|
|
|
|
function() { parser.parse("x", { startRule: "a" }); }
|
|
|
|
).toThrow();
|
|
|
|
expect(parser.parse("x", { startRule: "b" })).toBe("x");
|
|
|
|
expect(parser.parse("x", { startRule: "c" })).toBe("x");
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("intermediate results caching", function() {
|
|
|
|
var grammar = [
|
|
|
|
'{ var n = 0; }',
|
|
|
|
'start = (a "b") / (a "c") { return n; }',
|
|
|
|
'a = "a" { n++; }'
|
|
|
|
].join("\n");
|
|
|
|
|
|
|
|
describe("when |cache| is not set", function() {
|
2014-12-05 15:33:29 +01:00
|
|
|
it("generated parser doesn't cache intermediate parse results", function() {
|
2016-05-04 14:01:14 +02:00
|
|
|
var parser = peg.generate(grammar);
|
2014-05-17 07:52:49 +02:00
|
|
|
|
|
|
|
expect(parser.parse("ac")).toBe(2);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("when |cache| is set to |false|", function() {
|
2014-12-05 15:33:29 +01:00
|
|
|
it("generated parser doesn't cache intermediate parse results", function() {
|
2016-05-04 14:01:14 +02:00
|
|
|
var parser = peg.generate(grammar, { cache: false });
|
2014-05-17 07:52:49 +02:00
|
|
|
|
|
|
|
expect(parser.parse("ac")).toBe(2);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("when |cache| is set to |true|", function() {
|
2014-12-05 15:33:29 +01:00
|
|
|
it("generated parser caches intermediate parse results", function() {
|
2016-05-04 14:01:14 +02:00
|
|
|
var parser = peg.generate(grammar, { cache: true });
|
2014-05-17 07:52:49 +02:00
|
|
|
|
|
|
|
expect(parser.parse("ac")).toBe(1);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2015-02-24 05:12:24 +01:00
|
|
|
describe("tracing", function() {
|
|
|
|
var grammar = 'start = "a"';
|
|
|
|
|
|
|
|
describe("when |trace| is not set", function() {
|
|
|
|
it("generated parser doesn't trace", function() {
|
2016-05-04 14:01:14 +02:00
|
|
|
var parser = peg.generate(grammar),
|
2015-08-21 17:20:40 +02:00
|
|
|
tracer = jasmine.createSpyObj("tracer", ["trace"]);
|
2015-02-24 05:12:24 +01:00
|
|
|
|
2015-08-21 17:20:40 +02:00
|
|
|
parser.parse("a", { tracer: tracer });
|
2015-02-24 05:12:24 +01:00
|
|
|
|
2015-08-21 17:20:40 +02:00
|
|
|
expect(tracer.trace).not.toHaveBeenCalled();
|
2015-02-24 05:12:24 +01:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("when |trace| is set to |false|", function() {
|
|
|
|
it("generated parser doesn't trace", function() {
|
2016-05-04 14:01:14 +02:00
|
|
|
var parser = peg.generate(grammar, { trace: false }),
|
2015-08-21 17:20:40 +02:00
|
|
|
tracer = jasmine.createSpyObj("tracer", ["trace"]);
|
2015-02-24 05:12:24 +01:00
|
|
|
|
2015-08-21 17:20:40 +02:00
|
|
|
parser.parse("a", { tracer: tracer });
|
2015-02-24 05:12:24 +01:00
|
|
|
|
2015-08-21 17:20:40 +02:00
|
|
|
expect(tracer.trace).not.toHaveBeenCalled();
|
2015-02-24 05:12:24 +01:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("when |trace| is set to |true|", function() {
|
|
|
|
it("generated parser traces", function() {
|
2016-05-04 14:01:14 +02:00
|
|
|
var parser = peg.generate(grammar, { trace: true }),
|
2015-08-21 17:20:40 +02:00
|
|
|
tracer = jasmine.createSpyObj("tracer", ["trace"]);
|
2015-02-24 05:12:24 +01:00
|
|
|
|
2015-08-21 17:20:40 +02:00
|
|
|
parser.parse("a", { tracer: tracer });
|
2015-02-24 05:12:24 +01:00
|
|
|
|
2015-08-21 17:20:40 +02:00
|
|
|
expect(tracer.trace).toHaveBeenCalled();
|
2015-02-24 05:12:24 +01:00
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2014-05-17 07:52:49 +02:00
|
|
|
/*
|
|
|
|
* The |optimize| option isn't tested because there is no meaningful way to
|
|
|
|
* write the specs without turning this into a performance test.
|
|
|
|
*/
|
|
|
|
|
|
|
|
describe("output", function() {
|
|
|
|
var grammar = 'start = "a"';
|
|
|
|
|
|
|
|
describe("when |output| is not set", function() {
|
2014-12-05 15:33:29 +01:00
|
|
|
it("returns generated parser object", function() {
|
2016-05-04 14:01:14 +02:00
|
|
|
var parser = peg.generate(grammar);
|
2014-05-17 07:52:49 +02:00
|
|
|
|
|
|
|
expect(typeof parser).toBe("object");
|
|
|
|
expect(parser.parse("a")).toBe("a");
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("when |output| is set to |\"parser\"|", function() {
|
|
|
|
it("returns generated parser object", function() {
|
2016-05-04 14:01:14 +02:00
|
|
|
var parser = peg.generate(grammar, { output: "parser" });
|
2014-05-17 07:52:49 +02:00
|
|
|
|
|
|
|
expect(typeof parser).toBe("object");
|
|
|
|
expect(parser.parse("a")).toBe("a");
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("when |output| is set to |\"source\"|", function() {
|
|
|
|
it("returns generated parser source code", function() {
|
2016-05-04 14:01:14 +02:00
|
|
|
var source = peg.generate(grammar, { output: "source" });
|
2014-05-17 07:52:49 +02:00
|
|
|
|
|
|
|
expect(typeof source).toBe("string");
|
|
|
|
expect(eval(source).parse("a")).toBe("a");
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2016-05-02 16:05:09 +02:00
|
|
|
/*
|
2016-05-03 11:16:03 +02:00
|
|
|
* 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.
|
2016-05-02 16:05:09 +02:00
|
|
|
*/
|
|
|
|
|
2014-05-17 07:52:49 +02:00
|
|
|
/* The |plugins| option is tested in plugin API specs. */
|
|
|
|
|
|
|
|
it("accepts custom options", function() {
|
2016-05-04 14:01:14 +02:00
|
|
|
peg.generate('start = "a"', { foo: 42 });
|
2014-05-17 07:52:49 +02:00
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|