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,379 @@
1
+ /**
2
+ * Circuit Breaker implementation for HTTP requests.
3
+ *
4
+ * Prevents repeated failures from cascading by automatically detecting
5
+ * failure patterns and "opening" the circuit to fail fast.
6
+ *
7
+ * States:
8
+ * - CLOSED: Normal operation, requests pass through
9
+ * - OPEN: Circuit tripped, requests fail immediately
10
+ * - HALF_OPEN: Testing if service recovered, limited requests allowed
11
+ */
12
+
13
+ export type CircuitState = 'closed' | 'open' | 'half_open';
14
+
15
+ export interface CircuitBreakerConfig {
16
+ /** Number of failures before opening circuit (default: 5) */
17
+ failureThreshold?: number;
18
+ /** Time in ms before attempting recovery (default: 30000 = 30s) */
19
+ resetTimeout?: number;
20
+ /** Number of successful requests in half-open to close circuit (default: 2) */
21
+ successThreshold?: number;
22
+ /** Time window in ms for counting failures (default: 60000 = 60s) */
23
+ failureWindow?: number;
24
+ /** HTTP status codes to count as failures (default: 500-599) */
25
+ failureStatusCodes?: number[];
26
+ /** Whether to count network errors as failures (default: true) */
27
+ countNetworkErrors?: boolean;
28
+ }
29
+
30
+ export interface CircuitBreakerStatus {
31
+ state: CircuitState;
32
+ failures: number;
33
+ successes: number;
34
+ lastFailureTime?: Date;
35
+ nextAttemptTime?: Date;
36
+ isOpen: boolean;
37
+ }
38
+
39
+ export interface CircuitBreakerEvent {
40
+ source: string;
41
+ endpoint?: string;
42
+ state: CircuitState;
43
+ previousState: CircuitState;
44
+ failures: number;
45
+ reason?: string;
46
+ }
47
+
48
+ export interface CircuitBreakerCallbacks {
49
+ /** Called when circuit opens (too many failures) */
50
+ onOpen?: (event: CircuitBreakerEvent) => void;
51
+ /** Called when circuit closes (recovery successful) */
52
+ onClose?: (event: CircuitBreakerEvent) => void;
53
+ /** Called when circuit enters half-open state */
54
+ onHalfOpen?: (event: CircuitBreakerEvent) => void;
55
+ /** Called when a request is rejected due to open circuit */
56
+ onRejected?: (event: { source: string; endpoint?: string; nextAttemptIn: number }) => void;
57
+ }
58
+
59
+ interface CircuitEntry {
60
+ state: CircuitState;
61
+ failures: number;
62
+ successes: number;
63
+ failureTimestamps: number[];
64
+ lastFailureTime?: number;
65
+ openedAt?: number;
66
+ config: Required<CircuitBreakerConfig>;
67
+ }
68
+
69
+ const DEFAULT_CONFIG: Required<CircuitBreakerConfig> = {
70
+ failureThreshold: 5,
71
+ resetTimeout: 30000,
72
+ successThreshold: 2,
73
+ failureWindow: 60000,
74
+ failureStatusCodes: [500, 501, 502, 503, 504],
75
+ countNetworkErrors: true,
76
+ };
77
+
78
+ /**
79
+ * Error thrown when circuit breaker is open
80
+ */
81
+ export class CircuitBreakerError extends Error {
82
+ constructor(
83
+ public readonly source: string,
84
+ public readonly endpoint: string | undefined,
85
+ public readonly nextAttemptIn: number
86
+ ) {
87
+ super(
88
+ `Circuit breaker open for ${source}${endpoint ? `:${endpoint}` : ''}. ` +
89
+ `Next attempt in ${Math.ceil(nextAttemptIn / 1000)}s`
90
+ );
91
+ this.name = 'CircuitBreakerError';
92
+ }
93
+ }
94
+
95
+ /**
96
+ * Circuit breaker for managing failure detection and recovery
97
+ */
98
+ export class CircuitBreaker {
99
+ private circuits: Map<string, CircuitEntry> = new Map();
100
+ private callbacks: CircuitBreakerCallbacks = {};
101
+ private defaultConfig: Required<CircuitBreakerConfig>;
102
+
103
+ constructor(defaultConfig?: CircuitBreakerConfig) {
104
+ this.defaultConfig = { ...DEFAULT_CONFIG, ...defaultConfig };
105
+ }
106
+
107
+ /**
108
+ * Configure circuit breaker for a specific source
109
+ */
110
+ configure(source: string, config: CircuitBreakerConfig): void {
111
+ const key = this.getKey(source);
112
+ const existing = this.circuits.get(key);
113
+
114
+ if (existing) {
115
+ existing.config = { ...this.defaultConfig, ...config };
116
+ } else {
117
+ this.circuits.set(key, this.createEntry({ ...this.defaultConfig, ...config }));
118
+ }
119
+ }
120
+
121
+ /**
122
+ * Set event callbacks
123
+ */
124
+ setCallbacks(callbacks: CircuitBreakerCallbacks): void {
125
+ this.callbacks = callbacks;
126
+ }
127
+
128
+ /**
129
+ * Check if a request can proceed (throws if circuit is open)
130
+ */
131
+ canProceed(source: string, endpoint?: string): boolean {
132
+ const entry = this.getOrCreateEntry(source, endpoint);
133
+ const now = Date.now();
134
+
135
+ if (entry.state === 'closed') {
136
+ return true;
137
+ }
138
+
139
+ if (entry.state === 'open') {
140
+ const timeSinceOpen = now - (entry.openedAt ?? now);
141
+
142
+ if (timeSinceOpen >= entry.config.resetTimeout) {
143
+ // Transition to half-open
144
+ this.transitionTo(entry, 'half_open', source, endpoint);
145
+ return true;
146
+ }
147
+
148
+ return false;
149
+ }
150
+
151
+ // Half-open: allow the request through for testing
152
+ return true;
153
+ }
154
+
155
+ /**
156
+ * Ensure request can proceed, throwing CircuitBreakerError if not
157
+ */
158
+ ensureCanProceed(source: string, endpoint?: string): void {
159
+ if (!this.canProceed(source, endpoint)) {
160
+ const entry = this.getOrCreateEntry(source, endpoint);
161
+ const now = Date.now();
162
+ const nextAttemptIn = entry.config.resetTimeout - (now - (entry.openedAt ?? now));
163
+
164
+ this.callbacks.onRejected?.({
165
+ source,
166
+ endpoint,
167
+ nextAttemptIn,
168
+ });
169
+
170
+ throw new CircuitBreakerError(source, endpoint, nextAttemptIn);
171
+ }
172
+ }
173
+
174
+ /**
175
+ * Record a successful request
176
+ */
177
+ recordSuccess(source: string, endpoint?: string): void {
178
+ const entry = this.getOrCreateEntry(source, endpoint);
179
+
180
+ if (entry.state === 'half_open') {
181
+ entry.successes++;
182
+
183
+ if (entry.successes >= entry.config.successThreshold) {
184
+ // Recovery successful, close the circuit
185
+ this.transitionTo(entry, 'closed', source, endpoint);
186
+ }
187
+ } else if (entry.state === 'closed') {
188
+ // Clear old failures from window
189
+ this.pruneOldFailures(entry);
190
+ }
191
+ }
192
+
193
+ /**
194
+ * Record a failed request
195
+ */
196
+ recordFailure(source: string, endpoint?: string, statusCode?: number, isNetworkError = false): void {
197
+ const entry = this.getOrCreateEntry(source, endpoint);
198
+ const config = entry.config;
199
+
200
+ // Check if this failure type should be counted
201
+ const isFailureStatus = statusCode !== undefined && config.failureStatusCodes.includes(statusCode);
202
+ const shouldCount = isFailureStatus || (isNetworkError && config.countNetworkErrors);
203
+
204
+ if (!shouldCount) {
205
+ return;
206
+ }
207
+
208
+ const now = Date.now();
209
+
210
+ if (entry.state === 'half_open') {
211
+ // Any failure in half-open immediately re-opens circuit
212
+ this.transitionTo(entry, 'open', source, endpoint, 'Failure during recovery attempt');
213
+ return;
214
+ }
215
+
216
+ if (entry.state === 'closed') {
217
+ // Prune old failures and add new one
218
+ this.pruneOldFailures(entry);
219
+ entry.failureTimestamps.push(now);
220
+ entry.failures = entry.failureTimestamps.length;
221
+ entry.lastFailureTime = now;
222
+
223
+ // Check if we should open the circuit
224
+ if (entry.failures >= config.failureThreshold) {
225
+ this.transitionTo(
226
+ entry,
227
+ 'open',
228
+ source,
229
+ endpoint,
230
+ `${entry.failures} failures in ${config.failureWindow}ms window`
231
+ );
232
+ }
233
+ }
234
+ }
235
+
236
+ /**
237
+ * Get current status for a source/endpoint
238
+ */
239
+ getStatus(source: string, endpoint?: string): CircuitBreakerStatus {
240
+ const entry = this.getOrCreateEntry(source, endpoint);
241
+ const now = Date.now();
242
+
243
+ let nextAttemptTime: Date | undefined;
244
+ if (entry.state === 'open' && entry.openedAt) {
245
+ const nextAttemptMs = entry.openedAt + entry.config.resetTimeout;
246
+ nextAttemptTime = new Date(nextAttemptMs);
247
+ }
248
+
249
+ return {
250
+ state: entry.state,
251
+ failures: entry.failures,
252
+ successes: entry.successes,
253
+ lastFailureTime: entry.lastFailureTime ? new Date(entry.lastFailureTime) : undefined,
254
+ nextAttemptTime,
255
+ isOpen: entry.state === 'open',
256
+ };
257
+ }
258
+
259
+ /**
260
+ * Force reset a circuit to closed state
261
+ */
262
+ reset(source: string, endpoint?: string): void {
263
+ const key = this.getKey(source, endpoint);
264
+ const entry = this.circuits.get(key);
265
+
266
+ if (entry) {
267
+ const previousState = entry.state;
268
+ entry.state = 'closed';
269
+ entry.failures = 0;
270
+ entry.successes = 0;
271
+ entry.failureTimestamps = [];
272
+ entry.lastFailureTime = undefined;
273
+ entry.openedAt = undefined;
274
+
275
+ if (previousState !== 'closed') {
276
+ this.callbacks.onClose?.({
277
+ source,
278
+ endpoint,
279
+ state: 'closed',
280
+ previousState,
281
+ failures: 0,
282
+ reason: 'Manual reset',
283
+ });
284
+ }
285
+ }
286
+ }
287
+
288
+ /**
289
+ * Get all circuit statuses
290
+ */
291
+ getAllStatuses(): Map<string, CircuitBreakerStatus> {
292
+ const result = new Map<string, CircuitBreakerStatus>();
293
+
294
+ for (const [key, entry] of this.circuits) {
295
+ const [source, endpoint] = key.split(':');
296
+ result.set(key, this.getStatus(source, endpoint === '' ? undefined : endpoint));
297
+ }
298
+
299
+ return result;
300
+ }
301
+
302
+ private getKey(source: string, endpoint?: string): string {
303
+ return endpoint ? `${source}:${endpoint}` : source;
304
+ }
305
+
306
+ private createEntry(config: Required<CircuitBreakerConfig>): CircuitEntry {
307
+ return {
308
+ state: 'closed',
309
+ failures: 0,
310
+ successes: 0,
311
+ failureTimestamps: [],
312
+ config,
313
+ };
314
+ }
315
+
316
+ private getOrCreateEntry(source: string, endpoint?: string): CircuitEntry {
317
+ const key = this.getKey(source, endpoint);
318
+ let entry = this.circuits.get(key);
319
+
320
+ if (!entry) {
321
+ // Check for source-level config
322
+ const sourceEntry = this.circuits.get(source);
323
+ const config = sourceEntry?.config ?? this.defaultConfig;
324
+ entry = this.createEntry(config);
325
+ this.circuits.set(key, entry);
326
+ }
327
+
328
+ return entry;
329
+ }
330
+
331
+ private pruneOldFailures(entry: CircuitEntry): void {
332
+ const now = Date.now();
333
+ const windowStart = now - entry.config.failureWindow;
334
+ entry.failureTimestamps = entry.failureTimestamps.filter((ts) => ts >= windowStart);
335
+ entry.failures = entry.failureTimestamps.length;
336
+ }
337
+
338
+ private transitionTo(
339
+ entry: CircuitEntry,
340
+ newState: CircuitState,
341
+ source: string,
342
+ endpoint?: string,
343
+ reason?: string
344
+ ): void {
345
+ const previousState = entry.state;
346
+ entry.state = newState;
347
+
348
+ const event: CircuitBreakerEvent = {
349
+ source,
350
+ endpoint,
351
+ state: newState,
352
+ previousState,
353
+ failures: entry.failures,
354
+ reason,
355
+ };
356
+
357
+ switch (newState) {
358
+ case 'open':
359
+ entry.openedAt = Date.now();
360
+ entry.successes = 0;
361
+ this.callbacks.onOpen?.(event);
362
+ break;
363
+
364
+ case 'half_open':
365
+ entry.successes = 0;
366
+ this.callbacks.onHalfOpen?.(event);
367
+ break;
368
+
369
+ case 'closed':
370
+ entry.failures = 0;
371
+ entry.successes = 0;
372
+ entry.failureTimestamps = [];
373
+ entry.lastFailureTime = undefined;
374
+ entry.openedAt = undefined;
375
+ this.callbacks.onClose?.(event);
376
+ break;
377
+ }
378
+ }
379
+ }
@@ -0,0 +1,273 @@
1
+ import { describe, it, expect, beforeEach, afterEach } from 'vitest';
2
+ import {
3
+ resolveEnvString,
4
+ hasEnvReference,
5
+ resolveCredentials,
6
+ credentialsFromEnv,
7
+ } from './credentials.js';
8
+
9
+ describe('resolveEnvString', () => {
10
+ const originalEnv = process.env;
11
+
12
+ beforeEach(() => {
13
+ process.env = { ...originalEnv };
14
+ });
15
+
16
+ afterEach(() => {
17
+ process.env = originalEnv;
18
+ });
19
+
20
+ it('should resolve $VAR_NAME pattern', () => {
21
+ process.env.MY_TOKEN = 'secret123';
22
+ expect(resolveEnvString('Bearer $MY_TOKEN')).toBe('Bearer secret123');
23
+ });
24
+
25
+ it('should resolve ${VAR_NAME} pattern', () => {
26
+ process.env.API_KEY = 'key456';
27
+ expect(resolveEnvString('${API_KEY}')).toBe('key456');
28
+ });
29
+
30
+ it('should resolve ${VAR_NAME:-default} pattern with value present', () => {
31
+ process.env.DB_HOST = 'production.db';
32
+ expect(resolveEnvString('${DB_HOST:-localhost}')).toBe('production.db');
33
+ });
34
+
35
+ it('should use default when env var is missing', () => {
36
+ delete process.env.MISSING_VAR;
37
+ expect(resolveEnvString('${MISSING_VAR:-fallback}')).toBe('fallback');
38
+ });
39
+
40
+ it('should return empty string for missing var without default', () => {
41
+ delete process.env.MISSING_VAR;
42
+ expect(resolveEnvString('prefix-${MISSING_VAR}-suffix')).toBe('prefix--suffix');
43
+ });
44
+
45
+ it('should resolve multiple variables in one string', () => {
46
+ process.env.HOST = 'api.example.com';
47
+ process.env.PORT = '8080';
48
+ expect(resolveEnvString('https://${HOST}:${PORT}/path')).toBe(
49
+ 'https://api.example.com:8080/path'
50
+ );
51
+ });
52
+
53
+ it('should handle mixed patterns', () => {
54
+ process.env.USER = 'admin';
55
+ process.env.PASS = 'secret';
56
+ expect(resolveEnvString('$USER:${PASS}')).toBe('admin:secret');
57
+ });
58
+
59
+ it('should not modify strings without env references', () => {
60
+ expect(resolveEnvString('plain text')).toBe('plain text');
61
+ });
62
+
63
+ it('should handle empty default value', () => {
64
+ delete process.env.EMPTY_DEFAULT;
65
+ expect(resolveEnvString('${EMPTY_DEFAULT:-}')).toBe('');
66
+ });
67
+ });
68
+
69
+ describe('hasEnvReference', () => {
70
+ it('should detect $VAR pattern', () => {
71
+ expect(hasEnvReference('token: $API_KEY')).toBe(true);
72
+ });
73
+
74
+ it('should detect ${VAR} pattern', () => {
75
+ expect(hasEnvReference('${SECRET}')).toBe(true);
76
+ });
77
+
78
+ it('should detect ${VAR:-default} pattern', () => {
79
+ expect(hasEnvReference('${VAR:-default}')).toBe(true);
80
+ });
81
+
82
+ it('should return false for plain strings', () => {
83
+ expect(hasEnvReference('no variables here')).toBe(false);
84
+ });
85
+
86
+ it('should return false for partial patterns', () => {
87
+ expect(hasEnvReference('$ not a var')).toBe(false);
88
+ });
89
+ });
90
+
91
+ describe('resolveCredentials', () => {
92
+ const originalEnv = process.env;
93
+
94
+ beforeEach(() => {
95
+ process.env = { ...originalEnv };
96
+ });
97
+
98
+ afterEach(() => {
99
+ process.env = originalEnv;
100
+ });
101
+
102
+ it('should resolve env vars in nested objects', () => {
103
+ process.env.GITHUB_TOKEN = 'ghp_xxx';
104
+ process.env.XERO_CLIENT_ID = 'client123';
105
+ process.env.XERO_CLIENT_SECRET = 'secret456';
106
+
107
+ const config = {
108
+ GitHub: {
109
+ type: 'bearer',
110
+ token: '$GITHUB_TOKEN',
111
+ },
112
+ Xero: {
113
+ type: 'oauth2',
114
+ clientId: '${XERO_CLIENT_ID}',
115
+ clientSecret: '${XERO_CLIENT_SECRET}',
116
+ tokenEndpoint: 'https://identity.xero.com/connect/token',
117
+ },
118
+ };
119
+
120
+ const resolved = resolveCredentials(config);
121
+
122
+ expect(resolved).toEqual({
123
+ GitHub: {
124
+ type: 'bearer',
125
+ token: 'ghp_xxx',
126
+ },
127
+ Xero: {
128
+ type: 'oauth2',
129
+ clientId: 'client123',
130
+ clientSecret: 'secret456',
131
+ tokenEndpoint: 'https://identity.xero.com/connect/token',
132
+ },
133
+ });
134
+ });
135
+
136
+ it('should resolve env vars in arrays', () => {
137
+ process.env.ITEM1 = 'first';
138
+ process.env.ITEM2 = 'second';
139
+
140
+ const config = ['$ITEM1', '${ITEM2}', 'static'];
141
+ const resolved = resolveCredentials(config);
142
+
143
+ expect(resolved).toEqual(['first', 'second', 'static']);
144
+ });
145
+
146
+ it('should handle null and undefined', () => {
147
+ expect(resolveCredentials(null)).toBe(null);
148
+ expect(resolveCredentials(undefined)).toBe(undefined);
149
+ });
150
+
151
+ it('should preserve non-string values', () => {
152
+ const config = {
153
+ timeout: 5000,
154
+ enabled: true,
155
+ items: [1, 2, 3],
156
+ };
157
+
158
+ expect(resolveCredentials(config)).toEqual(config);
159
+ });
160
+ });
161
+
162
+ describe('credentialsFromEnv', () => {
163
+ const originalEnv = process.env;
164
+
165
+ beforeEach(() => {
166
+ process.env = { ...originalEnv };
167
+ });
168
+
169
+ afterEach(() => {
170
+ process.env = originalEnv;
171
+ });
172
+
173
+ it('should build credentials from REQON_* env vars', () => {
174
+ process.env.REQON_GITHUB_TOKEN = 'ghp_xxx';
175
+ process.env.REQON_GITHUB_TYPE = 'bearer';
176
+
177
+ const creds = credentialsFromEnv(['GitHub']);
178
+
179
+ expect(creds).toEqual({
180
+ GitHub: {
181
+ type: 'bearer',
182
+ token: 'ghp_xxx',
183
+ },
184
+ });
185
+ });
186
+
187
+ it('should default to bearer when token is present without type', () => {
188
+ process.env.REQON_STRIPE_TOKEN = 'sk_live_xxx';
189
+
190
+ const creds = credentialsFromEnv(['Stripe']);
191
+
192
+ expect(creds).toEqual({
193
+ Stripe: {
194
+ type: 'bearer',
195
+ token: 'sk_live_xxx',
196
+ },
197
+ });
198
+ });
199
+
200
+ it('should handle oauth2 credentials', () => {
201
+ process.env.REQON_XERO_TYPE = 'oauth2';
202
+ process.env.REQON_XERO_ACCESS_TOKEN = 'access123';
203
+ process.env.REQON_XERO_REFRESH_TOKEN = 'refresh456';
204
+ process.env.REQON_XERO_CLIENT_ID = 'client789';
205
+ process.env.REQON_XERO_TOKEN_ENDPOINT = 'https://identity.xero.com/connect/token';
206
+
207
+ const creds = credentialsFromEnv(['Xero']);
208
+
209
+ expect(creds).toEqual({
210
+ Xero: {
211
+ type: 'oauth2',
212
+ accessToken: 'access123',
213
+ refreshToken: 'refresh456',
214
+ clientId: 'client789',
215
+ tokenEndpoint: 'https://identity.xero.com/connect/token',
216
+ },
217
+ });
218
+ });
219
+
220
+ it('should handle api_key credentials', () => {
221
+ process.env.REQON_SENDGRID_TYPE = 'api_key';
222
+ process.env.REQON_SENDGRID_API_KEY = 'SG.xxx';
223
+ process.env.REQON_SENDGRID_HEADER_NAME = 'Authorization';
224
+
225
+ const creds = credentialsFromEnv(['SendGrid']);
226
+
227
+ expect(creds).toEqual({
228
+ SendGrid: {
229
+ type: 'api_key',
230
+ apiKey: 'SG.xxx',
231
+ headerName: 'Authorization',
232
+ },
233
+ });
234
+ });
235
+
236
+ it('should skip sources with no credentials', () => {
237
+ const creds = credentialsFromEnv(['Unknown', 'Missing']);
238
+
239
+ expect(creds).toEqual({});
240
+ });
241
+
242
+ it('should handle multiple sources', () => {
243
+ process.env.REQON_GITHUB_TOKEN = 'ghp_xxx';
244
+ process.env.REQON_STRIPE_TOKEN = 'sk_xxx';
245
+
246
+ const creds = credentialsFromEnv(['GitHub', 'Stripe', 'NoCredentials']);
247
+
248
+ expect(creds).toEqual({
249
+ GitHub: {
250
+ type: 'bearer',
251
+ token: 'ghp_xxx',
252
+ },
253
+ Stripe: {
254
+ type: 'bearer',
255
+ token: 'sk_xxx',
256
+ },
257
+ });
258
+ });
259
+
260
+ it('should be case-insensitive for source names', () => {
261
+ process.env.REQON_MYAPI_TOKEN = 'token123';
262
+
263
+ // Uppercase source name
264
+ const creds = credentialsFromEnv(['MyApi']);
265
+
266
+ expect(creds).toEqual({
267
+ MyApi: {
268
+ type: 'bearer',
269
+ token: 'token123',
270
+ },
271
+ });
272
+ });
273
+ });