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,246 @@
1
+ /**
2
+ * Credentials resolver with dotenv support
3
+ *
4
+ * Supports loading credentials from:
5
+ * - Environment variables (via .env files or process.env)
6
+ * - JSON config files with env var interpolation
7
+ *
8
+ * Env var reference patterns:
9
+ * - $VAR_NAME
10
+ * - ${VAR_NAME}
11
+ * - ${VAR_NAME:-default} (with default value)
12
+ */
13
+
14
+ import { config as dotenvConfig } from 'dotenv';
15
+ import { resolve } from 'node:path';
16
+ import { existsSync } from 'node:fs';
17
+
18
+ export interface CredentialsConfig {
19
+ /** Path to .env file (default: .env in cwd) */
20
+ envFile?: string;
21
+ /** Additional .env files to load (loaded in order, later files override) */
22
+ envFiles?: string[];
23
+ /** Whether to load from process.env (default: true) */
24
+ useProcessEnv?: boolean;
25
+ /** Base directory for resolving relative paths */
26
+ baseDir?: string;
27
+ }
28
+
29
+ export interface LoadEnvResult {
30
+ /** Whether any .env files were loaded */
31
+ loaded: boolean;
32
+ /** Paths of loaded .env files */
33
+ files: string[];
34
+ /** Number of variables loaded */
35
+ count: number;
36
+ }
37
+
38
+ /**
39
+ * Load environment variables from .env files
40
+ */
41
+ export function loadEnv(options: CredentialsConfig = {}): LoadEnvResult {
42
+ const baseDir = options.baseDir || process.cwd();
43
+ const files: string[] = [];
44
+ let totalCount = 0;
45
+
46
+ // Determine which .env files to load
47
+ const envFilePaths: string[] = [];
48
+
49
+ if (options.envFiles) {
50
+ envFilePaths.push(...options.envFiles.map((f) => resolve(baseDir, f)));
51
+ } else if (options.envFile) {
52
+ envFilePaths.push(resolve(baseDir, options.envFile));
53
+ } else {
54
+ // Default: look for .env, .env.local in order
55
+ const defaultFiles = ['.env', '.env.local'];
56
+ for (const file of defaultFiles) {
57
+ const path = resolve(baseDir, file);
58
+ if (existsSync(path)) {
59
+ envFilePaths.push(path);
60
+ }
61
+ }
62
+ }
63
+
64
+ // Load each file (later files override earlier ones)
65
+ for (const envPath of envFilePaths) {
66
+ if (existsSync(envPath)) {
67
+ const result = dotenvConfig({ path: envPath, override: true });
68
+ if (!result.error && result.parsed) {
69
+ files.push(envPath);
70
+ totalCount += Object.keys(result.parsed).length;
71
+ }
72
+ }
73
+ }
74
+
75
+ return {
76
+ loaded: files.length > 0,
77
+ files,
78
+ count: totalCount,
79
+ };
80
+ }
81
+
82
+ /**
83
+ * Resolve environment variable references in a string
84
+ *
85
+ * Supports:
86
+ * - $VAR_NAME
87
+ * - ${VAR_NAME}
88
+ * - ${VAR_NAME:-default}
89
+ */
90
+ export function resolveEnvString(value: string): string {
91
+ // Pattern for ${VAR:-default} or ${VAR}
92
+ const bracketPattern = /\$\{([^}:]+)(?::-([^}]*))?\}/g;
93
+ // Pattern for $VAR (word characters only, not followed by {)
94
+ const simplePattern = /\$([A-Za-z_][A-Za-z0-9_]*)/g;
95
+
96
+ let result = value;
97
+
98
+ // First resolve ${VAR} and ${VAR:-default} patterns
99
+ result = result.replace(bracketPattern, (_match, varName, defaultValue) => {
100
+ const envValue = process.env[varName];
101
+ if (envValue !== undefined) {
102
+ return envValue;
103
+ }
104
+ if (defaultValue !== undefined) {
105
+ return defaultValue;
106
+ }
107
+ // Return empty string if no value and no default
108
+ return '';
109
+ });
110
+
111
+ // Then resolve $VAR patterns (only if not already resolved)
112
+ result = result.replace(simplePattern, (_match, varName) => {
113
+ const envValue = process.env[varName];
114
+ return envValue !== undefined ? envValue : '';
115
+ });
116
+
117
+ return result;
118
+ }
119
+
120
+ /**
121
+ * Check if a string contains env var references
122
+ */
123
+ export function hasEnvReference(value: string): boolean {
124
+ return /\$\{?[A-Za-z_][A-Za-z0-9_]*/.test(value);
125
+ }
126
+
127
+ /**
128
+ * Recursively resolve all env var references in an object
129
+ */
130
+ export function resolveCredentials<T>(config: T): T {
131
+ if (config === null || config === undefined) {
132
+ return config;
133
+ }
134
+
135
+ if (typeof config === 'string') {
136
+ return resolveEnvString(config) as T;
137
+ }
138
+
139
+ if (Array.isArray(config)) {
140
+ return config.map((item) => resolveCredentials(item)) as T;
141
+ }
142
+
143
+ if (typeof config === 'object') {
144
+ const result: Record<string, unknown> = {};
145
+ for (const [key, value] of Object.entries(config)) {
146
+ result[key] = resolveCredentials(value);
147
+ }
148
+ return result as T;
149
+ }
150
+
151
+ return config;
152
+ }
153
+
154
+ /**
155
+ * Auth configuration types that can be resolved from env vars
156
+ */
157
+ export interface AuthCredentials {
158
+ [sourceName: string]: SourceCredentials;
159
+ }
160
+
161
+ export interface SourceCredentials {
162
+ type: 'bearer' | 'oauth2' | 'api_key' | 'basic';
163
+ /** Bearer token (for type: bearer) */
164
+ token?: string;
165
+ /** Access token (for type: oauth2) */
166
+ accessToken?: string;
167
+ /** Refresh token (for type: oauth2) */
168
+ refreshToken?: string;
169
+ /** Token endpoint URL (for type: oauth2) */
170
+ tokenEndpoint?: string;
171
+ /** OAuth2 client ID */
172
+ clientId?: string;
173
+ /** OAuth2 client secret */
174
+ clientSecret?: string;
175
+ /** API key value (for type: api_key) */
176
+ apiKey?: string;
177
+ /** API key header name (default: X-API-Key) */
178
+ headerName?: string;
179
+ /** Username (for type: basic) */
180
+ username?: string;
181
+ /** Password (for type: basic) */
182
+ password?: string;
183
+ }
184
+
185
+ /**
186
+ * Load and resolve credentials from a JSON file with env var interpolation
187
+ */
188
+ export function loadCredentials(
189
+ config: Record<string, unknown>,
190
+ options: CredentialsConfig = {}
191
+ ): AuthCredentials {
192
+ // Load .env files first
193
+ loadEnv(options);
194
+
195
+ // Resolve env var references in the config
196
+ return resolveCredentials(config) as AuthCredentials;
197
+ }
198
+
199
+ /**
200
+ * Build credentials directly from environment variables using a naming convention
201
+ *
202
+ * Convention: REQON_{SOURCE}_{FIELD}
203
+ * Example: REQON_GITHUB_TOKEN, REQON_XERO_CLIENT_ID
204
+ */
205
+ export function credentialsFromEnv(sourceNames: string[]): AuthCredentials {
206
+ const credentials: AuthCredentials = {};
207
+
208
+ for (const sourceName of sourceNames) {
209
+ const prefix = `REQON_${sourceName.toUpperCase()}_`;
210
+ const sourceCredentials: Partial<SourceCredentials> = {};
211
+
212
+ // Map env var suffixes to credential fields
213
+ const fieldMappings: Record<string, keyof SourceCredentials> = {
214
+ TYPE: 'type',
215
+ TOKEN: 'token',
216
+ ACCESS_TOKEN: 'accessToken',
217
+ REFRESH_TOKEN: 'refreshToken',
218
+ TOKEN_ENDPOINT: 'tokenEndpoint',
219
+ CLIENT_ID: 'clientId',
220
+ CLIENT_SECRET: 'clientSecret',
221
+ API_KEY: 'apiKey',
222
+ HEADER_NAME: 'headerName',
223
+ USERNAME: 'username',
224
+ PASSWORD: 'password',
225
+ };
226
+
227
+ for (const [suffix, field] of Object.entries(fieldMappings)) {
228
+ const envVar = `${prefix}${suffix}`;
229
+ const value = process.env[envVar];
230
+ if (value !== undefined) {
231
+ (sourceCredentials as Record<string, string>)[field] = value;
232
+ }
233
+ }
234
+
235
+ // Only add if we found at least a type or token
236
+ if (sourceCredentials.type || sourceCredentials.token || sourceCredentials.accessToken) {
237
+ // Default to bearer if we have a token but no type
238
+ if (!sourceCredentials.type && sourceCredentials.token) {
239
+ sourceCredentials.type = 'bearer';
240
+ }
241
+ credentials[sourceName] = sourceCredentials as SourceCredentials;
242
+ }
243
+ }
244
+
245
+ return credentials;
246
+ }
@@ -0,0 +1,40 @@
1
+ // Types
2
+ export type {
3
+ AuthProvider,
4
+ TokenInfo,
5
+ OAuth2Tokens,
6
+ TokenStore,
7
+ OAuth2Config,
8
+ RateLimitInfo,
9
+ RateLimiter,
10
+ RateLimitStatus,
11
+ RateLimitConfig,
12
+ RateLimitCallbacks,
13
+ RateLimitEvent,
14
+ RateLimitStrategy,
15
+ } from './types.js';
16
+
17
+ // Rate limiting
18
+ export {
19
+ AdaptiveRateLimiter,
20
+ parseRateLimitHeaders,
21
+ RateLimitError,
22
+ RateLimitTimeoutError,
23
+ } from './rate-limiter.js';
24
+
25
+ // Token stores
26
+ export { InMemoryTokenStore, FileTokenStore } from './token-store.js';
27
+
28
+ // Auth providers
29
+ export { OAuth2AuthProvider, BearerTokenProvider, ApiKeyProvider } from './oauth2-provider.js';
30
+
31
+ // Circuit breaker
32
+ export {
33
+ CircuitBreaker,
34
+ CircuitBreakerError,
35
+ type CircuitBreakerConfig,
36
+ type CircuitBreakerStatus,
37
+ type CircuitBreakerEvent,
38
+ type CircuitBreakerCallbacks,
39
+ type CircuitState,
40
+ } from './circuit-breaker.js';
@@ -0,0 +1,177 @@
1
+ import type { AuthProvider, TokenInfo, TokenStore, OAuth2Tokens, OAuth2Config } from './types.js';
2
+
3
+ interface TokenResponse {
4
+ access_token: string;
5
+ refresh_token?: string;
6
+ expires_in?: number;
7
+ token_type?: string;
8
+ scope?: string;
9
+ }
10
+
11
+ /**
12
+ * OAuth2 auth provider with automatic token refresh
13
+ */
14
+ export class OAuth2AuthProvider implements AuthProvider {
15
+ private connectionId: string;
16
+ private store: TokenStore;
17
+ private config: OAuth2Config;
18
+ private refreshBuffer: number;
19
+ private refreshPromise: Promise<string> | null = null;
20
+
21
+ constructor(options: {
22
+ connectionId: string;
23
+ store: TokenStore;
24
+ config: OAuth2Config;
25
+ }) {
26
+ this.connectionId = options.connectionId;
27
+ this.store = options.store;
28
+ this.config = options.config;
29
+ this.refreshBuffer = (options.config.refreshBuffer ?? 300) * 1000; // Convert to ms
30
+ }
31
+
32
+ async getToken(): Promise<string> {
33
+ const tokens = await this.store.get(this.connectionId);
34
+
35
+ if (!tokens) {
36
+ throw new Error(`No tokens found for connection: ${this.connectionId}`);
37
+ }
38
+
39
+ // Check if we need to refresh
40
+ if (this.shouldRefresh(tokens)) {
41
+ return this.refreshToken();
42
+ }
43
+
44
+ // Update last used time
45
+ await this.store.touch(this.connectionId);
46
+
47
+ return tokens.accessToken;
48
+ }
49
+
50
+ async refreshToken(): Promise<string> {
51
+ // Deduplicate concurrent refresh requests
52
+ if (this.refreshPromise) {
53
+ return this.refreshPromise;
54
+ }
55
+
56
+ this.refreshPromise = this.doRefresh();
57
+
58
+ try {
59
+ return await this.refreshPromise;
60
+ } finally {
61
+ this.refreshPromise = null;
62
+ }
63
+ }
64
+
65
+ private async doRefresh(): Promise<string> {
66
+ const tokens = await this.store.get(this.connectionId);
67
+
68
+ if (!tokens?.refreshToken) {
69
+ throw new Error(`No refresh token available for connection: ${this.connectionId}`);
70
+ }
71
+
72
+ const body = new URLSearchParams({
73
+ grant_type: 'refresh_token',
74
+ refresh_token: tokens.refreshToken,
75
+ client_id: this.config.clientId,
76
+ });
77
+
78
+ if (this.config.clientSecret) {
79
+ body.set('client_secret', this.config.clientSecret);
80
+ }
81
+
82
+ const response = await fetch(this.config.tokenEndpoint, {
83
+ method: 'POST',
84
+ headers: {
85
+ 'Content-Type': 'application/x-www-form-urlencoded',
86
+ },
87
+ body,
88
+ });
89
+
90
+ if (!response.ok) {
91
+ const error = await response.text();
92
+ throw new Error(`Token refresh failed: ${response.status} ${error}`);
93
+ }
94
+
95
+ const data = (await response.json()) as TokenResponse;
96
+
97
+ const newTokens: OAuth2Tokens = {
98
+ accessToken: data.access_token,
99
+ refreshToken: data.refresh_token ?? tokens.refreshToken, // Keep old if not rotated
100
+ tokenType: data.token_type ?? 'Bearer',
101
+ scope: data.scope,
102
+ };
103
+
104
+ // Calculate expiry
105
+ if (data.expires_in) {
106
+ newTokens.expiresAt = new Date(Date.now() + data.expires_in * 1000);
107
+ }
108
+
109
+ await this.store.set(this.connectionId, newTokens);
110
+ await this.store.touch(this.connectionId);
111
+
112
+ return newTokens.accessToken;
113
+ }
114
+
115
+ private shouldRefresh(tokens: OAuth2Tokens): boolean {
116
+ if (!tokens.expiresAt) {
117
+ return false; // No expiry info, assume valid
118
+ }
119
+
120
+ const now = Date.now();
121
+ const expiresAt = tokens.expiresAt.getTime();
122
+
123
+ // Refresh if within buffer of expiry
124
+ return expiresAt - now < this.refreshBuffer;
125
+ }
126
+
127
+ getTokenInfo(): TokenInfo {
128
+ // Synchronous - returns cached info
129
+ return {
130
+ connectionId: this.connectionId,
131
+ };
132
+ }
133
+ }
134
+
135
+ /**
136
+ * Simple bearer token provider (no refresh)
137
+ */
138
+ export class BearerTokenProvider implements AuthProvider {
139
+ private token: string;
140
+
141
+ constructor(token: string) {
142
+ this.token = token;
143
+ }
144
+
145
+ async getToken(): Promise<string> {
146
+ return this.token;
147
+ }
148
+
149
+ getTokenInfo(): TokenInfo {
150
+ return {};
151
+ }
152
+ }
153
+
154
+ /**
155
+ * API key provider
156
+ */
157
+ export class ApiKeyProvider implements AuthProvider {
158
+ private apiKey: string;
159
+ private headerName: string;
160
+
161
+ constructor(apiKey: string, headerName = 'X-API-Key') {
162
+ this.apiKey = apiKey;
163
+ this.headerName = headerName;
164
+ }
165
+
166
+ async getToken(): Promise<string> {
167
+ return this.apiKey;
168
+ }
169
+
170
+ getHeaderName(): string {
171
+ return this.headerName;
172
+ }
173
+
174
+ getTokenInfo(): TokenInfo {
175
+ return {};
176
+ }
177
+ }