reqon-dsl 0.2.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (388) hide show
  1. package/.claude/settings.local.json +31 -0
  2. package/.claude/skills/api-integration.md +125 -0
  3. package/.claude/skills/database-schema.md +51 -0
  4. package/.claude/skills/dsl-design.md +80 -0
  5. package/.claude/skills/property-testing.md +143 -0
  6. package/.claude/skills/reqon/SKILL.md +44 -0
  7. package/.claude/skills/reqon/references/examples.md +206 -0
  8. package/.claude/skills/reqon/references/syntax.md +263 -0
  9. package/.claude/skills/vscode-extension.md +113 -0
  10. package/.github/dependabot.yml +32 -0
  11. package/.github/pull_request_template.md +21 -0
  12. package/.github/workflows/ci.yml +174 -0
  13. package/.github/workflows/release.yml +73 -0
  14. package/CLAUDE.md +72 -0
  15. package/CONTRIBUTING.md +161 -0
  16. package/README.md +235 -0
  17. package/TODO.md +51 -0
  18. package/dist/ast/index.d.ts +1 -0
  19. package/dist/ast/index.js +1 -0
  20. package/dist/ast/nodes.d.ts +237 -0
  21. package/dist/ast/nodes.js +12 -0
  22. package/dist/auth/auth.test.d.ts +1 -0
  23. package/dist/auth/auth.test.js +255 -0
  24. package/dist/auth/circuit-breaker.d.ts +115 -0
  25. package/dist/auth/circuit-breaker.js +267 -0
  26. package/dist/auth/credentials.d.ts +91 -0
  27. package/dist/auth/credentials.js +169 -0
  28. package/dist/auth/index.d.ts +5 -0
  29. package/dist/auth/index.js +8 -0
  30. package/dist/auth/oauth2-provider.d.ts +41 -0
  31. package/dist/auth/oauth2-provider.js +131 -0
  32. package/dist/auth/rate-limiter.d.ts +61 -0
  33. package/dist/auth/rate-limiter.js +380 -0
  34. package/dist/auth/token-store.d.ts +30 -0
  35. package/dist/auth/token-store.js +148 -0
  36. package/dist/auth/types.d.ts +142 -0
  37. package/dist/auth/types.js +1 -0
  38. package/dist/cli.d.ts +2 -0
  39. package/dist/cli.js +270 -0
  40. package/dist/errors/errors.test.d.ts +1 -0
  41. package/dist/errors/errors.test.js +165 -0
  42. package/dist/errors/index.d.ts +83 -0
  43. package/dist/errors/index.js +159 -0
  44. package/dist/execution/execution.test.d.ts +1 -0
  45. package/dist/execution/execution.test.js +246 -0
  46. package/dist/execution/index.d.ts +4 -0
  47. package/dist/execution/index.js +2 -0
  48. package/dist/execution/state.d.ts +136 -0
  49. package/dist/execution/state.js +82 -0
  50. package/dist/execution/store.d.ts +52 -0
  51. package/dist/execution/store.js +120 -0
  52. package/dist/index.d.ts +27 -0
  53. package/dist/index.js +57 -0
  54. package/dist/integration.test.d.ts +1 -0
  55. package/dist/integration.test.js +168 -0
  56. package/dist/interpreter/context.d.ts +15 -0
  57. package/dist/interpreter/context.js +29 -0
  58. package/dist/interpreter/evaluator.d.ts +5 -0
  59. package/dist/interpreter/evaluator.js +223 -0
  60. package/dist/interpreter/evaluator.test.d.ts +1 -0
  61. package/dist/interpreter/evaluator.test.js +512 -0
  62. package/dist/interpreter/executor.d.ts +131 -0
  63. package/dist/interpreter/executor.js +663 -0
  64. package/dist/interpreter/fetch-handler.d.ts +43 -0
  65. package/dist/interpreter/fetch-handler.js +203 -0
  66. package/dist/interpreter/http.d.ts +57 -0
  67. package/dist/interpreter/http.js +210 -0
  68. package/dist/interpreter/http.test.d.ts +1 -0
  69. package/dist/interpreter/http.test.js +299 -0
  70. package/dist/interpreter/index.d.ts +7 -0
  71. package/dist/interpreter/index.js +7 -0
  72. package/dist/interpreter/pagination.d.ts +63 -0
  73. package/dist/interpreter/pagination.js +155 -0
  74. package/dist/interpreter/progress.test.d.ts +1 -0
  75. package/dist/interpreter/progress.test.js +216 -0
  76. package/dist/interpreter/schema-matcher.d.ts +16 -0
  77. package/dist/interpreter/schema-matcher.js +136 -0
  78. package/dist/interpreter/schema-matcher.test.d.ts +1 -0
  79. package/dist/interpreter/schema-matcher.test.js +122 -0
  80. package/dist/interpreter/signals.d.ts +57 -0
  81. package/dist/interpreter/signals.js +73 -0
  82. package/dist/interpreter/step-handlers/for-handler.d.ts +17 -0
  83. package/dist/interpreter/step-handlers/for-handler.js +51 -0
  84. package/dist/interpreter/step-handlers/index.d.ts +8 -0
  85. package/dist/interpreter/step-handlers/index.js +8 -0
  86. package/dist/interpreter/step-handlers/map-handler.d.ts +10 -0
  87. package/dist/interpreter/step-handlers/map-handler.js +20 -0
  88. package/dist/interpreter/step-handlers/match-handler.d.ts +27 -0
  89. package/dist/interpreter/step-handlers/match-handler.js +61 -0
  90. package/dist/interpreter/step-handlers/store-handler.d.ts +13 -0
  91. package/dist/interpreter/step-handlers/store-handler.js +66 -0
  92. package/dist/interpreter/step-handlers/types.d.ts +15 -0
  93. package/dist/interpreter/step-handlers/types.js +1 -0
  94. package/dist/interpreter/step-handlers/validate-handler.d.ts +10 -0
  95. package/dist/interpreter/step-handlers/validate-handler.js +26 -0
  96. package/dist/interpreter/step-handlers/webhook-handler.d.ts +36 -0
  97. package/dist/interpreter/step-handlers/webhook-handler.js +104 -0
  98. package/dist/lexer/index.d.ts +10 -0
  99. package/dist/lexer/index.js +12 -0
  100. package/dist/lexer/lexer.d.ts +24 -0
  101. package/dist/lexer/lexer.js +264 -0
  102. package/dist/lexer/lexer.test.d.ts +1 -0
  103. package/dist/lexer/lexer.test.js +259 -0
  104. package/dist/lexer/tokens.d.ts +69 -0
  105. package/dist/lexer/tokens.js +146 -0
  106. package/dist/loader/index.d.ts +36 -0
  107. package/dist/loader/index.js +220 -0
  108. package/dist/loader/loader.test.d.ts +1 -0
  109. package/dist/loader/loader.test.js +287 -0
  110. package/dist/oas/index.d.ts +4 -0
  111. package/dist/oas/index.js +2 -0
  112. package/dist/oas/loader.d.ts +21 -0
  113. package/dist/oas/loader.js +82 -0
  114. package/dist/oas/oas.test.d.ts +1 -0
  115. package/dist/oas/oas.test.js +218 -0
  116. package/dist/oas/validator.d.ts +12 -0
  117. package/dist/oas/validator.js +227 -0
  118. package/dist/parser/base.d.ts +33 -0
  119. package/dist/parser/base.js +97 -0
  120. package/dist/parser/expressions.d.ts +27 -0
  121. package/dist/parser/expressions.js +248 -0
  122. package/dist/parser/expressions.test.d.ts +1 -0
  123. package/dist/parser/expressions.test.js +378 -0
  124. package/dist/parser/index.d.ts +3 -0
  125. package/dist/parser/index.js +3 -0
  126. package/dist/parser/match.test.d.ts +1 -0
  127. package/dist/parser/match.test.js +254 -0
  128. package/dist/parser/parser.d.ts +68 -0
  129. package/dist/parser/parser.js +1229 -0
  130. package/dist/parser/parser.test.d.ts +1 -0
  131. package/dist/parser/parser.test.js +333 -0
  132. package/dist/parser/schedule.test.d.ts +1 -0
  133. package/dist/parser/schedule.test.js +241 -0
  134. package/dist/plugin.d.ts +35 -0
  135. package/dist/plugin.js +68 -0
  136. package/dist/scheduler/cron-parser.d.ts +32 -0
  137. package/dist/scheduler/cron-parser.js +198 -0
  138. package/dist/scheduler/cron-parser.test.d.ts +1 -0
  139. package/dist/scheduler/cron-parser.test.js +188 -0
  140. package/dist/scheduler/index.d.ts +3 -0
  141. package/dist/scheduler/index.js +2 -0
  142. package/dist/scheduler/scheduler.d.ts +81 -0
  143. package/dist/scheduler/scheduler.js +376 -0
  144. package/dist/scheduler/types.d.ts +65 -0
  145. package/dist/scheduler/types.js +1 -0
  146. package/dist/stores/factory.d.ts +36 -0
  147. package/dist/stores/factory.js +73 -0
  148. package/dist/stores/file.d.ts +60 -0
  149. package/dist/stores/file.js +173 -0
  150. package/dist/stores/file.test.d.ts +1 -0
  151. package/dist/stores/file.test.js +165 -0
  152. package/dist/stores/index.d.ts +6 -0
  153. package/dist/stores/index.js +5 -0
  154. package/dist/stores/memory.d.ts +19 -0
  155. package/dist/stores/memory.js +51 -0
  156. package/dist/stores/memory.test.d.ts +1 -0
  157. package/dist/stores/memory.test.js +157 -0
  158. package/dist/stores/postgrest.d.ts +55 -0
  159. package/dist/stores/postgrest.js +217 -0
  160. package/dist/stores/stores.test.d.ts +1 -0
  161. package/dist/stores/stores.test.js +158 -0
  162. package/dist/stores/types.d.ts +31 -0
  163. package/dist/stores/types.js +26 -0
  164. package/dist/sync/index.d.ts +4 -0
  165. package/dist/sync/index.js +2 -0
  166. package/dist/sync/state.d.ts +69 -0
  167. package/dist/sync/state.js +66 -0
  168. package/dist/sync/store.d.ts +49 -0
  169. package/dist/sync/store.js +93 -0
  170. package/dist/sync/sync.test.d.ts +1 -0
  171. package/dist/sync/sync.test.js +221 -0
  172. package/dist/utils/async.d.ts +7 -0
  173. package/dist/utils/async.js +9 -0
  174. package/dist/utils/file.d.ts +38 -0
  175. package/dist/utils/file.js +92 -0
  176. package/dist/utils/index.d.ts +4 -0
  177. package/dist/utils/index.js +4 -0
  178. package/dist/utils/logger.d.ts +34 -0
  179. package/dist/utils/logger.js +39 -0
  180. package/dist/utils/path.d.ts +12 -0
  181. package/dist/utils/path.js +41 -0
  182. package/dist/webhook/index.d.ts +8 -0
  183. package/dist/webhook/index.js +7 -0
  184. package/dist/webhook/server.d.ts +84 -0
  185. package/dist/webhook/server.js +319 -0
  186. package/dist/webhook/store.d.ts +67 -0
  187. package/dist/webhook/store.js +193 -0
  188. package/dist/webhook/types.d.ts +88 -0
  189. package/dist/webhook/types.js +6 -0
  190. package/docusaurus/README.md +41 -0
  191. package/docusaurus/docs/advanced/execution-state.md +283 -0
  192. package/docusaurus/docs/advanced/extending-reqon.md +388 -0
  193. package/docusaurus/docs/advanced/multi-file-missions.md +250 -0
  194. package/docusaurus/docs/advanced/parallel-execution.md +353 -0
  195. package/docusaurus/docs/api-reference.md +443 -0
  196. package/docusaurus/docs/authentication/api-key.md +339 -0
  197. package/docusaurus/docs/authentication/basic.md +276 -0
  198. package/docusaurus/docs/authentication/bearer.md +282 -0
  199. package/docusaurus/docs/authentication/oauth2.md +317 -0
  200. package/docusaurus/docs/authentication/overview.md +251 -0
  201. package/docusaurus/docs/cli.md +229 -0
  202. package/docusaurus/docs/core-concepts/actions.md +286 -0
  203. package/docusaurus/docs/core-concepts/missions.md +264 -0
  204. package/docusaurus/docs/core-concepts/schemas.md +353 -0
  205. package/docusaurus/docs/core-concepts/sources.md +339 -0
  206. package/docusaurus/docs/core-concepts/stores.md +332 -0
  207. package/docusaurus/docs/dsl-syntax/expressions.md +361 -0
  208. package/docusaurus/docs/dsl-syntax/fetch.md +293 -0
  209. package/docusaurus/docs/dsl-syntax/for-loops.md +324 -0
  210. package/docusaurus/docs/dsl-syntax/map.md +345 -0
  211. package/docusaurus/docs/dsl-syntax/match.md +387 -0
  212. package/docusaurus/docs/dsl-syntax/pipelines.md +397 -0
  213. package/docusaurus/docs/dsl-syntax/validate.md +401 -0
  214. package/docusaurus/docs/error-handling/dead-letter-queues.md +399 -0
  215. package/docusaurus/docs/error-handling/flow-control.md +337 -0
  216. package/docusaurus/docs/error-handling/retry-strategies.md +368 -0
  217. package/docusaurus/docs/examples.md +488 -0
  218. package/docusaurus/docs/getting-started.md +256 -0
  219. package/docusaurus/docs/http/circuit-breaker.md +401 -0
  220. package/docusaurus/docs/http/incremental-sync.md +394 -0
  221. package/docusaurus/docs/http/pagination.md +361 -0
  222. package/docusaurus/docs/http/rate-limiting.md +383 -0
  223. package/docusaurus/docs/http/requests.md +328 -0
  224. package/docusaurus/docs/http/retry.md +402 -0
  225. package/docusaurus/docs/intro.md +90 -0
  226. package/docusaurus/docs/openapi/loading-specs.md +305 -0
  227. package/docusaurus/docs/openapi/operation-calls.md +314 -0
  228. package/docusaurus/docs/openapi/overview.md +212 -0
  229. package/docusaurus/docs/openapi/response-validation.md +344 -0
  230. package/docusaurus/docs/scheduling/cron.md +305 -0
  231. package/docusaurus/docs/scheduling/daemon-mode.md +317 -0
  232. package/docusaurus/docs/scheduling/intervals.md +289 -0
  233. package/docusaurus/docs/scheduling/overview.md +231 -0
  234. package/docusaurus/docs/stores/custom-adapters.md +376 -0
  235. package/docusaurus/docs/stores/file.md +236 -0
  236. package/docusaurus/docs/stores/memory.md +193 -0
  237. package/docusaurus/docs/stores/overview.md +274 -0
  238. package/docusaurus/docs/stores/postgrest.md +316 -0
  239. package/docusaurus/docusaurus.config.ts +148 -0
  240. package/docusaurus/package-lock.json +18029 -0
  241. package/docusaurus/package.json +47 -0
  242. package/docusaurus/sidebars.ts +155 -0
  243. package/docusaurus/src/components/HomepageFeatures/index.tsx +105 -0
  244. package/docusaurus/src/components/HomepageFeatures/styles.module.css +12 -0
  245. package/docusaurus/src/css/custom.css +169 -0
  246. package/docusaurus/src/pages/index.module.css +48 -0
  247. package/docusaurus/src/pages/index.tsx +110 -0
  248. package/docusaurus/src/pages/markdown-page.md +7 -0
  249. package/docusaurus/static/.nojekyll +0 -0
  250. package/docusaurus/static/img/docusaurus-social-card.jpg +0 -0
  251. package/docusaurus/static/img/docusaurus.png +0 -0
  252. package/docusaurus/static/img/favicon.ico +0 -0
  253. package/docusaurus/static/img/logo.svg +10 -0
  254. package/docusaurus/static/img/undraw_docusaurus_mountain.svg +171 -0
  255. package/docusaurus/static/img/undraw_docusaurus_react.svg +170 -0
  256. package/docusaurus/static/img/undraw_docusaurus_tree.svg +40 -0
  257. package/docusaurus/tsconfig.json +8 -0
  258. package/examples/README.md +112 -0
  259. package/examples/error-handling/README.md +150 -0
  260. package/examples/error-handling/payment-processor.vague +287 -0
  261. package/examples/github-sync/README.md +74 -0
  262. package/examples/github-sync/fetch-issues.vague +47 -0
  263. package/examples/github-sync/fetch-prs.vague +40 -0
  264. package/examples/github-sync/mission.vague +101 -0
  265. package/examples/github-sync/normalize.vague +70 -0
  266. package/examples/jsonplaceholder/README.md +28 -0
  267. package/examples/jsonplaceholder/posts.vague +48 -0
  268. package/examples/petstore/README.md +35 -0
  269. package/examples/petstore/openapi.yaml +97 -0
  270. package/examples/petstore/sync.vague +52 -0
  271. package/examples/temporal-comparison/README.md +297 -0
  272. package/examples/temporal-comparison/reconciliation.vague +355 -0
  273. package/examples/temporal-comparison/temporal/activities/index.ts +8 -0
  274. package/examples/temporal-comparison/temporal/activities/shipstation.ts +225 -0
  275. package/examples/temporal-comparison/temporal/activities/shopify.ts +257 -0
  276. package/examples/temporal-comparison/temporal/activities/storage.ts +198 -0
  277. package/examples/temporal-comparison/temporal/activities/stripe.ts +169 -0
  278. package/examples/temporal-comparison/temporal/activities/validation.ts +205 -0
  279. package/examples/temporal-comparison/temporal/client/schedule.ts +218 -0
  280. package/examples/temporal-comparison/temporal/config/retry.ts +63 -0
  281. package/examples/temporal-comparison/temporal/types/index.ts +129 -0
  282. package/examples/temporal-comparison/temporal/workers/main.ts +130 -0
  283. package/examples/temporal-comparison/temporal/workflows/orderReconciliation.ts +262 -0
  284. package/examples/xero/README.md +88 -0
  285. package/examples/xero/invoices.vague +189 -0
  286. package/package.json +40 -0
  287. package/src/api-integration.test.ts +954 -0
  288. package/src/ast/index.ts +1 -0
  289. package/src/ast/nodes.ts +310 -0
  290. package/src/auth/auth.test.ts +326 -0
  291. package/src/auth/circuit-breaker.test.ts +390 -0
  292. package/src/auth/circuit-breaker.ts +379 -0
  293. package/src/auth/credentials.test.ts +273 -0
  294. package/src/auth/credentials.ts +246 -0
  295. package/src/auth/index.ts +40 -0
  296. package/src/auth/oauth2-provider.ts +177 -0
  297. package/src/auth/rate-limiter.ts +459 -0
  298. package/src/auth/token-store.ts +177 -0
  299. package/src/auth/types.ts +159 -0
  300. package/src/benchmark/e2e.bench.ts +288 -0
  301. package/src/benchmark/evaluator.bench.ts +331 -0
  302. package/src/benchmark/fixtures.ts +295 -0
  303. package/src/benchmark/index.ts +108 -0
  304. package/src/benchmark/lexer.bench.ts +69 -0
  305. package/src/benchmark/parser.bench.ts +103 -0
  306. package/src/benchmark/resilience.bench.ts +193 -0
  307. package/src/benchmark/store.bench.ts +147 -0
  308. package/src/benchmark/utils.ts +230 -0
  309. package/src/cli.ts +313 -0
  310. package/src/errors/errors.test.ts +234 -0
  311. package/src/errors/index.ts +223 -0
  312. package/src/execution/execution.test.ts +307 -0
  313. package/src/execution/index.ts +21 -0
  314. package/src/execution/state.ts +207 -0
  315. package/src/execution/store.ts +188 -0
  316. package/src/index.ts +169 -0
  317. package/src/integration.test.ts +192 -0
  318. package/src/interpreter/context.ts +57 -0
  319. package/src/interpreter/evaluator.test.ts +796 -0
  320. package/src/interpreter/evaluator.ts +245 -0
  321. package/src/interpreter/executor.ts +946 -0
  322. package/src/interpreter/fetch-handler.ts +302 -0
  323. package/src/interpreter/http.test.ts +423 -0
  324. package/src/interpreter/http.ts +308 -0
  325. package/src/interpreter/index.ts +32 -0
  326. package/src/interpreter/pagination.ts +207 -0
  327. package/src/interpreter/progress.test.ts +276 -0
  328. package/src/interpreter/schema-matcher.test.ts +160 -0
  329. package/src/interpreter/schema-matcher.ts +168 -0
  330. package/src/interpreter/signals.ts +73 -0
  331. package/src/interpreter/step-handlers/for-handler.ts +65 -0
  332. package/src/interpreter/step-handlers/index.ts +17 -0
  333. package/src/interpreter/step-handlers/map-handler.ts +24 -0
  334. package/src/interpreter/step-handlers/match-handler.ts +101 -0
  335. package/src/interpreter/step-handlers/store-handler.ts +78 -0
  336. package/src/interpreter/step-handlers/types.ts +17 -0
  337. package/src/interpreter/step-handlers/validate-handler.ts +30 -0
  338. package/src/interpreter/step-handlers/webhook-handler.ts +142 -0
  339. package/src/lexer/index.ts +18 -0
  340. package/src/lexer/lexer.test.ts +316 -0
  341. package/src/lexer/tokens.ts +179 -0
  342. package/src/loader/index.ts +288 -0
  343. package/src/loader/loader.test.ts +360 -0
  344. package/src/oas/index.ts +4 -0
  345. package/src/oas/loader.ts +126 -0
  346. package/src/oas/oas.test.ts +254 -0
  347. package/src/oas/validator.ts +299 -0
  348. package/src/parser/base.ts +124 -0
  349. package/src/parser/expressions.test.ts +525 -0
  350. package/src/parser/expressions.ts +314 -0
  351. package/src/parser/index.ts +3 -0
  352. package/src/parser/match.test.ts +296 -0
  353. package/src/parser/parser.test.ts +739 -0
  354. package/src/parser/parser.ts +1469 -0
  355. package/src/parser/schedule.test.ts +287 -0
  356. package/src/parser/webhook.test.ts +248 -0
  357. package/src/plugin.ts +83 -0
  358. package/src/scheduler/cron-parser.test.ts +236 -0
  359. package/src/scheduler/cron-parser.ts +236 -0
  360. package/src/scheduler/index.ts +10 -0
  361. package/src/scheduler/scheduler.ts +443 -0
  362. package/src/scheduler/types.ts +71 -0
  363. package/src/stores/factory.ts +104 -0
  364. package/src/stores/file.test.ts +276 -0
  365. package/src/stores/file.ts +211 -0
  366. package/src/stores/index.ts +6 -0
  367. package/src/stores/memory.test.ts +238 -0
  368. package/src/stores/memory.ts +63 -0
  369. package/src/stores/postgrest.test.ts +488 -0
  370. package/src/stores/postgrest.ts +263 -0
  371. package/src/stores/stores.test.ts +197 -0
  372. package/src/stores/types.ts +58 -0
  373. package/src/sync/index.ts +16 -0
  374. package/src/sync/state.ts +126 -0
  375. package/src/sync/store.ts +139 -0
  376. package/src/sync/sync.test.ts +271 -0
  377. package/src/utils/async.ts +10 -0
  378. package/src/utils/file.ts +106 -0
  379. package/src/utils/index.ts +14 -0
  380. package/src/utils/logger.ts +53 -0
  381. package/src/utils/path.ts +47 -0
  382. package/src/webhook/index.ts +15 -0
  383. package/src/webhook/server.test.ts +253 -0
  384. package/src/webhook/server.ts +389 -0
  385. package/src/webhook/store.ts +239 -0
  386. package/src/webhook/types.ts +93 -0
  387. package/tsconfig.json +17 -0
  388. package/vitest.config.ts +39 -0
@@ -0,0 +1,124 @@
1
+ import { TokenType, type Token } from 'vague-lang';
2
+ import { ReqonTokenType } from '../lexer/tokens.js';
3
+ import { ParseError, type ErrorContext } from '../errors/index.js';
4
+
5
+ // Token type can be Vague's TokenType, Reqon's ReqonTokenType, or a plugin string
6
+ type AnyTokenType = TokenType | ReqonTokenType | string;
7
+
8
+ export class ReqonParserBase {
9
+ protected tokens: Token[];
10
+ protected pos = 0;
11
+ protected source?: string;
12
+ protected filePath?: string;
13
+
14
+ constructor(tokens: Token[], source?: string, filePath?: string) {
15
+ this.tokens = tokens.filter((t) => t.type !== TokenType.NEWLINE);
16
+ this.source = source;
17
+ this.filePath = filePath;
18
+ }
19
+
20
+ protected peek(): Token {
21
+ return this.tokens[this.pos];
22
+ }
23
+
24
+ protected peekNext(): Token | undefined {
25
+ return this.tokens[this.pos + 1];
26
+ }
27
+
28
+ protected check(type: AnyTokenType): boolean {
29
+ return !this.isAtEnd() && this.peek().type === type;
30
+ }
31
+
32
+ protected checkAny(...types: AnyTokenType[]): boolean {
33
+ return types.some((t) => this.check(t));
34
+ }
35
+
36
+ protected match(type: AnyTokenType): boolean {
37
+ if (this.check(type)) {
38
+ this.advance();
39
+ return true;
40
+ }
41
+ return false;
42
+ }
43
+
44
+ protected matchAny(...types: AnyTokenType[]): AnyTokenType | null {
45
+ for (const type of types) {
46
+ if (this.check(type)) {
47
+ this.advance();
48
+ return type;
49
+ }
50
+ }
51
+ return null;
52
+ }
53
+
54
+ protected advance(): Token {
55
+ if (!this.isAtEnd()) this.pos++;
56
+ return this.tokens[this.pos - 1];
57
+ }
58
+
59
+ protected consume(type: AnyTokenType, message: string): Token {
60
+ if (this.check(type)) return this.advance();
61
+ throw this.error(message);
62
+ }
63
+
64
+ /**
65
+ * Consume an identifier, allowing HTTP method tokens to be used as identifiers.
66
+ * This is needed because 'get', 'post', etc. are valid variable/store names.
67
+ */
68
+ protected consumeIdentifier(message: string): Token {
69
+ const token = this.peek();
70
+ // Accept both regular identifiers and HTTP method tokens as identifiers
71
+ if (
72
+ token.type === TokenType.IDENTIFIER ||
73
+ token.type === ReqonTokenType.GET ||
74
+ token.type === ReqonTokenType.POST ||
75
+ token.type === ReqonTokenType.PUT ||
76
+ token.type === ReqonTokenType.PATCH ||
77
+ token.type === ReqonTokenType.DELETE
78
+ ) {
79
+ return this.advance();
80
+ }
81
+ throw this.error(message);
82
+ }
83
+
84
+ /**
85
+ * Check if current token is an identifier (including HTTP methods as identifiers)
86
+ */
87
+ protected checkIdentifier(): boolean {
88
+ const type = this.peek().type;
89
+ return (
90
+ type === TokenType.IDENTIFIER ||
91
+ type === ReqonTokenType.GET ||
92
+ type === ReqonTokenType.POST ||
93
+ type === ReqonTokenType.PUT ||
94
+ type === ReqonTokenType.PATCH ||
95
+ type === ReqonTokenType.DELETE
96
+ );
97
+ }
98
+
99
+ protected isAtEnd(): boolean {
100
+ return this.peek().type === TokenType.EOF;
101
+ }
102
+
103
+ protected error(message: string): ParseError {
104
+ const token = this.peek();
105
+ const context: ErrorContext | undefined = this.source
106
+ ? { source: this.source, filePath: this.filePath }
107
+ : undefined;
108
+
109
+ return new ParseError(
110
+ message,
111
+ { line: token.line, column: token.column },
112
+ context,
113
+ token.value
114
+ );
115
+ }
116
+
117
+ protected savePosition(): number {
118
+ return this.pos;
119
+ }
120
+
121
+ protected restorePosition(saved: number): void {
122
+ this.pos = saved;
123
+ }
124
+ }
@@ -0,0 +1,525 @@
1
+ import { describe, it, expect } from 'vitest';
2
+ import { ReqonLexer } from '../lexer/index.js';
3
+ import { ReqonExpressionParser } from './expressions.js';
4
+ import type { ReqonToken } from '../lexer/tokens.js';
5
+
6
+ describe('ReqonExpressionParser', () => {
7
+ function parseExpr(source: string) {
8
+ const lexer = new ReqonLexer(source);
9
+ const tokens = lexer.tokenize();
10
+ const parser = new TestableExpressionParser(tokens);
11
+ return parser.parseExpression();
12
+ }
13
+
14
+ // Create a testable subclass that exposes parseExpression
15
+ class TestableExpressionParser extends ReqonExpressionParser {
16
+ constructor(tokens: ReqonToken[]) {
17
+ super(tokens);
18
+ }
19
+ }
20
+
21
+ describe('literals', () => {
22
+ it('parses number literals', () => {
23
+ const expr = parseExpr('42');
24
+
25
+ expect(expr.type).toBe('Literal');
26
+ if (expr.type === 'Literal') {
27
+ expect(expr.value).toBe(42);
28
+ expect(expr.dataType).toBe('number');
29
+ }
30
+ });
31
+
32
+ it('parses decimal number literals', () => {
33
+ const expr = parseExpr('3.14');
34
+
35
+ expect(expr.type).toBe('Literal');
36
+ if (expr.type === 'Literal') {
37
+ expect(expr.value).toBe(3.14);
38
+ }
39
+ });
40
+
41
+ it('parses string literals', () => {
42
+ const expr = parseExpr('"hello world"');
43
+
44
+ expect(expr.type).toBe('Literal');
45
+ if (expr.type === 'Literal') {
46
+ expect(expr.value).toBe('hello world');
47
+ expect(expr.dataType).toBe('string');
48
+ }
49
+ });
50
+
51
+ it('parses true literal', () => {
52
+ const expr = parseExpr('true');
53
+
54
+ expect(expr.type).toBe('Literal');
55
+ if (expr.type === 'Literal') {
56
+ expect(expr.value).toBe(true);
57
+ expect(expr.dataType).toBe('boolean');
58
+ }
59
+ });
60
+
61
+ it('parses false literal', () => {
62
+ const expr = parseExpr('false');
63
+
64
+ expect(expr.type).toBe('Literal');
65
+ if (expr.type === 'Literal') {
66
+ expect(expr.value).toBe(false);
67
+ expect(expr.dataType).toBe('boolean');
68
+ }
69
+ });
70
+
71
+ it('parses null literal', () => {
72
+ const expr = parseExpr('null');
73
+
74
+ expect(expr.type).toBe('Literal');
75
+ if (expr.type === 'Literal') {
76
+ expect(expr.value).toBe(null);
77
+ expect(expr.dataType).toBe('null');
78
+ }
79
+ });
80
+ });
81
+
82
+ describe('identifiers', () => {
83
+ it('parses simple identifiers', () => {
84
+ const expr = parseExpr('myVariable');
85
+
86
+ expect(expr.type).toBe('Identifier');
87
+ if (expr.type === 'Identifier') {
88
+ expect(expr.name).toBe('myVariable');
89
+ }
90
+ });
91
+
92
+ it('parses dot-prefixed field shorthand', () => {
93
+ const expr = parseExpr('.fieldName');
94
+
95
+ expect(expr.type).toBe('Identifier');
96
+ if (expr.type === 'Identifier') {
97
+ expect(expr.name).toBe('fieldName');
98
+ }
99
+ });
100
+ });
101
+
102
+ describe('binary arithmetic expressions', () => {
103
+ it('parses addition', () => {
104
+ const expr = parseExpr('1 + 2');
105
+
106
+ expect(expr.type).toBe('BinaryExpression');
107
+ if (expr.type === 'BinaryExpression') {
108
+ expect(expr.operator).toBe('+');
109
+ expect(expr.left).toMatchObject({ type: 'Literal', value: 1 });
110
+ expect(expr.right).toMatchObject({ type: 'Literal', value: 2 });
111
+ }
112
+ });
113
+
114
+ it('parses subtraction', () => {
115
+ const expr = parseExpr('10 - 5');
116
+
117
+ expect(expr.type).toBe('BinaryExpression');
118
+ if (expr.type === 'BinaryExpression') {
119
+ expect(expr.operator).toBe('-');
120
+ }
121
+ });
122
+
123
+ it('parses multiplication', () => {
124
+ const expr = parseExpr('3 * 4');
125
+
126
+ expect(expr.type).toBe('BinaryExpression');
127
+ if (expr.type === 'BinaryExpression') {
128
+ expect(expr.operator).toBe('*');
129
+ }
130
+ });
131
+
132
+ it('parses division', () => {
133
+ const expr = parseExpr('20 / 4');
134
+
135
+ expect(expr.type).toBe('BinaryExpression');
136
+ if (expr.type === 'BinaryExpression') {
137
+ expect(expr.operator).toBe('/');
138
+ }
139
+ });
140
+
141
+ it('respects operator precedence (multiplication before addition)', () => {
142
+ const expr = parseExpr('1 + 2 * 3');
143
+
144
+ expect(expr.type).toBe('BinaryExpression');
145
+ if (expr.type === 'BinaryExpression') {
146
+ expect(expr.operator).toBe('+');
147
+ expect(expr.left).toMatchObject({ type: 'Literal', value: 1 });
148
+ expect(expr.right.type).toBe('BinaryExpression');
149
+ if (expr.right.type === 'BinaryExpression') {
150
+ expect(expr.right.operator).toBe('*');
151
+ }
152
+ }
153
+ });
154
+
155
+ it('respects parentheses', () => {
156
+ const expr = parseExpr('(1 + 2) * 3');
157
+
158
+ expect(expr.type).toBe('BinaryExpression');
159
+ if (expr.type === 'BinaryExpression') {
160
+ expect(expr.operator).toBe('*');
161
+ expect(expr.left.type).toBe('BinaryExpression');
162
+ expect(expr.right).toMatchObject({ type: 'Literal', value: 3 });
163
+ }
164
+ });
165
+ });
166
+
167
+ describe('comparison expressions', () => {
168
+ it('parses equality', () => {
169
+ const expr = parseExpr('x == 5');
170
+
171
+ expect(expr.type).toBe('BinaryExpression');
172
+ if (expr.type === 'BinaryExpression') {
173
+ expect(expr.operator).toBe('==');
174
+ }
175
+ });
176
+
177
+ it('parses less than', () => {
178
+ const expr = parseExpr('x < 10');
179
+
180
+ expect(expr.type).toBe('BinaryExpression');
181
+ if (expr.type === 'BinaryExpression') {
182
+ expect(expr.operator).toBe('<');
183
+ }
184
+ });
185
+
186
+ it('parses greater than', () => {
187
+ const expr = parseExpr('x > 0');
188
+
189
+ expect(expr.type).toBe('BinaryExpression');
190
+ if (expr.type === 'BinaryExpression') {
191
+ expect(expr.operator).toBe('>');
192
+ }
193
+ });
194
+
195
+ it('parses less than or equal', () => {
196
+ const expr = parseExpr('x <= 100');
197
+
198
+ expect(expr.type).toBe('BinaryExpression');
199
+ if (expr.type === 'BinaryExpression') {
200
+ expect(expr.operator).toBe('<=');
201
+ }
202
+ });
203
+
204
+ it('parses greater than or equal', () => {
205
+ const expr = parseExpr('x >= 0');
206
+
207
+ expect(expr.type).toBe('BinaryExpression');
208
+ if (expr.type === 'BinaryExpression') {
209
+ expect(expr.operator).toBe('>=');
210
+ }
211
+ });
212
+ });
213
+
214
+ describe('logical expressions', () => {
215
+ it('parses and expression', () => {
216
+ const expr = parseExpr('a and b');
217
+
218
+ expect(expr.type).toBe('LogicalExpression');
219
+ if (expr.type === 'LogicalExpression') {
220
+ expect(expr.operator).toBe('and');
221
+ }
222
+ });
223
+
224
+ it('parses or expression', () => {
225
+ const expr = parseExpr('a or b');
226
+
227
+ expect(expr.type).toBe('LogicalExpression');
228
+ if (expr.type === 'LogicalExpression') {
229
+ expect(expr.operator).toBe('or');
230
+ }
231
+ });
232
+
233
+ it('parses not expression', () => {
234
+ const expr = parseExpr('not a');
235
+
236
+ expect(expr.type).toBe('NotExpression');
237
+ if (expr.type === 'NotExpression') {
238
+ expect(expr.operand).toMatchObject({ type: 'Identifier', name: 'a' });
239
+ }
240
+ });
241
+
242
+ it('parses combined logical expressions', () => {
243
+ const expr = parseExpr('a and b or c');
244
+
245
+ // 'or' has lower precedence, so it's the root
246
+ expect(expr.type).toBe('LogicalExpression');
247
+ if (expr.type === 'LogicalExpression') {
248
+ expect(expr.operator).toBe('or');
249
+ expect(expr.left.type).toBe('LogicalExpression');
250
+ if (expr.left.type === 'LogicalExpression') {
251
+ expect(expr.left.operator).toBe('and');
252
+ }
253
+ }
254
+ });
255
+
256
+ it('parses double negation', () => {
257
+ const expr = parseExpr('not not a');
258
+
259
+ expect(expr.type).toBe('NotExpression');
260
+ if (expr.type === 'NotExpression') {
261
+ expect(expr.operand.type).toBe('NotExpression');
262
+ }
263
+ });
264
+ });
265
+
266
+ describe('ternary expressions', () => {
267
+ it('parses simple ternary', () => {
268
+ const expr = parseExpr('true ? 1 : 0');
269
+
270
+ expect(expr.type).toBe('TernaryExpression');
271
+ if (expr.type === 'TernaryExpression') {
272
+ expect(expr.condition).toMatchObject({ type: 'Literal', value: true });
273
+ expect(expr.consequent).toMatchObject({ type: 'Literal', value: 1 });
274
+ expect(expr.alternate).toMatchObject({ type: 'Literal', value: 0 });
275
+ }
276
+ });
277
+
278
+ it('parses nested ternary', () => {
279
+ const expr = parseExpr('a ? b ? 1 : 2 : 3');
280
+
281
+ expect(expr.type).toBe('TernaryExpression');
282
+ if (expr.type === 'TernaryExpression') {
283
+ expect(expr.consequent.type).toBe('TernaryExpression');
284
+ }
285
+ });
286
+ });
287
+
288
+ describe('unary expressions', () => {
289
+ it('parses negative number', () => {
290
+ const expr = parseExpr('-5');
291
+
292
+ expect(expr.type).toBe('UnaryExpression');
293
+ if (expr.type === 'UnaryExpression') {
294
+ expect(expr.operator).toBe('-');
295
+ expect(expr.operand).toMatchObject({ type: 'Literal', value: 5 });
296
+ }
297
+ });
298
+
299
+ it('parses positive number', () => {
300
+ const expr = parseExpr('+5');
301
+
302
+ expect(expr.type).toBe('UnaryExpression');
303
+ if (expr.type === 'UnaryExpression') {
304
+ expect(expr.operator).toBe('+');
305
+ }
306
+ });
307
+
308
+ it('parses double negative', () => {
309
+ const expr = parseExpr('--x');
310
+
311
+ expect(expr.type).toBe('UnaryExpression');
312
+ if (expr.type === 'UnaryExpression') {
313
+ expect(expr.operand.type).toBe('UnaryExpression');
314
+ }
315
+ });
316
+ });
317
+
318
+ describe('member access (qualified names)', () => {
319
+ it('parses single property access', () => {
320
+ const expr = parseExpr('obj.prop');
321
+
322
+ expect(expr.type).toBe('QualifiedName');
323
+ if (expr.type === 'QualifiedName') {
324
+ expect(expr.parts).toEqual(['obj', 'prop']);
325
+ }
326
+ });
327
+
328
+ it('parses chained property access', () => {
329
+ const expr = parseExpr('a.b.c.d');
330
+
331
+ expect(expr.type).toBe('QualifiedName');
332
+ if (expr.type === 'QualifiedName') {
333
+ expect(expr.parts).toEqual(['a', 'b', 'c', 'd']);
334
+ }
335
+ });
336
+ });
337
+
338
+ describe('function calls', () => {
339
+ it('parses function call without arguments', () => {
340
+ const expr = parseExpr('fn()');
341
+
342
+ expect(expr.type).toBe('CallExpression');
343
+ if (expr.type === 'CallExpression') {
344
+ expect(expr.callee).toBe('fn');
345
+ expect(expr.arguments).toEqual([]);
346
+ }
347
+ });
348
+
349
+ it('parses function call with single argument', () => {
350
+ const expr = parseExpr('length(items)');
351
+
352
+ expect(expr.type).toBe('CallExpression');
353
+ if (expr.type === 'CallExpression') {
354
+ expect(expr.callee).toBe('length');
355
+ expect(expr.arguments).toHaveLength(1);
356
+ }
357
+ });
358
+
359
+ it('parses function call with multiple arguments', () => {
360
+ const expr = parseExpr('func(1, 2, 3)');
361
+
362
+ expect(expr.type).toBe('CallExpression');
363
+ if (expr.type === 'CallExpression') {
364
+ expect(expr.arguments).toHaveLength(3);
365
+ }
366
+ });
367
+ });
368
+
369
+ describe('match expressions', () => {
370
+ it('parses simple match expression', () => {
371
+ const expr = parseExpr('match x { 1 => "one", 2 => "two" }');
372
+
373
+ expect(expr.type).toBe('MatchExpression');
374
+ if (expr.type === 'MatchExpression') {
375
+ expect(expr.arms).toHaveLength(2);
376
+ expect(expr.arms[0].pattern).toMatchObject({ type: 'Literal', value: 1 });
377
+ expect(expr.arms[0].result).toMatchObject({ type: 'Literal', value: 'one' });
378
+ }
379
+ });
380
+
381
+ it('parses match with wildcard', () => {
382
+ const expr = parseExpr('match status { "A" => "active", _ => "other" }');
383
+
384
+ expect(expr.type).toBe('MatchExpression');
385
+ if (expr.type === 'MatchExpression') {
386
+ expect(expr.arms).toHaveLength(2);
387
+ expect(expr.arms[1].pattern).toMatchObject({ type: 'Identifier', name: '_' });
388
+ }
389
+ });
390
+ });
391
+
392
+ describe('any of expressions', () => {
393
+ it('parses any of expression', () => {
394
+ const expr = parseExpr('any of items');
395
+
396
+ expect(expr.type).toBe('AnyOfExpression');
397
+ if (expr.type === 'AnyOfExpression') {
398
+ expect(expr.collection).toMatchObject({ type: 'Identifier', name: 'items' });
399
+ expect(expr.condition).toBeUndefined();
400
+ }
401
+ });
402
+
403
+ it('parses any of with where condition', () => {
404
+ const expr = parseExpr('any of items where active');
405
+
406
+ expect(expr.type).toBe('AnyOfExpression');
407
+ if (expr.type === 'AnyOfExpression') {
408
+ expect(expr.condition).toBeDefined();
409
+ }
410
+ });
411
+ });
412
+
413
+ describe('range expressions', () => {
414
+ it('parses range expression', () => {
415
+ const expr = parseExpr('1..10');
416
+
417
+ expect(expr.type).toBe('RangeExpression');
418
+ if (expr.type === 'RangeExpression') {
419
+ expect(expr.min).toMatchObject({ type: 'Literal', value: 1 });
420
+ expect(expr.max).toMatchObject({ type: 'Literal', value: 10 });
421
+ }
422
+ });
423
+
424
+ it('parses open-ended range', () => {
425
+ const expr = parseExpr('5..');
426
+
427
+ expect(expr.type).toBe('RangeExpression');
428
+ if (expr.type === 'RangeExpression') {
429
+ expect(expr.min).toMatchObject({ type: 'Literal', value: 5 });
430
+ expect(expr.max).toBeUndefined();
431
+ }
432
+ });
433
+ });
434
+
435
+ describe('is type checking expressions', () => {
436
+ it('parses is array', () => {
437
+ const expr = parseExpr('items is array');
438
+
439
+ expect(expr.type).toBe('IsExpression');
440
+ if (expr.type === 'IsExpression') {
441
+ expect((expr as any).operand).toMatchObject({ type: 'Identifier', name: 'items' });
442
+ expect((expr as any).typeCheck).toBe('array');
443
+ }
444
+ });
445
+
446
+ it('parses is string', () => {
447
+ const expr = parseExpr('.name is string');
448
+
449
+ expect(expr.type).toBe('IsExpression');
450
+ if (expr.type === 'IsExpression') {
451
+ expect((expr as any).typeCheck).toBe('string');
452
+ }
453
+ });
454
+
455
+ it('parses is number', () => {
456
+ const expr = parseExpr('value is number');
457
+
458
+ expect(expr.type).toBe('IsExpression');
459
+ if (expr.type === 'IsExpression') {
460
+ expect((expr as any).typeCheck).toBe('number');
461
+ }
462
+ });
463
+
464
+ it('parses is object', () => {
465
+ const expr = parseExpr('data is object');
466
+
467
+ expect(expr.type).toBe('IsExpression');
468
+ if (expr.type === 'IsExpression') {
469
+ expect((expr as any).typeCheck).toBe('object');
470
+ }
471
+ });
472
+
473
+ it('parses is boolean', () => {
474
+ const expr = parseExpr('flag is boolean');
475
+
476
+ expect(expr.type).toBe('IsExpression');
477
+ if (expr.type === 'IsExpression') {
478
+ expect((expr as any).typeCheck).toBe('boolean');
479
+ }
480
+ });
481
+
482
+ it('parses is with property access', () => {
483
+ const expr = parseExpr('response.items is array');
484
+
485
+ expect(expr.type).toBe('IsExpression');
486
+ if (expr.type === 'IsExpression') {
487
+ expect((expr as any).operand.type).toBe('QualifiedName');
488
+ expect((expr as any).typeCheck).toBe('array');
489
+ }
490
+ });
491
+ });
492
+
493
+ describe('complex expressions', () => {
494
+ it('parses complex arithmetic with comparisons', () => {
495
+ const expr = parseExpr('(a + b) * 2 > 100');
496
+
497
+ expect(expr.type).toBe('BinaryExpression');
498
+ if (expr.type === 'BinaryExpression') {
499
+ expect(expr.operator).toBe('>');
500
+ }
501
+ });
502
+
503
+ it('parses logical expression with comparisons', () => {
504
+ const expr = parseExpr('x > 0 and x < 100');
505
+
506
+ expect(expr.type).toBe('LogicalExpression');
507
+ if (expr.type === 'LogicalExpression') {
508
+ expect(expr.operator).toBe('and');
509
+ expect(expr.left.type).toBe('BinaryExpression');
510
+ expect(expr.right.type).toBe('BinaryExpression');
511
+ }
512
+ });
513
+
514
+ it('parses ternary with function call', () => {
515
+ const expr = parseExpr('length(items) > 0 ? first(items) : null');
516
+
517
+ expect(expr.type).toBe('TernaryExpression');
518
+ if (expr.type === 'TernaryExpression') {
519
+ expect(expr.condition.type).toBe('BinaryExpression');
520
+ expect(expr.consequent.type).toBe('CallExpression');
521
+ expect(expr.alternate).toMatchObject({ type: 'Literal', value: null });
522
+ }
523
+ });
524
+ });
525
+ });