You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1862 lines
46 KiB
JavaScript
1862 lines
46 KiB
JavaScript
7 years ago
|
'use strict';
|
||
|
|
||
|
Object.defineProperty(exports, '__esModule', { value: true });
|
||
|
|
||
|
function _interopDefault (ex) { return (ex && (typeof ex === 'object') && 'default' in ex) ? ex['default'] : ex; }
|
||
|
|
||
|
var _Array$from = _interopDefault(require('babel-runtime/core-js/array/from'));
|
||
|
var _getIterator = _interopDefault(require('babel-runtime/core-js/get-iterator'));
|
||
|
var _get = _interopDefault(require('babel-runtime/helpers/get'));
|
||
|
var _Object$getPrototypeOf = _interopDefault(require('babel-runtime/core-js/object/get-prototype-of'));
|
||
|
var _possibleConstructorReturn = _interopDefault(require('babel-runtime/helpers/possibleConstructorReturn'));
|
||
|
var _inherits = _interopDefault(require('babel-runtime/helpers/inherits'));
|
||
|
var _Set = _interopDefault(require('babel-runtime/core-js/set'));
|
||
|
var _classCallCheck = _interopDefault(require('babel-runtime/helpers/classCallCheck'));
|
||
|
var _createClass = _interopDefault(require('babel-runtime/helpers/createClass'));
|
||
|
var _slicedToArray = _interopDefault(require('babel-runtime/helpers/slicedToArray'));
|
||
|
var _defineProperty = _interopDefault(require('babel-runtime/helpers/defineProperty'));
|
||
|
var _regeneratorRuntime = _interopDefault(require('babel-runtime/regenerator'));
|
||
|
var _Symbol$iterator = _interopDefault(require('babel-runtime/core-js/symbol/iterator'));
|
||
|
|
||
|
/**
|
||
|
* Returns a new set representing the union of a and b.
|
||
|
*/
|
||
|
function union(a, b) {
|
||
|
var s = new _Set(a);
|
||
|
addAll(s, b);
|
||
|
return s;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Adds all items from the set b to a.
|
||
|
*/
|
||
|
function addAll(a, b) {
|
||
|
var _iteratorNormalCompletion = true;
|
||
|
var _didIteratorError = false;
|
||
|
var _iteratorError = undefined;
|
||
|
|
||
|
try {
|
||
|
for (var _iterator = _getIterator(b), _step; !(_iteratorNormalCompletion = (_step = _iterator.next()).done); _iteratorNormalCompletion = true) {
|
||
|
var x = _step.value;
|
||
|
|
||
|
a.add(x);
|
||
|
}
|
||
|
} catch (err) {
|
||
|
_didIteratorError = true;
|
||
|
_iteratorError = err;
|
||
|
} finally {
|
||
|
try {
|
||
|
if (!_iteratorNormalCompletion && _iterator.return) {
|
||
|
_iterator.return();
|
||
|
}
|
||
|
} finally {
|
||
|
if (_didIteratorError) {
|
||
|
throw _iteratorError;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Returns whether two sets are equal
|
||
|
*/
|
||
|
function equal(a, b) {
|
||
|
if (a === b) return true;
|
||
|
|
||
|
if (a.size !== b.size) return false;
|
||
|
|
||
|
var _iteratorNormalCompletion2 = true;
|
||
|
var _didIteratorError2 = false;
|
||
|
var _iteratorError2 = undefined;
|
||
|
|
||
|
try {
|
||
|
for (var _iterator2 = _getIterator(a), _step2; !(_iteratorNormalCompletion2 = (_step2 = _iterator2.next()).done); _iteratorNormalCompletion2 = true) {
|
||
|
var x = _step2.value;
|
||
|
|
||
|
if (!b.has(x)) {
|
||
|
return false;
|
||
|
}
|
||
|
}
|
||
|
} catch (err) {
|
||
|
_didIteratorError2 = true;
|
||
|
_iteratorError2 = err;
|
||
|
} finally {
|
||
|
try {
|
||
|
if (!_iteratorNormalCompletion2 && _iterator2.return) {
|
||
|
_iterator2.return();
|
||
|
}
|
||
|
} finally {
|
||
|
if (_didIteratorError2) {
|
||
|
throw _iteratorError2;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Base AST node
|
||
|
*/
|
||
|
var Node = function () {
|
||
|
function Node() {
|
||
|
_classCallCheck(this, Node);
|
||
|
|
||
|
Object.defineProperty(this, 'followpos', { value: new _Set() });
|
||
|
}
|
||
|
|
||
|
_createClass(Node, [{
|
||
|
key: 'calcFollowpos',
|
||
|
value: function calcFollowpos() {
|
||
|
for (var key in this) {
|
||
|
if (this[key] instanceof Node) {
|
||
|
this[key].calcFollowpos();
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}]);
|
||
|
|
||
|
return Node;
|
||
|
}();
|
||
|
|
||
|
/**
|
||
|
* Represents a variable reference
|
||
|
*/
|
||
|
var Variable = function (_Node) {
|
||
|
_inherits(Variable, _Node);
|
||
|
|
||
|
function Variable(name) {
|
||
|
_classCallCheck(this, Variable);
|
||
|
|
||
|
var _this = _possibleConstructorReturn(this, (Variable.__proto__ || _Object$getPrototypeOf(Variable)).call(this));
|
||
|
|
||
|
_this.name = name;
|
||
|
return _this;
|
||
|
}
|
||
|
|
||
|
return Variable;
|
||
|
}(Node);
|
||
|
|
||
|
/**
|
||
|
* Represents a comment
|
||
|
*/
|
||
|
var Comment = function (_Node2) {
|
||
|
_inherits(Comment, _Node2);
|
||
|
|
||
|
function Comment(value) {
|
||
|
_classCallCheck(this, Comment);
|
||
|
|
||
|
var _this2 = _possibleConstructorReturn(this, (Comment.__proto__ || _Object$getPrototypeOf(Comment)).call(this));
|
||
|
|
||
|
_this2.value = value;
|
||
|
return _this2;
|
||
|
}
|
||
|
|
||
|
return Comment;
|
||
|
}(Node);
|
||
|
|
||
|
/**
|
||
|
* Represents an assignment statement.
|
||
|
* e.g. `variable = expression;`
|
||
|
*/
|
||
|
var Assignment = function (_Node3) {
|
||
|
_inherits(Assignment, _Node3);
|
||
|
|
||
|
function Assignment(variable, expression) {
|
||
|
_classCallCheck(this, Assignment);
|
||
|
|
||
|
var _this3 = _possibleConstructorReturn(this, (Assignment.__proto__ || _Object$getPrototypeOf(Assignment)).call(this));
|
||
|
|
||
|
_this3.variable = variable;
|
||
|
_this3.expression = expression;
|
||
|
return _this3;
|
||
|
}
|
||
|
|
||
|
return Assignment;
|
||
|
}(Node);
|
||
|
|
||
|
/**
|
||
|
* Represents an alternation.
|
||
|
* e.g. `a | b`
|
||
|
*/
|
||
|
var Alternation = function (_Node4) {
|
||
|
_inherits(Alternation, _Node4);
|
||
|
|
||
|
function Alternation(a, b) {
|
||
|
_classCallCheck(this, Alternation);
|
||
|
|
||
|
var _this4 = _possibleConstructorReturn(this, (Alternation.__proto__ || _Object$getPrototypeOf(Alternation)).call(this));
|
||
|
|
||
|
_this4.a = a;
|
||
|
_this4.b = b;
|
||
|
return _this4;
|
||
|
}
|
||
|
|
||
|
_createClass(Alternation, [{
|
||
|
key: 'copy',
|
||
|
value: function copy() {
|
||
|
return new Alternation(this.a.copy(), this.b.copy());
|
||
|
}
|
||
|
}, {
|
||
|
key: 'nullable',
|
||
|
get: function get() {
|
||
|
return this.a.nullable || this.b.nullable;
|
||
|
}
|
||
|
}, {
|
||
|
key: 'firstpos',
|
||
|
get: function get() {
|
||
|
return union(this.a.firstpos, this.b.firstpos);
|
||
|
}
|
||
|
}, {
|
||
|
key: 'lastpos',
|
||
|
get: function get() {
|
||
|
return union(this.a.lastpos, this.b.lastpos);
|
||
|
}
|
||
|
}]);
|
||
|
|
||
|
return Alternation;
|
||
|
}(Node);
|
||
|
|
||
|
/**
|
||
|
* Represents a concatenation, or chain.
|
||
|
* e.g. `a b c`
|
||
|
*/
|
||
|
var Concatenation = function (_Node5) {
|
||
|
_inherits(Concatenation, _Node5);
|
||
|
|
||
|
function Concatenation(a, b) {
|
||
|
_classCallCheck(this, Concatenation);
|
||
|
|
||
|
var _this5 = _possibleConstructorReturn(this, (Concatenation.__proto__ || _Object$getPrototypeOf(Concatenation)).call(this));
|
||
|
|
||
|
_this5.a = a;
|
||
|
_this5.b = b;
|
||
|
return _this5;
|
||
|
}
|
||
|
|
||
|
_createClass(Concatenation, [{
|
||
|
key: 'calcFollowpos',
|
||
|
value: function calcFollowpos() {
|
||
|
_get(Concatenation.prototype.__proto__ || _Object$getPrototypeOf(Concatenation.prototype), 'calcFollowpos', this).call(this);
|
||
|
var _iteratorNormalCompletion = true;
|
||
|
var _didIteratorError = false;
|
||
|
var _iteratorError = undefined;
|
||
|
|
||
|
try {
|
||
|
for (var _iterator = _getIterator(this.a.lastpos), _step; !(_iteratorNormalCompletion = (_step = _iterator.next()).done); _iteratorNormalCompletion = true) {
|
||
|
var n = _step.value;
|
||
|
|
||
|
addAll(n.followpos, this.b.firstpos);
|
||
|
}
|
||
|
} catch (err) {
|
||
|
_didIteratorError = true;
|
||
|
_iteratorError = err;
|
||
|
} finally {
|
||
|
try {
|
||
|
if (!_iteratorNormalCompletion && _iterator.return) {
|
||
|
_iterator.return();
|
||
|
}
|
||
|
} finally {
|
||
|
if (_didIteratorError) {
|
||
|
throw _iteratorError;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}, {
|
||
|
key: 'copy',
|
||
|
value: function copy() {
|
||
|
return new Concatenation(this.a.copy(), this.b.copy());
|
||
|
}
|
||
|
}, {
|
||
|
key: 'nullable',
|
||
|
get: function get() {
|
||
|
return this.a.nullable && this.b.nullable;
|
||
|
}
|
||
|
}, {
|
||
|
key: 'firstpos',
|
||
|
get: function get() {
|
||
|
var s = this.a.firstpos;
|
||
|
if (this.a.nullable) {
|
||
|
s = union(s, this.b.firstpos);
|
||
|
}
|
||
|
|
||
|
return s;
|
||
|
}
|
||
|
}, {
|
||
|
key: 'lastpos',
|
||
|
get: function get() {
|
||
|
var s = this.b.lastpos;
|
||
|
if (this.b.nullable) {
|
||
|
s = union(s, this.a.lastpos);
|
||
|
}
|
||
|
|
||
|
return s;
|
||
|
}
|
||
|
}]);
|
||
|
|
||
|
return Concatenation;
|
||
|
}(Node);
|
||
|
|
||
|
/**
|
||
|
* Represents a repetition.
|
||
|
* e.g. `a+`, `b*`, or `c?`
|
||
|
*/
|
||
|
var Repeat = function (_Node6) {
|
||
|
_inherits(Repeat, _Node6);
|
||
|
|
||
|
function Repeat(expression, op) {
|
||
|
_classCallCheck(this, Repeat);
|
||
|
|
||
|
var _this6 = _possibleConstructorReturn(this, (Repeat.__proto__ || _Object$getPrototypeOf(Repeat)).call(this));
|
||
|
|
||
|
_this6.expression = expression;
|
||
|
_this6.op = op;
|
||
|
return _this6;
|
||
|
}
|
||
|
|
||
|
_createClass(Repeat, [{
|
||
|
key: 'calcFollowpos',
|
||
|
value: function calcFollowpos() {
|
||
|
_get(Repeat.prototype.__proto__ || _Object$getPrototypeOf(Repeat.prototype), 'calcFollowpos', this).call(this);
|
||
|
if (this.op === '*' || this.op === '+') {
|
||
|
var _iteratorNormalCompletion2 = true;
|
||
|
var _didIteratorError2 = false;
|
||
|
var _iteratorError2 = undefined;
|
||
|
|
||
|
try {
|
||
|
for (var _iterator2 = _getIterator(this.lastpos), _step2; !(_iteratorNormalCompletion2 = (_step2 = _iterator2.next()).done); _iteratorNormalCompletion2 = true) {
|
||
|
var n = _step2.value;
|
||
|
|
||
|
addAll(n.followpos, this.firstpos);
|
||
|
}
|
||
|
} catch (err) {
|
||
|
_didIteratorError2 = true;
|
||
|
_iteratorError2 = err;
|
||
|
} finally {
|
||
|
try {
|
||
|
if (!_iteratorNormalCompletion2 && _iterator2.return) {
|
||
|
_iterator2.return();
|
||
|
}
|
||
|
} finally {
|
||
|
if (_didIteratorError2) {
|
||
|
throw _iteratorError2;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}, {
|
||
|
key: 'copy',
|
||
|
value: function copy() {
|
||
|
return new Repeat(this.expression.copy(), this.op);
|
||
|
}
|
||
|
}, {
|
||
|
key: 'nullable',
|
||
|
get: function get() {
|
||
|
return this.op === '*' || this.op === '?';
|
||
|
}
|
||
|
}, {
|
||
|
key: 'firstpos',
|
||
|
get: function get() {
|
||
|
return this.expression.firstpos;
|
||
|
}
|
||
|
}, {
|
||
|
key: 'lastpos',
|
||
|
get: function get() {
|
||
|
return this.expression.lastpos;
|
||
|
}
|
||
|
}]);
|
||
|
|
||
|
return Repeat;
|
||
|
}(Node);
|
||
|
|
||
|
/**
|
||
|
* Base class for leaf nodes
|
||
|
*/
|
||
|
|
||
|
var Leaf = function (_Node7) {
|
||
|
_inherits(Leaf, _Node7);
|
||
|
|
||
|
function Leaf() {
|
||
|
_classCallCheck(this, Leaf);
|
||
|
|
||
|
return _possibleConstructorReturn(this, (Leaf.__proto__ || _Object$getPrototypeOf(Leaf)).apply(this, arguments));
|
||
|
}
|
||
|
|
||
|
_createClass(Leaf, [{
|
||
|
key: 'nullable',
|
||
|
get: function get() {
|
||
|
return false;
|
||
|
}
|
||
|
}, {
|
||
|
key: 'firstpos',
|
||
|
get: function get() {
|
||
|
return new _Set([this]);
|
||
|
}
|
||
|
}, {
|
||
|
key: 'lastpos',
|
||
|
get: function get() {
|
||
|
return new _Set([this]);
|
||
|
}
|
||
|
}]);
|
||
|
|
||
|
return Leaf;
|
||
|
}(Node);
|
||
|
|
||
|
/**
|
||
|
* Represents a literal value, e.g. a number
|
||
|
*/
|
||
|
|
||
|
|
||
|
var Literal = function (_Leaf) {
|
||
|
_inherits(Literal, _Leaf);
|
||
|
|
||
|
function Literal(value) {
|
||
|
_classCallCheck(this, Literal);
|
||
|
|
||
|
var _this8 = _possibleConstructorReturn(this, (Literal.__proto__ || _Object$getPrototypeOf(Literal)).call(this));
|
||
|
|
||
|
_this8.value = value;
|
||
|
return _this8;
|
||
|
}
|
||
|
|
||
|
_createClass(Literal, [{
|
||
|
key: 'copy',
|
||
|
value: function copy() {
|
||
|
return new Literal(this.value);
|
||
|
}
|
||
|
}]);
|
||
|
|
||
|
return Literal;
|
||
|
}(Leaf);
|
||
|
|
||
|
/**
|
||
|
* Marks the end of an expression
|
||
|
*/
|
||
|
var EndMarker = function (_Leaf2) {
|
||
|
_inherits(EndMarker, _Leaf2);
|
||
|
|
||
|
function EndMarker() {
|
||
|
_classCallCheck(this, EndMarker);
|
||
|
|
||
|
return _possibleConstructorReturn(this, (EndMarker.__proto__ || _Object$getPrototypeOf(EndMarker)).apply(this, arguments));
|
||
|
}
|
||
|
|
||
|
return EndMarker;
|
||
|
}(Leaf);
|
||
|
|
||
|
/**
|
||
|
* Represents a tag
|
||
|
* e.g. `a:(a b)`
|
||
|
*/
|
||
|
var Tag = function (_Leaf3) {
|
||
|
_inherits(Tag, _Leaf3);
|
||
|
|
||
|
function Tag(name) {
|
||
|
_classCallCheck(this, Tag);
|
||
|
|
||
|
var _this10 = _possibleConstructorReturn(this, (Tag.__proto__ || _Object$getPrototypeOf(Tag)).call(this));
|
||
|
|
||
|
_this10.name = name;
|
||
|
return _this10;
|
||
|
}
|
||
|
|
||
|
_createClass(Tag, [{
|
||
|
key: 'copy',
|
||
|
value: function copy() {
|
||
|
return new Tag(this.name);
|
||
|
}
|
||
|
}, {
|
||
|
key: 'nullable',
|
||
|
get: function get() {
|
||
|
return true;
|
||
|
}
|
||
|
}]);
|
||
|
|
||
|
return Tag;
|
||
|
}(Leaf);
|
||
|
|
||
|
var nodes = Object.freeze({
|
||
|
Node: Node,
|
||
|
Variable: Variable,
|
||
|
Comment: Comment,
|
||
|
Assignment: Assignment,
|
||
|
Alternation: Alternation,
|
||
|
Concatenation: Concatenation,
|
||
|
Repeat: Repeat,
|
||
|
Literal: Literal,
|
||
|
EndMarker: EndMarker,
|
||
|
Tag: Tag
|
||
|
});
|
||
|
|
||
|
var require$$0 = ( nodes && nodes['default'] ) || nodes;
|
||
|
|
||
|
function peg$subclass(child, parent) {
|
||
|
function ctor() {
|
||
|
this.constructor = child;
|
||
|
}
|
||
|
ctor.prototype = parent.prototype;
|
||
|
child.prototype = new ctor();
|
||
|
}
|
||
|
|
||
|
function peg$SyntaxError(message, expected, found, location) {
|
||
|
this.message = message;
|
||
|
this.expected = expected;
|
||
|
this.found = found;
|
||
|
this.location = location;
|
||
|
this.name = "SyntaxError";
|
||
|
|
||
|
if (typeof Error.captureStackTrace === "function") {
|
||
|
Error.captureStackTrace(this, peg$SyntaxError);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
peg$subclass(peg$SyntaxError, Error);
|
||
|
|
||
|
peg$SyntaxError.buildMessage = function (expected, found) {
|
||
|
var DESCRIBE_EXPECTATION_FNS = {
|
||
|
literal: function literal(expectation) {
|
||
|
return "\"" + literalEscape(expectation.text) + "\"";
|
||
|
},
|
||
|
|
||
|
"class": function _class(expectation) {
|
||
|
var escapedParts = "",
|
||
|
i;
|
||
|
|
||
|
for (i = 0; i < expectation.parts.length; i++) {
|
||
|
escapedParts += expectation.parts[i] instanceof Array ? classEscape(expectation.parts[i][0]) + "-" + classEscape(expectation.parts[i][1]) : classEscape(expectation.parts[i]);
|
||
|
}
|
||
|
|
||
|
return "[" + (expectation.inverted ? "^" : "") + escapedParts + "]";
|
||
|
},
|
||
|
|
||
|
any: function any(expectation) {
|
||
|
return "any character";
|
||
|
},
|
||
|
|
||
|
end: function end(expectation) {
|
||
|
return "end of input";
|
||
|
},
|
||
|
|
||
|
other: function other(expectation) {
|
||
|
return expectation.description;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
function hex(ch) {
|
||
|
return ch.charCodeAt(0).toString(16).toUpperCase();
|
||
|
}
|
||
|
|
||
|
function literalEscape(s) {
|
||
|
return s.replace(/\\/g, '\\\\').replace(/"/g, '\\"').replace(/\0/g, '\\0').replace(/\t/g, '\\t').replace(/\n/g, '\\n').replace(/\r/g, '\\r').replace(/[\x00-\x0F]/g, function (ch) {
|
||
|
return '\\x0' + hex(ch);
|
||
|
}).replace(/[\x10-\x1F\x7F-\x9F]/g, function (ch) {
|
||
|
return '\\x' + hex(ch);
|
||
|
});
|
||
|
}
|
||
|
|
||
|
function classEscape(s) {
|
||
|
return s.replace(/\\/g, '\\\\').replace(/\]/g, '\\]').replace(/\^/g, '\\^').replace(/-/g, '\\-').replace(/\0/g, '\\0').replace(/\t/g, '\\t').replace(/\n/g, '\\n').replace(/\r/g, '\\r').replace(/[\x00-\x0F]/g, function (ch) {
|
||
|
return '\\x0' + hex(ch);
|
||
|
}).replace(/[\x10-\x1F\x7F-\x9F]/g, function (ch) {
|
||
|
return '\\x' + hex(ch);
|
||
|
});
|
||
|
}
|
||
|
|
||
|
function describeExpectation(expectation) {
|
||
|
return DESCRIBE_EXPECTATION_FNS[expectation.type](expectation);
|
||
|
}
|
||
|
|
||
|
function describeExpected(expected) {
|
||
|
var descriptions = new Array(expected.length),
|
||
|
i,
|
||
|
j;
|
||
|
|
||
|
for (i = 0; i < expected.length; i++) {
|
||
|
descriptions[i] = describeExpectation(expected[i]);
|
||
|
}
|
||
|
|
||
|
descriptions.sort();
|
||
|
|
||
|
if (descriptions.length > 0) {
|
||
|
for (i = 1, j = 1; i < descriptions.length; i++) {
|
||
|
if (descriptions[i - 1] !== descriptions[i]) {
|
||
|
descriptions[j] = descriptions[i];
|
||
|
j++;
|
||
|
}
|
||
|
}
|
||
|
descriptions.length = j;
|
||
|
}
|
||
|
|
||
|
switch (descriptions.length) {
|
||
|
case 1:
|
||
|
return descriptions[0];
|
||
|
|
||
|
case 2:
|
||
|
return descriptions[0] + " or " + descriptions[1];
|
||
|
|
||
|
default:
|
||
|
return descriptions.slice(0, -1).join(", ") + ", or " + descriptions[descriptions.length - 1];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
function describeFound(found) {
|
||
|
return found ? "\"" + literalEscape(found) + "\"" : "end of input";
|
||
|
}
|
||
|
|
||
|
return "Expected " + describeExpected(expected) + " but " + describeFound(found) + " found.";
|
||
|
};
|
||
|
|
||
|
function peg$parse(input, options) {
|
||
|
options = options !== void 0 ? options : {};
|
||
|
|
||
|
var peg$FAILED = {},
|
||
|
peg$startRuleFunctions = { rules: peg$parserules },
|
||
|
peg$startRuleFunction = peg$parserules,
|
||
|
peg$c0 = function peg$c0(s) {
|
||
|
return s;
|
||
|
},
|
||
|
peg$c1 = "#",
|
||
|
peg$c2 = peg$literalExpectation("#", false),
|
||
|
peg$c3 = /^[^\r\n]/,
|
||
|
peg$c4 = peg$classExpectation(["\r", "\n"], true, false),
|
||
|
peg$c5 = /^[\r\n]/,
|
||
|
peg$c6 = peg$classExpectation(["\r", "\n"], false, false),
|
||
|
peg$c7 = function peg$c7(v) {
|
||
|
return new n.Comment(v.join(''));
|
||
|
},
|
||
|
peg$c8 = "=",
|
||
|
peg$c9 = peg$literalExpectation("=", false),
|
||
|
peg$c10 = ";",
|
||
|
peg$c11 = peg$literalExpectation(";", false),
|
||
|
peg$c12 = function peg$c12(v, e) {
|
||
|
return new n.Assignment(v, e);
|
||
|
},
|
||
|
peg$c13 = function peg$c13(v) {
|
||
|
return new n.Variable(v);
|
||
|
},
|
||
|
peg$c14 = "|",
|
||
|
peg$c15 = peg$literalExpectation("|", false),
|
||
|
peg$c16 = function peg$c16(a, b) {
|
||
|
return new n.Alternation(a, b);
|
||
|
},
|
||
|
peg$c17 = function peg$c17(a, b) {
|
||
|
return new n.Concatenation(a, b);
|
||
|
},
|
||
|
peg$c18 = ":",
|
||
|
peg$c19 = peg$literalExpectation(":", false),
|
||
|
peg$c20 = function peg$c20(t, e) {
|
||
|
return new n.Concatenation(e, new n.Tag(t));
|
||
|
},
|
||
|
peg$c21 = "*",
|
||
|
peg$c22 = peg$literalExpectation("*", false),
|
||
|
peg$c23 = function peg$c23(t) {
|
||
|
return new n.Repeat(t, '*');
|
||
|
},
|
||
|
peg$c24 = "?",
|
||
|
peg$c25 = peg$literalExpectation("?", false),
|
||
|
peg$c26 = function peg$c26(t) {
|
||
|
return new n.Repeat(t, '?');
|
||
|
},
|
||
|
peg$c27 = "+",
|
||
|
peg$c28 = peg$literalExpectation("+", false),
|
||
|
peg$c29 = function peg$c29(t) {
|
||
|
return new n.Repeat(t, '+');
|
||
|
},
|
||
|
peg$c30 = function peg$c30(x) {
|
||
|
return new n.Literal(x);
|
||
|
},
|
||
|
peg$c31 = "(",
|
||
|
peg$c32 = peg$literalExpectation("(", false),
|
||
|
peg$c33 = ")",
|
||
|
peg$c34 = peg$literalExpectation(")", false),
|
||
|
peg$c35 = function peg$c35(e) {
|
||
|
return e;
|
||
|
},
|
||
|
peg$c36 = function peg$c36(a, b) {
|
||
|
return a + b.join('');
|
||
|
},
|
||
|
peg$c37 = "_",
|
||
|
peg$c38 = peg$literalExpectation("_", false),
|
||
|
peg$c39 = /^[a-zA-Z]/,
|
||
|
peg$c40 = peg$classExpectation([["a", "z"], ["A", "Z"]], false, false),
|
||
|
peg$c41 = /^[0-9]/,
|
||
|
peg$c42 = peg$classExpectation([["0", "9"]], false, false),
|
||
|
peg$c43 = function peg$c43(num) {
|
||
|
return parseInt(num.join(''));
|
||
|
},
|
||
|
peg$c44 = /^[ \t\r\n]/,
|
||
|
peg$c45 = peg$classExpectation([" ", "\t", "\r", "\n"], false, false),
|
||
|
peg$currPos = 0,
|
||
|
peg$savedPos = 0,
|
||
|
peg$posDetailsCache = [{ line: 1, column: 1 }],
|
||
|
peg$maxFailPos = 0,
|
||
|
peg$maxFailExpected = [],
|
||
|
peg$silentFails = 0,
|
||
|
peg$result;
|
||
|
|
||
|
if ("startRule" in options) {
|
||
|
if (!(options.startRule in peg$startRuleFunctions)) {
|
||
|
throw new Error("Can't start parsing from rule \"" + options.startRule + "\".");
|
||
|
}
|
||
|
|
||
|
peg$startRuleFunction = peg$startRuleFunctions[options.startRule];
|
||
|
}
|
||
|
|
||
|
function text() {
|
||
|
return input.substring(peg$savedPos, peg$currPos);
|
||
|
}
|
||
|
|
||
|
function location() {
|
||
|
return peg$computeLocation(peg$savedPos, peg$currPos);
|
||
|
}
|
||
|
|
||
|
function expected(description, location) {
|
||
|
location = location !== void 0 ? location : peg$computeLocation(peg$savedPos, peg$currPos);
|
||
|
|
||
|
throw peg$buildStructuredError([peg$otherExpectation(description)], input.substring(peg$savedPos, peg$currPos), location);
|
||
|
}
|
||
|
|
||
|
function error(message, location) {
|
||
|
location = location !== void 0 ? location : peg$computeLocation(peg$savedPos, peg$currPos);
|
||
|
|
||
|
throw peg$buildSimpleError(message, location);
|
||
|
}
|
||
|
|
||
|
function peg$literalExpectation(text, ignoreCase) {
|
||
|
return { type: "literal", text: text, ignoreCase: ignoreCase };
|
||
|
}
|
||
|
|
||
|
function peg$classExpectation(parts, inverted, ignoreCase) {
|
||
|
return { type: "class", parts: parts, inverted: inverted, ignoreCase: ignoreCase };
|
||
|
}
|
||
|
|
||
|
function peg$anyExpectation() {
|
||
|
return { type: "any" };
|
||
|
}
|
||
|
|
||
|
function peg$endExpectation() {
|
||
|
return { type: "end" };
|
||
|
}
|
||
|
|
||
|
function peg$otherExpectation(description) {
|
||
|
return { type: "other", description: description };
|
||
|
}
|
||
|
|
||
|
function peg$computePosDetails(pos) {
|
||
|
var details = peg$posDetailsCache[pos],
|
||
|
p;
|
||
|
|
||
|
if (details) {
|
||
|
return details;
|
||
|
} else {
|
||
|
p = pos - 1;
|
||
|
while (!peg$posDetailsCache[p]) {
|
||
|
p--;
|
||
|
}
|
||
|
|
||
|
details = peg$posDetailsCache[p];
|
||
|
details = {
|
||
|
line: details.line,
|
||
|
column: details.column
|
||
|
};
|
||
|
|
||
|
while (p < pos) {
|
||
|
if (input.charCodeAt(p) === 10) {
|
||
|
details.line++;
|
||
|
details.column = 1;
|
||
|
} else {
|
||
|
details.column++;
|
||
|
}
|
||
|
|
||
|
p++;
|
||
|
}
|
||
|
|
||
|
peg$posDetailsCache[pos] = details;
|
||
|
return details;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
function peg$computeLocation(startPos, endPos) {
|
||
|
var startPosDetails = peg$computePosDetails(startPos),
|
||
|
endPosDetails = peg$computePosDetails(endPos);
|
||
|
|
||
|
return {
|
||
|
start: {
|
||
|
offset: startPos,
|
||
|
line: startPosDetails.line,
|
||
|
column: startPosDetails.column
|
||
|
},
|
||
|
end: {
|
||
|
offset: endPos,
|
||
|
line: endPosDetails.line,
|
||
|
column: endPosDetails.column
|
||
|
}
|
||
|
};
|
||
|
}
|
||
|
|
||
|
function peg$fail(expected) {
|
||
|
if (peg$currPos < peg$maxFailPos) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (peg$currPos > peg$maxFailPos) {
|
||
|
peg$maxFailPos = peg$currPos;
|
||
|
peg$maxFailExpected = [];
|
||
|
}
|
||
|
|
||
|
peg$maxFailExpected.push(expected);
|
||
|
}
|
||
|
|
||
|
function peg$buildSimpleError(message, location) {
|
||
|
return new peg$SyntaxError(message, null, null, location);
|
||
|
}
|
||
|
|
||
|
function peg$buildStructuredError(expected, found, location) {
|
||
|
return new peg$SyntaxError(peg$SyntaxError.buildMessage(expected, found), expected, found, location);
|
||
|
}
|
||
|
|
||
|
function peg$parserules() {
|
||
|
var s0, s1;
|
||
|
|
||
|
s0 = [];
|
||
|
s1 = peg$parsestatement();
|
||
|
if (s1 !== peg$FAILED) {
|
||
|
while (s1 !== peg$FAILED) {
|
||
|
s0.push(s1);
|
||
|
s1 = peg$parsestatement();
|
||
|
}
|
||
|
} else {
|
||
|
s0 = peg$FAILED;
|
||
|
}
|
||
|
|
||
|
return s0;
|
||
|
}
|
||
|
|
||
|
function peg$parsestatement() {
|
||
|
var s0, s1, s2;
|
||
|
|
||
|
s0 = peg$currPos;
|
||
|
s1 = peg$parsestatement_type();
|
||
|
if (s1 !== peg$FAILED) {
|
||
|
s2 = peg$parse_();
|
||
|
if (s2 !== peg$FAILED) {
|
||
|
peg$savedPos = s0;
|
||
|
s1 = peg$c0(s1);
|
||
|
s0 = s1;
|
||
|
} else {
|
||
|
peg$currPos = s0;
|
||
|
s0 = peg$FAILED;
|
||
|
}
|
||
|
} else {
|
||
|
peg$currPos = s0;
|
||
|
s0 = peg$FAILED;
|
||
|
}
|
||
|
|
||
|
return s0;
|
||
|
}
|
||
|
|
||
|
function peg$parsestatement_type() {
|
||
|
var s0;
|
||
|
|
||
|
s0 = peg$parseassignment();
|
||
|
if (s0 === peg$FAILED) {
|
||
|
s0 = peg$parsecomment();
|
||
|
}
|
||
|
|
||
|
return s0;
|
||
|
}
|
||
|
|
||
|
function peg$parsecomment() {
|
||
|
var s0, s1, s2, s3;
|
||
|
|
||
|
s0 = peg$currPos;
|
||
|
if (input.charCodeAt(peg$currPos) === 35) {
|
||
|
s1 = peg$c1;
|
||
|
peg$currPos++;
|
||
|
} else {
|
||
|
s1 = peg$FAILED;
|
||
|
if (peg$silentFails === 0) {
|
||
|
peg$fail(peg$c2);
|
||
|
}
|
||
|
}
|
||
|
if (s1 !== peg$FAILED) {
|
||
|
s2 = [];
|
||
|
if (peg$c3.test(input.charAt(peg$currPos))) {
|
||
|
s3 = input.charAt(peg$currPos);
|
||
|
peg$currPos++;
|
||
|
} else {
|
||
|
s3 = peg$FAILED;
|
||
|
if (peg$silentFails === 0) {
|
||
|
peg$fail(peg$c4);
|
||
|
}
|
||
|
}
|
||
|
while (s3 !== peg$FAILED) {
|
||
|
s2.push(s3);
|
||
|
if (peg$c3.test(input.charAt(peg$currPos))) {
|
||
|
s3 = input.charAt(peg$currPos);
|
||
|
peg$currPos++;
|
||
|
} else {
|
||
|
s3 = peg$FAILED;
|
||
|
if (peg$silentFails === 0) {
|
||
|
peg$fail(peg$c4);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
if (s2 !== peg$FAILED) {
|
||
|
if (peg$c5.test(input.charAt(peg$currPos))) {
|
||
|
s3 = input.charAt(peg$currPos);
|
||
|
peg$currPos++;
|
||
|
} else {
|
||
|
s3 = peg$FAILED;
|
||
|
if (peg$silentFails === 0) {
|
||
|
peg$fail(peg$c6);
|
||
|
}
|
||
|
}
|
||
|
if (s3 !== peg$FAILED) {
|
||
|
peg$savedPos = s0;
|
||
|
s1 = peg$c7(s2);
|
||
|
s0 = s1;
|
||
|
} else {
|
||
|
peg$currPos = s0;
|
||
|
s0 = peg$FAILED;
|
||
|
}
|
||
|
} else {
|
||
|
peg$currPos = s0;
|
||
|
s0 = peg$FAILED;
|
||
|
}
|
||
|
} else {
|
||
|
peg$currPos = s0;
|
||
|
s0 = peg$FAILED;
|
||
|
}
|
||
|
|
||
|
return s0;
|
||
|
}
|
||
|
|
||
|
function peg$parseassignment() {
|
||
|
var s0, s1, s2, s3, s4, s5, s6, s7;
|
||
|
|
||
|
s0 = peg$currPos;
|
||
|
s1 = peg$parsevariable();
|
||
|
if (s1 !== peg$FAILED) {
|
||
|
s2 = peg$parse_();
|
||
|
if (s2 !== peg$FAILED) {
|
||
|
if (input.charCodeAt(peg$currPos) === 61) {
|
||
|
s3 = peg$c8;
|
||
|
peg$currPos++;
|
||
|
} else {
|
||
|
s3 = peg$FAILED;
|
||
|
if (peg$silentFails === 0) {
|
||
|
peg$fail(peg$c9);
|
||
|
}
|
||
|
}
|
||
|
if (s3 !== peg$FAILED) {
|
||
|
s4 = peg$parse_();
|
||
|
if (s4 !== peg$FAILED) {
|
||
|
s5 = peg$parseexpr();
|
||
|
if (s5 !== peg$FAILED) {
|
||
|
s6 = peg$parse_();
|
||
|
if (s6 !== peg$FAILED) {
|
||
|
if (input.charCodeAt(peg$currPos) === 59) {
|
||
|
s7 = peg$c10;
|
||
|
peg$currPos++;
|
||
|
} else {
|
||
|
s7 = peg$FAILED;
|
||
|
if (peg$silentFails === 0) {
|
||
|
peg$fail(peg$c11);
|
||
|
}
|
||
|
}
|
||
|
if (s7 !== peg$FAILED) {
|
||
|
peg$savedPos = s0;
|
||
|
s1 = peg$c12(s1, s5);
|
||
|
s0 = s1;
|
||
|
} else {
|
||
|
peg$currPos = s0;
|
||
|
s0 = peg$FAILED;
|
||
|
}
|
||
|
} else {
|
||
|
peg$currPos = s0;
|
||
|
s0 = peg$FAILED;
|
||
|
}
|
||
|
} else {
|
||
|
peg$currPos = s0;
|
||
|
s0 = peg$FAILED;
|
||
|
}
|
||
|
} else {
|
||
|
peg$currPos = s0;
|
||
|
s0 = peg$FAILED;
|
||
|
}
|
||
|
} else {
|
||
|
peg$currPos = s0;
|
||
|
s0 = peg$FAILED;
|
||
|
}
|
||
|
} else {
|
||
|
peg$currPos = s0;
|
||
|
s0 = peg$FAILED;
|
||
|
}
|
||
|
} else {
|
||
|
peg$currPos = s0;
|
||
|
s0 = peg$FAILED;
|
||
|
}
|
||
|
|
||
|
return s0;
|
||
|
}
|
||
|
|
||
|
function peg$parsevariable() {
|
||
|
var s0, s1;
|
||
|
|
||
|
s0 = peg$currPos;
|
||
|
s1 = peg$parsename();
|
||
|
if (s1 !== peg$FAILED) {
|
||
|
peg$savedPos = s0;
|
||
|
s1 = peg$c13(s1);
|
||
|
}
|
||
|
s0 = s1;
|
||
|
|
||
|
return s0;
|
||
|
}
|
||
|
|
||
|
function peg$parseexpr() {
|
||
|
var s0, s1, s2, s3, s4, s5;
|
||
|
|
||
|
s0 = peg$currPos;
|
||
|
s1 = peg$parseexpr_q();
|
||
|
if (s1 !== peg$FAILED) {
|
||
|
s2 = peg$parse_();
|
||
|
if (s2 !== peg$FAILED) {
|
||
|
if (input.charCodeAt(peg$currPos) === 124) {
|
||
|
s3 = peg$c14;
|
||
|
peg$currPos++;
|
||
|
} else {
|
||
|
s3 = peg$FAILED;
|
||
|
if (peg$silentFails === 0) {
|
||
|
peg$fail(peg$c15);
|
||
|
}
|
||
|
}
|
||
|
if (s3 !== peg$FAILED) {
|
||
|
s4 = peg$parse_();
|
||
|
if (s4 !== peg$FAILED) {
|
||
|
s5 = peg$parseexpr();
|
||
|
if (s5 !== peg$FAILED) {
|
||
|
peg$savedPos = s0;
|
||
|
s1 = peg$c16(s1, s5);
|
||
|
s0 = s1;
|
||
|
} else {
|
||
|
peg$currPos = s0;
|
||
|
s0 = peg$FAILED;
|
||
|
}
|
||
|
} else {
|
||
|
peg$currPos = s0;
|
||
|
s0 = peg$FAILED;
|
||
|
}
|
||
|
} else {
|
||
|
peg$currPos = s0;
|
||
|
s0 = peg$FAILED;
|
||
|
}
|
||
|
} else {
|
||
|
peg$currPos = s0;
|
||
|
s0 = peg$FAILED;
|
||
|
}
|
||
|
} else {
|
||
|
peg$currPos = s0;
|
||
|
s0 = peg$FAILED;
|
||
|
}
|
||
|
if (s0 === peg$FAILED) {
|
||
|
s0 = peg$currPos;
|
||
|
s1 = peg$parseexpr_q();
|
||
|
if (s1 !== peg$FAILED) {
|
||
|
s2 = peg$parse_();
|
||
|
if (s2 !== peg$FAILED) {
|
||
|
s3 = peg$parseexpr();
|
||
|
if (s3 !== peg$FAILED) {
|
||
|
peg$savedPos = s0;
|
||
|
s1 = peg$c17(s1, s3);
|
||
|
s0 = s1;
|
||
|
} else {
|
||
|
peg$currPos = s0;
|
||
|
s0 = peg$FAILED;
|
||
|
}
|
||
|
} else {
|
||
|
peg$currPos = s0;
|
||
|
s0 = peg$FAILED;
|
||
|
}
|
||
|
} else {
|
||
|
peg$currPos = s0;
|
||
|
s0 = peg$FAILED;
|
||
|
}
|
||
|
if (s0 === peg$FAILED) {
|
||
|
s0 = peg$parseexpr_q();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return s0;
|
||
|
}
|
||
|
|
||
|
function peg$parseexpr_q() {
|
||
|
var s0, s1, s2, s3;
|
||
|
|
||
|
s0 = peg$currPos;
|
||
|
s1 = peg$parsename();
|
||
|
if (s1 !== peg$FAILED) {
|
||
|
if (input.charCodeAt(peg$currPos) === 58) {
|
||
|
s2 = peg$c18;
|
||
|
peg$currPos++;
|
||
|
} else {
|
||
|
s2 = peg$FAILED;
|
||
|
if (peg$silentFails === 0) {
|
||
|
peg$fail(peg$c19);
|
||
|
}
|
||
|
}
|
||
|
if (s2 !== peg$FAILED) {
|
||
|
s3 = peg$parseexpr_q();
|
||
|
if (s3 !== peg$FAILED) {
|
||
|
peg$savedPos = s0;
|
||
|
s1 = peg$c20(s1, s3);
|
||
|
s0 = s1;
|
||
|
} else {
|
||
|
peg$currPos = s0;
|
||
|
s0 = peg$FAILED;
|
||
|
}
|
||
|
} else {
|
||
|
peg$currPos = s0;
|
||
|
s0 = peg$FAILED;
|
||
|
}
|
||
|
} else {
|
||
|
peg$currPos = s0;
|
||
|
s0 = peg$FAILED;
|
||
|
}
|
||
|
if (s0 === peg$FAILED) {
|
||
|
s0 = peg$currPos;
|
||
|
s1 = peg$parseterm();
|
||
|
if (s1 !== peg$FAILED) {
|
||
|
if (input.charCodeAt(peg$currPos) === 42) {
|
||
|
s2 = peg$c21;
|
||
|
peg$currPos++;
|
||
|
} else {
|
||
|
s2 = peg$FAILED;
|
||
|
if (peg$silentFails === 0) {
|
||
|
peg$fail(peg$c22);
|
||
|
}
|
||
|
}
|
||
|
if (s2 !== peg$FAILED) {
|
||
|
peg$savedPos = s0;
|
||
|
s1 = peg$c23(s1);
|
||
|
s0 = s1;
|
||
|
} else {
|
||
|
peg$currPos = s0;
|
||
|
s0 = peg$FAILED;
|
||
|
}
|
||
|
} else {
|
||
|
peg$currPos = s0;
|
||
|
s0 = peg$FAILED;
|
||
|
}
|
||
|
if (s0 === peg$FAILED) {
|
||
|
s0 = peg$currPos;
|
||
|
s1 = peg$parseterm();
|
||
|
if (s1 !== peg$FAILED) {
|
||
|
if (input.charCodeAt(peg$currPos) === 63) {
|
||
|
s2 = peg$c24;
|
||
|
peg$currPos++;
|
||
|
} else {
|
||
|
s2 = peg$FAILED;
|
||
|
if (peg$silentFails === 0) {
|
||
|
peg$fail(peg$c25);
|
||
|
}
|
||
|
}
|
||
|
if (s2 !== peg$FAILED) {
|
||
|
peg$savedPos = s0;
|
||
|
s1 = peg$c26(s1);
|
||
|
s0 = s1;
|
||
|
} else {
|
||
|
peg$currPos = s0;
|
||
|
s0 = peg$FAILED;
|
||
|
}
|
||
|
} else {
|
||
|
peg$currPos = s0;
|
||
|
s0 = peg$FAILED;
|
||
|
}
|
||
|
if (s0 === peg$FAILED) {
|
||
|
s0 = peg$currPos;
|
||
|
s1 = peg$parseterm();
|
||
|
if (s1 !== peg$FAILED) {
|
||
|
if (input.charCodeAt(peg$currPos) === 43) {
|
||
|
s2 = peg$c27;
|
||
|
peg$currPos++;
|
||
|
} else {
|
||
|
s2 = peg$FAILED;
|
||
|
if (peg$silentFails === 0) {
|
||
|
peg$fail(peg$c28);
|
||
|
}
|
||
|
}
|
||
|
if (s2 !== peg$FAILED) {
|
||
|
peg$savedPos = s0;
|
||
|
s1 = peg$c29(s1);
|
||
|
s0 = s1;
|
||
|
} else {
|
||
|
peg$currPos = s0;
|
||
|
s0 = peg$FAILED;
|
||
|
}
|
||
|
} else {
|
||
|
peg$currPos = s0;
|
||
|
s0 = peg$FAILED;
|
||
|
}
|
||
|
if (s0 === peg$FAILED) {
|
||
|
s0 = peg$parseterm();
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return s0;
|
||
|
}
|
||
|
|
||
|
function peg$parseterm() {
|
||
|
var s0, s1, s2, s3;
|
||
|
|
||
|
s0 = peg$parsevariable();
|
||
|
if (s0 === peg$FAILED) {
|
||
|
s0 = peg$currPos;
|
||
|
s1 = peg$parsenumber();
|
||
|
if (s1 !== peg$FAILED) {
|
||
|
peg$savedPos = s0;
|
||
|
s1 = peg$c30(s1);
|
||
|
}
|
||
|
s0 = s1;
|
||
|
if (s0 === peg$FAILED) {
|
||
|
s0 = peg$currPos;
|
||
|
if (input.charCodeAt(peg$currPos) === 40) {
|
||
|
s1 = peg$c31;
|
||
|
peg$currPos++;
|
||
|
} else {
|
||
|
s1 = peg$FAILED;
|
||
|
if (peg$silentFails === 0) {
|
||
|
peg$fail(peg$c32);
|
||
|
}
|
||
|
}
|
||
|
if (s1 !== peg$FAILED) {
|
||
|
s2 = peg$parseexpr();
|
||
|
if (s2 !== peg$FAILED) {
|
||
|
if (input.charCodeAt(peg$currPos) === 41) {
|
||
|
s3 = peg$c33;
|
||
|
peg$currPos++;
|
||
|
} else {
|
||
|
s3 = peg$FAILED;
|
||
|
if (peg$silentFails === 0) {
|
||
|
peg$fail(peg$c34);
|
||
|
}
|
||
|
}
|
||
|
if (s3 !== peg$FAILED) {
|
||
|
peg$savedPos = s0;
|
||
|
s1 = peg$c35(s2);
|
||
|
s0 = s1;
|
||
|
} else {
|
||
|
peg$currPos = s0;
|
||
|
s0 = peg$FAILED;
|
||
|
}
|
||
|
} else {
|
||
|
peg$currPos = s0;
|
||
|
s0 = peg$FAILED;
|
||
|
}
|
||
|
} else {
|
||
|
peg$currPos = s0;
|
||
|
s0 = peg$FAILED;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return s0;
|
||
|
}
|
||
|
|
||
|
function peg$parsename() {
|
||
|
var s0, s1, s2, s3;
|
||
|
|
||
|
s0 = peg$currPos;
|
||
|
s1 = peg$parsename_start_char();
|
||
|
if (s1 !== peg$FAILED) {
|
||
|
s2 = [];
|
||
|
s3 = peg$parsename_char();
|
||
|
while (s3 !== peg$FAILED) {
|
||
|
s2.push(s3);
|
||
|
s3 = peg$parsename_char();
|
||
|
}
|
||
|
if (s2 !== peg$FAILED) {
|
||
|
peg$savedPos = s0;
|
||
|
s1 = peg$c36(s1, s2);
|
||
|
s0 = s1;
|
||
|
} else {
|
||
|
peg$currPos = s0;
|
||
|
s0 = peg$FAILED;
|
||
|
}
|
||
|
} else {
|
||
|
peg$currPos = s0;
|
||
|
s0 = peg$FAILED;
|
||
|
}
|
||
|
|
||
|
return s0;
|
||
|
}
|
||
|
|
||
|
function peg$parsename_start_char() {
|
||
|
var s0;
|
||
|
|
||
|
if (input.charCodeAt(peg$currPos) === 95) {
|
||
|
s0 = peg$c37;
|
||
|
peg$currPos++;
|
||
|
} else {
|
||
|
s0 = peg$FAILED;
|
||
|
if (peg$silentFails === 0) {
|
||
|
peg$fail(peg$c38);
|
||
|
}
|
||
|
}
|
||
|
if (s0 === peg$FAILED) {
|
||
|
if (peg$c39.test(input.charAt(peg$currPos))) {
|
||
|
s0 = input.charAt(peg$currPos);
|
||
|
peg$currPos++;
|
||
|
} else {
|
||
|
s0 = peg$FAILED;
|
||
|
if (peg$silentFails === 0) {
|
||
|
peg$fail(peg$c40);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return s0;
|
||
|
}
|
||
|
|
||
|
function peg$parsename_char() {
|
||
|
var s0;
|
||
|
|
||
|
s0 = peg$parsename_start_char();
|
||
|
if (s0 === peg$FAILED) {
|
||
|
if (peg$c41.test(input.charAt(peg$currPos))) {
|
||
|
s0 = input.charAt(peg$currPos);
|
||
|
peg$currPos++;
|
||
|
} else {
|
||
|
s0 = peg$FAILED;
|
||
|
if (peg$silentFails === 0) {
|
||
|
peg$fail(peg$c42);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return s0;
|
||
|
}
|
||
|
|
||
|
function peg$parsenumber() {
|
||
|
var s0, s1, s2;
|
||
|
|
||
|
s0 = peg$currPos;
|
||
|
s1 = [];
|
||
|
if (peg$c41.test(input.charAt(peg$currPos))) {
|
||
|
s2 = input.charAt(peg$currPos);
|
||
|
peg$currPos++;
|
||
|
} else {
|
||
|
s2 = peg$FAILED;
|
||
|
if (peg$silentFails === 0) {
|
||
|
peg$fail(peg$c42);
|
||
|
}
|
||
|
}
|
||
|
if (s2 !== peg$FAILED) {
|
||
|
while (s2 !== peg$FAILED) {
|
||
|
s1.push(s2);
|
||
|
if (peg$c41.test(input.charAt(peg$currPos))) {
|
||
|
s2 = input.charAt(peg$currPos);
|
||
|
peg$currPos++;
|
||
|
} else {
|
||
|
s2 = peg$FAILED;
|
||
|
if (peg$silentFails === 0) {
|
||
|
peg$fail(peg$c42);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
} else {
|
||
|
s1 = peg$FAILED;
|
||
|
}
|
||
|
if (s1 !== peg$FAILED) {
|
||
|
peg$savedPos = s0;
|
||
|
s1 = peg$c43(s1);
|
||
|
}
|
||
|
s0 = s1;
|
||
|
|
||
|
return s0;
|
||
|
}
|
||
|
|
||
|
function peg$parse_() {
|
||
|
var s0, s1;
|
||
|
|
||
|
s0 = [];
|
||
|
if (peg$c44.test(input.charAt(peg$currPos))) {
|
||
|
s1 = input.charAt(peg$currPos);
|
||
|
peg$currPos++;
|
||
|
} else {
|
||
|
s1 = peg$FAILED;
|
||
|
if (peg$silentFails === 0) {
|
||
|
peg$fail(peg$c45);
|
||
|
}
|
||
|
}
|
||
|
while (s1 !== peg$FAILED) {
|
||
|
s0.push(s1);
|
||
|
if (peg$c44.test(input.charAt(peg$currPos))) {
|
||
|
s1 = input.charAt(peg$currPos);
|
||
|
peg$currPos++;
|
||
|
} else {
|
||
|
s1 = peg$FAILED;
|
||
|
if (peg$silentFails === 0) {
|
||
|
peg$fail(peg$c45);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return s0;
|
||
|
}
|
||
|
|
||
|
var n = require$$0;
|
||
|
|
||
|
peg$result = peg$startRuleFunction();
|
||
|
|
||
|
if (peg$result !== peg$FAILED && peg$currPos === input.length) {
|
||
|
return peg$result;
|
||
|
} else {
|
||
|
if (peg$result !== peg$FAILED && peg$currPos < input.length) {
|
||
|
peg$fail(peg$endExpectation());
|
||
|
}
|
||
|
|
||
|
throw peg$buildStructuredError(peg$maxFailExpected, peg$maxFailPos < input.length ? input.charAt(peg$maxFailPos) : null, peg$maxFailPos < input.length ? peg$computeLocation(peg$maxFailPos, peg$maxFailPos + 1) : peg$computeLocation(peg$maxFailPos, peg$maxFailPos));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
var grammar = {
|
||
|
SyntaxError: peg$SyntaxError,
|
||
|
parse: peg$parse
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Processes a list of statements into a symbol table
|
||
|
*/
|
||
|
|
||
|
var SymbolTable = function () {
|
||
|
function SymbolTable(statements) {
|
||
|
var externalSymbols = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {};
|
||
|
|
||
|
_classCallCheck(this, SymbolTable);
|
||
|
|
||
|
this.variables = {};
|
||
|
this.symbols = {};
|
||
|
this.main = null;
|
||
|
this.size = 0;
|
||
|
|
||
|
this.addExternalSymbols(externalSymbols);
|
||
|
this.process(statements);
|
||
|
}
|
||
|
|
||
|
_createClass(SymbolTable, [{
|
||
|
key: 'addExternalSymbols',
|
||
|
value: function addExternalSymbols(externalSymbols) {
|
||
|
for (var key in externalSymbols) {
|
||
|
this.variables[key] = new Literal(externalSymbols[key]);
|
||
|
this.symbols[key] = externalSymbols[key];
|
||
|
this.size++;
|
||
|
}
|
||
|
}
|
||
|
}, {
|
||
|
key: 'process',
|
||
|
value: function process(statements) {
|
||
|
var _iteratorNormalCompletion = true;
|
||
|
var _didIteratorError = false;
|
||
|
var _iteratorError = undefined;
|
||
|
|
||
|
try {
|
||
|
for (var _iterator = _getIterator(statements), _step; !(_iteratorNormalCompletion = (_step = _iterator.next()).done); _iteratorNormalCompletion = true) {
|
||
|
var statement = _step.value;
|
||
|
|
||
|
if (statement instanceof Assignment) {
|
||
|
this.variables[statement.variable.name] = this.processExpression(statement.expression);
|
||
|
|
||
|
if (statement.expression instanceof Literal) {
|
||
|
this.symbols[statement.variable.name] = statement.expression.value;
|
||
|
this.size++;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
} catch (err) {
|
||
|
_didIteratorError = true;
|
||
|
_iteratorError = err;
|
||
|
} finally {
|
||
|
try {
|
||
|
if (!_iteratorNormalCompletion && _iterator.return) {
|
||
|
_iterator.return();
|
||
|
}
|
||
|
} finally {
|
||
|
if (_didIteratorError) {
|
||
|
throw _iteratorError;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
this.main = this.variables.main;
|
||
|
if (!this.main) {
|
||
|
throw new Error('No main variable declaration found');
|
||
|
}
|
||
|
}
|
||
|
}, {
|
||
|
key: 'processExpression',
|
||
|
value: function processExpression(expr) {
|
||
|
// Process children
|
||
|
for (var key in expr) {
|
||
|
if (expr[key] instanceof Node) {
|
||
|
expr[key] = this.processExpression(expr[key]);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Replace variable references with their values
|
||
|
if (expr instanceof Variable) {
|
||
|
var value = this.variables[expr.name];
|
||
|
if (value == null) throw new Error('Undeclared indentifier ' + expr.name);
|
||
|
|
||
|
expr = this.processExpression(value.copy());
|
||
|
}
|
||
|
|
||
|
return expr;
|
||
|
}
|
||
|
}]);
|
||
|
|
||
|
return SymbolTable;
|
||
|
}();
|
||
|
|
||
|
var END_MARKER = new EndMarker();
|
||
|
|
||
|
/**
|
||
|
* This is an implementation of the direct regular expression to DFA algorithm described
|
||
|
* in section 3.9.5 of "Compilers: Principles, Techniques, and Tools" by Aho,
|
||
|
* Lam, Sethi, and Ullman. http://dragonbook.stanford.edu
|
||
|
* There is a PDF of the book here:
|
||
|
* http://www.informatik.uni-bremen.de/agbkb/lehre/ccfl/Material/ALSUdragonbook.pdf
|
||
|
*/
|
||
|
function buildDFA(root, numSymbols) {
|
||
|
root = new Concatenation(root, END_MARKER);
|
||
|
root.calcFollowpos();
|
||
|
|
||
|
var failState = new State(new _Set(), numSymbols);
|
||
|
var initialState = new State(root.firstpos, numSymbols);
|
||
|
var dstates = [failState, initialState];
|
||
|
|
||
|
// while there is an unmarked state S in dstates
|
||
|
while (1) {
|
||
|
var s = null;
|
||
|
|
||
|
for (var j = 1; j < dstates.length; j++) {
|
||
|
if (!dstates[j].marked) {
|
||
|
s = dstates[j];
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (s == null) {
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
// mark S
|
||
|
s.marked = true;
|
||
|
|
||
|
// for each input symbol a
|
||
|
for (var a = 0; a < numSymbols; a++) {
|
||
|
// let U be the union of followpos(p) for all
|
||
|
// p in S that correspond to a
|
||
|
var u = new _Set();
|
||
|
var _iteratorNormalCompletion = true;
|
||
|
var _didIteratorError = false;
|
||
|
var _iteratorError = undefined;
|
||
|
|
||
|
try {
|
||
|
for (var _iterator = _getIterator(s.positions), _step; !(_iteratorNormalCompletion = (_step = _iterator.next()).done); _iteratorNormalCompletion = true) {
|
||
|
var p = _step.value;
|
||
|
|
||
|
if (p instanceof Literal && p.value === a) {
|
||
|
addAll(u, p.followpos);
|
||
|
}
|
||
|
}
|
||
|
} catch (err) {
|
||
|
_didIteratorError = true;
|
||
|
_iteratorError = err;
|
||
|
} finally {
|
||
|
try {
|
||
|
if (!_iteratorNormalCompletion && _iterator.return) {
|
||
|
_iterator.return();
|
||
|
}
|
||
|
} finally {
|
||
|
if (_didIteratorError) {
|
||
|
throw _iteratorError;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (u.size === 0) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
// if U is not in dstates
|
||
|
var ux = -1;
|
||
|
for (var i = 0; i < dstates.length; i++) {
|
||
|
if (equal(u, dstates[i].positions)) {
|
||
|
ux = i;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (ux === -1) {
|
||
|
// Add U as an unmarked state to dstates
|
||
|
dstates.push(new State(u, numSymbols));
|
||
|
ux = dstates.length - 1;
|
||
|
}
|
||
|
|
||
|
s.transitions[a] = ux;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return dstates;
|
||
|
}
|
||
|
|
||
|
var State = function State(positions, len) {
|
||
|
_classCallCheck(this, State);
|
||
|
|
||
|
this.positions = positions;
|
||
|
this.transitions = new Uint16Array(len);
|
||
|
this.accepting = positions.has(END_MARKER);
|
||
|
this.marked = false;
|
||
|
this.tags = new _Set();
|
||
|
|
||
|
var _iteratorNormalCompletion2 = true;
|
||
|
var _didIteratorError2 = false;
|
||
|
var _iteratorError2 = undefined;
|
||
|
|
||
|
try {
|
||
|
for (var _iterator2 = _getIterator(positions), _step2; !(_iteratorNormalCompletion2 = (_step2 = _iterator2.next()).done); _iteratorNormalCompletion2 = true) {
|
||
|
var pos = _step2.value;
|
||
|
|
||
|
if (pos instanceof Tag) {
|
||
|
this.tags.add(pos.name);
|
||
|
}
|
||
|
}
|
||
|
} catch (err) {
|
||
|
_didIteratorError2 = true;
|
||
|
_iteratorError2 = err;
|
||
|
} finally {
|
||
|
try {
|
||
|
if (!_iteratorNormalCompletion2 && _iterator2.return) {
|
||
|
_iterator2.return();
|
||
|
}
|
||
|
} finally {
|
||
|
if (_didIteratorError2) {
|
||
|
throw _iteratorError2;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
};
|
||
|
|
||
|
var INITIAL_STATE = 1;
|
||
|
var FAIL_STATE = 0;
|
||
|
|
||
|
/**
|
||
|
* A StateMachine represents a deterministic finite automaton.
|
||
|
* It can perform matches over a sequence of values, similar to a regular expression.
|
||
|
*/
|
||
|
|
||
|
var StateMachine = function () {
|
||
|
function StateMachine(dfa) {
|
||
|
_classCallCheck(this, StateMachine);
|
||
|
|
||
|
this.stateTable = dfa.stateTable;
|
||
|
this.accepting = dfa.accepting;
|
||
|
this.tags = dfa.tags;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Returns an iterable object that yields pattern matches over the input sequence.
|
||
|
* Matches are of the form [startIndex, endIndex, tags].
|
||
|
*/
|
||
|
|
||
|
|
||
|
_createClass(StateMachine, [{
|
||
|
key: 'match',
|
||
|
value: function match(str) {
|
||
|
var self = this;
|
||
|
return _defineProperty({}, _Symbol$iterator, _regeneratorRuntime.mark(function _callee() {
|
||
|
var state, startRun, lastAccepting, lastState, p, c;
|
||
|
return _regeneratorRuntime.wrap(function _callee$(_context) {
|
||
|
while (1) {
|
||
|
switch (_context.prev = _context.next) {
|
||
|
case 0:
|
||
|
state = INITIAL_STATE;
|
||
|
startRun = null;
|
||
|
lastAccepting = null;
|
||
|
lastState = null;
|
||
|
p = 0;
|
||
|
|
||
|
case 5:
|
||
|
if (!(p < str.length)) {
|
||
|
_context.next = 21;
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
c = str[p];
|
||
|
|
||
|
|
||
|
lastState = state;
|
||
|
state = self.stateTable[state][c];
|
||
|
|
||
|
if (!(state === FAIL_STATE)) {
|
||
|
_context.next = 15;
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
if (!(startRun != null && lastAccepting != null && lastAccepting >= startRun)) {
|
||
|
_context.next = 13;
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
_context.next = 13;
|
||
|
return [startRun, lastAccepting, self.tags[lastState]];
|
||
|
|
||
|
case 13:
|
||
|
|
||
|
// reset the state as if we started over from the initial state
|
||
|
state = self.stateTable[INITIAL_STATE][c];
|
||
|
startRun = null;
|
||
|
|
||
|
case 15:
|
||
|
|
||
|
// start a run if not in the failure state
|
||
|
if (state !== FAIL_STATE && startRun == null) {
|
||
|
startRun = p;
|
||
|
}
|
||
|
|
||
|
// if accepting, mark the potential match end
|
||
|
if (self.accepting[state]) {
|
||
|
lastAccepting = p;
|
||
|
}
|
||
|
|
||
|
// reset the state to the initial state if we get into the failure state
|
||
|
if (state === FAIL_STATE) {
|
||
|
state = INITIAL_STATE;
|
||
|
}
|
||
|
|
||
|
case 18:
|
||
|
p++;
|
||
|
_context.next = 5;
|
||
|
break;
|
||
|
|
||
|
case 21:
|
||
|
if (!(startRun != null && lastAccepting != null && lastAccepting >= startRun)) {
|
||
|
_context.next = 24;
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
_context.next = 24;
|
||
|
return [startRun, lastAccepting, self.tags[state]];
|
||
|
|
||
|
case 24:
|
||
|
case 'end':
|
||
|
return _context.stop();
|
||
|
}
|
||
|
}
|
||
|
}, _callee, this);
|
||
|
}));
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* For each match over the input sequence, action functions matching
|
||
|
* the tag definitions in the input pattern are called with the startIndex,
|
||
|
* endIndex, and sub-match sequence.
|
||
|
*/
|
||
|
|
||
|
}, {
|
||
|
key: 'apply',
|
||
|
value: function apply(str, actions) {
|
||
|
var _iteratorNormalCompletion = true;
|
||
|
var _didIteratorError = false;
|
||
|
var _iteratorError = undefined;
|
||
|
|
||
|
try {
|
||
|
for (var _iterator = _getIterator(this.match(str)), _step; !(_iteratorNormalCompletion = (_step = _iterator.next()).done); _iteratorNormalCompletion = true) {
|
||
|
var _step$value = _slicedToArray(_step.value, 3);
|
||
|
|
||
|
var start = _step$value[0];
|
||
|
var end = _step$value[1];
|
||
|
var tags = _step$value[2];
|
||
|
var _iteratorNormalCompletion2 = true;
|
||
|
var _didIteratorError2 = false;
|
||
|
var _iteratorError2 = undefined;
|
||
|
|
||
|
try {
|
||
|
for (var _iterator2 = _getIterator(tags), _step2; !(_iteratorNormalCompletion2 = (_step2 = _iterator2.next()).done); _iteratorNormalCompletion2 = true) {
|
||
|
var tag = _step2.value;
|
||
|
|
||
|
if (typeof actions[tag] === 'function') {
|
||
|
actions[tag](start, end, str.slice(start, end + 1));
|
||
|
}
|
||
|
}
|
||
|
} catch (err) {
|
||
|
_didIteratorError2 = true;
|
||
|
_iteratorError2 = err;
|
||
|
} finally {
|
||
|
try {
|
||
|
if (!_iteratorNormalCompletion2 && _iterator2.return) {
|
||
|
_iterator2.return();
|
||
|
}
|
||
|
} finally {
|
||
|
if (_didIteratorError2) {
|
||
|
throw _iteratorError2;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
} catch (err) {
|
||
|
_didIteratorError = true;
|
||
|
_iteratorError = err;
|
||
|
} finally {
|
||
|
try {
|
||
|
if (!_iteratorNormalCompletion && _iterator.return) {
|
||
|
_iterator.return();
|
||
|
}
|
||
|
} finally {
|
||
|
if (_didIteratorError) {
|
||
|
throw _iteratorError;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}]);
|
||
|
|
||
|
return StateMachine;
|
||
|
}();
|
||
|
|
||
|
function parse(string, externalSymbols) {
|
||
|
var ast = grammar.parse(string);
|
||
|
return new SymbolTable(ast, externalSymbols);
|
||
|
}
|
||
|
|
||
|
function build(symbolTable) {
|
||
|
var states = buildDFA(symbolTable.main, symbolTable.size);
|
||
|
|
||
|
return new StateMachine({
|
||
|
stateTable: states.map(function (s) {
|
||
|
return _Array$from(s.transitions);
|
||
|
}),
|
||
|
accepting: states.map(function (s) {
|
||
|
return s.accepting;
|
||
|
}),
|
||
|
tags: states.map(function (s) {
|
||
|
return _Array$from(s.tags);
|
||
|
})
|
||
|
});
|
||
|
}
|
||
|
|
||
|
function compile(string, externalSymbols) {
|
||
|
return build(parse(string, externalSymbols));
|
||
|
}
|
||
|
|
||
|
exports.parse = parse;
|
||
|
exports.build = build;
|
||
|
exports['default'] = compile;
|
||
|
//# sourceMappingURL=compile.js.map
|