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,390 @@
1
+ import { describe, it, expect, beforeEach, vi } from 'vitest';
2
+ import {
3
+ CircuitBreaker,
4
+ CircuitBreakerError,
5
+ type CircuitBreakerCallbacks,
6
+ } from './circuit-breaker.js';
7
+
8
+ describe('CircuitBreaker', () => {
9
+ let breaker: CircuitBreaker;
10
+
11
+ beforeEach(() => {
12
+ breaker = new CircuitBreaker();
13
+ });
14
+
15
+ describe('closed state (default)', () => {
16
+ it('allows requests when circuit is closed', () => {
17
+ expect(breaker.canProceed('TestAPI')).toBe(true);
18
+ });
19
+
20
+ it('reports closed state in status', () => {
21
+ const status = breaker.getStatus('TestAPI');
22
+ expect(status.state).toBe('closed');
23
+ expect(status.isOpen).toBe(false);
24
+ expect(status.failures).toBe(0);
25
+ });
26
+
27
+ it('tracks failures within window', () => {
28
+ breaker.recordFailure('TestAPI', undefined, 500);
29
+ breaker.recordFailure('TestAPI', undefined, 502);
30
+
31
+ const status = breaker.getStatus('TestAPI');
32
+ expect(status.failures).toBe(2);
33
+ expect(status.state).toBe('closed');
34
+ });
35
+
36
+ it('counts network errors as failures', () => {
37
+ breaker.recordFailure('TestAPI', undefined, undefined, true);
38
+ breaker.recordFailure('TestAPI', undefined, undefined, true);
39
+
40
+ const status = breaker.getStatus('TestAPI');
41
+ expect(status.failures).toBe(2);
42
+ });
43
+
44
+ it('does not count non-failure status codes', () => {
45
+ breaker.recordFailure('TestAPI', undefined, 400);
46
+ breaker.recordFailure('TestAPI', undefined, 404);
47
+
48
+ const status = breaker.getStatus('TestAPI');
49
+ expect(status.failures).toBe(0);
50
+ });
51
+ });
52
+
53
+ describe('opening circuit', () => {
54
+ it('opens circuit after reaching failure threshold', () => {
55
+ // Default threshold is 5
56
+ for (let i = 0; i < 5; i++) {
57
+ breaker.recordFailure('TestAPI', undefined, 500);
58
+ }
59
+
60
+ const status = breaker.getStatus('TestAPI');
61
+ expect(status.state).toBe('open');
62
+ expect(status.isOpen).toBe(true);
63
+ });
64
+
65
+ it('respects custom failure threshold', () => {
66
+ breaker.configure('TestAPI', { failureThreshold: 3 });
67
+
68
+ breaker.recordFailure('TestAPI', undefined, 500);
69
+ breaker.recordFailure('TestAPI', undefined, 500);
70
+ expect(breaker.getStatus('TestAPI').state).toBe('closed');
71
+
72
+ breaker.recordFailure('TestAPI', undefined, 500);
73
+ expect(breaker.getStatus('TestAPI').state).toBe('open');
74
+ });
75
+
76
+ it('rejects requests when circuit is open', () => {
77
+ breaker.configure('TestAPI', { failureThreshold: 1 });
78
+ breaker.recordFailure('TestAPI', undefined, 500);
79
+
80
+ expect(breaker.canProceed('TestAPI')).toBe(false);
81
+ });
82
+
83
+ it('throws CircuitBreakerError when ensureCanProceed is called on open circuit', () => {
84
+ breaker.configure('TestAPI', { failureThreshold: 1 });
85
+ breaker.recordFailure('TestAPI', undefined, 500);
86
+
87
+ expect(() => breaker.ensureCanProceed('TestAPI')).toThrow(CircuitBreakerError);
88
+ });
89
+
90
+ it('includes retry information in CircuitBreakerError', () => {
91
+ breaker.configure('TestAPI', { failureThreshold: 1, resetTimeout: 30000 });
92
+ breaker.recordFailure('TestAPI', undefined, 500);
93
+
94
+ try {
95
+ breaker.ensureCanProceed('TestAPI');
96
+ expect.fail('Should have thrown');
97
+ } catch (error) {
98
+ expect(error).toBeInstanceOf(CircuitBreakerError);
99
+ const cbError = error as CircuitBreakerError;
100
+ expect(cbError.source).toBe('TestAPI');
101
+ expect(cbError.nextAttemptIn).toBeGreaterThan(0);
102
+ expect(cbError.nextAttemptIn).toBeLessThanOrEqual(30000);
103
+ }
104
+ });
105
+ });
106
+
107
+ describe('half-open state', () => {
108
+ it('transitions to half-open after reset timeout', async () => {
109
+ breaker.configure('TestAPI', { failureThreshold: 1, resetTimeout: 50 });
110
+ breaker.recordFailure('TestAPI', undefined, 500);
111
+
112
+ expect(breaker.getStatus('TestAPI').state).toBe('open');
113
+
114
+ // Wait for reset timeout
115
+ await new Promise((resolve) => setTimeout(resolve, 60));
116
+
117
+ // canProceed should transition to half-open
118
+ expect(breaker.canProceed('TestAPI')).toBe(true);
119
+ expect(breaker.getStatus('TestAPI').state).toBe('half_open');
120
+ });
121
+
122
+ it('closes circuit after success threshold in half-open', async () => {
123
+ breaker.configure('TestAPI', {
124
+ failureThreshold: 1,
125
+ resetTimeout: 50,
126
+ successThreshold: 2,
127
+ });
128
+ breaker.recordFailure('TestAPI', undefined, 500);
129
+
130
+ await new Promise((resolve) => setTimeout(resolve, 60));
131
+ breaker.canProceed('TestAPI'); // Trigger transition to half-open
132
+
133
+ breaker.recordSuccess('TestAPI');
134
+ expect(breaker.getStatus('TestAPI').state).toBe('half_open');
135
+
136
+ breaker.recordSuccess('TestAPI');
137
+ expect(breaker.getStatus('TestAPI').state).toBe('closed');
138
+ });
139
+
140
+ it('re-opens circuit on any failure in half-open', async () => {
141
+ breaker.configure('TestAPI', {
142
+ failureThreshold: 1,
143
+ resetTimeout: 50,
144
+ successThreshold: 2,
145
+ });
146
+ breaker.recordFailure('TestAPI', undefined, 500);
147
+
148
+ await new Promise((resolve) => setTimeout(resolve, 60));
149
+ breaker.canProceed('TestAPI'); // Trigger transition to half-open
150
+
151
+ // One success, then a failure
152
+ breaker.recordSuccess('TestAPI');
153
+ breaker.recordFailure('TestAPI', undefined, 500);
154
+
155
+ expect(breaker.getStatus('TestAPI').state).toBe('open');
156
+ });
157
+ });
158
+
159
+ describe('failure window', () => {
160
+ it('prunes old failures outside window', async () => {
161
+ breaker.configure('TestAPI', { failureThreshold: 3, failureWindow: 100 });
162
+
163
+ breaker.recordFailure('TestAPI', undefined, 500);
164
+ breaker.recordFailure('TestAPI', undefined, 500);
165
+
166
+ // Wait for failures to age out
167
+ await new Promise((resolve) => setTimeout(resolve, 150));
168
+
169
+ // Record a new failure - old ones should be pruned
170
+ breaker.recordFailure('TestAPI', undefined, 500);
171
+
172
+ const status = breaker.getStatus('TestAPI');
173
+ expect(status.failures).toBe(1);
174
+ expect(status.state).toBe('closed');
175
+ });
176
+
177
+ it('does not open circuit if failures are spread across windows', async () => {
178
+ breaker.configure('TestAPI', { failureThreshold: 3, failureWindow: 50 });
179
+
180
+ breaker.recordFailure('TestAPI', undefined, 500);
181
+ await new Promise((resolve) => setTimeout(resolve, 60));
182
+
183
+ breaker.recordFailure('TestAPI', undefined, 500);
184
+ await new Promise((resolve) => setTimeout(resolve, 60));
185
+
186
+ breaker.recordFailure('TestAPI', undefined, 500);
187
+
188
+ // Each failure was in a different window, so should not open
189
+ expect(breaker.getStatus('TestAPI').state).toBe('closed');
190
+ });
191
+ });
192
+
193
+ describe('per-endpoint tracking', () => {
194
+ it('tracks failures per endpoint', () => {
195
+ breaker.configure('TestAPI', { failureThreshold: 2 });
196
+
197
+ breaker.recordFailure('TestAPI', '/invoices', 500);
198
+ breaker.recordFailure('TestAPI', '/invoices', 500);
199
+ breaker.recordFailure('TestAPI', '/contacts', 500);
200
+
201
+ expect(breaker.canProceed('TestAPI', '/invoices')).toBe(false);
202
+ expect(breaker.canProceed('TestAPI', '/contacts')).toBe(true);
203
+ });
204
+
205
+ it('uses source-level config for endpoints', () => {
206
+ breaker.configure('TestAPI', { failureThreshold: 1 });
207
+
208
+ breaker.recordFailure('TestAPI', '/invoices', 500);
209
+
210
+ expect(breaker.canProceed('TestAPI', '/invoices')).toBe(false);
211
+ });
212
+ });
213
+
214
+ describe('reset', () => {
215
+ it('resets circuit to closed state', () => {
216
+ breaker.configure('TestAPI', { failureThreshold: 1 });
217
+ breaker.recordFailure('TestAPI', undefined, 500);
218
+
219
+ expect(breaker.getStatus('TestAPI').state).toBe('open');
220
+
221
+ breaker.reset('TestAPI');
222
+
223
+ const status = breaker.getStatus('TestAPI');
224
+ expect(status.state).toBe('closed');
225
+ expect(status.failures).toBe(0);
226
+ });
227
+
228
+ it('resets specific endpoint', () => {
229
+ breaker.configure('TestAPI', { failureThreshold: 1 });
230
+ breaker.recordFailure('TestAPI', '/invoices', 500);
231
+ breaker.recordFailure('TestAPI', '/contacts', 500);
232
+
233
+ breaker.reset('TestAPI', '/invoices');
234
+
235
+ expect(breaker.canProceed('TestAPI', '/invoices')).toBe(true);
236
+ expect(breaker.canProceed('TestAPI', '/contacts')).toBe(false);
237
+ });
238
+ });
239
+
240
+ describe('custom failure status codes', () => {
241
+ it('uses custom failure status codes', () => {
242
+ breaker.configure('TestAPI', {
243
+ failureThreshold: 2,
244
+ failureStatusCodes: [503, 504],
245
+ });
246
+
247
+ breaker.recordFailure('TestAPI', undefined, 500); // Not counted
248
+ breaker.recordFailure('TestAPI', undefined, 503);
249
+ breaker.recordFailure('TestAPI', undefined, 504);
250
+
251
+ expect(breaker.getStatus('TestAPI').state).toBe('open');
252
+ });
253
+ });
254
+
255
+ describe('network error handling', () => {
256
+ it('respects countNetworkErrors setting', () => {
257
+ breaker.configure('TestAPI', {
258
+ failureThreshold: 1,
259
+ countNetworkErrors: false,
260
+ });
261
+
262
+ breaker.recordFailure('TestAPI', undefined, undefined, true);
263
+
264
+ expect(breaker.getStatus('TestAPI').state).toBe('closed');
265
+ });
266
+ });
267
+
268
+ describe('callbacks', () => {
269
+ it('calls onOpen when circuit opens', () => {
270
+ const onOpen = vi.fn();
271
+ breaker.setCallbacks({ onOpen });
272
+ breaker.configure('TestAPI', { failureThreshold: 1 });
273
+
274
+ breaker.recordFailure('TestAPI', undefined, 500);
275
+
276
+ expect(onOpen).toHaveBeenCalledTimes(1);
277
+ expect(onOpen).toHaveBeenCalledWith(
278
+ expect.objectContaining({
279
+ source: 'TestAPI',
280
+ state: 'open',
281
+ previousState: 'closed',
282
+ })
283
+ );
284
+ });
285
+
286
+ it('calls onHalfOpen when transitioning to half-open', async () => {
287
+ const onHalfOpen = vi.fn();
288
+ breaker.setCallbacks({ onHalfOpen });
289
+ breaker.configure('TestAPI', { failureThreshold: 1, resetTimeout: 50 });
290
+
291
+ breaker.recordFailure('TestAPI', undefined, 500);
292
+ await new Promise((resolve) => setTimeout(resolve, 60));
293
+ breaker.canProceed('TestAPI');
294
+
295
+ expect(onHalfOpen).toHaveBeenCalledTimes(1);
296
+ expect(onHalfOpen).toHaveBeenCalledWith(
297
+ expect.objectContaining({
298
+ source: 'TestAPI',
299
+ state: 'half_open',
300
+ previousState: 'open',
301
+ })
302
+ );
303
+ });
304
+
305
+ it('calls onClose when circuit closes', async () => {
306
+ const onClose = vi.fn();
307
+ breaker.setCallbacks({ onClose });
308
+ breaker.configure('TestAPI', {
309
+ failureThreshold: 1,
310
+ resetTimeout: 50,
311
+ successThreshold: 1,
312
+ });
313
+
314
+ breaker.recordFailure('TestAPI', undefined, 500);
315
+ await new Promise((resolve) => setTimeout(resolve, 60));
316
+ breaker.canProceed('TestAPI');
317
+ breaker.recordSuccess('TestAPI');
318
+
319
+ expect(onClose).toHaveBeenCalledTimes(1);
320
+ expect(onClose).toHaveBeenCalledWith(
321
+ expect.objectContaining({
322
+ source: 'TestAPI',
323
+ state: 'closed',
324
+ previousState: 'half_open',
325
+ })
326
+ );
327
+ });
328
+
329
+ it('calls onRejected when request is rejected', () => {
330
+ const onRejected = vi.fn();
331
+ breaker.setCallbacks({ onRejected });
332
+ breaker.configure('TestAPI', { failureThreshold: 1 });
333
+
334
+ breaker.recordFailure('TestAPI', undefined, 500);
335
+
336
+ try {
337
+ breaker.ensureCanProceed('TestAPI');
338
+ } catch {
339
+ // Expected
340
+ }
341
+
342
+ expect(onRejected).toHaveBeenCalledTimes(1);
343
+ expect(onRejected).toHaveBeenCalledWith(
344
+ expect.objectContaining({
345
+ source: 'TestAPI',
346
+ nextAttemptIn: expect.any(Number),
347
+ })
348
+ );
349
+ });
350
+ });
351
+
352
+ describe('getAllStatuses', () => {
353
+ it('returns all circuit statuses', () => {
354
+ breaker.configure('API1', { failureThreshold: 1 });
355
+ breaker.configure('API2', { failureThreshold: 1 });
356
+
357
+ breaker.recordFailure('API1', undefined, 500);
358
+ breaker.recordSuccess('API2');
359
+
360
+ const statuses = breaker.getAllStatuses();
361
+
362
+ expect(statuses.size).toBe(2);
363
+ expect(statuses.get('API1')?.state).toBe('open');
364
+ expect(statuses.get('API2')?.state).toBe('closed');
365
+ });
366
+ });
367
+
368
+ describe('default configuration', () => {
369
+ it('uses default config when not configured', () => {
370
+ // 5 failures needed by default
371
+ for (let i = 0; i < 4; i++) {
372
+ breaker.recordFailure('TestAPI', undefined, 500);
373
+ }
374
+ expect(breaker.getStatus('TestAPI').state).toBe('closed');
375
+
376
+ breaker.recordFailure('TestAPI', undefined, 500);
377
+ expect(breaker.getStatus('TestAPI').state).toBe('open');
378
+ });
379
+
380
+ it('allows setting default config via constructor', () => {
381
+ const customBreaker = new CircuitBreaker({ failureThreshold: 2 });
382
+
383
+ customBreaker.recordFailure('TestAPI', undefined, 500);
384
+ expect(customBreaker.getStatus('TestAPI').state).toBe('closed');
385
+
386
+ customBreaker.recordFailure('TestAPI', undefined, 500);
387
+ expect(customBreaker.getStatus('TestAPI').state).toBe('open');
388
+ });
389
+ });
390
+ });