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,216 @@
1
+ import { describe, it, expect, vi } from 'vitest';
2
+ import { MissionExecutor } from './executor.js';
3
+ function createTestProgram(stages) {
4
+ const actions = stages.map((name) => ({
5
+ type: 'ActionDefinition',
6
+ name,
7
+ steps: [],
8
+ }));
9
+ const pipeline = {
10
+ type: 'PipelineDefinition',
11
+ stages: stages.map((action) => ({ action })),
12
+ };
13
+ const mission = {
14
+ type: 'MissionDefinition',
15
+ name: 'TestMission',
16
+ sources: [],
17
+ stores: [],
18
+ schemas: [],
19
+ actions,
20
+ pipeline,
21
+ };
22
+ return {
23
+ type: 'ReqonProgram',
24
+ statements: [mission],
25
+ };
26
+ }
27
+ describe('Progress Callbacks', () => {
28
+ describe('onExecutionStart', () => {
29
+ it('is called when execution begins', async () => {
30
+ const onExecutionStart = vi.fn();
31
+ const executor = new MissionExecutor({
32
+ progress: { onExecutionStart },
33
+ });
34
+ await executor.execute(createTestProgram(['StepA', 'StepB']));
35
+ expect(onExecutionStart).toHaveBeenCalledOnce();
36
+ expect(onExecutionStart).toHaveBeenCalledWith(expect.objectContaining({
37
+ mission: 'TestMission',
38
+ stageCount: 2,
39
+ isResume: false,
40
+ }));
41
+ });
42
+ it('includes executionId from ephemeral execution', async () => {
43
+ const onExecutionStart = vi.fn();
44
+ const executor = new MissionExecutor({
45
+ progress: { onExecutionStart },
46
+ });
47
+ await executor.execute(createTestProgram(['StepA']));
48
+ expect(onExecutionStart).toHaveBeenCalledWith(expect.objectContaining({
49
+ executionId: 'ephemeral',
50
+ }));
51
+ });
52
+ it('includes executionId when persistence enabled', async () => {
53
+ const onExecutionStart = vi.fn();
54
+ const executor = new MissionExecutor({
55
+ persistState: true,
56
+ dataDir: '.reqon-test-progress',
57
+ progress: { onExecutionStart },
58
+ });
59
+ await executor.execute(createTestProgram(['StepA']));
60
+ expect(onExecutionStart).toHaveBeenCalledWith(expect.objectContaining({
61
+ executionId: expect.stringMatching(/^exec_/),
62
+ }));
63
+ });
64
+ it('includes metadata if provided', async () => {
65
+ const onExecutionStart = vi.fn();
66
+ const executor = new MissionExecutor({
67
+ metadata: { tenant: 'acme', userId: '123' },
68
+ progress: { onExecutionStart },
69
+ });
70
+ await executor.execute(createTestProgram(['StepA']));
71
+ expect(onExecutionStart).toHaveBeenCalledWith(expect.objectContaining({
72
+ metadata: { tenant: 'acme', userId: '123' },
73
+ }));
74
+ });
75
+ });
76
+ describe('onExecutionComplete', () => {
77
+ it('is called when execution finishes successfully', async () => {
78
+ const onExecutionComplete = vi.fn();
79
+ const executor = new MissionExecutor({
80
+ progress: { onExecutionComplete },
81
+ });
82
+ await executor.execute(createTestProgram(['StepA', 'StepB']));
83
+ expect(onExecutionComplete).toHaveBeenCalledOnce();
84
+ expect(onExecutionComplete).toHaveBeenCalledWith(expect.objectContaining({
85
+ mission: 'TestMission',
86
+ success: true,
87
+ stagesCompleted: 2,
88
+ stagesFailed: 0,
89
+ errors: [],
90
+ }));
91
+ });
92
+ it('includes duration', async () => {
93
+ const onExecutionComplete = vi.fn();
94
+ const executor = new MissionExecutor({
95
+ progress: { onExecutionComplete },
96
+ });
97
+ await executor.execute(createTestProgram(['StepA']));
98
+ const event = onExecutionComplete.mock.calls[0][0];
99
+ expect(event.duration).toBeGreaterThanOrEqual(0);
100
+ });
101
+ });
102
+ describe('onStageStart', () => {
103
+ it('is called for each stage', async () => {
104
+ const onStageStart = vi.fn();
105
+ const executor = new MissionExecutor({
106
+ progress: { onStageStart },
107
+ });
108
+ await executor.execute(createTestProgram(['StepA', 'StepB', 'StepC']));
109
+ expect(onStageStart).toHaveBeenCalledTimes(3);
110
+ });
111
+ it('includes stage index and name', async () => {
112
+ const events = [];
113
+ const executor = new MissionExecutor({
114
+ progress: {
115
+ onStageStart: (e) => events.push(e),
116
+ },
117
+ });
118
+ await executor.execute(createTestProgram(['Fetch', 'Process', 'Store']));
119
+ expect(events[0]).toMatchObject({
120
+ stageIndex: 0,
121
+ stageName: 'Fetch',
122
+ totalStages: 3,
123
+ });
124
+ expect(events[1]).toMatchObject({
125
+ stageIndex: 1,
126
+ stageName: 'Process',
127
+ totalStages: 3,
128
+ });
129
+ expect(events[2]).toMatchObject({
130
+ stageIndex: 2,
131
+ stageName: 'Store',
132
+ totalStages: 3,
133
+ });
134
+ });
135
+ });
136
+ describe('onStageComplete', () => {
137
+ it('is called for each stage', async () => {
138
+ const onStageComplete = vi.fn();
139
+ const executor = new MissionExecutor({
140
+ progress: { onStageComplete },
141
+ });
142
+ await executor.execute(createTestProgram(['StepA', 'StepB']));
143
+ expect(onStageComplete).toHaveBeenCalledTimes(2);
144
+ });
145
+ it('includes success status and duration', async () => {
146
+ const events = [];
147
+ const executor = new MissionExecutor({
148
+ progress: {
149
+ onStageComplete: (e) => events.push(e),
150
+ },
151
+ });
152
+ await executor.execute(createTestProgram(['StepA']));
153
+ expect(events[0]).toMatchObject({
154
+ stageIndex: 0,
155
+ stageName: 'StepA',
156
+ success: true,
157
+ });
158
+ expect(events[0].duration).toBeGreaterThanOrEqual(0);
159
+ expect(events[0].error).toBeUndefined();
160
+ });
161
+ });
162
+ describe('callback order', () => {
163
+ it('fires callbacks in correct order', async () => {
164
+ const callOrder = [];
165
+ const executor = new MissionExecutor({
166
+ progress: {
167
+ onExecutionStart: () => callOrder.push('execStart'),
168
+ onExecutionComplete: () => callOrder.push('execComplete'),
169
+ onStageStart: (e) => callOrder.push(`stageStart:${e.stageName}`),
170
+ onStageComplete: (e) => callOrder.push(`stageComplete:${e.stageName}`),
171
+ },
172
+ });
173
+ await executor.execute(createTestProgram(['A', 'B']));
174
+ expect(callOrder).toEqual([
175
+ 'execStart',
176
+ 'stageStart:A',
177
+ 'stageComplete:A',
178
+ 'stageStart:B',
179
+ 'stageComplete:B',
180
+ 'execComplete',
181
+ ]);
182
+ });
183
+ });
184
+ describe('real-time UI example', () => {
185
+ it('can build a progress tracker', async () => {
186
+ const progressLog = [];
187
+ const executor = new MissionExecutor({
188
+ progress: {
189
+ onExecutionStart: (e) => {
190
+ progressLog.push(`Starting ${e.mission} (${e.stageCount} stages)`);
191
+ },
192
+ onStageStart: (e) => {
193
+ const pct = Math.round((e.stageIndex / e.totalStages) * 100);
194
+ progressLog.push(`[${pct}%] Running ${e.stageName}...`);
195
+ },
196
+ onStageComplete: (e) => {
197
+ const pct = Math.round(((e.stageIndex + 1) / e.totalStages) * 100);
198
+ progressLog.push(`[${pct}%] ${e.stageName} ${e.success ? 'done' : 'failed'}`);
199
+ },
200
+ onExecutionComplete: (e) => {
201
+ progressLog.push(`Finished: ${e.success ? 'SUCCESS' : 'FAILED'} in ${e.duration}ms`);
202
+ },
203
+ },
204
+ });
205
+ await executor.execute(createTestProgram(['Fetch', 'Process', 'Store']));
206
+ expect(progressLog[0]).toBe('Starting TestMission (3 stages)');
207
+ expect(progressLog[1]).toBe('[0%] Running Fetch...');
208
+ expect(progressLog[2]).toBe('[33%] Fetch done');
209
+ expect(progressLog[3]).toBe('[33%] Running Process...');
210
+ expect(progressLog[4]).toBe('[67%] Process done');
211
+ expect(progressLog[5]).toBe('[67%] Running Store...');
212
+ expect(progressLog[6]).toBe('[100%] Store done');
213
+ expect(progressLog[7]).toMatch(/^Finished: SUCCESS in \d+ms$/);
214
+ });
215
+ });
216
+ });
@@ -0,0 +1,16 @@
1
+ import type { SchemaDefinition } from 'vague-lang';
2
+ /**
3
+ * Check if a value matches a schema definition.
4
+ *
5
+ * Matching rules:
6
+ * - All required fields (non-optional) must be present
7
+ * - Field types must match (string, int, decimal, boolean, date)
8
+ * - Extra fields are allowed (open schema)
9
+ * - Nested objects are not deeply validated (future enhancement)
10
+ */
11
+ export declare function matchesSchema(value: unknown, schema: SchemaDefinition): boolean;
12
+ /**
13
+ * Find the first matching schema from a list.
14
+ * Returns the schema name or undefined if no match.
15
+ */
16
+ export declare function findMatchingSchema(value: unknown, schemas: Map<string, SchemaDefinition>, schemaNames: string[]): string | undefined;
@@ -0,0 +1,136 @@
1
+ /**
2
+ * Check if a value matches a schema definition.
3
+ *
4
+ * Matching rules:
5
+ * - All required fields (non-optional) must be present
6
+ * - Field types must match (string, int, decimal, boolean, date)
7
+ * - Extra fields are allowed (open schema)
8
+ * - Nested objects are not deeply validated (future enhancement)
9
+ */
10
+ export function matchesSchema(value, schema) {
11
+ if (typeof value !== 'object' || value === null) {
12
+ return false;
13
+ }
14
+ const obj = value;
15
+ for (const field of schema.fields) {
16
+ const fieldValue = obj[field.name];
17
+ const isOptional = field.optional === true;
18
+ // Required field must be present
19
+ if (fieldValue === undefined) {
20
+ if (!isOptional) {
21
+ return false;
22
+ }
23
+ continue;
24
+ }
25
+ // Null is allowed for optional fields
26
+ if (fieldValue === null && isOptional) {
27
+ continue;
28
+ }
29
+ // Check type if present
30
+ if (!matchesFieldType(fieldValue, field.fieldType)) {
31
+ return false;
32
+ }
33
+ }
34
+ return true;
35
+ }
36
+ /**
37
+ * Check if a value matches the expected field type
38
+ */
39
+ function matchesFieldType(value, fieldType) {
40
+ // Handle primitive types
41
+ if (fieldType.type === 'PrimitiveType') {
42
+ return matchesPrimitiveType(value, fieldType.name);
43
+ }
44
+ // Handle collection types (arrays)
45
+ if (fieldType.type === 'CollectionType') {
46
+ if (!Array.isArray(value)) {
47
+ return false;
48
+ }
49
+ // For now, don't validate element types
50
+ return true;
51
+ }
52
+ // Handle object/reference types
53
+ if (fieldType.type === 'ReferenceType') {
54
+ // For now, just check it's an object
55
+ return typeof value === 'object' && value !== null;
56
+ }
57
+ // Handle superposition types (unions) - any option matching is ok
58
+ if (fieldType.type === 'SuperpositionType') {
59
+ // Would need to check each option - be permissive for now
60
+ return true;
61
+ }
62
+ // Handle generator types (faker, etc.) - can't validate statically
63
+ if (fieldType.type === 'GeneratorType') {
64
+ return true;
65
+ }
66
+ // Handle expression types - can't validate statically
67
+ if (fieldType.type === 'ExpressionType') {
68
+ return true;
69
+ }
70
+ // Handle range types - check it's a number in range
71
+ if (fieldType.type === 'RangeType') {
72
+ return typeof value === 'number';
73
+ }
74
+ // Handle ordered sequence types (tuples)
75
+ if (fieldType.type === 'OrderedSequenceType') {
76
+ return Array.isArray(value);
77
+ }
78
+ // Unknown type - be permissive
79
+ return true;
80
+ }
81
+ /**
82
+ * Check if a value matches a primitive type
83
+ */
84
+ function matchesPrimitiveType(value, typeName) {
85
+ switch (typeName) {
86
+ case 'string':
87
+ return typeof value === 'string';
88
+ case 'int':
89
+ case 'integer':
90
+ return typeof value === 'number' && Number.isInteger(value);
91
+ case 'decimal':
92
+ case 'number':
93
+ case 'float':
94
+ case 'double':
95
+ return typeof value === 'number';
96
+ case 'boolean':
97
+ case 'bool':
98
+ return typeof value === 'boolean';
99
+ case 'date':
100
+ case 'datetime':
101
+ // Accept strings (ISO format) or Date objects
102
+ if (value instanceof Date)
103
+ return true;
104
+ if (typeof value === 'string') {
105
+ const parsed = Date.parse(value);
106
+ return !isNaN(parsed);
107
+ }
108
+ return false;
109
+ case 'any':
110
+ return true;
111
+ default:
112
+ // Unknown primitive - be permissive
113
+ return true;
114
+ }
115
+ }
116
+ /**
117
+ * Find the first matching schema from a list.
118
+ * Returns the schema name or undefined if no match.
119
+ */
120
+ export function findMatchingSchema(value, schemas, schemaNames) {
121
+ for (const name of schemaNames) {
122
+ // Handle wildcard
123
+ if (name === '_') {
124
+ return '_';
125
+ }
126
+ const schema = schemas.get(name);
127
+ if (!schema) {
128
+ // Schema not found - skip
129
+ continue;
130
+ }
131
+ if (matchesSchema(value, schema)) {
132
+ return name;
133
+ }
134
+ }
135
+ return undefined;
136
+ }
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,122 @@
1
+ import { describe, it, expect } from 'vitest';
2
+ import { matchesSchema, findMatchingSchema } from './schema-matcher.js';
3
+ describe('Schema Matcher', () => {
4
+ describe('matchesSchema', () => {
5
+ const userSchema = {
6
+ type: 'SchemaDefinition',
7
+ name: 'User',
8
+ fields: [
9
+ {
10
+ type: 'FieldDefinition',
11
+ name: 'id',
12
+ fieldType: { type: 'PrimitiveType', name: 'int' },
13
+ },
14
+ {
15
+ type: 'FieldDefinition',
16
+ name: 'name',
17
+ fieldType: { type: 'PrimitiveType', name: 'string' },
18
+ },
19
+ {
20
+ type: 'FieldDefinition',
21
+ name: 'email',
22
+ fieldType: { type: 'PrimitiveType', name: 'string' },
23
+ optional: true,
24
+ },
25
+ ],
26
+ };
27
+ it('matches valid object with all required fields', () => {
28
+ const value = { id: 1, name: 'Alice' };
29
+ expect(matchesSchema(value, userSchema)).toBe(true);
30
+ });
31
+ it('matches valid object with optional field', () => {
32
+ const value = { id: 1, name: 'Alice', email: 'alice@example.com' };
33
+ expect(matchesSchema(value, userSchema)).toBe(true);
34
+ });
35
+ it('matches object with extra fields (open schema)', () => {
36
+ const value = { id: 1, name: 'Alice', extra: 'ignored' };
37
+ expect(matchesSchema(value, userSchema)).toBe(true);
38
+ });
39
+ it('rejects object missing required field', () => {
40
+ const value = { id: 1 }; // missing 'name'
41
+ expect(matchesSchema(value, userSchema)).toBe(false);
42
+ });
43
+ it('rejects wrong type for field', () => {
44
+ const value = { id: 'not-a-number', name: 'Alice' };
45
+ expect(matchesSchema(value, userSchema)).toBe(false);
46
+ });
47
+ it('rejects null value', () => {
48
+ expect(matchesSchema(null, userSchema)).toBe(false);
49
+ });
50
+ it('rejects primitive value', () => {
51
+ expect(matchesSchema('string', userSchema)).toBe(false);
52
+ });
53
+ it('allows null for optional fields', () => {
54
+ const value = { id: 1, name: 'Alice', email: null };
55
+ expect(matchesSchema(value, userSchema)).toBe(true);
56
+ });
57
+ });
58
+ describe('findMatchingSchema', () => {
59
+ const successSchema = {
60
+ type: 'SchemaDefinition',
61
+ name: 'SuccessResponse',
62
+ fields: [
63
+ {
64
+ type: 'FieldDefinition',
65
+ name: 'data',
66
+ fieldType: { type: 'ReferenceType', path: { type: 'QualifiedName', parts: ['object'] } },
67
+ },
68
+ {
69
+ type: 'FieldDefinition',
70
+ name: 'status',
71
+ fieldType: { type: 'PrimitiveType', name: 'string' },
72
+ },
73
+ ],
74
+ };
75
+ const errorSchema = {
76
+ type: 'SchemaDefinition',
77
+ name: 'ErrorResponse',
78
+ fields: [
79
+ {
80
+ type: 'FieldDefinition',
81
+ name: 'error',
82
+ fieldType: { type: 'PrimitiveType', name: 'string' },
83
+ },
84
+ {
85
+ type: 'FieldDefinition',
86
+ name: 'code',
87
+ fieldType: { type: 'PrimitiveType', name: 'int' },
88
+ },
89
+ ],
90
+ };
91
+ const schemas = new Map([
92
+ ['SuccessResponse', successSchema],
93
+ ['ErrorResponse', errorSchema],
94
+ ]);
95
+ it('finds matching schema in order', () => {
96
+ const successValue = { data: { id: 1 }, status: 'ok' };
97
+ const result = findMatchingSchema(successValue, schemas, ['SuccessResponse', 'ErrorResponse']);
98
+ expect(result).toBe('SuccessResponse');
99
+ });
100
+ it('finds second schema if first does not match', () => {
101
+ const errorValue = { error: 'Not found', code: 404 };
102
+ const result = findMatchingSchema(errorValue, schemas, ['SuccessResponse', 'ErrorResponse']);
103
+ expect(result).toBe('ErrorResponse');
104
+ });
105
+ it('returns undefined if no schema matches', () => {
106
+ const value = { random: 'value' };
107
+ const result = findMatchingSchema(value, schemas, ['SuccessResponse', 'ErrorResponse']);
108
+ expect(result).toBeUndefined();
109
+ });
110
+ it('handles wildcard pattern', () => {
111
+ const value = { random: 'value' };
112
+ const result = findMatchingSchema(value, schemas, ['SuccessResponse', 'ErrorResponse', '_']);
113
+ expect(result).toBe('_');
114
+ });
115
+ it('respects schema order (first match wins)', () => {
116
+ // Create a schema that matches both
117
+ const ambiguousValue = { data: {}, status: 'ok', error: 'msg', code: 500 };
118
+ const result = findMatchingSchema(ambiguousValue, schemas, ['SuccessResponse', 'ErrorResponse']);
119
+ expect(result).toBe('SuccessResponse'); // First match wins
120
+ });
121
+ });
122
+ });
@@ -0,0 +1,57 @@
1
+ /**
2
+ * Execution control flow signals
3
+ *
4
+ * These are used for non-exceptional control flow during pipeline execution.
5
+ * They extend Error for easy stack-based propagation but represent expected
6
+ * flow control, not actual errors.
7
+ */
8
+ /**
9
+ * Signal thrown when a match arm triggers skip
10
+ */
11
+ export declare class SkipSignal extends Error {
12
+ constructor();
13
+ }
14
+ /**
15
+ * Signal thrown when a match arm triggers retry
16
+ */
17
+ export declare class RetrySignal extends Error {
18
+ backoff?: {
19
+ maxAttempts: number;
20
+ backoff: string;
21
+ initialDelay: number;
22
+ } | undefined;
23
+ constructor(backoff?: {
24
+ maxAttempts: number;
25
+ backoff: string;
26
+ initialDelay: number;
27
+ } | undefined);
28
+ }
29
+ /**
30
+ * Signal thrown when a match arm triggers jump
31
+ */
32
+ export declare class JumpSignal extends Error {
33
+ action: string;
34
+ then?: "retry" | "continue" | undefined;
35
+ constructor(action: string, then?: "retry" | "continue" | undefined);
36
+ }
37
+ /**
38
+ * Signal thrown when a match arm triggers queue
39
+ */
40
+ export declare class QueueSignal extends Error {
41
+ value: unknown;
42
+ target?: string | undefined;
43
+ constructor(value: unknown, target?: string | undefined);
44
+ }
45
+ /**
46
+ * Error thrown when a match step has no matching arm
47
+ */
48
+ export declare class NoMatchError extends Error {
49
+ value: unknown;
50
+ constructor(value: unknown);
51
+ }
52
+ /**
53
+ * Error thrown when a match arm triggers an abort
54
+ */
55
+ export declare class AbortError extends Error {
56
+ constructor(message?: string);
57
+ }
@@ -0,0 +1,73 @@
1
+ /**
2
+ * Execution control flow signals
3
+ *
4
+ * These are used for non-exceptional control flow during pipeline execution.
5
+ * They extend Error for easy stack-based propagation but represent expected
6
+ * flow control, not actual errors.
7
+ */
8
+ /**
9
+ * Signal thrown when a match arm triggers skip
10
+ */
11
+ export class SkipSignal extends Error {
12
+ constructor() {
13
+ super('Skip remaining steps');
14
+ this.name = 'SkipSignal';
15
+ }
16
+ }
17
+ /**
18
+ * Signal thrown when a match arm triggers retry
19
+ */
20
+ export class RetrySignal extends Error {
21
+ backoff;
22
+ constructor(backoff) {
23
+ super('Retry action');
24
+ this.backoff = backoff;
25
+ this.name = 'RetrySignal';
26
+ }
27
+ }
28
+ /**
29
+ * Signal thrown when a match arm triggers jump
30
+ */
31
+ export class JumpSignal extends Error {
32
+ action;
33
+ then;
34
+ constructor(action, then) {
35
+ super(`Jump to action: ${action}`);
36
+ this.action = action;
37
+ this.then = then;
38
+ this.name = 'JumpSignal';
39
+ }
40
+ }
41
+ /**
42
+ * Signal thrown when a match arm triggers queue
43
+ */
44
+ export class QueueSignal extends Error {
45
+ value;
46
+ target;
47
+ constructor(value, target) {
48
+ super('Queue for later processing');
49
+ this.value = value;
50
+ this.target = target;
51
+ this.name = 'QueueSignal';
52
+ }
53
+ }
54
+ /**
55
+ * Error thrown when a match step has no matching arm
56
+ */
57
+ export class NoMatchError extends Error {
58
+ value;
59
+ constructor(value) {
60
+ super('No matching schema found for response');
61
+ this.value = value;
62
+ this.name = 'NoMatchError';
63
+ }
64
+ }
65
+ /**
66
+ * Error thrown when a match arm triggers an abort
67
+ */
68
+ export class AbortError extends Error {
69
+ constructor(message) {
70
+ super(message ?? 'Execution aborted');
71
+ this.name = 'AbortError';
72
+ }
73
+ }
@@ -0,0 +1,17 @@
1
+ import type { ForStep, ActionStep } from '../../ast/nodes.js';
2
+ import type { StepHandler, StepHandlerDeps } from './types.js';
3
+ import type { ExecutionContext } from '../context.js';
4
+ export interface ForHandlerDeps extends StepHandlerDeps {
5
+ executeStep: (step: ActionStep, actionName: string, ctx: ExecutionContext) => Promise<void>;
6
+ actionName: string;
7
+ }
8
+ /**
9
+ * Handles for...in...where iteration steps
10
+ */
11
+ export declare class ForHandler implements StepHandler<ForStep> {
12
+ private deps;
13
+ constructor(deps: ForHandlerDeps);
14
+ execute(step: ForStep): Promise<void>;
15
+ private getCollection;
16
+ private executeForItem;
17
+ }