xfa_formcalc_spec.js 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683
  1. /**
  2. * @licstart The following is the entire license notice for the
  3. * Javascript code in this page
  4. *
  5. * Copyright 2021 Mozilla Foundation
  6. *
  7. * Licensed under the Apache License, Version 2.0 (the "License");
  8. * you may not use this file except in compliance with the License.
  9. * You may obtain a copy of the License at
  10. *
  11. * http://www.apache.org/licenses/LICENSE-2.0
  12. *
  13. * Unless required by applicable law or agreed to in writing, software
  14. * distributed under the License is distributed on an "AS IS" BASIS,
  15. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  16. * See the License for the specific language governing permissions and
  17. * limitations under the License.
  18. *
  19. * @licend The above is the entire license notice for the
  20. * Javascript code in this page
  21. */
  22. "use strict";
  23. var _formcalc_parser = require("../../core/xfa/formcalc_parser.js");
  24. var _formcalc_lexer = require("../../core/xfa/formcalc_lexer.js");
  25. describe("FormCalc expression parser", function () {
  26. const EOF = new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.eof);
  27. describe("FormCalc lexer", function () {
  28. it("should lex numbers", function () {
  29. const lexer = new _formcalc_lexer.Lexer("12 1.2345 .7 .12345 1e-2 1.2E+3 1e2 1.2E3 nan 12. 2.e3 infinity 99999999999999999 123456789.012345678 9e99999");
  30. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.number, 12));
  31. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.number, 1.2345));
  32. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.number, 0.7));
  33. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.number, 0.12345));
  34. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.number, 1e-2));
  35. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.number, 1.2e3));
  36. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.number, 1e2));
  37. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.number, 1.2e3));
  38. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.number, NaN));
  39. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.number, 12));
  40. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.number, 2e3));
  41. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.number, Infinity));
  42. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.number, 100000000000000000));
  43. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.number, 123456789.01234567));
  44. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.number, Infinity));
  45. expect(lexer.next()).toEqual(EOF);
  46. });
  47. it("should lex strings", function () {
  48. const lexer = new _formcalc_lexer.Lexer(`"hello world" "hello ""world" "hello ""world"" ""world""""hello""" "hello \\uabcdeh \\Uabcd \\u00000123abc" "a \\a \\ub \\Uc \\b"`);
  49. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.string, `hello world`));
  50. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.string, `hello "world`));
  51. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.string, `hello "world" "world""hello"`));
  52. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.string, `hello \uabcdeh \uabcd \u0123abc`));
  53. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.string, `a \\a \\ub \\Uc \\b`));
  54. expect(lexer.next()).toEqual(EOF);
  55. });
  56. it("should lex operators", function () {
  57. const lexer = new _formcalc_lexer.Lexer("( , ) <= <> = == >= < > / * . .* .# [ ] & |");
  58. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.leftParen));
  59. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.comma));
  60. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.rightParen));
  61. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.le));
  62. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.ne));
  63. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.assign));
  64. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.eq));
  65. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.ge));
  66. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.lt));
  67. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.gt));
  68. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.divide));
  69. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.times));
  70. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.dot));
  71. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.dotStar));
  72. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.dotHash));
  73. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.leftBracket));
  74. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.rightBracket));
  75. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.and));
  76. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.or));
  77. expect(lexer.next()).toEqual(EOF);
  78. });
  79. it("should skip comments", function () {
  80. const lexer = new _formcalc_lexer.Lexer(`
  81. \t\t 1 \r\n\r\n
  82. ; blah blah blah
  83. 2
  84. // blah blah blah blah blah
  85. 3
  86. `);
  87. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.number, 1));
  88. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.number, 2));
  89. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.number, 3));
  90. expect(lexer.next()).toEqual(EOF);
  91. });
  92. it("should lex identifiers", function () {
  93. const lexer = new _formcalc_lexer.Lexer("eq for fore while continue hello こんにちは世界 $!hello今日は12今日は");
  94. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.eq));
  95. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.for));
  96. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.identifier, "fore"));
  97. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.while));
  98. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.continue));
  99. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.identifier, "hello"));
  100. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.identifier, "こんにちは世界"));
  101. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.identifier, "$"));
  102. expect(lexer.next()).toEqual(new _formcalc_lexer.Token(_formcalc_lexer.TOKEN.identifier, "!hello今日は12今日は"));
  103. expect(lexer.next()).toEqual(EOF);
  104. });
  105. });
  106. describe("FormCalc parser", function () {
  107. it("should parse basic arithmetic expression", function () {
  108. const parser = new _formcalc_parser.Parser("1 + 2 * 3");
  109. expect(parser.parse().dump()[0]).toEqual(7);
  110. });
  111. it("should parse basic arithmetic expression with the same operator", function () {
  112. const parser = new _formcalc_parser.Parser("1 + a + 3");
  113. expect(parser.parse().dump()[0]).toEqual({
  114. operator: "+",
  115. left: {
  116. operator: "+",
  117. left: 1,
  118. right: {
  119. id: "a"
  120. }
  121. },
  122. right: 3
  123. });
  124. });
  125. it("should parse expressions with unary operators", function () {
  126. const parser = new _formcalc_parser.Parser(`
  127. s = +x + 1
  128. t = -+u * 2
  129. t = +-u * 2
  130. u = -foo()
  131. `);
  132. expect(parser.parse().dump()).toEqual([{
  133. assignment: "s",
  134. expr: {
  135. operator: "+",
  136. left: {
  137. operator: "+",
  138. arg: {
  139. id: "x"
  140. }
  141. },
  142. right: 1
  143. }
  144. }, {
  145. assignment: "t",
  146. expr: {
  147. operator: "*",
  148. left: {
  149. operator: "-",
  150. arg: {
  151. operator: "+",
  152. arg: {
  153. id: "u"
  154. }
  155. }
  156. },
  157. right: 2
  158. }
  159. }, {
  160. assignment: "t",
  161. expr: {
  162. operator: "*",
  163. left: {
  164. operator: "+",
  165. arg: {
  166. operator: "-",
  167. arg: {
  168. id: "u"
  169. }
  170. }
  171. },
  172. right: 2
  173. }
  174. }, {
  175. assignment: "u",
  176. expr: {
  177. operator: "-",
  178. arg: {
  179. callee: {
  180. id: "foo"
  181. },
  182. params: []
  183. }
  184. }
  185. }]);
  186. });
  187. it("should parse basic expression with a string", function () {
  188. const parser = new _formcalc_parser.Parser(`(5 - "abc") * 3`);
  189. expect(parser.parse().dump()[0]).toEqual(15);
  190. });
  191. it("should parse basic expression with a calls", function () {
  192. const parser = new _formcalc_parser.Parser(`foo(2, 3, a & b) or c * d + 1.234 / e`);
  193. expect(parser.parse().dump()[0]).toEqual({
  194. operator: "||",
  195. left: {
  196. callee: {
  197. id: "foo"
  198. },
  199. params: [2, 3, {
  200. operator: "&&",
  201. left: {
  202. id: "a"
  203. },
  204. right: {
  205. id: "b"
  206. }
  207. }]
  208. },
  209. right: {
  210. operator: "+",
  211. left: {
  212. operator: "*",
  213. left: {
  214. id: "c"
  215. },
  216. right: {
  217. id: "d"
  218. }
  219. },
  220. right: {
  221. operator: "/",
  222. left: 1.234,
  223. right: {
  224. id: "e"
  225. }
  226. }
  227. }
  228. });
  229. });
  230. it("should parse basic expression with a subscript", function () {
  231. let parser = new _formcalc_parser.Parser(`こんにちは世界[-0]`);
  232. let dump = parser.parse().dump()[0];
  233. expect(dump).toEqual({
  234. operand: {
  235. id: "こんにちは世界"
  236. },
  237. index: -0
  238. });
  239. expect(Object.is(-0, dump.index)).toBe(true);
  240. parser = new _formcalc_parser.Parser(`こんにちは世界[+0]`);
  241. dump = parser.parse().dump()[0];
  242. expect(dump).toEqual({
  243. operand: {
  244. id: "こんにちは世界"
  245. },
  246. index: +0
  247. });
  248. expect(Object.is(+0, dump.index)).toBe(true);
  249. parser = new _formcalc_parser.Parser(`こんにちは世界[*]`);
  250. expect(parser.parse().dump()[0]).toEqual({
  251. operand: {
  252. id: "こんにちは世界"
  253. },
  254. index: {
  255. special: "*"
  256. }
  257. });
  258. });
  259. it("should parse basic expression with dots", function () {
  260. const parser = new _formcalc_parser.Parser("a.b.c.#d..e.f..g.*");
  261. expect(parser.parse().dump()[0]).toEqual({
  262. operator: ".",
  263. left: {
  264. id: "a"
  265. },
  266. right: {
  267. operator: ".",
  268. left: {
  269. id: "b"
  270. },
  271. right: {
  272. operator: ".#",
  273. left: {
  274. id: "c"
  275. },
  276. right: {
  277. operator: "..",
  278. left: {
  279. id: "d"
  280. },
  281. right: {
  282. operator: ".",
  283. left: {
  284. id: "e"
  285. },
  286. right: {
  287. operator: "..",
  288. left: {
  289. id: "f"
  290. },
  291. right: {
  292. operator: ".",
  293. left: {
  294. id: "g"
  295. },
  296. right: {
  297. special: "*"
  298. }
  299. }
  300. }
  301. }
  302. }
  303. }
  304. }
  305. });
  306. });
  307. it("should parse var declaration with error", function () {
  308. let parser = new _formcalc_parser.Parser("var 123 = a");
  309. expect(() => parser.parse()).toThrow(new Error(_formcalc_parser.Errors.var));
  310. parser = new _formcalc_parser.Parser(`var "123" = a`);
  311. expect(() => parser.parse()).toThrow(new Error(_formcalc_parser.Errors.var));
  312. parser = new _formcalc_parser.Parser(`var for var a`);
  313. expect(() => parser.parse()).toThrow(new Error(_formcalc_parser.Errors.var));
  314. });
  315. it("should parse for declaration with a step", function () {
  316. const parser = new _formcalc_parser.Parser(`
  317. var s = 0
  318. for var i = 1 upto 10 + x step 1 do
  319. s = s + i * 2
  320. endfor`);
  321. expect(parser.parse().dump()).toEqual([{
  322. var: "s",
  323. expr: 0
  324. }, {
  325. decl: "for",
  326. assignment: {
  327. var: "i",
  328. expr: 1
  329. },
  330. type: "upto",
  331. end: {
  332. operator: "+",
  333. left: 10,
  334. right: {
  335. id: "x"
  336. }
  337. },
  338. step: 1,
  339. body: [{
  340. assignment: "s",
  341. expr: {
  342. operator: "+",
  343. left: {
  344. id: "s"
  345. },
  346. right: {
  347. operator: "*",
  348. left: {
  349. id: "i"
  350. },
  351. right: 2
  352. }
  353. }
  354. }]
  355. }]);
  356. });
  357. it("should parse for declaration without a step", function () {
  358. const parser = new _formcalc_parser.Parser(`
  359. for i = 1 + 2 downto 10 do
  360. s = foo()
  361. endfor`);
  362. expect(parser.parse().dump()).toEqual([{
  363. decl: "for",
  364. assignment: {
  365. assignment: "i",
  366. expr: 3
  367. },
  368. type: "downto",
  369. end: 10,
  370. step: null,
  371. body: [{
  372. assignment: "s",
  373. expr: {
  374. callee: {
  375. id: "foo"
  376. },
  377. params: []
  378. }
  379. }]
  380. }]);
  381. });
  382. it("should parse for declaration with error", function () {
  383. let parser = new _formcalc_parser.Parser("for 123 = i upto 1 do a = 1 endfor");
  384. expect(() => parser.parse()).toThrow(new Error(_formcalc_parser.Errors.assignment));
  385. parser = new _formcalc_parser.Parser("for var 123 = i upto 1 do a = 1 endfor");
  386. expect(() => parser.parse()).toThrow(new Error(_formcalc_parser.Errors.assignment));
  387. parser = new _formcalc_parser.Parser("for var i = 123 upt 1 do a = 1 endfor");
  388. expect(() => parser.parse()).toThrow(new Error(_formcalc_parser.Errors.for));
  389. parser = new _formcalc_parser.Parser("for var i = 123 var 1 do a = 1 endfor");
  390. expect(() => parser.parse()).toThrow(new Error(_formcalc_parser.Errors.for));
  391. parser = new _formcalc_parser.Parser("for var i = 123 upto 1 step for var j = 1 do endfor do a = 1 endfor");
  392. expect(() => parser.parse()).toThrow(new Error(_formcalc_parser.Errors.for));
  393. parser = new _formcalc_parser.Parser("for var i = 123 downto 1 do a = 1 endfunc");
  394. expect(() => parser.parse()).toThrow(new Error(_formcalc_parser.Errors.for));
  395. parser = new _formcalc_parser.Parser("for var i = 123 downto 1 do a = 1");
  396. expect(() => parser.parse()).toThrow(new Error(_formcalc_parser.Errors.for));
  397. });
  398. it("should parse foreach declaration", function () {
  399. const parser = new _formcalc_parser.Parser(`
  400. foreach i in (a, b, c, d) do
  401. s = foo()[i]
  402. endfor`);
  403. expect(parser.parse().dump()).toEqual([{
  404. decl: "foreach",
  405. id: "i",
  406. params: [{
  407. id: "a"
  408. }, {
  409. id: "b"
  410. }, {
  411. id: "c"
  412. }, {
  413. id: "d"
  414. }],
  415. body: [{
  416. assignment: "s",
  417. expr: {
  418. operand: {
  419. callee: {
  420. id: "foo"
  421. },
  422. params: []
  423. },
  424. index: {
  425. id: "i"
  426. }
  427. }
  428. }]
  429. }]);
  430. });
  431. it("should parse foreach declaration with error", function () {
  432. let parser = new _formcalc_parser.Parser("foreach 123 in (1, 2, 3) do a = 1 endfor");
  433. expect(() => parser.parse()).toThrow(new Error(_formcalc_parser.Errors.foreach));
  434. parser = new _formcalc_parser.Parser("foreach foo in 1, 2, 3) do a = 1 endfor");
  435. expect(() => parser.parse()).toThrow(new Error(_formcalc_parser.Errors.foreach));
  436. parser = new _formcalc_parser.Parser("foreach foo in (1, 2, 3 do a = 1 endfor");
  437. expect(() => parser.parse()).toThrow(new Error(_formcalc_parser.Errors.params));
  438. parser = new _formcalc_parser.Parser("foreach foo in (1, 2 3) do a = 1 endfor");
  439. expect(() => parser.parse()).toThrow(new Error(_formcalc_parser.Errors.params));
  440. parser = new _formcalc_parser.Parser("foreach foo in (1, 2, 3) od a = 1 endfor");
  441. expect(() => parser.parse()).toThrow(new Error(_formcalc_parser.Errors.foreach));
  442. parser = new _formcalc_parser.Parser("foreach foo in (1, 2, 3) do a = 1 endforeach");
  443. expect(() => parser.parse()).toThrow(new Error(_formcalc_parser.Errors.foreach));
  444. parser = new _formcalc_parser.Parser("foreach foo in (1, 2, 3) do a = 1 123");
  445. expect(() => parser.parse()).toThrow(new Error(_formcalc_parser.Errors.foreach));
  446. });
  447. it("should parse while declaration", function () {
  448. const parser = new _formcalc_parser.Parser(`
  449. while (1) do
  450. if (0) then
  451. break
  452. else
  453. continue
  454. endif
  455. endwhile
  456. `);
  457. expect(parser.parse().dump()).toEqual([{
  458. decl: "while",
  459. condition: 1,
  460. body: [{
  461. decl: "if",
  462. condition: 0,
  463. then: [{
  464. special: "break"
  465. }],
  466. elseif: null,
  467. else: [{
  468. special: "continue"
  469. }]
  470. }]
  471. }]);
  472. });
  473. it("should parse while declaration with error", function () {
  474. let parser = new _formcalc_parser.Parser("while a == 1 do a = 2 endwhile");
  475. expect(() => parser.parse()).toThrow(new Error(_formcalc_parser.Errors.while));
  476. parser = new _formcalc_parser.Parser("while (a == 1 do a = 2 endwhile");
  477. expect(() => parser.parse()).toThrow(new Error(_formcalc_parser.Errors.while));
  478. parser = new _formcalc_parser.Parser("while (a == 1) var a = 2 endwhile");
  479. expect(() => parser.parse()).toThrow(new Error(_formcalc_parser.Errors.while));
  480. parser = new _formcalc_parser.Parser("while (a == 1) do var a = 2 end");
  481. expect(() => parser.parse()).toThrow(new Error(_formcalc_parser.Errors.while));
  482. });
  483. it("should parse do declaration", function () {
  484. const parser = new _formcalc_parser.Parser(`
  485. do
  486. x = 1
  487. ; a comment in the middle of the block
  488. y = 2
  489. end
  490. `);
  491. expect(parser.parse().dump()).toEqual([{
  492. decl: "block",
  493. body: [{
  494. assignment: "x",
  495. expr: 1
  496. }, {
  497. assignment: "y",
  498. expr: 2
  499. }]
  500. }]);
  501. });
  502. it("should parse do declaration with error", function () {
  503. const parser = new _formcalc_parser.Parser(`
  504. do
  505. x = 1
  506. y = 2
  507. endfunc
  508. `);
  509. expect(() => parser.parse()).toThrow(new Error(_formcalc_parser.Errors.block));
  510. });
  511. it("should parse func declaration", function () {
  512. const parser = new _formcalc_parser.Parser(`
  513. func こんにちは世界123(a, b) do
  514. a + b
  515. endfunc
  516. `);
  517. expect(parser.parse().dump()).toEqual([{
  518. func: "こんにちは世界123",
  519. params: ["a", "b"],
  520. body: [{
  521. operator: "+",
  522. left: {
  523. id: "a"
  524. },
  525. right: {
  526. id: "b"
  527. }
  528. }]
  529. }]);
  530. });
  531. it("should parse func declaration with error", function () {
  532. let parser = new _formcalc_parser.Parser("func 123(a, b) do a = 1 endfunc");
  533. expect(() => parser.parse()).toThrow(new Error(_formcalc_parser.Errors.func));
  534. parser = new _formcalc_parser.Parser("func foo(a, b) for a = 1 endfunc");
  535. expect(() => parser.parse()).toThrow(new Error(_formcalc_parser.Errors.func));
  536. parser = new _formcalc_parser.Parser("func foo(a, b) do a = 1 endfun");
  537. expect(() => parser.parse()).toThrow(new Error(_formcalc_parser.Errors.func));
  538. parser = new _formcalc_parser.Parser("func foo(a, b, c do a = 1 endfunc");
  539. expect(() => parser.parse()).toThrow(new Error(_formcalc_parser.Errors.func));
  540. parser = new _formcalc_parser.Parser("func foo(a, b, 123) do a = 1 endfunc");
  541. expect(() => parser.parse()).toThrow(new Error(_formcalc_parser.Errors.func));
  542. });
  543. it("should parse if declaration", function () {
  544. const parser = new _formcalc_parser.Parser(`
  545. if (a & b) then
  546. var s = 1
  547. endif
  548. if (a or b) then
  549. var s = 1
  550. else
  551. var x = 2
  552. endif
  553. if (0) then
  554. s = 1
  555. elseif (1) then
  556. s = 2
  557. elseif (2) then
  558. s = 3
  559. elseif (3) then
  560. s = 4
  561. else
  562. s = 5
  563. endif
  564. // a comment
  565. if (0) then
  566. s = 1
  567. elseif (1) then
  568. s = 2
  569. endif
  570. `);
  571. expect(parser.parse().dump()).toEqual([{
  572. decl: "if",
  573. condition: {
  574. operator: "&&",
  575. left: {
  576. id: "a"
  577. },
  578. right: {
  579. id: "b"
  580. }
  581. },
  582. then: [{
  583. var: "s",
  584. expr: 1
  585. }],
  586. elseif: null,
  587. else: null
  588. }, {
  589. decl: "if",
  590. condition: {
  591. operator: "||",
  592. left: {
  593. id: "a"
  594. },
  595. right: {
  596. id: "b"
  597. }
  598. },
  599. then: [{
  600. var: "s",
  601. expr: 1
  602. }],
  603. elseif: null,
  604. else: [{
  605. var: "x",
  606. expr: 2
  607. }]
  608. }, {
  609. decl: "if",
  610. condition: 0,
  611. then: [{
  612. assignment: "s",
  613. expr: 1
  614. }],
  615. elseif: [{
  616. decl: "elseif",
  617. condition: 1,
  618. then: [{
  619. assignment: "s",
  620. expr: 2
  621. }]
  622. }, {
  623. decl: "elseif",
  624. condition: 2,
  625. then: [{
  626. assignment: "s",
  627. expr: 3
  628. }]
  629. }, {
  630. decl: "elseif",
  631. condition: 3,
  632. then: [{
  633. assignment: "s",
  634. expr: 4
  635. }]
  636. }],
  637. else: [{
  638. assignment: "s",
  639. expr: 5
  640. }]
  641. }, {
  642. decl: "if",
  643. condition: 0,
  644. then: [{
  645. assignment: "s",
  646. expr: 1
  647. }],
  648. elseif: [{
  649. decl: "elseif",
  650. condition: 1,
  651. then: [{
  652. assignment: "s",
  653. expr: 2
  654. }]
  655. }],
  656. else: null
  657. }]);
  658. });
  659. it("should parse if declaration with error", function () {
  660. let parser = new _formcalc_parser.Parser("if foo == 1 then a = 1 endif");
  661. expect(() => parser.parse()).toThrow(new Error(_formcalc_parser.Errors.if));
  662. parser = new _formcalc_parser.Parser("if (foo == 1 then a = 1 endif");
  663. expect(() => parser.parse()).toThrow(new Error(_formcalc_parser.Errors.if));
  664. parser = new _formcalc_parser.Parser("if (foo == 1) then a = 1 elseiff (foo == 2) then a = 2 endif");
  665. expect(() => parser.parse()).toThrow(new Error(_formcalc_parser.Errors.if));
  666. parser = new _formcalc_parser.Parser("if (foo == 1) then a = 1 elseif (foo == 2) then a = 2 end");
  667. expect(() => parser.parse()).toThrow(new Error(_formcalc_parser.Errors.if));
  668. });
  669. });
  670. });