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,108 @@
1
+ /**
2
+ * Benchmark suite for Reqon
3
+ *
4
+ * Run all benchmarks: npx tsx src/benchmark/index.ts
5
+ * Run specific suite: npx tsx src/benchmark/index.ts --suite lexer
6
+ */
7
+
8
+ import { runLexerBenchmarks } from './lexer.bench.js';
9
+ import { runParserBenchmarks } from './parser.bench.js';
10
+ import { runEvaluatorBenchmarks } from './evaluator.bench.js';
11
+ import { runStoreBenchmarks } from './store.bench.js';
12
+ import { runResilienceBenchmarks } from './resilience.bench.js';
13
+ import { runE2EBenchmarks } from './e2e.bench.js';
14
+
15
+ const SUITES: Record<string, () => Promise<void>> = {
16
+ lexer: runLexerBenchmarks,
17
+ parser: runParserBenchmarks,
18
+ evaluator: runEvaluatorBenchmarks,
19
+ store: runStoreBenchmarks,
20
+ resilience: runResilienceBenchmarks,
21
+ e2e: runE2EBenchmarks,
22
+ };
23
+
24
+ function printUsage(): void {
25
+ console.log(`
26
+ Reqon Benchmark Suite
27
+ =====================
28
+
29
+ Usage:
30
+ npx tsx src/benchmark/index.ts [options]
31
+
32
+ Options:
33
+ --suite <name> Run a specific benchmark suite
34
+ --list List available benchmark suites
35
+ --help Show this help message
36
+
37
+ Available Suites:
38
+ lexer - Tokenization benchmarks
39
+ parser - Parsing benchmarks
40
+ evaluator - Expression evaluation benchmarks
41
+ store - Store adapter benchmarks
42
+ resilience - Circuit breaker and rate limiter benchmarks
43
+ e2e - End-to-end execution benchmarks
44
+ all - Run all benchmarks (default)
45
+
46
+ Examples:
47
+ npx tsx src/benchmark/index.ts # Run all benchmarks
48
+ npx tsx src/benchmark/index.ts --suite lexer # Run only lexer benchmarks
49
+ npx tsx src/benchmark/index.ts --suite parser # Run only parser benchmarks
50
+ `);
51
+ }
52
+
53
+ async function main(): Promise<void> {
54
+ const args = process.argv.slice(2);
55
+
56
+ // Parse arguments
57
+ let suiteName = 'all';
58
+ for (let i = 0; i < args.length; i++) {
59
+ if (args[i] === '--help' || args[i] === '-h') {
60
+ printUsage();
61
+ process.exit(0);
62
+ }
63
+ if (args[i] === '--list') {
64
+ console.log('\nAvailable benchmark suites:');
65
+ for (const name of Object.keys(SUITES)) {
66
+ console.log(` - ${name}`);
67
+ }
68
+ console.log(' - all (runs all suites)');
69
+ process.exit(0);
70
+ }
71
+ if (args[i] === '--suite' || args[i] === '-s') {
72
+ suiteName = args[++i];
73
+ }
74
+ }
75
+
76
+ console.log(`
77
+ ╔═══════════════════════════════════════════════════════════════╗
78
+ ║ REQON BENCHMARK SUITE ║
79
+ ╚═══════════════════════════════════════════════════════════════╝
80
+ `);
81
+
82
+ const startTime = Date.now();
83
+
84
+ if (suiteName === 'all') {
85
+ console.log('Running all benchmark suites...\n');
86
+ for (const [name, runFn] of Object.entries(SUITES)) {
87
+ console.log(`\n>>> Running ${name} benchmarks...`);
88
+ await runFn();
89
+ }
90
+ } else if (SUITES[suiteName]) {
91
+ console.log(`Running ${suiteName} benchmarks...\n`);
92
+ await SUITES[suiteName]();
93
+ } else {
94
+ console.error(`Unknown suite: ${suiteName}`);
95
+ console.error(`Available suites: ${Object.keys(SUITES).join(', ')}, all`);
96
+ process.exit(1);
97
+ }
98
+
99
+ const totalTime = ((Date.now() - startTime) / 1000).toFixed(2);
100
+ console.log(`\n${'═'.repeat(60)}`);
101
+ console.log(`Benchmarks completed in ${totalTime}s`);
102
+ console.log(`${'═'.repeat(60)}`);
103
+ }
104
+
105
+ main().catch((err) => {
106
+ console.error('Benchmark failed:', err);
107
+ process.exit(1);
108
+ });
@@ -0,0 +1,69 @@
1
+ /**
2
+ * Lexer benchmarks
3
+ */
4
+
5
+ import { ReqonLexer } from '../lexer/lexer.js';
6
+ import { BenchmarkSuite } from './utils.js';
7
+ import {
8
+ SIMPLE_DSL,
9
+ MEDIUM_DSL,
10
+ COMPLEX_DSL,
11
+ EXPRESSION_HEAVY_DSL,
12
+ generateLargeDSL,
13
+ } from './fixtures.js';
14
+
15
+ export async function runLexerBenchmarks(): Promise<void> {
16
+ const suite = new BenchmarkSuite('Lexer');
17
+
18
+ // Simple DSL tokenization
19
+ suite.addSync('simple-dsl', () => {
20
+ const lexer = new ReqonLexer(SIMPLE_DSL);
21
+ return lexer.tokenize();
22
+ });
23
+
24
+ // Medium complexity DSL
25
+ suite.addSync('medium-dsl', () => {
26
+ const lexer = new ReqonLexer(MEDIUM_DSL);
27
+ return lexer.tokenize();
28
+ });
29
+
30
+ // Complex DSL
31
+ suite.addSync('complex-dsl', () => {
32
+ const lexer = new ReqonLexer(COMPLEX_DSL);
33
+ return lexer.tokenize();
34
+ });
35
+
36
+ // Expression-heavy DSL
37
+ suite.addSync('expression-heavy', () => {
38
+ const lexer = new ReqonLexer(EXPRESSION_HEAVY_DSL);
39
+ return lexer.tokenize();
40
+ });
41
+
42
+ // Large generated DSL (10 actions)
43
+ const largeDsl10 = generateLargeDSL(10);
44
+ suite.addSync('large-10-actions', () => {
45
+ const lexer = new ReqonLexer(largeDsl10);
46
+ return lexer.tokenize();
47
+ });
48
+
49
+ // Large generated DSL (50 actions)
50
+ const largeDsl50 = generateLargeDSL(50);
51
+ suite.addSync('large-50-actions', () => {
52
+ const lexer = new ReqonLexer(largeDsl50);
53
+ return lexer.tokenize();
54
+ }, { iterations: 500 });
55
+
56
+ // Large generated DSL (100 actions) - stress test
57
+ const largeDsl100 = generateLargeDSL(100);
58
+ suite.addSync('large-100-actions', () => {
59
+ const lexer = new ReqonLexer(largeDsl100);
60
+ return lexer.tokenize();
61
+ }, { iterations: 100, warmupIterations: 10 });
62
+
63
+ suite.print();
64
+ }
65
+
66
+ // Run if executed directly
67
+ if (import.meta.url === `file://${process.argv[1]}`) {
68
+ runLexerBenchmarks().catch(console.error);
69
+ }
@@ -0,0 +1,103 @@
1
+ /**
2
+ * Parser benchmarks
3
+ */
4
+
5
+ import { ReqonLexer } from '../lexer/lexer.js';
6
+ import { ReqonParser } from '../parser/parser.js';
7
+ import { BenchmarkSuite } from './utils.js';
8
+ import {
9
+ SIMPLE_DSL,
10
+ MEDIUM_DSL,
11
+ COMPLEX_DSL,
12
+ EXPRESSION_HEAVY_DSL,
13
+ MATCH_HEAVY_DSL,
14
+ generateLargeDSL,
15
+ } from './fixtures.js';
16
+
17
+ // Pre-tokenize for parser-only benchmarks
18
+ function preTokenize(source: string) {
19
+ const lexer = new ReqonLexer(source);
20
+ return lexer.tokenize();
21
+ }
22
+
23
+ export async function runParserBenchmarks(): Promise<void> {
24
+ const suite = new BenchmarkSuite('Parser');
25
+
26
+ // Pre-tokenize all fixtures
27
+ const simpleTokens = preTokenize(SIMPLE_DSL);
28
+ const mediumTokens = preTokenize(MEDIUM_DSL);
29
+ const complexTokens = preTokenize(COMPLEX_DSL);
30
+ const expressionTokens = preTokenize(EXPRESSION_HEAVY_DSL);
31
+ const matchTokens = preTokenize(MATCH_HEAVY_DSL);
32
+ const large10Tokens = preTokenize(generateLargeDSL(10));
33
+ const large50Tokens = preTokenize(generateLargeDSL(50));
34
+
35
+ // Parser-only benchmarks (with pre-tokenized input)
36
+ suite.addSync('simple-dsl', () => {
37
+ const parser = new ReqonParser([...simpleTokens]);
38
+ return parser.parse();
39
+ });
40
+
41
+ suite.addSync('medium-dsl', () => {
42
+ const parser = new ReqonParser([...mediumTokens]);
43
+ return parser.parse();
44
+ });
45
+
46
+ suite.addSync('complex-dsl', () => {
47
+ const parser = new ReqonParser([...complexTokens]);
48
+ return parser.parse();
49
+ });
50
+
51
+ suite.addSync('expression-heavy', () => {
52
+ const parser = new ReqonParser([...expressionTokens]);
53
+ return parser.parse();
54
+ });
55
+
56
+ suite.addSync('match-heavy', () => {
57
+ const parser = new ReqonParser([...matchTokens]);
58
+ return parser.parse();
59
+ });
60
+
61
+ suite.addSync('large-10-actions', () => {
62
+ const parser = new ReqonParser([...large10Tokens]);
63
+ return parser.parse();
64
+ });
65
+
66
+ suite.addSync('large-50-actions', () => {
67
+ const parser = new ReqonParser([...large50Tokens]);
68
+ return parser.parse();
69
+ }, { iterations: 500 });
70
+
71
+ suite.print();
72
+
73
+ // Full pipeline benchmarks (lexer + parser)
74
+ const pipelineSuite = new BenchmarkSuite('Lexer + Parser Pipeline');
75
+
76
+ pipelineSuite.addSync('simple-full', () => {
77
+ const lexer = new ReqonLexer(SIMPLE_DSL);
78
+ const tokens = lexer.tokenize();
79
+ const parser = new ReqonParser(tokens);
80
+ return parser.parse();
81
+ });
82
+
83
+ pipelineSuite.addSync('medium-full', () => {
84
+ const lexer = new ReqonLexer(MEDIUM_DSL);
85
+ const tokens = lexer.tokenize();
86
+ const parser = new ReqonParser(tokens);
87
+ return parser.parse();
88
+ });
89
+
90
+ pipelineSuite.addSync('complex-full', () => {
91
+ const lexer = new ReqonLexer(COMPLEX_DSL);
92
+ const tokens = lexer.tokenize();
93
+ const parser = new ReqonParser(tokens);
94
+ return parser.parse();
95
+ });
96
+
97
+ pipelineSuite.print();
98
+ }
99
+
100
+ // Run if executed directly
101
+ if (import.meta.url === `file://${process.argv[1]}`) {
102
+ runParserBenchmarks().catch(console.error);
103
+ }
@@ -0,0 +1,193 @@
1
+ /**
2
+ * Resilience component benchmarks (Circuit Breaker and Rate Limiter)
3
+ */
4
+
5
+ import { CircuitBreaker } from '../auth/circuit-breaker.js';
6
+ import { AdaptiveRateLimiter } from '../auth/rate-limiter.js';
7
+ import { BenchmarkSuite } from './utils.js';
8
+
9
+ export async function runResilienceBenchmarks(): Promise<void> {
10
+ // Circuit Breaker benchmarks
11
+ const cbSuite = new BenchmarkSuite('CircuitBreaker');
12
+
13
+ // Basic operations in closed state
14
+ const closedBreaker = new CircuitBreaker();
15
+
16
+ cbSuite.addSync('canProceed-closed', () => {
17
+ return closedBreaker.canProceed('test-source', '/endpoint');
18
+ });
19
+
20
+ cbSuite.addSync('ensureCanProceed-closed', () => {
21
+ closedBreaker.ensureCanProceed('test-source', '/endpoint');
22
+ });
23
+
24
+ cbSuite.addSync('recordSuccess-closed', () => {
25
+ closedBreaker.recordSuccess('test-source', '/endpoint');
26
+ });
27
+
28
+ cbSuite.addSync('recordFailure-closed', () => {
29
+ const breaker = new CircuitBreaker({ failureThreshold: 1000 }); // High threshold to stay closed
30
+ breaker.recordFailure('test-source', '/endpoint');
31
+ });
32
+
33
+ // Configure operation
34
+ cbSuite.addSync('configure', () => {
35
+ const breaker = new CircuitBreaker();
36
+ breaker.configure('new-source', {
37
+ failureThreshold: 10,
38
+ resetTimeout: 60000,
39
+ });
40
+ });
41
+
42
+ // State transition to open
43
+ cbSuite.addSync('transition-to-open', () => {
44
+ const breaker = new CircuitBreaker({ failureThreshold: 3 });
45
+ breaker.recordFailure('source', '/ep');
46
+ breaker.recordFailure('source', '/ep');
47
+ breaker.recordFailure('source', '/ep');
48
+ return breaker.getStatus('source', '/ep');
49
+ });
50
+
51
+ // Multiple sources/endpoints
52
+ cbSuite.addSync('multi-source-check', () => {
53
+ const breaker = new CircuitBreaker();
54
+ for (let i = 0; i < 10; i++) {
55
+ breaker.canProceed(`source-${i}`, `/endpoint-${i}`);
56
+ }
57
+ });
58
+
59
+ // Simulate realistic request pattern
60
+ cbSuite.addSync('realistic-request-pattern', () => {
61
+ const breaker = new CircuitBreaker({ failureThreshold: 5 });
62
+ // 80% success, 20% failure
63
+ for (let i = 0; i < 100; i++) {
64
+ breaker.ensureCanProceed('api', '/data');
65
+ if (Math.random() > 0.2) {
66
+ breaker.recordSuccess('api', '/data');
67
+ } else {
68
+ breaker.recordFailure('api', '/data');
69
+ }
70
+ }
71
+ }, { iterations: 100 });
72
+
73
+ cbSuite.print();
74
+
75
+ // Rate Limiter benchmarks
76
+ const rlSuite = new BenchmarkSuite('AdaptiveRateLimiter');
77
+
78
+ const limiter = new AdaptiveRateLimiter();
79
+
80
+ await rlSuite.addAsync('canProceed-fresh', async () => {
81
+ const rl = new AdaptiveRateLimiter();
82
+ return await rl.canProceed('api', '/endpoint');
83
+ });
84
+
85
+ await rlSuite.addAsync('canProceed-existing', async () => {
86
+ return await limiter.canProceed('test-api', '/test-endpoint');
87
+ });
88
+
89
+ // Configure operation
90
+ rlSuite.addSync('configure', () => {
91
+ const rl = new AdaptiveRateLimiter();
92
+ rl.configure('api', {
93
+ strategy: 'pause',
94
+ maxWait: 60,
95
+ notifyAt: 5,
96
+ fallbackRpm: 100,
97
+ });
98
+ });
99
+
100
+ // Record response with rate limit info
101
+ await rlSuite.addAsync('recordResponse-with-headers', async () => {
102
+ const rl = new AdaptiveRateLimiter();
103
+ rl.recordResponse('api', {
104
+ remaining: 50,
105
+ limit: 100,
106
+ resetAt: new Date(Date.now() + 3600000),
107
+ }, '/endpoint');
108
+ });
109
+
110
+ // Record response without rate limit info
111
+ await rlSuite.addAsync('recordResponse-no-headers', async () => {
112
+ const rl = new AdaptiveRateLimiter();
113
+ rl.recordResponse('api', {}, '/endpoint');
114
+ });
115
+
116
+ // Record 429 response with retry-after
117
+ await rlSuite.addAsync('recordResponse-429', async () => {
118
+ const rl = new AdaptiveRateLimiter();
119
+ rl.recordResponse('api', {
120
+ retryAfter: 60,
121
+ }, '/endpoint');
122
+ });
123
+
124
+ // getStatus operation
125
+ await rlSuite.addAsync('getStatus', async () => {
126
+ return limiter.getStatus('test-api', '/test-endpoint');
127
+ });
128
+
129
+ // Multiple endpoints check
130
+ await rlSuite.addAsync('multi-endpoint-check', async () => {
131
+ const rl = new AdaptiveRateLimiter();
132
+ for (let i = 0; i < 10; i++) {
133
+ await rl.canProceed(`source-${i}`, `/endpoint-${i}`);
134
+ }
135
+ });
136
+
137
+ // Simulate realistic request pattern
138
+ await rlSuite.addAsync('realistic-request-pattern', async () => {
139
+ const rl = new AdaptiveRateLimiter();
140
+
141
+ for (let i = 0; i < 20; i++) {
142
+ await rl.canProceed('api', '/data');
143
+ rl.recordResponse('api', {
144
+ remaining: 95 - i,
145
+ limit: 100,
146
+ resetAt: new Date(Date.now() + 60000),
147
+ }, '/data');
148
+ }
149
+ }, { iterations: 100 });
150
+
151
+ rlSuite.print();
152
+
153
+ // Combined resilience pattern
154
+ const combinedSuite = new BenchmarkSuite('Combined Resilience Pattern');
155
+
156
+ await combinedSuite.addAsync('circuit-breaker-plus-rate-limiter', async () => {
157
+ const cb = new CircuitBreaker({ failureThreshold: 5 });
158
+ const rl = new AdaptiveRateLimiter();
159
+
160
+ // Simulate 50 requests
161
+ for (let i = 0; i < 50; i++) {
162
+ // Check circuit breaker
163
+ try {
164
+ cb.ensureCanProceed('api', '/data');
165
+ } catch {
166
+ continue;
167
+ }
168
+
169
+ // Check rate limiter
170
+ if (!(await rl.canProceed('api', '/data'))) {
171
+ continue;
172
+ }
173
+
174
+ // Simulate response
175
+ const success = Math.random() > 0.1;
176
+ if (success) {
177
+ cb.recordSuccess('api', '/data');
178
+ rl.recordResponse('api', {
179
+ remaining: 50,
180
+ }, '/data');
181
+ } else {
182
+ cb.recordFailure('api', '/data');
183
+ }
184
+ }
185
+ }, { iterations: 100 });
186
+
187
+ combinedSuite.print();
188
+ }
189
+
190
+ // Run if executed directly
191
+ if (import.meta.url === `file://${process.argv[1]}`) {
192
+ runResilienceBenchmarks().catch(console.error);
193
+ }
@@ -0,0 +1,147 @@
1
+ /**
2
+ * Store adapter benchmarks
3
+ */
4
+
5
+ import { MemoryStore } from '../stores/memory.js';
6
+ import { applyStoreFilter } from '../stores/types.js';
7
+ import { BenchmarkSuite, generateRecords } from './utils.js';
8
+
9
+ export async function runStoreBenchmarks(): Promise<void> {
10
+ const suite = new BenchmarkSuite('MemoryStore Operations');
11
+
12
+ // Prepare test data
13
+ const smallRecords = generateRecords(100);
14
+ const mediumRecords = generateRecords(1000);
15
+ const largeRecords = generateRecords(10000);
16
+
17
+ // Individual set operations
18
+ await suite.addAsync('set-single', async () => {
19
+ const store = new MemoryStore('test');
20
+ await store.set('key-1', { id: '1', value: 42 });
21
+ });
22
+
23
+ // Batch set operations
24
+ await suite.addAsync('set-100-records', async () => {
25
+ const store = new MemoryStore('test');
26
+ for (const record of smallRecords) {
27
+ await store.set(record.id as string, record);
28
+ }
29
+ }, { iterations: 100 });
30
+
31
+ await suite.addAsync('set-1000-records', async () => {
32
+ const store = new MemoryStore('test');
33
+ for (const record of mediumRecords) {
34
+ await store.set(record.id as string, record);
35
+ }
36
+ }, { iterations: 50, warmupIterations: 5 });
37
+
38
+ // Get operations (pre-populate store)
39
+ const prePopulatedStore = new MemoryStore('prepopulated');
40
+ for (const record of mediumRecords) {
41
+ await prePopulatedStore.set(record.id as string, record);
42
+ }
43
+
44
+ await suite.addAsync('get-existing', async () => {
45
+ await prePopulatedStore.get('record-500');
46
+ });
47
+
48
+ await suite.addAsync('get-nonexistent', async () => {
49
+ await prePopulatedStore.get('nonexistent-key');
50
+ });
51
+
52
+ await suite.addAsync('get-100-sequential', async () => {
53
+ for (let i = 0; i < 100; i++) {
54
+ await prePopulatedStore.get(`record-${i}`);
55
+ }
56
+ }, { iterations: 100 });
57
+
58
+ // Update operations
59
+ await suite.addAsync('update-existing', async () => {
60
+ await prePopulatedStore.update('record-500', { updated: true });
61
+ });
62
+
63
+ await suite.addAsync('update-100-records', async () => {
64
+ for (let i = 0; i < 100; i++) {
65
+ await prePopulatedStore.update(`record-${i}`, { updated: true });
66
+ }
67
+ }, { iterations: 100 });
68
+
69
+ // List operations
70
+ await suite.addAsync('list-all-1000', async () => {
71
+ await prePopulatedStore.list();
72
+ }, { iterations: 100 });
73
+
74
+ // Large store for list benchmarks
75
+ const largeStore = new MemoryStore('large');
76
+ for (const record of largeRecords) {
77
+ await largeStore.set(record.id as string, record);
78
+ }
79
+
80
+ await suite.addAsync('list-all-10000', async () => {
81
+ await largeStore.list();
82
+ }, { iterations: 20, warmupIterations: 5 });
83
+
84
+ // List with filters
85
+ await suite.addAsync('list-with-limit-100', async () => {
86
+ await largeStore.list({ limit: 100 });
87
+ }, { iterations: 100 });
88
+
89
+ await suite.addAsync('list-with-offset-and-limit', async () => {
90
+ await largeStore.list({ offset: 5000, limit: 100 });
91
+ }, { iterations: 100 });
92
+
93
+ await suite.addAsync('list-with-where-clause', async () => {
94
+ await largeStore.list({ where: { active: true } });
95
+ }, { iterations: 50, warmupIterations: 5 });
96
+
97
+ // Delete operations
98
+ await suite.addAsync('delete-single', async () => {
99
+ const store = new MemoryStore('test');
100
+ await store.set('key-1', { id: '1' });
101
+ await store.delete('key-1');
102
+ });
103
+
104
+ suite.print();
105
+
106
+ // Filter function benchmarks
107
+ const filterSuite = new BenchmarkSuite('applyStoreFilter');
108
+
109
+ filterSuite.addSync('no-filter-1000', () => {
110
+ return applyStoreFilter(mediumRecords);
111
+ });
112
+
113
+ filterSuite.addSync('no-filter-10000', () => {
114
+ return applyStoreFilter(largeRecords);
115
+ }, { iterations: 100 });
116
+
117
+ filterSuite.addSync('limit-only-1000', () => {
118
+ return applyStoreFilter(mediumRecords, { limit: 100 });
119
+ });
120
+
121
+ filterSuite.addSync('offset-limit-1000', () => {
122
+ return applyStoreFilter(mediumRecords, { offset: 500, limit: 100 });
123
+ });
124
+
125
+ filterSuite.addSync('where-simple-1000', () => {
126
+ return applyStoreFilter(mediumRecords, { where: { active: true } });
127
+ });
128
+
129
+ filterSuite.addSync('where-simple-10000', () => {
130
+ return applyStoreFilter(largeRecords, { where: { active: true } });
131
+ }, { iterations: 100 });
132
+
133
+ filterSuite.addSync('where-multiple-conditions', () => {
134
+ return applyStoreFilter(largeRecords, {
135
+ where: { active: true },
136
+ offset: 100,
137
+ limit: 50,
138
+ });
139
+ }, { iterations: 100 });
140
+
141
+ filterSuite.print();
142
+ }
143
+
144
+ // Run if executed directly
145
+ if (import.meta.url === `file://${process.argv[1]}`) {
146
+ runStoreBenchmarks().catch(console.error);
147
+ }