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,287 @@
1
+ import { describe, it, expect } from 'vitest';
2
+ import { ReqonLexer } from '../lexer/index.js';
3
+ import { ReqonParser } from './parser.js';
4
+ import type { MissionDefinition, ScheduleDefinition } from '../ast/nodes.js';
5
+
6
+ function parseSchedule(source: string): ScheduleDefinition | undefined {
7
+ const lexer = new ReqonLexer(source);
8
+ const tokens = lexer.tokenize();
9
+ const parser = new ReqonParser(tokens);
10
+ const program = parser.parse();
11
+ const mission = program.statements.find((s) => s.type === 'MissionDefinition') as
12
+ | MissionDefinition
13
+ | undefined;
14
+ return mission?.schedule;
15
+ }
16
+
17
+ describe('Schedule parsing', () => {
18
+ describe('interval schedules', () => {
19
+ it('should parse schedule: every N hours', () => {
20
+ const source = `
21
+ mission Test {
22
+ schedule: every 6 hours
23
+ source API { auth: none, base: "http://api.example.com" }
24
+ action Sync { get "/data" }
25
+ run Sync
26
+ }
27
+ `;
28
+
29
+ const schedule = parseSchedule(source);
30
+
31
+ expect(schedule).toBeDefined();
32
+ expect(schedule!.scheduleType).toBe('interval');
33
+ expect(schedule!.interval).toEqual({ value: 6, unit: 'hours' });
34
+ });
35
+
36
+ it('should parse schedule: every N minutes', () => {
37
+ const source = `
38
+ mission Test {
39
+ schedule: every 30 minutes
40
+ source API { auth: none, base: "http://api.example.com" }
41
+ action Sync { get "/data" }
42
+ run Sync
43
+ }
44
+ `;
45
+
46
+ const schedule = parseSchedule(source);
47
+
48
+ expect(schedule).toBeDefined();
49
+ expect(schedule!.scheduleType).toBe('interval');
50
+ expect(schedule!.interval).toEqual({ value: 30, unit: 'minutes' });
51
+ });
52
+
53
+ it('should parse schedule: every N seconds', () => {
54
+ const source = `
55
+ mission Test {
56
+ schedule: every 60 seconds
57
+ source API { auth: none, base: "http://api.example.com" }
58
+ action Sync { get "/data" }
59
+ run Sync
60
+ }
61
+ `;
62
+
63
+ const schedule = parseSchedule(source);
64
+
65
+ expect(schedule).toBeDefined();
66
+ expect(schedule!.scheduleType).toBe('interval');
67
+ expect(schedule!.interval).toEqual({ value: 60, unit: 'seconds' });
68
+ });
69
+
70
+ it('should parse schedule: every N days', () => {
71
+ const source = `
72
+ mission Test {
73
+ schedule: every 1 days
74
+ source API { auth: none, base: "http://api.example.com" }
75
+ action Sync { get "/data" }
76
+ run Sync
77
+ }
78
+ `;
79
+
80
+ const schedule = parseSchedule(source);
81
+
82
+ expect(schedule).toBeDefined();
83
+ expect(schedule!.scheduleType).toBe('interval');
84
+ expect(schedule!.interval).toEqual({ value: 1, unit: 'days' });
85
+ });
86
+
87
+ it('should parse schedule: every N weeks', () => {
88
+ const source = `
89
+ mission Test {
90
+ schedule: every 2 weeks
91
+ source API { auth: none, base: "http://api.example.com" }
92
+ action Sync { get "/data" }
93
+ run Sync
94
+ }
95
+ `;
96
+
97
+ const schedule = parseSchedule(source);
98
+
99
+ expect(schedule).toBeDefined();
100
+ expect(schedule!.scheduleType).toBe('interval');
101
+ expect(schedule!.interval).toEqual({ value: 2, unit: 'weeks' });
102
+ });
103
+ });
104
+
105
+ describe('cron schedules', () => {
106
+ it('should parse schedule: cron expression', () => {
107
+ const source = `
108
+ mission Test {
109
+ schedule: cron "0 */6 * * *"
110
+ source API { auth: none, base: "http://api.example.com" }
111
+ action Sync { get "/data" }
112
+ run Sync
113
+ }
114
+ `;
115
+
116
+ const schedule = parseSchedule(source);
117
+
118
+ expect(schedule).toBeDefined();
119
+ expect(schedule!.scheduleType).toBe('cron');
120
+ expect(schedule!.cronExpression).toBe('0 */6 * * *');
121
+ });
122
+
123
+ it('should parse complex cron expressions', () => {
124
+ const source = `
125
+ mission Test {
126
+ schedule: cron "30 9 15 * 1-5"
127
+ source API { auth: none, base: "http://api.example.com" }
128
+ action Sync { get "/data" }
129
+ run Sync
130
+ }
131
+ `;
132
+
133
+ const schedule = parseSchedule(source);
134
+
135
+ expect(schedule).toBeDefined();
136
+ expect(schedule!.scheduleType).toBe('cron');
137
+ expect(schedule!.cronExpression).toBe('30 9 15 * 1-5');
138
+ });
139
+ });
140
+
141
+ describe('one-time schedules', () => {
142
+ it('should parse schedule: at datetime', () => {
143
+ const source = `
144
+ mission Test {
145
+ schedule: at "2025-01-25T15:00:00Z"
146
+ source API { auth: none, base: "http://api.example.com" }
147
+ action Sync { get "/data" }
148
+ run Sync
149
+ }
150
+ `;
151
+
152
+ const schedule = parseSchedule(source);
153
+
154
+ expect(schedule).toBeDefined();
155
+ expect(schedule!.scheduleType).toBe('once');
156
+ expect(schedule!.runAt).toBe('2025-01-25T15:00:00Z');
157
+ });
158
+ });
159
+
160
+ describe('schedule options', () => {
161
+ it('should parse schedule with timezone option', () => {
162
+ const source = `
163
+ mission Test {
164
+ schedule: every 6 hours {
165
+ timezone: "America/New_York"
166
+ }
167
+ source API { auth: none, base: "http://api.example.com" }
168
+ action Sync { get "/data" }
169
+ run Sync
170
+ }
171
+ `;
172
+
173
+ const schedule = parseSchedule(source);
174
+
175
+ expect(schedule).toBeDefined();
176
+ expect(schedule!.timezone).toBe('America/New_York');
177
+ });
178
+
179
+ it('should parse schedule with maxConcurrency option', () => {
180
+ const source = `
181
+ mission Test {
182
+ schedule: every 6 hours {
183
+ maxConcurrency: 2
184
+ }
185
+ source API { auth: none, base: "http://api.example.com" }
186
+ action Sync { get "/data" }
187
+ run Sync
188
+ }
189
+ `;
190
+
191
+ const schedule = parseSchedule(source);
192
+
193
+ expect(schedule).toBeDefined();
194
+ expect(schedule!.maxConcurrency).toBe(2);
195
+ });
196
+
197
+ it('should parse schedule with skipIfRunning option', () => {
198
+ const source = `
199
+ mission Test {
200
+ schedule: every 6 hours {
201
+ skipIfRunning: false
202
+ }
203
+ source API { auth: none, base: "http://api.example.com" }
204
+ action Sync { get "/data" }
205
+ run Sync
206
+ }
207
+ `;
208
+
209
+ const schedule = parseSchedule(source);
210
+
211
+ expect(schedule).toBeDefined();
212
+ expect(schedule!.skipIfRunning).toBe(false);
213
+ });
214
+
215
+ it('should parse schedule with retry config', () => {
216
+ const source = `
217
+ mission Test {
218
+ schedule: every 6 hours {
219
+ retry: {
220
+ maxRetries: 5,
221
+ delaySeconds: 120
222
+ }
223
+ }
224
+ source API { auth: none, base: "http://api.example.com" }
225
+ action Sync { get "/data" }
226
+ run Sync
227
+ }
228
+ `;
229
+
230
+ const schedule = parseSchedule(source);
231
+
232
+ expect(schedule).toBeDefined();
233
+ expect(schedule!.retryOnFailure).toEqual({
234
+ maxRetries: 5,
235
+ delaySeconds: 120,
236
+ });
237
+ });
238
+
239
+ it('should parse schedule with multiple options', () => {
240
+ const source = `
241
+ mission Test {
242
+ schedule: cron "0 9 * * 1-5" {
243
+ timezone: "Europe/London",
244
+ maxConcurrency: 1,
245
+ skipIfRunning: true,
246
+ retry: {
247
+ maxRetries: 3,
248
+ delaySeconds: 60
249
+ }
250
+ }
251
+ source API { auth: none, base: "http://api.example.com" }
252
+ action Sync { get "/data" }
253
+ run Sync
254
+ }
255
+ `;
256
+
257
+ const schedule = parseSchedule(source);
258
+
259
+ expect(schedule).toBeDefined();
260
+ expect(schedule!.scheduleType).toBe('cron');
261
+ expect(schedule!.cronExpression).toBe('0 9 * * 1-5');
262
+ expect(schedule!.timezone).toBe('Europe/London');
263
+ expect(schedule!.maxConcurrency).toBe(1);
264
+ expect(schedule!.skipIfRunning).toBe(true);
265
+ expect(schedule!.retryOnFailure).toEqual({
266
+ maxRetries: 3,
267
+ delaySeconds: 60,
268
+ });
269
+ });
270
+ });
271
+
272
+ describe('mission without schedule', () => {
273
+ it('should parse mission without schedule', () => {
274
+ const source = `
275
+ mission Test {
276
+ source API { auth: none, base: "http://api.example.com" }
277
+ action Sync { get "/data" }
278
+ run Sync
279
+ }
280
+ `;
281
+
282
+ const schedule = parseSchedule(source);
283
+
284
+ expect(schedule).toBeUndefined();
285
+ });
286
+ });
287
+ });
@@ -0,0 +1,248 @@
1
+ import { describe, it, expect } from 'vitest';
2
+ import { ReqonLexer } from '../lexer/index.js';
3
+ import { ReqonParser } from './parser.js';
4
+ import type { MissionDefinition, ActionDefinition, WebhookStep } from '../ast/nodes.js';
5
+
6
+ function parse(source: string) {
7
+ const lexer = new ReqonLexer(source);
8
+ const tokens = lexer.tokenize();
9
+ const parser = new ReqonParser(tokens, source);
10
+ return parser.parse();
11
+ }
12
+
13
+ describe('WebhookStep Parser', () => {
14
+ it('should parse a basic wait step', () => {
15
+ const source = `
16
+ mission TestWebhook {
17
+ source API { auth: none, base: "http://localhost" }
18
+ store events: memory("events")
19
+
20
+ action WaitForCallback {
21
+ wait {
22
+ timeout: 60000
23
+ }
24
+ }
25
+
26
+ run WaitForCallback
27
+ }
28
+ `;
29
+
30
+ const program = parse(source);
31
+ const mission = program.statements[0] as MissionDefinition;
32
+ const action = mission.actions[0] as ActionDefinition;
33
+ const step = action.steps[0] as WebhookStep;
34
+
35
+ expect(step.type).toBe('WebhookStep');
36
+ expect(step.timeout).toBe(60000);
37
+ });
38
+
39
+ it('should parse wait step with path', () => {
40
+ const source = `
41
+ mission TestWebhook {
42
+ source API { auth: none, base: "http://localhost" }
43
+
44
+ action WaitForCallback {
45
+ wait {
46
+ timeout: 30000,
47
+ path: "/webhooks/callback"
48
+ }
49
+ }
50
+
51
+ run WaitForCallback
52
+ }
53
+ `;
54
+
55
+ const program = parse(source);
56
+ const mission = program.statements[0] as MissionDefinition;
57
+ const action = mission.actions[0] as ActionDefinition;
58
+ const step = action.steps[0] as WebhookStep;
59
+
60
+ expect(step.type).toBe('WebhookStep');
61
+ expect(step.timeout).toBe(30000);
62
+ expect(step.path).toBe('/webhooks/callback');
63
+ });
64
+
65
+ it('should parse wait step with expectedEvents', () => {
66
+ const source = `
67
+ mission TestWebhook {
68
+ source API { auth: none, base: "http://localhost" }
69
+
70
+ action WaitForMultiple {
71
+ wait {
72
+ timeout: 120000,
73
+ expectedEvents: 3
74
+ }
75
+ }
76
+
77
+ run WaitForMultiple
78
+ }
79
+ `;
80
+
81
+ const program = parse(source);
82
+ const mission = program.statements[0] as MissionDefinition;
83
+ const action = mission.actions[0] as ActionDefinition;
84
+ const step = action.steps[0] as WebhookStep;
85
+
86
+ expect(step.expectedEvents).toBe(3);
87
+ });
88
+
89
+ it('should parse wait step with storage configuration', () => {
90
+ const source = `
91
+ mission TestWebhook {
92
+ source API { auth: none, base: "http://localhost" }
93
+ store webhook_events: memory("events")
94
+
95
+ action WaitAndStore {
96
+ wait {
97
+ timeout: 60000,
98
+ storage: {
99
+ target: webhook_events,
100
+ key: .id
101
+ }
102
+ }
103
+ }
104
+
105
+ run WaitAndStore
106
+ }
107
+ `;
108
+
109
+ const program = parse(source);
110
+ const mission = program.statements[0] as MissionDefinition;
111
+ const action = mission.actions[0] as ActionDefinition;
112
+ const step = action.steps[0] as WebhookStep;
113
+
114
+ expect(step.storage).toBeDefined();
115
+ expect(step.storage?.target).toBe('webhook_events');
116
+ expect(step.storage?.key).toBeDefined();
117
+ });
118
+
119
+ it('should parse wait step with retry configuration', () => {
120
+ const source = `
121
+ mission TestWebhook {
122
+ source API { auth: none, base: "http://localhost" }
123
+
124
+ action WaitWithRetry {
125
+ wait {
126
+ timeout: 30000,
127
+ retry: {
128
+ maxAttempts: 3,
129
+ backoff: exponential,
130
+ initialDelay: 1000
131
+ }
132
+ }
133
+ }
134
+
135
+ run WaitWithRetry
136
+ }
137
+ `;
138
+
139
+ const program = parse(source);
140
+ const mission = program.statements[0] as MissionDefinition;
141
+ const action = mission.actions[0] as ActionDefinition;
142
+ const step = action.steps[0] as WebhookStep;
143
+
144
+ expect(step.retryOnTimeout).toBeDefined();
145
+ expect(step.retryOnTimeout?.maxAttempts).toBe(3);
146
+ expect(step.retryOnTimeout?.backoff).toBe('exponential');
147
+ expect(step.retryOnTimeout?.initialDelay).toBe(1000);
148
+ });
149
+
150
+ it('should parse wait step with eventFilter', () => {
151
+ const source = `
152
+ mission TestWebhook {
153
+ source API { auth: none, base: "http://localhost" }
154
+
155
+ action WaitFiltered {
156
+ wait {
157
+ timeout: 60000,
158
+ eventFilter: .type == "payment.completed"
159
+ }
160
+ }
161
+
162
+ run WaitFiltered
163
+ }
164
+ `;
165
+
166
+ const program = parse(source);
167
+ const mission = program.statements[0] as MissionDefinition;
168
+ const action = mission.actions[0] as ActionDefinition;
169
+ const step = action.steps[0] as WebhookStep;
170
+
171
+ expect(step.eventFilter).toBeDefined();
172
+ });
173
+
174
+ it('should parse wait step with all options', () => {
175
+ const source = `
176
+ mission FullWebhook {
177
+ source API { auth: none, base: "http://localhost" }
178
+ store events: memory("events")
179
+
180
+ action CompleteWait {
181
+ wait {
182
+ timeout: 300000,
183
+ path: "/webhooks/payment",
184
+ expectedEvents: 1,
185
+ eventFilter: .status == "success",
186
+ storage: {
187
+ target: events,
188
+ key: .id
189
+ },
190
+ retry: {
191
+ maxAttempts: 5,
192
+ backoff: linear,
193
+ initialDelay: 2000
194
+ }
195
+ }
196
+ }
197
+
198
+ run CompleteWait
199
+ }
200
+ `;
201
+
202
+ const program = parse(source);
203
+ const mission = program.statements[0] as MissionDefinition;
204
+ const action = mission.actions[0] as ActionDefinition;
205
+ const step = action.steps[0] as WebhookStep;
206
+
207
+ expect(step.type).toBe('WebhookStep');
208
+ expect(step.timeout).toBe(300000);
209
+ expect(step.path).toBe('/webhooks/payment');
210
+ expect(step.expectedEvents).toBe(1);
211
+ expect(step.eventFilter).toBeDefined();
212
+ expect(step.storage?.target).toBe('events');
213
+ expect(step.retryOnTimeout?.maxAttempts).toBe(5);
214
+ });
215
+
216
+ it('should parse wait step in a multi-step action', () => {
217
+ const source = `
218
+ mission WorkflowWithWebhook {
219
+ source API { auth: none, base: "http://localhost" }
220
+ store orders: memory("orders")
221
+
222
+ action ProcessOrder {
223
+ post "/orders" {
224
+ body: { item: "test" }
225
+ }
226
+
227
+ wait {
228
+ timeout: 60000,
229
+ path: "/webhooks/order-confirmed"
230
+ }
231
+
232
+ store response -> orders { key: .id }
233
+ }
234
+
235
+ run ProcessOrder
236
+ }
237
+ `;
238
+
239
+ const program = parse(source);
240
+ const mission = program.statements[0] as MissionDefinition;
241
+ const action = mission.actions[0] as ActionDefinition;
242
+
243
+ expect(action.steps).toHaveLength(3);
244
+ expect(action.steps[0].type).toBe('FetchStep');
245
+ expect(action.steps[1].type).toBe('WebhookStep');
246
+ expect(action.steps[2].type).toBe('StoreStep');
247
+ });
248
+ });
package/src/plugin.ts ADDED
@@ -0,0 +1,83 @@
1
+ /**
2
+ * Reqon plugin for Vague.
3
+ *
4
+ * This plugin registers Reqon's keywords and statement parsers with Vague,
5
+ * allowing Vague to parse Reqon syntax when the plugin is registered.
6
+ *
7
+ * Usage:
8
+ * import { registerPlugin } from 'vague-lang';
9
+ * import { reqonPlugin } from 'reqon';
10
+ * registerPlugin(reqonPlugin);
11
+ *
12
+ * Or simply import the plugin module to auto-register:
13
+ * import 'reqon/plugin';
14
+ */
15
+
16
+ import {
17
+ registerPlugin,
18
+ unregisterPlugin,
19
+ type VaguePlugin,
20
+ type PluginKeyword,
21
+ } from 'vague-lang';
22
+ import { REQON_KEYWORDS } from './lexer/tokens.js';
23
+
24
+ /**
25
+ * Convert REQON_KEYWORDS map to PluginKeyword array for Vague plugin system.
26
+ */
27
+ function buildKeywords(): PluginKeyword[] {
28
+ const keywords: PluginKeyword[] = [];
29
+
30
+ for (const [keyword, tokenType] of Object.entries(REQON_KEYWORDS)) {
31
+ keywords.push({
32
+ keyword,
33
+ tokenType: tokenType as string,
34
+ });
35
+ }
36
+
37
+ return keywords;
38
+ }
39
+
40
+ /**
41
+ * The Reqon plugin for Vague.
42
+ *
43
+ * Registers all Reqon keywords with Vague's lexer, allowing Vague's
44
+ * lexer to tokenize Reqon source code.
45
+ */
46
+ export const reqonPlugin: VaguePlugin = {
47
+ name: 'reqon',
48
+ keywords: buildKeywords(),
49
+ // Statement parsers will be added when we refactor ReqonParser
50
+ };
51
+
52
+ let isRegistered = false;
53
+
54
+ /**
55
+ * Register Reqon with Vague's plugin system.
56
+ * Safe to call multiple times - will only register once.
57
+ */
58
+ export function registerReqonPlugin(): void {
59
+ if (!isRegistered) {
60
+ registerPlugin(reqonPlugin);
61
+ isRegistered = true;
62
+ }
63
+ }
64
+
65
+ /**
66
+ * Unregister Reqon from Vague's plugin system.
67
+ */
68
+ export function unregisterReqonPlugin(): void {
69
+ if (isRegistered) {
70
+ unregisterPlugin('reqon');
71
+ isRegistered = false;
72
+ }
73
+ }
74
+
75
+ /**
76
+ * Check if Reqon plugin is currently registered.
77
+ */
78
+ export function isReqonPluginRegistered(): boolean {
79
+ return isRegistered;
80
+ }
81
+
82
+ // Auto-register on import
83
+ registerReqonPlugin();