XFA - Add a lexer/parser for FormCalc language (#12936)
- the language specifications are: http://citeseerx.ist.psu.edu/viewdoc/download?doi=10.1.1.364.2157&rep=rep1&type=pdf#page=1049 - it can be used to: * as a scripting language for calculation, validations, ... * in SOM expressions to select nodes: http://citeseerx.ist.psu.edu/viewdoc/download?doi=10.1.1.364.2157&rep=rep1&type=pdf#page=101
This commit is contained in:
parent
4de8b7e433
commit
b5be515375
385
src/core/xfa/formcalc_lexer.js
Normal file
385
src/core/xfa/formcalc_lexer.js
Normal file
@ -0,0 +1,385 @@
|
||||
/* Copyright 2021 Mozilla Foundation
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
const KEYWORDS = new Set([
|
||||
"and",
|
||||
"break",
|
||||
"continue",
|
||||
"do",
|
||||
"downto",
|
||||
"else",
|
||||
"elseif",
|
||||
"end",
|
||||
"endfor",
|
||||
"endfunc",
|
||||
"endif",
|
||||
"endwhile",
|
||||
"eq",
|
||||
"exit",
|
||||
"for",
|
||||
"foreach",
|
||||
"func",
|
||||
"ge",
|
||||
"gt",
|
||||
"if",
|
||||
"in",
|
||||
"infinity",
|
||||
"le",
|
||||
"lt",
|
||||
"nan",
|
||||
"ne",
|
||||
"not",
|
||||
"null",
|
||||
"or",
|
||||
"return",
|
||||
"step",
|
||||
"then",
|
||||
"this",
|
||||
"throw",
|
||||
"upto",
|
||||
"var",
|
||||
"while",
|
||||
]);
|
||||
|
||||
const TOKEN = {
|
||||
/* Appears in expression */
|
||||
and: 0,
|
||||
divide: 1,
|
||||
dot: 2,
|
||||
dotDot: 3,
|
||||
dotHash: 4,
|
||||
dotStar: 5,
|
||||
eq: 6,
|
||||
ge: 7,
|
||||
gt: 8,
|
||||
le: 9,
|
||||
leftBracket: 10,
|
||||
leftParen: 11,
|
||||
lt: 12,
|
||||
minus: 13,
|
||||
ne: 14,
|
||||
not: 15,
|
||||
null: 16,
|
||||
number: 17,
|
||||
or: 18,
|
||||
plus: 19,
|
||||
rightBracket: 20,
|
||||
rightParen: 21,
|
||||
string: 22,
|
||||
this: 23,
|
||||
times: 24,
|
||||
identifier: 25, // in main statments too
|
||||
|
||||
/* Main statements */
|
||||
break: 26,
|
||||
continue: 27,
|
||||
do: 28,
|
||||
for: 29,
|
||||
foreach: 30,
|
||||
func: 31,
|
||||
if: 32,
|
||||
var: 33,
|
||||
while: 34,
|
||||
|
||||
/* Others */
|
||||
assign: 35,
|
||||
comma: 36,
|
||||
downto: 37,
|
||||
else: 38,
|
||||
elseif: 39,
|
||||
end: 40,
|
||||
endif: 41,
|
||||
endfor: 42,
|
||||
endfunc: 43,
|
||||
endwhile: 44,
|
||||
eof: 45,
|
||||
exit: 46,
|
||||
in: 47,
|
||||
infinity: 48,
|
||||
nan: 49,
|
||||
return: 50,
|
||||
step: 51,
|
||||
then: 52,
|
||||
throw: 53,
|
||||
upto: 54,
|
||||
};
|
||||
|
||||
const hexPattern = /^[uU]([0-9a-fA-F]{4,8})/;
|
||||
const numberPattern = /^[0-9]*(?:\.[0-9]*)?(?:[Ee][+-]?[0-9]+)?/;
|
||||
const dotNumberPattern = /^[0-9]*(?:[Ee][+-]?[0-9]+)?/;
|
||||
const eolPattern = /[\r\n]+/;
|
||||
const identifierPattern = new RegExp("^[\\p{L}_$!][\\p{L}\\p{N}_$]*", "u");
|
||||
|
||||
class Token {
|
||||
constructor(id, value = null) {
|
||||
this.id = id;
|
||||
this.value = value;
|
||||
}
|
||||
}
|
||||
|
||||
const Singletons = (function () {
|
||||
const obj = Object.create(null);
|
||||
const nonSingleton = new Set([
|
||||
"identifier",
|
||||
"string",
|
||||
"number",
|
||||
"nan",
|
||||
"infinity",
|
||||
]);
|
||||
for (const [name, id] of Object.entries(TOKEN)) {
|
||||
if (!nonSingleton.has(name)) {
|
||||
obj[name] = new Token(id);
|
||||
}
|
||||
}
|
||||
obj.nan = new Token(TOKEN.number, NaN);
|
||||
obj.infinity = new Token(TOKEN.number, Infinity);
|
||||
|
||||
return obj;
|
||||
})();
|
||||
|
||||
class Lexer {
|
||||
constructor(data) {
|
||||
this.data = data;
|
||||
this.pos = 0;
|
||||
this.len = data.length;
|
||||
this.strBuf = [];
|
||||
}
|
||||
|
||||
skipUntilEOL() {
|
||||
const match = this.data.slice(this.pos).match(eolPattern);
|
||||
if (match) {
|
||||
this.pos += match.index + match[0].length;
|
||||
} else {
|
||||
// No eol so consume all the chars.
|
||||
this.pos = this.len;
|
||||
}
|
||||
}
|
||||
|
||||
getIdentifier() {
|
||||
this.pos--;
|
||||
const match = this.data.slice(this.pos).match(identifierPattern);
|
||||
if (!match) {
|
||||
throw new Error(
|
||||
`Invalid token in FormCalc expression at position ${this.pos}.`
|
||||
);
|
||||
}
|
||||
|
||||
const identifier = this.data.slice(this.pos, this.pos + match[0].length);
|
||||
this.pos += match[0].length;
|
||||
|
||||
const lower = identifier.toLowerCase();
|
||||
if (!KEYWORDS.has(lower)) {
|
||||
return new Token(TOKEN.identifier, identifier);
|
||||
}
|
||||
|
||||
return Singletons[lower];
|
||||
}
|
||||
|
||||
getString() {
|
||||
const str = this.strBuf;
|
||||
const data = this.data;
|
||||
let start = this.pos;
|
||||
while (this.pos < this.len) {
|
||||
const char = data.charCodeAt(this.pos++);
|
||||
if (char === 0x22 /* = " */) {
|
||||
if (data.charCodeAt(this.pos) === 0x22 /* = " */) {
|
||||
// Escaped quote.
|
||||
str.push(data.slice(start, this.pos++));
|
||||
start = this.pos;
|
||||
continue;
|
||||
}
|
||||
// End of string
|
||||
break;
|
||||
}
|
||||
|
||||
if (char === 0x5c /* = \ */) {
|
||||
const match = data.substring(this.pos, this.pos + 10).match(hexPattern);
|
||||
if (!match) {
|
||||
continue;
|
||||
}
|
||||
|
||||
str.push(data.slice(start, this.pos - 1));
|
||||
const code = match[1];
|
||||
if (code.length === 4) {
|
||||
str.push(String.fromCharCode(parseInt(code, 16)));
|
||||
start = this.pos += 5;
|
||||
} else if (code.length !== 8) {
|
||||
str.push(String.fromCharCode(parseInt(code.slice(0, 4), 16)));
|
||||
start = this.pos += 5;
|
||||
} else {
|
||||
str.push(String.fromCharCode(parseInt(code, 16)));
|
||||
start = this.pos += 9;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
const lastChunk = data.slice(start, this.pos - 1);
|
||||
if (str.length === 0) {
|
||||
return new Token(TOKEN.string, lastChunk);
|
||||
}
|
||||
|
||||
str.push(lastChunk);
|
||||
const string = str.join("");
|
||||
str.length = 0;
|
||||
|
||||
return new Token(TOKEN.string, string);
|
||||
}
|
||||
|
||||
getNumber(first) {
|
||||
const match = this.data.substring(this.pos).match(numberPattern);
|
||||
if (!match) {
|
||||
return first - 0x30 /* = 0 */;
|
||||
}
|
||||
const number = parseFloat(
|
||||
this.data.substring(this.pos - 1, this.pos + match[0].length)
|
||||
);
|
||||
|
||||
this.pos += match[0].length;
|
||||
return new Token(TOKEN.number, number);
|
||||
}
|
||||
|
||||
getCompOperator(alt1, alt2) {
|
||||
if (this.data.charCodeAt(this.pos) === 0x3d /* = = */) {
|
||||
this.pos++;
|
||||
return alt1;
|
||||
}
|
||||
return alt2;
|
||||
}
|
||||
|
||||
getLower() {
|
||||
const char = this.data.charCodeAt(this.pos);
|
||||
if (char === 0x3d /* = = */) {
|
||||
this.pos++;
|
||||
return Singletons.le;
|
||||
}
|
||||
|
||||
if (char === 0x3e /* = > */) {
|
||||
this.pos++;
|
||||
return Singletons.ne;
|
||||
}
|
||||
|
||||
return Singletons.lt;
|
||||
}
|
||||
|
||||
getSlash() {
|
||||
if (this.data.charCodeAt(this.pos) === 0x2f /* = / */) {
|
||||
this.skipUntilEOL();
|
||||
return false;
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
getDot() {
|
||||
const char = this.data.charCodeAt(this.pos);
|
||||
if (char === 0x2e /* = . */) {
|
||||
this.pos++;
|
||||
return Singletons.dotDot;
|
||||
}
|
||||
|
||||
if (char === 0x2a /* = * */) {
|
||||
this.pos++;
|
||||
return Singletons.dotStar;
|
||||
}
|
||||
|
||||
if (char === 0x23 /* = # */) {
|
||||
this.pos++;
|
||||
return Singletons.dotHash;
|
||||
}
|
||||
|
||||
if (0x30 /* = 0 */ <= char && char <= 0x39 /* = 9 */) {
|
||||
this.pos++;
|
||||
const match = this.data.substring(this.pos).match(dotNumberPattern);
|
||||
if (!match) {
|
||||
return new Token(TOKEN.number, (char - 0x30) /* = 0 */ / 10);
|
||||
}
|
||||
const end = this.pos + match[0].length;
|
||||
const number = parseFloat(this.data.substring(this.pos - 2, end));
|
||||
this.pos = end;
|
||||
return new Token(TOKEN.number, number);
|
||||
}
|
||||
|
||||
return Singletons.dot;
|
||||
}
|
||||
|
||||
next() {
|
||||
while (this.pos < this.len) {
|
||||
const char = this.data.charCodeAt(this.pos++);
|
||||
switch (char) {
|
||||
case 0x09 /* = \t */:
|
||||
case 0x0a /* = \n */:
|
||||
case 0x0b /* = \v */:
|
||||
case 0x0c /* = \f */:
|
||||
case 0x0d /* = \r */:
|
||||
case 0x20 /* = */:
|
||||
break;
|
||||
case 0x22 /* = " */:
|
||||
return this.getString();
|
||||
case 0x26 /* = & */:
|
||||
return Singletons.and;
|
||||
case 0x28 /* = ( */:
|
||||
return Singletons.leftParen;
|
||||
case 0x29 /* = ) */:
|
||||
return Singletons.rightParen;
|
||||
case 0x2a /* = * */:
|
||||
return Singletons.times;
|
||||
case 0x2b /* = + */:
|
||||
return Singletons.plus;
|
||||
case 0x2c /* = , */:
|
||||
return Singletons.comma;
|
||||
case 0x2d /* = - */:
|
||||
return Singletons.minus;
|
||||
case 0x2e /* = . */:
|
||||
return this.getDot();
|
||||
case 0x2f /* = / */:
|
||||
if (this.getSlash()) {
|
||||
return Singletons.divide;
|
||||
}
|
||||
// It was a comment.
|
||||
break;
|
||||
case 0x30 /* = 0 */:
|
||||
case 0x31 /* = 1 */:
|
||||
case 0x32 /* = 2 */:
|
||||
case 0x33 /* = 3 */:
|
||||
case 0x34 /* = 4 */:
|
||||
case 0x35 /* = 5 */:
|
||||
case 0x36 /* = 6 */:
|
||||
case 0x37 /* = 7 */:
|
||||
case 0x38 /* = 8 */:
|
||||
case 0x39 /* = 9 */:
|
||||
return this.getNumber(char);
|
||||
case 0x3b /* = ; */:
|
||||
this.skipUntilEOL();
|
||||
break;
|
||||
case 0x3c /* = < */:
|
||||
return this.getLower();
|
||||
case 0x3d /* = = */:
|
||||
return this.getCompOperator(Singletons.eq, Singletons.assign);
|
||||
case 0x3e /* = > */:
|
||||
return this.getCompOperator(Singletons.ge, Singletons.gt);
|
||||
case 0x5b /* = [ */:
|
||||
return Singletons.leftBracket;
|
||||
case 0x5d /* = ] */:
|
||||
return Singletons.rightBracket;
|
||||
case 0x7c /* = | */:
|
||||
return Singletons.or;
|
||||
default:
|
||||
return this.getIdentifier();
|
||||
}
|
||||
}
|
||||
return Singletons.eof;
|
||||
}
|
||||
}
|
||||
|
||||
export { Lexer, Token, TOKEN };
|
1340
src/core/xfa/formcalc_parser.js
Normal file
1340
src/core/xfa/formcalc_parser.js
Normal file
File diff suppressed because it is too large
Load Diff
@ -39,6 +39,7 @@
|
||||
"unicode_spec.js",
|
||||
"util_spec.js",
|
||||
"writer_spec.js",
|
||||
"xfa_formcalc_spec.js",
|
||||
"xfa_parser_spec.js",
|
||||
"xml_spec.js"
|
||||
]
|
||||
|
@ -85,6 +85,7 @@ async function initializePDFJS(callback) {
|
||||
"pdfjs-test/unit/unicode_spec.js",
|
||||
"pdfjs-test/unit/util_spec.js",
|
||||
"pdfjs-test/unit/writer_spec.js",
|
||||
"pdfjs-test/unit/xfa_formcalc_spec.js",
|
||||
"pdfjs-test/unit/xfa_parser_spec.js",
|
||||
"pdfjs-test/unit/xml_spec.js",
|
||||
].map(function (moduleName) {
|
||||
|
729
test/unit/xfa_formcalc_spec.js
Normal file
729
test/unit/xfa_formcalc_spec.js
Normal file
@ -0,0 +1,729 @@
|
||||
/* Copyright 2020 Mozilla Foundation
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
import { Errors, Parser } from "../../src/core/xfa/formcalc_parser.js";
|
||||
import { Lexer, Token, TOKEN } from "../../src/core/xfa/formcalc_lexer.js";
|
||||
|
||||
describe("FormCalc expression parser", function () {
|
||||
const EOF = new Token(TOKEN.eof);
|
||||
|
||||
describe("FormCalc lexer", function () {
|
||||
it("should lex numbers", function () {
|
||||
const lexer = new Lexer(
|
||||
"12 1.2345 .7 .12345 1e-2 1.2E+3 1e2 1.2E3 nan 12. 2.e3 infinity 99999999999999999 123456789.012345678 9e99999"
|
||||
);
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.number, 12));
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.number, 1.2345));
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.number, 0.7));
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.number, 0.12345));
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.number, 1e-2));
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.number, 1.2e3));
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.number, 1e2));
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.number, 1.2e3));
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.number, NaN));
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.number, 12));
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.number, 2e3));
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.number, Infinity));
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.number, 100000000000000000));
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.number, 123456789.01234567));
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.number, Infinity));
|
||||
expect(lexer.next()).toEqual(EOF);
|
||||
});
|
||||
|
||||
it("should lex strings", function () {
|
||||
const lexer = new Lexer(
|
||||
`"hello world" "hello ""world" "hello ""world"" ""world""""hello""" "hello \\uabcdeh \\Uabcd \\u00000123abc" "a \\a \\ub \\Uc \\b"`
|
||||
);
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.string, `hello world`));
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.string, `hello "world`));
|
||||
expect(lexer.next()).toEqual(
|
||||
new Token(TOKEN.string, `hello "world" "world""hello"`)
|
||||
);
|
||||
expect(lexer.next()).toEqual(
|
||||
new Token(TOKEN.string, `hello \uabcdeh \uabcd \u0123abc`)
|
||||
);
|
||||
expect(lexer.next()).toEqual(
|
||||
new Token(TOKEN.string, `a \\a \\ub \\Uc \\b`)
|
||||
);
|
||||
expect(lexer.next()).toEqual(EOF);
|
||||
});
|
||||
|
||||
it("should lex operators", function () {
|
||||
const lexer = new Lexer("( , ) <= <> = == >= < > / * . .* .# [ ] & |");
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.leftParen));
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.comma));
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.rightParen));
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.le));
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.ne));
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.assign));
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.eq));
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.ge));
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.lt));
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.gt));
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.divide));
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.times));
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.dot));
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.dotStar));
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.dotHash));
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.leftBracket));
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.rightBracket));
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.and));
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.or));
|
||||
expect(lexer.next()).toEqual(EOF);
|
||||
});
|
||||
|
||||
it("should skip comments", function () {
|
||||
const lexer = new Lexer(`
|
||||
|
||||
\t\t 1 \r\n\r\n
|
||||
|
||||
; blah blah blah
|
||||
|
||||
2
|
||||
|
||||
// blah blah blah blah blah
|
||||
|
||||
|
||||
3
|
||||
`);
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.number, 1));
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.number, 2));
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.number, 3));
|
||||
expect(lexer.next()).toEqual(EOF);
|
||||
});
|
||||
|
||||
it("should lex identifiers", function () {
|
||||
const lexer = new Lexer(
|
||||
"eq for fore while continue hello こんにちは世界 $!hello今日は12今日は"
|
||||
);
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.eq));
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.for));
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.identifier, "fore"));
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.while));
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.continue));
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.identifier, "hello"));
|
||||
expect(lexer.next()).toEqual(
|
||||
new Token(TOKEN.identifier, "こんにちは世界")
|
||||
);
|
||||
expect(lexer.next()).toEqual(new Token(TOKEN.identifier, "$"));
|
||||
expect(lexer.next()).toEqual(
|
||||
new Token(TOKEN.identifier, "!hello今日は12今日は")
|
||||
);
|
||||
expect(lexer.next()).toEqual(EOF);
|
||||
});
|
||||
});
|
||||
|
||||
describe("FormCalc parser", function () {
|
||||
it("should parse basic arithmetic expression", function () {
|
||||
const parser = new Parser("1 + 2 * 3");
|
||||
expect(parser.parse().dump()[0]).toEqual(7);
|
||||
});
|
||||
|
||||
it("should parse basic arithmetic expression with the same operator", function () {
|
||||
const parser = new Parser("1 + a + 3");
|
||||
expect(parser.parse().dump()[0]).toEqual({
|
||||
operator: "+",
|
||||
left: {
|
||||
operator: "+",
|
||||
left: 1,
|
||||
right: { id: "a" },
|
||||
},
|
||||
right: 3,
|
||||
});
|
||||
});
|
||||
|
||||
it("should parse expressions with unary operators", function () {
|
||||
const parser = new Parser(`
|
||||
s = +x + 1
|
||||
t = -+u * 2
|
||||
t = +-u * 2
|
||||
u = -foo()
|
||||
`);
|
||||
expect(parser.parse().dump()).toEqual([
|
||||
{
|
||||
assignment: "s",
|
||||
expr: {
|
||||
operator: "+",
|
||||
left: { operator: "+", arg: { id: "x" } },
|
||||
right: 1,
|
||||
},
|
||||
},
|
||||
{
|
||||
assignment: "t",
|
||||
expr: {
|
||||
operator: "*",
|
||||
left: {
|
||||
operator: "-",
|
||||
arg: {
|
||||
operator: "+",
|
||||
arg: { id: "u" },
|
||||
},
|
||||
},
|
||||
right: 2,
|
||||
},
|
||||
},
|
||||
{
|
||||
assignment: "t",
|
||||
expr: {
|
||||
operator: "*",
|
||||
left: {
|
||||
operator: "+",
|
||||
arg: {
|
||||
operator: "-",
|
||||
arg: { id: "u" },
|
||||
},
|
||||
},
|
||||
right: 2,
|
||||
},
|
||||
},
|
||||
{
|
||||
assignment: "u",
|
||||
expr: {
|
||||
operator: "-",
|
||||
arg: {
|
||||
callee: { id: "foo" },
|
||||
params: [],
|
||||
},
|
||||
},
|
||||
},
|
||||
]);
|
||||
});
|
||||
|
||||
it("should parse basic expression with a string", function () {
|
||||
const parser = new Parser(`(5 - "abc") * 3`);
|
||||
expect(parser.parse().dump()[0]).toEqual(15);
|
||||
});
|
||||
|
||||
it("should parse basic expression with a calls", function () {
|
||||
const parser = new Parser(`foo(2, 3, a & b) or c * d + 1.234 / e`);
|
||||
expect(parser.parse().dump()[0]).toEqual({
|
||||
operator: "||",
|
||||
left: {
|
||||
callee: { id: "foo" },
|
||||
params: [
|
||||
2,
|
||||
3,
|
||||
{
|
||||
operator: "&&",
|
||||
left: { id: "a" },
|
||||
right: { id: "b" },
|
||||
},
|
||||
],
|
||||
},
|
||||
right: {
|
||||
operator: "+",
|
||||
left: {
|
||||
operator: "*",
|
||||
left: { id: "c" },
|
||||
right: { id: "d" },
|
||||
},
|
||||
right: {
|
||||
operator: "/",
|
||||
left: 1.234,
|
||||
right: { id: "e" },
|
||||
},
|
||||
},
|
||||
});
|
||||
});
|
||||
|
||||
it("should parse basic expression with a subscript", function () {
|
||||
let parser = new Parser(`こんにちは世界[-0]`);
|
||||
let dump = parser.parse().dump()[0];
|
||||
expect(dump).toEqual({
|
||||
operand: { id: "こんにちは世界" },
|
||||
index: -0,
|
||||
});
|
||||
expect(Object.is(-0, dump.index)).toBe(true);
|
||||
|
||||
parser = new Parser(`こんにちは世界[+0]`);
|
||||
dump = parser.parse().dump()[0];
|
||||
expect(dump).toEqual({
|
||||
operand: { id: "こんにちは世界" },
|
||||
index: +0,
|
||||
});
|
||||
expect(Object.is(+0, dump.index)).toBe(true);
|
||||
|
||||
parser = new Parser(`こんにちは世界[*]`);
|
||||
expect(parser.parse().dump()[0]).toEqual({
|
||||
operand: { id: "こんにちは世界" },
|
||||
index: { special: "*" },
|
||||
});
|
||||
});
|
||||
|
||||
it("should parse basic expression with dots", function () {
|
||||
const parser = new Parser("a.b.c.#d..e.f..g.*");
|
||||
expect(parser.parse().dump()[0]).toEqual({
|
||||
operator: ".",
|
||||
left: { id: "a" },
|
||||
right: {
|
||||
operator: ".",
|
||||
left: { id: "b" },
|
||||
right: {
|
||||
operator: ".#",
|
||||
left: { id: "c" },
|
||||
right: {
|
||||
operator: "..",
|
||||
left: { id: "d" },
|
||||
right: {
|
||||
operator: ".",
|
||||
left: { id: "e" },
|
||||
right: {
|
||||
operator: "..",
|
||||
left: { id: "f" },
|
||||
right: {
|
||||
operator: ".",
|
||||
left: { id: "g" },
|
||||
right: { special: "*" },
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
});
|
||||
});
|
||||
|
||||
it("should parse var declaration with error", function () {
|
||||
let parser = new Parser("var 123 = a");
|
||||
expect(() => parser.parse()).toThrow(new Error(Errors.var));
|
||||
|
||||
parser = new Parser(`var "123" = a`);
|
||||
expect(() => parser.parse()).toThrow(new Error(Errors.var));
|
||||
|
||||
parser = new Parser(`var for var a`);
|
||||
expect(() => parser.parse()).toThrow(new Error(Errors.var));
|
||||
});
|
||||
|
||||
it("should parse for declaration with a step", function () {
|
||||
const parser = new Parser(`
|
||||
var s = 0
|
||||
for var i = 1 upto 10 + x step 1 do
|
||||
s = s + i * 2
|
||||
endfor`);
|
||||
expect(parser.parse().dump()).toEqual([
|
||||
{
|
||||
var: "s",
|
||||
expr: 0,
|
||||
},
|
||||
{
|
||||
decl: "for",
|
||||
assignment: {
|
||||
var: "i",
|
||||
expr: 1,
|
||||
},
|
||||
type: "upto",
|
||||
end: {
|
||||
operator: "+",
|
||||
left: 10,
|
||||
right: { id: "x" },
|
||||
},
|
||||
step: 1,
|
||||
body: [
|
||||
{
|
||||
assignment: "s",
|
||||
expr: {
|
||||
operator: "+",
|
||||
left: { id: "s" },
|
||||
right: {
|
||||
operator: "*",
|
||||
left: { id: "i" },
|
||||
right: 2,
|
||||
},
|
||||
},
|
||||
},
|
||||
],
|
||||
},
|
||||
]);
|
||||
});
|
||||
|
||||
it("should parse for declaration without a step", function () {
|
||||
const parser = new Parser(`
|
||||
for i = 1 + 2 downto 10 do
|
||||
s = foo()
|
||||
endfor`);
|
||||
expect(parser.parse().dump()).toEqual([
|
||||
{
|
||||
decl: "for",
|
||||
assignment: {
|
||||
assignment: "i",
|
||||
expr: 3,
|
||||
},
|
||||
type: "downto",
|
||||
end: 10,
|
||||
step: null,
|
||||
body: [
|
||||
{
|
||||
assignment: "s",
|
||||
expr: {
|
||||
callee: { id: "foo" },
|
||||
params: [],
|
||||
},
|
||||
},
|
||||
],
|
||||
},
|
||||
]);
|
||||
});
|
||||
|
||||
it("should parse for declaration with error", function () {
|
||||
let parser = new Parser("for 123 = i upto 1 do a = 1 endfor");
|
||||
expect(() => parser.parse()).toThrow(new Error(Errors.assignment));
|
||||
|
||||
parser = new Parser("for var 123 = i upto 1 do a = 1 endfor");
|
||||
expect(() => parser.parse()).toThrow(new Error(Errors.assignment));
|
||||
|
||||
parser = new Parser("for var i = 123 upt 1 do a = 1 endfor");
|
||||
expect(() => parser.parse()).toThrow(new Error(Errors.for));
|
||||
|
||||
parser = new Parser("for var i = 123 var 1 do a = 1 endfor");
|
||||
expect(() => parser.parse()).toThrow(new Error(Errors.for));
|
||||
|
||||
parser = new Parser(
|
||||
"for var i = 123 upto 1 step for var j = 1 do endfor do a = 1 endfor"
|
||||
);
|
||||
expect(() => parser.parse()).toThrow(new Error(Errors.for));
|
||||
|
||||
parser = new Parser("for var i = 123 downto 1 do a = 1 endfunc");
|
||||
expect(() => parser.parse()).toThrow(new Error(Errors.for));
|
||||
|
||||
parser = new Parser("for var i = 123 downto 1 do a = 1");
|
||||
expect(() => parser.parse()).toThrow(new Error(Errors.for));
|
||||
});
|
||||
|
||||
it("should parse foreach declaration", function () {
|
||||
const parser = new Parser(`
|
||||
foreach i in (a, b, c, d) do
|
||||
s = foo()[i]
|
||||
endfor`);
|
||||
expect(parser.parse().dump()).toEqual([
|
||||
{
|
||||
decl: "foreach",
|
||||
id: "i",
|
||||
params: [{ id: "a" }, { id: "b" }, { id: "c" }, { id: "d" }],
|
||||
body: [
|
||||
{
|
||||
assignment: "s",
|
||||
expr: {
|
||||
operand: {
|
||||
callee: { id: "foo" },
|
||||
params: [],
|
||||
},
|
||||
index: { id: "i" },
|
||||
},
|
||||
},
|
||||
],
|
||||
},
|
||||
]);
|
||||
});
|
||||
|
||||
it("should parse foreach declaration with error", function () {
|
||||
let parser = new Parser("foreach 123 in (1, 2, 3) do a = 1 endfor");
|
||||
expect(() => parser.parse()).toThrow(new Error(Errors.foreach));
|
||||
|
||||
parser = new Parser("foreach foo in 1, 2, 3) do a = 1 endfor");
|
||||
expect(() => parser.parse()).toThrow(new Error(Errors.foreach));
|
||||
|
||||
parser = new Parser("foreach foo in (1, 2, 3 do a = 1 endfor");
|
||||
expect(() => parser.parse()).toThrow(new Error(Errors.params));
|
||||
|
||||
parser = new Parser("foreach foo in (1, 2 3) do a = 1 endfor");
|
||||
expect(() => parser.parse()).toThrow(new Error(Errors.params));
|
||||
|
||||
parser = new Parser("foreach foo in (1, 2, 3) od a = 1 endfor");
|
||||
expect(() => parser.parse()).toThrow(new Error(Errors.foreach));
|
||||
|
||||
parser = new Parser("foreach foo in (1, 2, 3) do a = 1 endforeach");
|
||||
expect(() => parser.parse()).toThrow(new Error(Errors.foreach));
|
||||
|
||||
parser = new Parser("foreach foo in (1, 2, 3) do a = 1 123");
|
||||
expect(() => parser.parse()).toThrow(new Error(Errors.foreach));
|
||||
});
|
||||
|
||||
it("should parse while declaration", function () {
|
||||
const parser = new Parser(`
|
||||
while (1) do
|
||||
if (0) then
|
||||
break
|
||||
else
|
||||
continue
|
||||
endif
|
||||
endwhile
|
||||
`);
|
||||
expect(parser.parse().dump()).toEqual([
|
||||
{
|
||||
decl: "while",
|
||||
condition: 1,
|
||||
body: [
|
||||
{
|
||||
decl: "if",
|
||||
condition: 0,
|
||||
then: [{ special: "break" }],
|
||||
elseif: null,
|
||||
else: [{ special: "continue" }],
|
||||
},
|
||||
],
|
||||
},
|
||||
]);
|
||||
});
|
||||
|
||||
it("should parse while declaration with error", function () {
|
||||
let parser = new Parser("while a == 1 do a = 2 endwhile");
|
||||
expect(() => parser.parse()).toThrow(new Error(Errors.while));
|
||||
|
||||
parser = new Parser("while (a == 1 do a = 2 endwhile");
|
||||
expect(() => parser.parse()).toThrow(new Error(Errors.while));
|
||||
|
||||
parser = new Parser("while (a == 1) var a = 2 endwhile");
|
||||
expect(() => parser.parse()).toThrow(new Error(Errors.while));
|
||||
|
||||
parser = new Parser("while (a == 1) do var a = 2 end");
|
||||
expect(() => parser.parse()).toThrow(new Error(Errors.while));
|
||||
});
|
||||
|
||||
it("should parse do declaration", function () {
|
||||
const parser = new Parser(`
|
||||
do
|
||||
x = 1
|
||||
; a comment in the middle of the block
|
||||
y = 2
|
||||
end
|
||||
`);
|
||||
expect(parser.parse().dump()).toEqual([
|
||||
{
|
||||
decl: "block",
|
||||
body: [
|
||||
{
|
||||
assignment: "x",
|
||||
expr: 1,
|
||||
},
|
||||
{
|
||||
assignment: "y",
|
||||
expr: 2,
|
||||
},
|
||||
],
|
||||
},
|
||||
]);
|
||||
});
|
||||
|
||||
it("should parse do declaration with error", function () {
|
||||
const parser = new Parser(`
|
||||
do
|
||||
x = 1
|
||||
y = 2
|
||||
endfunc
|
||||
`);
|
||||
expect(() => parser.parse()).toThrow(new Error(Errors.block));
|
||||
});
|
||||
|
||||
it("should parse func declaration", function () {
|
||||
const parser = new Parser(`
|
||||
func こんにちは世界123(a, b) do
|
||||
a + b
|
||||
endfunc
|
||||
`);
|
||||
expect(parser.parse().dump()).toEqual([
|
||||
{
|
||||
func: "こんにちは世界123",
|
||||
params: ["a", "b"],
|
||||
body: [
|
||||
{
|
||||
operator: "+",
|
||||
left: { id: "a" },
|
||||
right: { id: "b" },
|
||||
},
|
||||
],
|
||||
},
|
||||
]);
|
||||
});
|
||||
|
||||
it("should parse func declaration with error", function () {
|
||||
let parser = new Parser("func 123(a, b) do a = 1 endfunc");
|
||||
expect(() => parser.parse()).toThrow(new Error(Errors.func));
|
||||
|
||||
parser = new Parser("func foo(a, b) for a = 1 endfunc");
|
||||
expect(() => parser.parse()).toThrow(new Error(Errors.func));
|
||||
|
||||
parser = new Parser("func foo(a, b) do a = 1 endfun");
|
||||
expect(() => parser.parse()).toThrow(new Error(Errors.func));
|
||||
|
||||
parser = new Parser("func foo(a, b, c do a = 1 endfunc");
|
||||
expect(() => parser.parse()).toThrow(new Error(Errors.func));
|
||||
|
||||
parser = new Parser("func foo(a, b, 123) do a = 1 endfunc");
|
||||
expect(() => parser.parse()).toThrow(new Error(Errors.func));
|
||||
});
|
||||
|
||||
it("should parse if declaration", function () {
|
||||
const parser = new Parser(`
|
||||
if (a & b) then
|
||||
var s = 1
|
||||
endif
|
||||
|
||||
if (a or b) then
|
||||
var s = 1
|
||||
else
|
||||
var x = 2
|
||||
endif
|
||||
|
||||
if (0) then
|
||||
s = 1
|
||||
elseif (1) then
|
||||
s = 2
|
||||
elseif (2) then
|
||||
s = 3
|
||||
elseif (3) then
|
||||
s = 4
|
||||
else
|
||||
s = 5
|
||||
endif
|
||||
|
||||
// a comment
|
||||
|
||||
if (0) then
|
||||
s = 1
|
||||
elseif (1) then
|
||||
s = 2
|
||||
endif
|
||||
`);
|
||||
expect(parser.parse().dump()).toEqual([
|
||||
{
|
||||
decl: "if",
|
||||
condition: {
|
||||
operator: "&&",
|
||||
left: { id: "a" },
|
||||
right: { id: "b" },
|
||||
},
|
||||
then: [
|
||||
{
|
||||
var: "s",
|
||||
expr: 1,
|
||||
},
|
||||
],
|
||||
elseif: null,
|
||||
else: null,
|
||||
},
|
||||
{
|
||||
decl: "if",
|
||||
condition: {
|
||||
operator: "||",
|
||||
left: { id: "a" },
|
||||
right: { id: "b" },
|
||||
},
|
||||
then: [
|
||||
{
|
||||
var: "s",
|
||||
expr: 1,
|
||||
},
|
||||
],
|
||||
elseif: null,
|
||||
else: [
|
||||
{
|
||||
var: "x",
|
||||
expr: 2,
|
||||
},
|
||||
],
|
||||
},
|
||||
{
|
||||
decl: "if",
|
||||
condition: 0,
|
||||
then: [
|
||||
{
|
||||
assignment: "s",
|
||||
expr: 1,
|
||||
},
|
||||
],
|
||||
elseif: [
|
||||
{
|
||||
decl: "elseif",
|
||||
condition: 1,
|
||||
then: [
|
||||
{
|
||||
assignment: "s",
|
||||
expr: 2,
|
||||
},
|
||||
],
|
||||
},
|
||||
{
|
||||
decl: "elseif",
|
||||
condition: 2,
|
||||
then: [
|
||||
{
|
||||
assignment: "s",
|
||||
expr: 3,
|
||||
},
|
||||
],
|
||||
},
|
||||
{
|
||||
decl: "elseif",
|
||||
condition: 3,
|
||||
then: [
|
||||
{
|
||||
assignment: "s",
|
||||
expr: 4,
|
||||
},
|
||||
],
|
||||
},
|
||||
],
|
||||
else: [
|
||||
{
|
||||
assignment: "s",
|
||||
expr: 5,
|
||||
},
|
||||
],
|
||||
},
|
||||
{
|
||||
decl: "if",
|
||||
condition: 0,
|
||||
then: [
|
||||
{
|
||||
assignment: "s",
|
||||
expr: 1,
|
||||
},
|
||||
],
|
||||
elseif: [
|
||||
{
|
||||
decl: "elseif",
|
||||
condition: 1,
|
||||
then: [
|
||||
{
|
||||
assignment: "s",
|
||||
expr: 2,
|
||||
},
|
||||
],
|
||||
},
|
||||
],
|
||||
else: null,
|
||||
},
|
||||
]);
|
||||
});
|
||||
|
||||
it("should parse if declaration with error", function () {
|
||||
let parser = new Parser("if foo == 1 then a = 1 endif");
|
||||
expect(() => parser.parse()).toThrow(new Error(Errors.if));
|
||||
|
||||
parser = new Parser("if (foo == 1 then a = 1 endif");
|
||||
expect(() => parser.parse()).toThrow(new Error(Errors.if));
|
||||
|
||||
parser = new Parser(
|
||||
"if (foo == 1) then a = 1 elseiff (foo == 2) then a = 2 endif"
|
||||
);
|
||||
expect(() => parser.parse()).toThrow(new Error(Errors.if));
|
||||
|
||||
parser = new Parser(
|
||||
"if (foo == 1) then a = 1 elseif (foo == 2) then a = 2 end"
|
||||
);
|
||||
expect(() => parser.parse()).toThrow(new Error(Errors.if));
|
||||
});
|
||||
});
|
||||
});
|
Loading…
Reference in New Issue
Block a user