reqon-dsl 0.2.0 → 0.3.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 (396) hide show
  1. package/CHANGELOG.md +7 -0
  2. package/README.md +22 -0
  3. package/dist/ast/nodes.d.ts +83 -4
  4. package/dist/ast/nodes.js +14 -0
  5. package/dist/auth/circuit-breaker.js +7 -6
  6. package/dist/auth/rate-limiter.d.ts +4 -0
  7. package/dist/auth/rate-limiter.js +9 -16
  8. package/dist/cli.d.ts +13 -0
  9. package/dist/cli.js +84 -4
  10. package/dist/config/constants.d.ts +141 -0
  11. package/dist/config/constants.js +128 -0
  12. package/dist/config/index.d.ts +4 -0
  13. package/dist/config/index.js +4 -0
  14. package/dist/control/index.d.ts +2 -0
  15. package/dist/control/index.js +1 -0
  16. package/dist/control/server.d.ts +88 -0
  17. package/dist/control/server.js +238 -0
  18. package/dist/control/types.d.ts +55 -0
  19. package/dist/control/types.js +7 -0
  20. package/dist/debug/cli-debugger.d.ts +17 -0
  21. package/dist/debug/cli-debugger.js +180 -0
  22. package/dist/debug/controller.d.ts +94 -0
  23. package/dist/debug/controller.js +45 -0
  24. package/dist/debug/index.d.ts +6 -0
  25. package/dist/debug/index.js +5 -0
  26. package/dist/errors/index.d.ts +67 -0
  27. package/dist/errors/index.js +89 -1
  28. package/dist/execution/index.d.ts +1 -1
  29. package/dist/execution/state.d.ts +24 -0
  30. package/dist/index.d.ts +21 -1
  31. package/dist/index.js +33 -2
  32. package/dist/interpreter/context.d.ts +14 -0
  33. package/dist/interpreter/context.js +15 -0
  34. package/dist/interpreter/evaluator.d.ts +63 -1
  35. package/dist/interpreter/evaluator.js +186 -39
  36. package/dist/interpreter/executor.d.ts +70 -14
  37. package/dist/interpreter/executor.js +503 -174
  38. package/dist/interpreter/fetch-handler.d.ts +9 -0
  39. package/dist/interpreter/fetch-handler.js +133 -24
  40. package/dist/interpreter/http.d.ts +5 -0
  41. package/dist/interpreter/http.js +26 -12
  42. package/dist/interpreter/index.d.ts +3 -1
  43. package/dist/interpreter/index.js +2 -0
  44. package/dist/interpreter/pagination.d.ts +11 -2
  45. package/dist/interpreter/pagination.js +95 -31
  46. package/dist/interpreter/signals.d.ts +8 -0
  47. package/dist/interpreter/signals.js +12 -0
  48. package/dist/interpreter/source-manager.d.ts +75 -0
  49. package/dist/interpreter/source-manager.js +157 -0
  50. package/dist/interpreter/step-handlers/apply-handler.d.ts +29 -0
  51. package/dist/interpreter/step-handlers/apply-handler.js +79 -0
  52. package/dist/interpreter/step-handlers/for-handler.d.ts +13 -0
  53. package/dist/interpreter/step-handlers/for-handler.js +71 -4
  54. package/dist/interpreter/step-handlers/index.d.ts +4 -2
  55. package/dist/interpreter/step-handlers/index.js +4 -2
  56. package/dist/interpreter/step-handlers/match-handler.d.ts +9 -0
  57. package/dist/interpreter/step-handlers/match-handler.js +43 -16
  58. package/dist/interpreter/step-handlers/pause-handler.d.ts +52 -0
  59. package/dist/interpreter/step-handlers/pause-handler.js +87 -0
  60. package/dist/interpreter/step-handlers/store-handler.d.ts +11 -1
  61. package/dist/interpreter/step-handlers/store-handler.js +45 -13
  62. package/dist/interpreter/step-handlers/types.d.ts +3 -0
  63. package/dist/interpreter/step-handlers/validate-handler.d.ts +2 -1
  64. package/dist/interpreter/step-handlers/validate-handler.js +4 -2
  65. package/dist/interpreter/step-handlers/webhook-handler.d.ts +3 -0
  66. package/dist/interpreter/step-handlers/webhook-handler.js +18 -2
  67. package/dist/interpreter/store-manager.d.ts +46 -0
  68. package/dist/interpreter/store-manager.js +66 -0
  69. package/dist/lexer/index.d.ts +11 -4
  70. package/dist/lexer/index.js +11 -4
  71. package/dist/lexer/tokens.d.ts +17 -1
  72. package/dist/lexer/tokens.js +36 -0
  73. package/dist/mcp/index.d.ts +11 -0
  74. package/dist/mcp/index.js +11 -0
  75. package/dist/mcp/server.d.ts +17 -0
  76. package/dist/mcp/server.js +451 -0
  77. package/dist/oas/index.d.ts +2 -0
  78. package/dist/oas/index.js +1 -0
  79. package/dist/oas/mock-generator.d.ts +12 -0
  80. package/dist/oas/mock-generator.js +187 -0
  81. package/dist/observability/events.d.ts +244 -0
  82. package/dist/observability/events.js +90 -0
  83. package/dist/observability/index.d.ts +15 -0
  84. package/dist/observability/index.js +12 -0
  85. package/dist/observability/logger.d.ts +106 -0
  86. package/dist/observability/logger.js +259 -0
  87. package/dist/observability/otel.d.ts +135 -0
  88. package/dist/observability/otel.js +386 -0
  89. package/dist/parser/action-parser.d.ts +105 -0
  90. package/dist/parser/action-parser.js +645 -0
  91. package/dist/parser/expressions.d.ts +13 -0
  92. package/dist/parser/expressions.js +72 -2
  93. package/dist/parser/fetch-parser.d.ts +27 -0
  94. package/dist/parser/fetch-parser.js +269 -0
  95. package/dist/parser/index.d.ts +17 -0
  96. package/dist/parser/index.js +17 -0
  97. package/dist/parser/parser.d.ts +44 -46
  98. package/dist/parser/parser.js +122 -1070
  99. package/dist/parser/pipeline-parser.d.ts +12 -0
  100. package/dist/parser/pipeline-parser.js +52 -0
  101. package/dist/parser/schedule-parser.d.ts +7 -0
  102. package/dist/parser/schedule-parser.js +137 -0
  103. package/dist/parser/source-parser.d.ts +9 -0
  104. package/dist/parser/source-parser.js +151 -0
  105. package/dist/pause/index.d.ts +14 -0
  106. package/dist/pause/index.js +11 -0
  107. package/dist/pause/manager.d.ts +118 -0
  108. package/dist/pause/manager.js +245 -0
  109. package/dist/pause/state.d.ts +93 -0
  110. package/dist/pause/state.js +103 -0
  111. package/dist/pause/store.d.ts +61 -0
  112. package/dist/pause/store.js +156 -0
  113. package/dist/plugin.d.ts +9 -12
  114. package/dist/plugin.js +10 -13
  115. package/dist/stores/factory.d.ts +1 -1
  116. package/dist/stores/factory.js +3 -2
  117. package/dist/stores/file.d.ts +26 -0
  118. package/dist/stores/file.js +64 -10
  119. package/dist/stores/index.d.ts +16 -1
  120. package/dist/stores/index.js +16 -1
  121. package/dist/stores/memory.d.ts +4 -0
  122. package/dist/stores/memory.js +11 -0
  123. package/dist/stores/types.d.ts +17 -0
  124. package/dist/stores/types.js +12 -0
  125. package/dist/trace/index.d.ts +16 -0
  126. package/dist/trace/index.js +12 -0
  127. package/dist/trace/recorder.d.ts +71 -0
  128. package/dist/trace/recorder.js +144 -0
  129. package/dist/trace/replay.d.ts +132 -0
  130. package/dist/trace/replay.js +264 -0
  131. package/dist/trace/state.d.ts +102 -0
  132. package/dist/trace/state.js +86 -0
  133. package/dist/trace/store.d.ts +69 -0
  134. package/dist/trace/store.js +225 -0
  135. package/dist/utils/index.d.ts +1 -0
  136. package/dist/utils/index.js +1 -0
  137. package/dist/utils/type-guards.d.ts +58 -0
  138. package/dist/utils/type-guards.js +92 -0
  139. package/dist/webhook/server.js +7 -6
  140. package/package.json +55 -6
  141. package/.claude/settings.local.json +0 -31
  142. package/.claude/skills/api-integration.md +0 -125
  143. package/.claude/skills/database-schema.md +0 -51
  144. package/.claude/skills/dsl-design.md +0 -80
  145. package/.claude/skills/property-testing.md +0 -143
  146. package/.claude/skills/reqon/SKILL.md +0 -44
  147. package/.claude/skills/reqon/references/examples.md +0 -206
  148. package/.claude/skills/reqon/references/syntax.md +0 -263
  149. package/.claude/skills/vscode-extension.md +0 -113
  150. package/.github/dependabot.yml +0 -32
  151. package/.github/pull_request_template.md +0 -21
  152. package/.github/workflows/ci.yml +0 -174
  153. package/.github/workflows/release.yml +0 -73
  154. package/CLAUDE.md +0 -72
  155. package/CONTRIBUTING.md +0 -161
  156. package/TODO.md +0 -51
  157. package/dist/auth/auth.test.d.ts +0 -1
  158. package/dist/auth/auth.test.js +0 -255
  159. package/dist/errors/errors.test.d.ts +0 -1
  160. package/dist/errors/errors.test.js +0 -165
  161. package/dist/execution/execution.test.d.ts +0 -1
  162. package/dist/execution/execution.test.js +0 -246
  163. package/dist/integration.test.d.ts +0 -1
  164. package/dist/integration.test.js +0 -168
  165. package/dist/interpreter/evaluator.test.d.ts +0 -1
  166. package/dist/interpreter/evaluator.test.js +0 -512
  167. package/dist/interpreter/http.test.d.ts +0 -1
  168. package/dist/interpreter/http.test.js +0 -299
  169. package/dist/interpreter/progress.test.d.ts +0 -1
  170. package/dist/interpreter/progress.test.js +0 -216
  171. package/dist/interpreter/schema-matcher.test.d.ts +0 -1
  172. package/dist/interpreter/schema-matcher.test.js +0 -122
  173. package/dist/lexer/lexer.d.ts +0 -24
  174. package/dist/lexer/lexer.js +0 -264
  175. package/dist/lexer/lexer.test.d.ts +0 -1
  176. package/dist/lexer/lexer.test.js +0 -259
  177. package/dist/loader/loader.test.d.ts +0 -1
  178. package/dist/loader/loader.test.js +0 -287
  179. package/dist/oas/oas.test.d.ts +0 -1
  180. package/dist/oas/oas.test.js +0 -218
  181. package/dist/parser/expressions.test.d.ts +0 -1
  182. package/dist/parser/expressions.test.js +0 -378
  183. package/dist/parser/match.test.d.ts +0 -1
  184. package/dist/parser/match.test.js +0 -254
  185. package/dist/parser/parser.test.d.ts +0 -1
  186. package/dist/parser/parser.test.js +0 -333
  187. package/dist/parser/schedule.test.d.ts +0 -1
  188. package/dist/parser/schedule.test.js +0 -241
  189. package/dist/scheduler/cron-parser.test.d.ts +0 -1
  190. package/dist/scheduler/cron-parser.test.js +0 -188
  191. package/dist/stores/file.test.d.ts +0 -1
  192. package/dist/stores/file.test.js +0 -165
  193. package/dist/stores/memory.test.d.ts +0 -1
  194. package/dist/stores/memory.test.js +0 -157
  195. package/dist/stores/stores.test.d.ts +0 -1
  196. package/dist/stores/stores.test.js +0 -158
  197. package/dist/sync/sync.test.d.ts +0 -1
  198. package/dist/sync/sync.test.js +0 -221
  199. package/docusaurus/README.md +0 -41
  200. package/docusaurus/docs/advanced/execution-state.md +0 -283
  201. package/docusaurus/docs/advanced/extending-reqon.md +0 -388
  202. package/docusaurus/docs/advanced/multi-file-missions.md +0 -250
  203. package/docusaurus/docs/advanced/parallel-execution.md +0 -353
  204. package/docusaurus/docs/api-reference.md +0 -443
  205. package/docusaurus/docs/authentication/api-key.md +0 -339
  206. package/docusaurus/docs/authentication/basic.md +0 -276
  207. package/docusaurus/docs/authentication/bearer.md +0 -282
  208. package/docusaurus/docs/authentication/oauth2.md +0 -317
  209. package/docusaurus/docs/authentication/overview.md +0 -251
  210. package/docusaurus/docs/cli.md +0 -229
  211. package/docusaurus/docs/core-concepts/actions.md +0 -286
  212. package/docusaurus/docs/core-concepts/missions.md +0 -264
  213. package/docusaurus/docs/core-concepts/schemas.md +0 -353
  214. package/docusaurus/docs/core-concepts/sources.md +0 -339
  215. package/docusaurus/docs/core-concepts/stores.md +0 -332
  216. package/docusaurus/docs/dsl-syntax/expressions.md +0 -361
  217. package/docusaurus/docs/dsl-syntax/fetch.md +0 -293
  218. package/docusaurus/docs/dsl-syntax/for-loops.md +0 -324
  219. package/docusaurus/docs/dsl-syntax/map.md +0 -345
  220. package/docusaurus/docs/dsl-syntax/match.md +0 -387
  221. package/docusaurus/docs/dsl-syntax/pipelines.md +0 -397
  222. package/docusaurus/docs/dsl-syntax/validate.md +0 -401
  223. package/docusaurus/docs/error-handling/dead-letter-queues.md +0 -399
  224. package/docusaurus/docs/error-handling/flow-control.md +0 -337
  225. package/docusaurus/docs/error-handling/retry-strategies.md +0 -368
  226. package/docusaurus/docs/examples.md +0 -488
  227. package/docusaurus/docs/getting-started.md +0 -256
  228. package/docusaurus/docs/http/circuit-breaker.md +0 -401
  229. package/docusaurus/docs/http/incremental-sync.md +0 -394
  230. package/docusaurus/docs/http/pagination.md +0 -361
  231. package/docusaurus/docs/http/rate-limiting.md +0 -383
  232. package/docusaurus/docs/http/requests.md +0 -328
  233. package/docusaurus/docs/http/retry.md +0 -402
  234. package/docusaurus/docs/intro.md +0 -90
  235. package/docusaurus/docs/openapi/loading-specs.md +0 -305
  236. package/docusaurus/docs/openapi/operation-calls.md +0 -314
  237. package/docusaurus/docs/openapi/overview.md +0 -212
  238. package/docusaurus/docs/openapi/response-validation.md +0 -344
  239. package/docusaurus/docs/scheduling/cron.md +0 -305
  240. package/docusaurus/docs/scheduling/daemon-mode.md +0 -317
  241. package/docusaurus/docs/scheduling/intervals.md +0 -289
  242. package/docusaurus/docs/scheduling/overview.md +0 -231
  243. package/docusaurus/docs/stores/custom-adapters.md +0 -376
  244. package/docusaurus/docs/stores/file.md +0 -236
  245. package/docusaurus/docs/stores/memory.md +0 -193
  246. package/docusaurus/docs/stores/overview.md +0 -274
  247. package/docusaurus/docs/stores/postgrest.md +0 -316
  248. package/docusaurus/docusaurus.config.ts +0 -148
  249. package/docusaurus/package-lock.json +0 -18029
  250. package/docusaurus/package.json +0 -47
  251. package/docusaurus/sidebars.ts +0 -155
  252. package/docusaurus/src/components/HomepageFeatures/index.tsx +0 -105
  253. package/docusaurus/src/components/HomepageFeatures/styles.module.css +0 -12
  254. package/docusaurus/src/css/custom.css +0 -169
  255. package/docusaurus/src/pages/index.module.css +0 -48
  256. package/docusaurus/src/pages/index.tsx +0 -110
  257. package/docusaurus/src/pages/markdown-page.md +0 -7
  258. package/docusaurus/static/.nojekyll +0 -0
  259. package/docusaurus/static/img/docusaurus-social-card.jpg +0 -0
  260. package/docusaurus/static/img/docusaurus.png +0 -0
  261. package/docusaurus/static/img/favicon.ico +0 -0
  262. package/docusaurus/static/img/logo.svg +0 -10
  263. package/docusaurus/static/img/undraw_docusaurus_mountain.svg +0 -171
  264. package/docusaurus/static/img/undraw_docusaurus_react.svg +0 -170
  265. package/docusaurus/static/img/undraw_docusaurus_tree.svg +0 -40
  266. package/docusaurus/tsconfig.json +0 -8
  267. package/examples/README.md +0 -112
  268. package/examples/error-handling/README.md +0 -150
  269. package/examples/error-handling/payment-processor.vague +0 -287
  270. package/examples/github-sync/README.md +0 -74
  271. package/examples/github-sync/fetch-issues.vague +0 -47
  272. package/examples/github-sync/fetch-prs.vague +0 -40
  273. package/examples/github-sync/mission.vague +0 -101
  274. package/examples/github-sync/normalize.vague +0 -70
  275. package/examples/jsonplaceholder/README.md +0 -28
  276. package/examples/jsonplaceholder/posts.vague +0 -48
  277. package/examples/petstore/README.md +0 -35
  278. package/examples/petstore/openapi.yaml +0 -97
  279. package/examples/petstore/sync.vague +0 -52
  280. package/examples/temporal-comparison/README.md +0 -297
  281. package/examples/temporal-comparison/reconciliation.vague +0 -355
  282. package/examples/temporal-comparison/temporal/activities/index.ts +0 -8
  283. package/examples/temporal-comparison/temporal/activities/shipstation.ts +0 -225
  284. package/examples/temporal-comparison/temporal/activities/shopify.ts +0 -257
  285. package/examples/temporal-comparison/temporal/activities/storage.ts +0 -198
  286. package/examples/temporal-comparison/temporal/activities/stripe.ts +0 -169
  287. package/examples/temporal-comparison/temporal/activities/validation.ts +0 -205
  288. package/examples/temporal-comparison/temporal/client/schedule.ts +0 -218
  289. package/examples/temporal-comparison/temporal/config/retry.ts +0 -63
  290. package/examples/temporal-comparison/temporal/types/index.ts +0 -129
  291. package/examples/temporal-comparison/temporal/workers/main.ts +0 -130
  292. package/examples/temporal-comparison/temporal/workflows/orderReconciliation.ts +0 -262
  293. package/examples/xero/README.md +0 -88
  294. package/examples/xero/invoices.vague +0 -189
  295. package/src/api-integration.test.ts +0 -954
  296. package/src/ast/index.ts +0 -1
  297. package/src/ast/nodes.ts +0 -310
  298. package/src/auth/auth.test.ts +0 -326
  299. package/src/auth/circuit-breaker.test.ts +0 -390
  300. package/src/auth/circuit-breaker.ts +0 -379
  301. package/src/auth/credentials.test.ts +0 -273
  302. package/src/auth/credentials.ts +0 -246
  303. package/src/auth/index.ts +0 -40
  304. package/src/auth/oauth2-provider.ts +0 -177
  305. package/src/auth/rate-limiter.ts +0 -459
  306. package/src/auth/token-store.ts +0 -177
  307. package/src/auth/types.ts +0 -159
  308. package/src/benchmark/e2e.bench.ts +0 -288
  309. package/src/benchmark/evaluator.bench.ts +0 -331
  310. package/src/benchmark/fixtures.ts +0 -295
  311. package/src/benchmark/index.ts +0 -108
  312. package/src/benchmark/lexer.bench.ts +0 -69
  313. package/src/benchmark/parser.bench.ts +0 -103
  314. package/src/benchmark/resilience.bench.ts +0 -193
  315. package/src/benchmark/store.bench.ts +0 -147
  316. package/src/benchmark/utils.ts +0 -230
  317. package/src/cli.ts +0 -313
  318. package/src/errors/errors.test.ts +0 -234
  319. package/src/errors/index.ts +0 -223
  320. package/src/execution/execution.test.ts +0 -307
  321. package/src/execution/index.ts +0 -21
  322. package/src/execution/state.ts +0 -207
  323. package/src/execution/store.ts +0 -188
  324. package/src/index.ts +0 -169
  325. package/src/integration.test.ts +0 -192
  326. package/src/interpreter/context.ts +0 -57
  327. package/src/interpreter/evaluator.test.ts +0 -796
  328. package/src/interpreter/evaluator.ts +0 -245
  329. package/src/interpreter/executor.ts +0 -946
  330. package/src/interpreter/fetch-handler.ts +0 -302
  331. package/src/interpreter/http.test.ts +0 -423
  332. package/src/interpreter/http.ts +0 -308
  333. package/src/interpreter/index.ts +0 -32
  334. package/src/interpreter/pagination.ts +0 -207
  335. package/src/interpreter/progress.test.ts +0 -276
  336. package/src/interpreter/schema-matcher.test.ts +0 -160
  337. package/src/interpreter/schema-matcher.ts +0 -168
  338. package/src/interpreter/signals.ts +0 -73
  339. package/src/interpreter/step-handlers/for-handler.ts +0 -65
  340. package/src/interpreter/step-handlers/index.ts +0 -17
  341. package/src/interpreter/step-handlers/map-handler.ts +0 -24
  342. package/src/interpreter/step-handlers/match-handler.ts +0 -101
  343. package/src/interpreter/step-handlers/store-handler.ts +0 -78
  344. package/src/interpreter/step-handlers/types.ts +0 -17
  345. package/src/interpreter/step-handlers/validate-handler.ts +0 -30
  346. package/src/interpreter/step-handlers/webhook-handler.ts +0 -142
  347. package/src/lexer/index.ts +0 -18
  348. package/src/lexer/lexer.test.ts +0 -316
  349. package/src/lexer/tokens.ts +0 -179
  350. package/src/loader/index.ts +0 -288
  351. package/src/loader/loader.test.ts +0 -360
  352. package/src/oas/index.ts +0 -4
  353. package/src/oas/loader.ts +0 -126
  354. package/src/oas/oas.test.ts +0 -254
  355. package/src/oas/validator.ts +0 -299
  356. package/src/parser/base.ts +0 -124
  357. package/src/parser/expressions.test.ts +0 -525
  358. package/src/parser/expressions.ts +0 -314
  359. package/src/parser/index.ts +0 -3
  360. package/src/parser/match.test.ts +0 -296
  361. package/src/parser/parser.test.ts +0 -739
  362. package/src/parser/parser.ts +0 -1469
  363. package/src/parser/schedule.test.ts +0 -287
  364. package/src/parser/webhook.test.ts +0 -248
  365. package/src/plugin.ts +0 -83
  366. package/src/scheduler/cron-parser.test.ts +0 -236
  367. package/src/scheduler/cron-parser.ts +0 -236
  368. package/src/scheduler/index.ts +0 -10
  369. package/src/scheduler/scheduler.ts +0 -443
  370. package/src/scheduler/types.ts +0 -71
  371. package/src/stores/factory.ts +0 -104
  372. package/src/stores/file.test.ts +0 -276
  373. package/src/stores/file.ts +0 -211
  374. package/src/stores/index.ts +0 -6
  375. package/src/stores/memory.test.ts +0 -238
  376. package/src/stores/memory.ts +0 -63
  377. package/src/stores/postgrest.test.ts +0 -488
  378. package/src/stores/postgrest.ts +0 -263
  379. package/src/stores/stores.test.ts +0 -197
  380. package/src/stores/types.ts +0 -58
  381. package/src/sync/index.ts +0 -16
  382. package/src/sync/state.ts +0 -126
  383. package/src/sync/store.ts +0 -139
  384. package/src/sync/sync.test.ts +0 -271
  385. package/src/utils/async.ts +0 -10
  386. package/src/utils/file.ts +0 -106
  387. package/src/utils/index.ts +0 -14
  388. package/src/utils/logger.ts +0 -53
  389. package/src/utils/path.ts +0 -47
  390. package/src/webhook/index.ts +0 -15
  391. package/src/webhook/server.test.ts +0 -253
  392. package/src/webhook/server.ts +0 -389
  393. package/src/webhook/store.ts +0 -239
  394. package/src/webhook/types.ts +0 -93
  395. package/tsconfig.json +0 -17
  396. package/vitest.config.ts +0 -39
@@ -1,288 +0,0 @@
1
- import { readFile, readdir, stat } from 'node:fs/promises';
2
- import { resolve, join, dirname, basename } from 'node:path';
3
- import { ReqonLexer } from '../lexer/index.js';
4
- import { ReqonParser } from '../parser/index.js';
5
- import type { ReqonProgram, MissionDefinition, ActionDefinition, Statement } from '../ast/nodes.js';
6
-
7
- export interface LoadResult {
8
- program: ReqonProgram;
9
- /** Base directory for the mission (folder path or file's parent) */
10
- baseDir: string;
11
- /** All source files that were loaded */
12
- sourceFiles: string[];
13
- }
14
-
15
- /** Supported file extensions, in order of preference */
16
- export const SUPPORTED_EXTENSIONS = ['.vague', '.reqon'] as const;
17
- export type SupportedExtension = (typeof SUPPORTED_EXTENSIONS)[number];
18
-
19
- export interface LoadOptions {
20
- /** File extension to look for (default: tries '.vague' then '.reqon') */
21
- extension?: SupportedExtension;
22
- }
23
-
24
- /**
25
- * Load a mission from a file or folder.
26
- *
27
- * Single file mode:
28
- * loadMission('./sync-invoices.reqon')
29
- * Returns the parsed program from that file.
30
- *
31
- * Folder mode:
32
- * loadMission('./sync-invoices/')
33
- * Looks for mission.reqon (root) and merges all *.reqon action files.
34
- * Actions are resolved by name from the `run` pipeline.
35
- */
36
- export async function loadMission(
37
- path: string,
38
- options: LoadOptions = {}
39
- ): Promise<LoadResult> {
40
- const absolutePath = resolve(path);
41
- const stats = await stat(absolutePath);
42
-
43
- if (stats.isDirectory()) {
44
- return loadMissionFolder(absolutePath, options);
45
- } else {
46
- return loadMissionFile(absolutePath);
47
- }
48
- }
49
-
50
- /**
51
- * Load a single .reqon file
52
- */
53
- async function loadMissionFile(filePath: string): Promise<LoadResult> {
54
- const source = await readFile(filePath, 'utf-8');
55
- const program = parseSource(source, filePath);
56
-
57
- return {
58
- program,
59
- baseDir: dirname(filePath),
60
- sourceFiles: [filePath],
61
- };
62
- }
63
-
64
- /**
65
- * Load a mission from a folder.
66
- *
67
- * Expected structure:
68
- * mission-name/
69
- * ├── mission.reqon # Root file with mission definition (sources, stores, pipeline)
70
- * ├── fetch-list.reqon # Action file
71
- * ├── hydrate.reqon # Action file
72
- * └── normalize.reqon # Action file
73
- *
74
- * The root file (mission.reqon) contains:
75
- * - source definitions
76
- * - store definitions
77
- * - schema definitions
78
- * - the pipeline (run X then Y then Z)
79
- * - optionally inline actions
80
- *
81
- * Action files contain standalone action definitions that get merged in.
82
- */
83
- async function loadMissionFolder(
84
- folderPath: string,
85
- options: LoadOptions
86
- ): Promise<LoadResult> {
87
- // Find root file - try extensions in order of preference
88
- const extensionsToTry = options.extension
89
- ? [options.extension]
90
- : SUPPORTED_EXTENSIONS;
91
-
92
- let rootFilePath: string | null = null;
93
- let ext: SupportedExtension | null = null;
94
-
95
- for (const extension of extensionsToTry) {
96
- const candidatePath = join(folderPath, `mission${extension}`);
97
- try {
98
- await stat(candidatePath);
99
- rootFilePath = candidatePath;
100
- ext = extension;
101
- break;
102
- } catch {
103
- // Try next extension
104
- }
105
- }
106
-
107
- if (!rootFilePath || !ext) {
108
- const tried = extensionsToTry.map(e => `mission${e}`).join(' or ');
109
- throw new Error(
110
- `Mission folder must contain a root file (${tried}). ` +
111
- `Not found in: ${folderPath}`
112
- );
113
- }
114
-
115
- // Load root file
116
- const rootSource = await readFile(rootFilePath, 'utf-8');
117
- const rootProgram = parseSource(rootSource, rootFilePath);
118
-
119
- // Find all other action files in the folder (same extension as root)
120
- const rootFileName = basename(rootFilePath);
121
- const files = await readdir(folderPath);
122
- const actionFiles = files.filter(
123
- f => f.endsWith(ext) && f !== rootFileName
124
- );
125
-
126
- const sourceFiles = [rootFilePath];
127
- const externalActions: ActionDefinition[] = [];
128
-
129
- // Parse each action file
130
- for (const file of actionFiles) {
131
- const filePath = join(folderPath, file);
132
- sourceFiles.push(filePath);
133
-
134
- const source = await readFile(filePath, 'utf-8');
135
- const actionProgram = parseSource(source, filePath);
136
-
137
- // Extract action definitions from the file
138
- for (const stmt of actionProgram.statements) {
139
- if (stmt.type === 'ActionDefinition') {
140
- externalActions.push(stmt);
141
- } else if (stmt.type === 'MissionDefinition') {
142
- throw new Error(
143
- `Action file '${file}' should not contain a mission definition. ` +
144
- `Only '${rootFileName}' should define the mission.`
145
- );
146
- }
147
- }
148
- }
149
-
150
- // Merge external actions into the mission
151
- const mergedProgram = mergeActions(rootProgram, externalActions);
152
-
153
- // Validate that all actions referenced in the pipeline exist
154
- validatePipelineActions(mergedProgram);
155
-
156
- return {
157
- program: mergedProgram,
158
- baseDir: folderPath,
159
- sourceFiles,
160
- };
161
- }
162
-
163
- /**
164
- * Parse source code into a program
165
- */
166
- function parseSource(source: string, filePath: string): ReqonProgram {
167
- const lexer = new ReqonLexer(source);
168
- const tokens = lexer.tokenize();
169
- const parser = new ReqonParser(tokens, source, filePath);
170
- return parser.parse();
171
- }
172
-
173
- /**
174
- * Merge external actions into the mission definition
175
- */
176
- function mergeActions(
177
- program: ReqonProgram,
178
- externalActions: ActionDefinition[]
179
- ): ReqonProgram {
180
- // Find the mission in the program
181
- const missionIndex = program.statements.findIndex(
182
- (s): s is MissionDefinition => s.type === 'MissionDefinition'
183
- );
184
-
185
- if (missionIndex === -1) {
186
- throw new Error('Root file must contain a mission definition');
187
- }
188
-
189
- const mission = program.statements[missionIndex] as MissionDefinition;
190
-
191
- // Build a map of existing actions
192
- const actionMap = new Map<string, ActionDefinition>();
193
- for (const action of mission.actions) {
194
- actionMap.set(action.name, action);
195
- }
196
-
197
- // Add external actions (error if duplicate)
198
- for (const action of externalActions) {
199
- if (actionMap.has(action.name)) {
200
- throw new Error(
201
- `Duplicate action definition: '${action.name}'. ` +
202
- `Action names must be unique across all files.`
203
- );
204
- }
205
- actionMap.set(action.name, action);
206
- }
207
-
208
- // Create updated mission with all actions
209
- const updatedMission: MissionDefinition = {
210
- ...mission,
211
- actions: Array.from(actionMap.values()),
212
- };
213
-
214
- // Return updated program
215
- const updatedStatements = [...program.statements];
216
- updatedStatements[missionIndex] = updatedMission;
217
-
218
- return {
219
- type: 'ReqonProgram',
220
- statements: updatedStatements,
221
- };
222
- }
223
-
224
- /**
225
- * Validate that all actions referenced in the pipeline exist
226
- */
227
- function validatePipelineActions(program: ReqonProgram): void {
228
- const mission = program.statements.find(
229
- (s): s is MissionDefinition => s.type === 'MissionDefinition'
230
- );
231
-
232
- if (!mission) return;
233
-
234
- const actionNames = new Set(mission.actions.map(a => a.name));
235
-
236
- for (const stage of mission.pipeline.stages) {
237
- const stageActions = stage.actions ?? (stage.action ? [stage.action] : []);
238
-
239
- for (const actionName of stageActions) {
240
- if (!actionNames.has(actionName)) {
241
- throw new Error(
242
- `Pipeline references unknown action: '${actionName}'. ` +
243
- `Available actions: ${Array.from(actionNames).join(', ')}`
244
- );
245
- }
246
- }
247
- }
248
- }
249
-
250
- /**
251
- * Check if a path is a mission folder (contains mission.vague or mission.reqon)
252
- */
253
- export async function isMissionFolder(path: string): Promise<boolean> {
254
- try {
255
- const absolutePath = resolve(path);
256
- const stats = await stat(absolutePath);
257
-
258
- if (!stats.isDirectory()) return false;
259
-
260
- // Check for any supported root file
261
- for (const ext of SUPPORTED_EXTENSIONS) {
262
- try {
263
- await stat(join(absolutePath, `mission${ext}`));
264
- return true;
265
- } catch {
266
- // Try next extension
267
- }
268
- }
269
- return false;
270
- } catch {
271
- return false;
272
- }
273
- }
274
-
275
- /**
276
- * Get the mission name from a path (folder name or file name without extension)
277
- */
278
- export function getMissionName(path: string): string {
279
- const absolutePath = resolve(path);
280
- const name = basename(absolutePath);
281
-
282
- for (const ext of SUPPORTED_EXTENSIONS) {
283
- if (name.endsWith(ext)) {
284
- return name.slice(0, -ext.length);
285
- }
286
- }
287
- return name;
288
- }
@@ -1,360 +0,0 @@
1
- import { describe, it, expect, beforeEach, afterEach } from 'vitest';
2
- import { mkdir, writeFile, rm } from 'node:fs/promises';
3
- import { join } from 'node:path';
4
- import { loadMission, isMissionFolder, getMissionName } from './index.js';
5
-
6
- const TEST_DIR = '.test-missions';
7
-
8
- describe('Mission Loader', () => {
9
- beforeEach(async () => {
10
- await mkdir(TEST_DIR, { recursive: true });
11
- });
12
-
13
- afterEach(async () => {
14
- await rm(TEST_DIR, { recursive: true, force: true });
15
- });
16
-
17
- describe('single file loading', () => {
18
- it('loads a single .reqon file', async () => {
19
- const filePath = join(TEST_DIR, 'simple.reqon');
20
- await writeFile(filePath, `
21
- mission Simple {
22
- source Api {
23
- auth: none,
24
- base: "https://api.example.com"
25
- }
26
-
27
- store data: memory("items")
28
-
29
- action Fetch {
30
- get "/items" { source: Api }
31
- store response -> data
32
- }
33
-
34
- run Fetch
35
- }
36
- `);
37
-
38
- const result = await loadMission(filePath);
39
-
40
- expect(result.sourceFiles).toHaveLength(1);
41
- expect(result.sourceFiles[0]).toContain('simple.reqon');
42
-
43
- const mission = result.program.statements.find(
44
- s => s.type === 'MissionDefinition'
45
- );
46
- expect(mission).toBeDefined();
47
- expect((mission as any).name).toBe('Simple');
48
- expect((mission as any).actions).toHaveLength(1);
49
- });
50
-
51
- it('loads a single .vague file', async () => {
52
- const filePath = join(TEST_DIR, 'simple.vague');
53
- await writeFile(filePath, `
54
- mission Simple {
55
- source Api {
56
- auth: none,
57
- base: "https://api.example.com"
58
- }
59
-
60
- store data: memory("items")
61
-
62
- action Fetch {
63
- fetch GET "/items" { source: Api }
64
- store response -> data
65
- }
66
-
67
- run Fetch
68
- }
69
- `);
70
-
71
- const result = await loadMission(filePath);
72
-
73
- expect(result.sourceFiles).toHaveLength(1);
74
- expect(result.sourceFiles[0]).toContain('simple.vague');
75
-
76
- const mission = result.program.statements.find(
77
- s => s.type === 'MissionDefinition'
78
- );
79
- expect(mission).toBeDefined();
80
- expect((mission as any).name).toBe('Simple');
81
- });
82
- });
83
-
84
- describe('folder loading', () => {
85
- it('loads mission from folder with separate action files', async () => {
86
- const missionDir = join(TEST_DIR, 'sync-invoices');
87
- await mkdir(missionDir, { recursive: true });
88
-
89
- // Root mission file
90
- await writeFile(join(missionDir, 'mission.reqon'), `
91
- mission SyncInvoices {
92
- source Xero {
93
- auth: oauth2,
94
- base: "https://api.xero.com"
95
- }
96
-
97
- store invoices: memory("invoices")
98
- store details: memory("invoice_details")
99
-
100
- run FetchList then Hydrate
101
- }
102
- `);
103
-
104
- // Action files
105
- await writeFile(join(missionDir, 'fetch-list.reqon'), `
106
- action FetchList {
107
- get "/invoices" { source: Xero }
108
- store response -> invoices
109
- }
110
- `);
111
-
112
- await writeFile(join(missionDir, 'hydrate.reqon'), `
113
- action Hydrate {
114
- for invoice in invoices {
115
- get "/invoices/{invoice.id}" { source: Xero }
116
- store response -> details
117
- }
118
- }
119
- `);
120
-
121
- const result = await loadMission(missionDir);
122
-
123
- expect(result.sourceFiles).toHaveLength(3);
124
- expect(result.baseDir).toContain('sync-invoices');
125
-
126
- const mission = result.program.statements.find(
127
- s => s.type === 'MissionDefinition'
128
- );
129
- expect(mission).toBeDefined();
130
- expect((mission as any).name).toBe('SyncInvoices');
131
- expect((mission as any).actions).toHaveLength(2);
132
-
133
- const actionNames = (mission as any).actions.map((a: any) => a.name);
134
- expect(actionNames).toContain('FetchList');
135
- expect(actionNames).toContain('Hydrate');
136
- });
137
-
138
- it('errors if mission.reqon is missing from folder', async () => {
139
- const missionDir = join(TEST_DIR, 'no-root');
140
- await mkdir(missionDir, { recursive: true });
141
-
142
- await writeFile(join(missionDir, 'action.reqon'), `
143
- action SomeAction {
144
- get "/test" { source: Api }
145
- }
146
- `);
147
-
148
- await expect(loadMission(missionDir)).rejects.toThrow(
149
- /must contain a root file/
150
- );
151
- });
152
-
153
- it('errors if action file contains a mission definition', async () => {
154
- const missionDir = join(TEST_DIR, 'bad-action');
155
- await mkdir(missionDir, { recursive: true });
156
-
157
- await writeFile(join(missionDir, 'mission.reqon'), `
158
- mission Main {
159
- source Api { auth: none, base: "https://api.example.com" }
160
- store data: memory("data")
161
- run Fetch
162
- }
163
- `);
164
-
165
- await writeFile(join(missionDir, 'nested.reqon'), `
166
- mission NestedMission {
167
- source Api { auth: none, base: "https://other.com" }
168
- store data: memory("data")
169
- run Something
170
- }
171
- `);
172
-
173
- await expect(loadMission(missionDir)).rejects.toThrow(
174
- /should not contain a mission definition/
175
- );
176
- });
177
-
178
- it('errors on duplicate action names', async () => {
179
- const missionDir = join(TEST_DIR, 'duplicate');
180
- await mkdir(missionDir, { recursive: true });
181
-
182
- await writeFile(join(missionDir, 'mission.reqon'), `
183
- mission Main {
184
- source Api { auth: none, base: "https://api.example.com" }
185
- store data: memory("data")
186
-
187
- action Fetch {
188
- get "/a" { source: Api }
189
- }
190
-
191
- run Fetch
192
- }
193
- `);
194
-
195
- await writeFile(join(missionDir, 'fetch.reqon'), `
196
- action Fetch {
197
- get "/b" { source: Api }
198
- }
199
- `);
200
-
201
- await expect(loadMission(missionDir)).rejects.toThrow(
202
- /Duplicate action definition: 'Fetch'/
203
- );
204
- });
205
-
206
- it('errors if pipeline references unknown action', async () => {
207
- const missionDir = join(TEST_DIR, 'unknown-action');
208
- await mkdir(missionDir, { recursive: true });
209
-
210
- await writeFile(join(missionDir, 'mission.reqon'), `
211
- mission Main {
212
- source Api { auth: none, base: "https://api.example.com" }
213
- store data: memory("data")
214
-
215
- action Fetch {
216
- get "/items" { source: Api }
217
- }
218
-
219
- run Fetch then NonExistent
220
- }
221
- `);
222
-
223
- await expect(loadMission(missionDir)).rejects.toThrow(
224
- /Pipeline references unknown action: 'NonExistent'/
225
- );
226
- });
227
-
228
- it('supports inline actions alongside external files', async () => {
229
- const missionDir = join(TEST_DIR, 'mixed');
230
- await mkdir(missionDir, { recursive: true });
231
-
232
- await writeFile(join(missionDir, 'mission.reqon'), `
233
- mission Mixed {
234
- source Api { auth: none, base: "https://api.example.com" }
235
- store data: memory("data")
236
-
237
- action InlineAction {
238
- get "/inline" { source: Api }
239
- }
240
-
241
- run InlineAction then ExternalAction
242
- }
243
- `);
244
-
245
- await writeFile(join(missionDir, 'external.reqon'), `
246
- action ExternalAction {
247
- get "/external" { source: Api }
248
- store response -> data
249
- }
250
- `);
251
-
252
- const result = await loadMission(missionDir);
253
-
254
- const mission = result.program.statements.find(
255
- s => s.type === 'MissionDefinition'
256
- );
257
- expect((mission as any).actions).toHaveLength(2);
258
- });
259
- });
260
-
261
- describe('isMissionFolder', () => {
262
- it('returns true for folder with mission.reqon', async () => {
263
- const missionDir = join(TEST_DIR, 'valid-folder');
264
- await mkdir(missionDir, { recursive: true });
265
- await writeFile(join(missionDir, 'mission.reqon'), 'mission X { run A }');
266
-
267
- expect(await isMissionFolder(missionDir)).toBe(true);
268
- });
269
-
270
- it('returns true for folder with mission.vague', async () => {
271
- const missionDir = join(TEST_DIR, 'vague-folder');
272
- await mkdir(missionDir, { recursive: true });
273
- await writeFile(join(missionDir, 'mission.vague'), 'mission X { run A }');
274
-
275
- expect(await isMissionFolder(missionDir)).toBe(true);
276
- });
277
-
278
- it('returns false for folder without mission.reqon', async () => {
279
- const missionDir = join(TEST_DIR, 'invalid-folder');
280
- await mkdir(missionDir, { recursive: true });
281
-
282
- expect(await isMissionFolder(missionDir)).toBe(false);
283
- });
284
-
285
- it('returns false for a file', async () => {
286
- const filePath = join(TEST_DIR, 'file.reqon');
287
- await writeFile(filePath, 'mission X { run A }');
288
-
289
- expect(await isMissionFolder(filePath)).toBe(false);
290
- });
291
- });
292
-
293
- describe('getMissionName', () => {
294
- it('extracts name from folder path', () => {
295
- expect(getMissionName('/path/to/sync-invoices')).toBe('sync-invoices');
296
- });
297
-
298
- it('extracts name from .reqon file path', () => {
299
- expect(getMissionName('/path/to/sync-invoices.reqon')).toBe('sync-invoices');
300
- });
301
-
302
- it('extracts name from .vague file path', () => {
303
- expect(getMissionName('/path/to/sync-invoices.vague')).toBe('sync-invoices');
304
- });
305
- });
306
-
307
- describe('.vague file priority', () => {
308
- it('prefers .vague over .reqon in folder mode', async () => {
309
- const missionDir = join(TEST_DIR, 'both-extensions');
310
- await mkdir(missionDir, { recursive: true });
311
-
312
- // Create both files - .vague should be preferred
313
- await writeFile(join(missionDir, 'mission.vague'), `
314
- mission FromVague {
315
- source Api { auth: none, base: "https://api.example.com" }
316
- store data: memory("data")
317
- action Fetch { fetch GET "/vague" { source: Api } }
318
- run Fetch
319
- }
320
- `);
321
- await writeFile(join(missionDir, 'mission.reqon'), `
322
- mission FromReqon {
323
- source Api { auth: none, base: "https://api.example.com" }
324
- store data: memory("data")
325
- action Fetch { fetch GET "/reqon" { source: Api } }
326
- run Fetch
327
- }
328
- `);
329
-
330
- const result = await loadMission(missionDir);
331
- const mission = result.program.statements.find(
332
- s => s.type === 'MissionDefinition'
333
- );
334
- expect((mission as any).name).toBe('FromVague');
335
- });
336
-
337
- it('loads folder with mission.vague and .vague action files', async () => {
338
- const missionDir = join(TEST_DIR, 'vague-folder-mode');
339
- await mkdir(missionDir, { recursive: true });
340
-
341
- await writeFile(join(missionDir, 'mission.vague'), `
342
- mission VagueMode {
343
- source Api { auth: none, base: "https://api.example.com" }
344
- store data: memory("data")
345
- run Fetch
346
- }
347
- `);
348
- await writeFile(join(missionDir, 'fetch.vague'), `
349
- action Fetch {
350
- fetch GET "/items" { source: Api }
351
- }
352
- `);
353
-
354
- const result = await loadMission(missionDir);
355
- expect(result.sourceFiles).toHaveLength(2);
356
- expect(result.sourceFiles[0]).toContain('mission.vague');
357
- expect(result.sourceFiles[1]).toContain('fetch.vague');
358
- });
359
- });
360
- });
package/src/oas/index.ts DELETED
@@ -1,4 +0,0 @@
1
- export { loadOAS, resolveOperation, getResponseSchema, clearCache } from './loader.js';
2
- export type { OASSource, OASOperation, OpenAPISpec } from './loader.js';
3
- export { validateResponse } from './validator.js';
4
- export type { ValidationResult, ValidationError } from './validator.js';