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,276 @@
1
+ import { describe, it, expect, beforeEach, afterEach } from 'vitest';
2
+ import { existsSync, rmSync, mkdirSync, readFileSync } from 'node:fs';
3
+ import { join } from 'node:path';
4
+ import { FileStore } from './file.js';
5
+
6
+ describe('FileStore', () => {
7
+ const TEST_DIR = '.reqon-test-stores';
8
+
9
+ beforeEach(() => {
10
+ if (existsSync(TEST_DIR)) {
11
+ rmSync(TEST_DIR, { recursive: true, force: true });
12
+ }
13
+ });
14
+
15
+ afterEach(() => {
16
+ if (existsSync(TEST_DIR)) {
17
+ rmSync(TEST_DIR, { recursive: true, force: true });
18
+ }
19
+ });
20
+
21
+ it('should create directory and file on first write', async () => {
22
+ const store = new FileStore('test-store', { baseDir: TEST_DIR });
23
+
24
+ await store.set('1', { id: '1', name: 'Alice' });
25
+
26
+ expect(existsSync(join(TEST_DIR, 'test-store.json'))).toBe(true);
27
+ });
28
+
29
+ it('should create .gitignore in data directory', async () => {
30
+ const store = new FileStore('test-store', { baseDir: TEST_DIR });
31
+ // Trigger initialization by calling any store method
32
+ await store.get('nonexistent');
33
+
34
+ const gitignorePath = join(TEST_DIR, '.gitignore');
35
+ expect(existsSync(gitignorePath)).toBe(true);
36
+ const content = readFileSync(gitignorePath, 'utf-8');
37
+ expect(content).toContain('*.json');
38
+ });
39
+
40
+ it('should get a record by key', async () => {
41
+ const store = new FileStore('test-store', { baseDir: TEST_DIR });
42
+
43
+ await store.set('1', { id: '1', name: 'Alice' });
44
+ const record = await store.get('1');
45
+
46
+ expect(record).toEqual({ id: '1', name: 'Alice' });
47
+ });
48
+
49
+ it('should return null for non-existent key', async () => {
50
+ const store = new FileStore('test-store', { baseDir: TEST_DIR });
51
+
52
+ const record = await store.get('nonexistent');
53
+ expect(record).toBeNull();
54
+ });
55
+
56
+ it('should update existing record', async () => {
57
+ const store = new FileStore('test-store', { baseDir: TEST_DIR });
58
+
59
+ await store.set('1', { id: '1', name: 'Alice', age: 30 });
60
+ await store.update('1', { age: 31 });
61
+
62
+ const record = await store.get('1');
63
+ expect(record).toEqual({ id: '1', name: 'Alice', age: 31 });
64
+ });
65
+
66
+ it('should create record on update if not exists', async () => {
67
+ const store = new FileStore('test-store', { baseDir: TEST_DIR });
68
+
69
+ await store.update('1', { name: 'Alice' });
70
+
71
+ const record = await store.get('1');
72
+ expect(record).toEqual({ name: 'Alice' });
73
+ });
74
+
75
+ it('should delete a record', async () => {
76
+ const store = new FileStore('test-store', { baseDir: TEST_DIR });
77
+
78
+ await store.set('1', { id: '1', name: 'Alice' });
79
+ await store.delete('1');
80
+
81
+ const record = await store.get('1');
82
+ expect(record).toBeNull();
83
+ });
84
+
85
+ it('should list all records', async () => {
86
+ const store = new FileStore('test-store', { baseDir: TEST_DIR });
87
+
88
+ await store.set('1', { id: '1', name: 'Alice' });
89
+ await store.set('2', { id: '2', name: 'Bob' });
90
+
91
+ const records = await store.list();
92
+ expect(records).toHaveLength(2);
93
+ });
94
+
95
+ it('should filter records with where clause', async () => {
96
+ const store = new FileStore('test-store', { baseDir: TEST_DIR });
97
+
98
+ await store.set('1', { id: '1', name: 'Alice', active: true });
99
+ await store.set('2', { id: '2', name: 'Bob', active: false });
100
+ await store.set('3', { id: '3', name: 'Charlie', active: true });
101
+
102
+ const records = await store.list({ where: { active: true } });
103
+ expect(records).toHaveLength(2);
104
+ expect(records.map(r => r.name)).toContain('Alice');
105
+ expect(records.map(r => r.name)).toContain('Charlie');
106
+ });
107
+
108
+ it('should support limit and offset', async () => {
109
+ const store = new FileStore('test-store', { baseDir: TEST_DIR });
110
+
111
+ await store.set('1', { id: '1', name: 'Alice' });
112
+ await store.set('2', { id: '2', name: 'Bob' });
113
+ await store.set('3', { id: '3', name: 'Charlie' });
114
+
115
+ const records = await store.list({ offset: 1, limit: 1 });
116
+ expect(records).toHaveLength(1);
117
+ });
118
+
119
+ it('should clear all records', async () => {
120
+ const store = new FileStore('test-store', { baseDir: TEST_DIR });
121
+
122
+ await store.set('1', { id: '1', name: 'Alice' });
123
+ await store.set('2', { id: '2', name: 'Bob' });
124
+ await store.clear();
125
+
126
+ const records = await store.list();
127
+ expect(records).toHaveLength(0);
128
+ });
129
+
130
+ describe('batch mode', () => {
131
+ it('should not write to disk until flush in batch mode', async () => {
132
+ const store = new FileStore('test-store', { baseDir: TEST_DIR, persist: 'batch' });
133
+ const filePath = store.getFilePath();
134
+
135
+ await store.set('1', { id: '1', name: 'Alice' });
136
+
137
+ // File exists but should be empty or not contain data yet
138
+ const contentBefore = existsSync(filePath) ? readFileSync(filePath, 'utf-8') : '{}';
139
+ const parsedBefore = JSON.parse(contentBefore);
140
+ expect(Object.keys(parsedBefore)).toHaveLength(0);
141
+
142
+ // After flush, data should be persisted
143
+ store.flush();
144
+
145
+ const contentAfter = readFileSync(filePath, 'utf-8');
146
+ const parsedAfter = JSON.parse(contentAfter);
147
+ expect(parsedAfter['1']).toEqual({ id: '1', name: 'Alice' });
148
+ });
149
+ });
150
+
151
+ describe('persistence', () => {
152
+ it('should persist data across store instances', async () => {
153
+ const store1 = new FileStore('test-store', { baseDir: TEST_DIR });
154
+ await store1.set('1', { id: '1', name: 'Alice' });
155
+
156
+ const store2 = new FileStore('test-store', { baseDir: TEST_DIR });
157
+ const record = await store2.get('1');
158
+
159
+ expect(record).toEqual({ id: '1', name: 'Alice' });
160
+ });
161
+
162
+ it('should reload data from disk', async () => {
163
+ const store = new FileStore('test-store', { baseDir: TEST_DIR });
164
+ await store.set('1', { id: '1', name: 'Alice' });
165
+
166
+ // Manually modify the file
167
+ const filePath = store.getFilePath();
168
+ const content = JSON.parse(readFileSync(filePath, 'utf-8'));
169
+ content['1'].name = 'Modified';
170
+ require('fs').writeFileSync(filePath, JSON.stringify(content));
171
+
172
+ // Reload and check
173
+ await store.reload();
174
+ const record = await store.get('1');
175
+ expect(record?.name).toBe('Modified');
176
+ });
177
+ });
178
+
179
+ describe('pretty printing', () => {
180
+ it('should pretty-print JSON by default', async () => {
181
+ const store = new FileStore('test-store', { baseDir: TEST_DIR });
182
+ await store.set('1', { id: '1', name: 'Alice' });
183
+
184
+ const content = readFileSync(store.getFilePath(), 'utf-8');
185
+ expect(content).toContain('\n'); // Pretty printed has newlines
186
+ });
187
+
188
+ it('should compact JSON when pretty is false', async () => {
189
+ const store = new FileStore('test-store', { baseDir: TEST_DIR, pretty: false });
190
+ await store.set('1', { id: '1', name: 'Alice' });
191
+
192
+ const content = readFileSync(store.getFilePath(), 'utf-8');
193
+ expect(content).not.toContain('\n'); // Compact has no newlines
194
+ });
195
+ });
196
+
197
+ describe('count', () => {
198
+ it('should return total count when no filter', async () => {
199
+ const store = new FileStore('test-store', { baseDir: TEST_DIR });
200
+
201
+ await store.set('1', { id: '1', name: 'Alice', status: 'active' });
202
+ await store.set('2', { id: '2', name: 'Bob', status: 'inactive' });
203
+ await store.set('3', { id: '3', name: 'Charlie', status: 'active' });
204
+
205
+ const count = await store.count();
206
+ expect(count).toBe(3);
207
+ });
208
+
209
+ it('should return count matching where clause', async () => {
210
+ const store = new FileStore('test-store', { baseDir: TEST_DIR });
211
+
212
+ await store.set('1', { id: '1', name: 'Alice', status: 'active' });
213
+ await store.set('2', { id: '2', name: 'Bob', status: 'inactive' });
214
+ await store.set('3', { id: '3', name: 'Charlie', status: 'active' });
215
+
216
+ const count = await store.count({ where: { status: 'active' } });
217
+ expect(count).toBe(2);
218
+ });
219
+
220
+ it('should return 0 when no records match', async () => {
221
+ const store = new FileStore('test-store', { baseDir: TEST_DIR });
222
+
223
+ await store.set('1', { id: '1', status: 'active' });
224
+
225
+ const count = await store.count({ where: { status: 'unknown' } });
226
+ expect(count).toBe(0);
227
+ });
228
+
229
+ it('should ignore limit and offset for counting', async () => {
230
+ const store = new FileStore('test-store', { baseDir: TEST_DIR });
231
+
232
+ await store.set('1', { id: '1', status: 'active' });
233
+ await store.set('2', { id: '2', status: 'active' });
234
+ await store.set('3', { id: '3', status: 'active' });
235
+
236
+ // limit/offset should not affect count - count returns total matching records
237
+ const count = await store.count({ where: { status: 'active' }, limit: 1, offset: 1 });
238
+ expect(count).toBe(3);
239
+ });
240
+
241
+ it('should return 0 for empty store', async () => {
242
+ const store = new FileStore('test-store', { baseDir: TEST_DIR });
243
+
244
+ const count = await store.count();
245
+ expect(count).toBe(0);
246
+ });
247
+ });
248
+
249
+ describe('utilities', () => {
250
+ it('should report size correctly', async () => {
251
+ const store = new FileStore('test-store', { baseDir: TEST_DIR });
252
+
253
+ expect(store.size()).toBe(0);
254
+
255
+ await store.set('1', { id: '1' });
256
+ await store.set('2', { id: '2' });
257
+
258
+ expect(store.size()).toBe(2);
259
+ });
260
+
261
+ it('should dump all records', async () => {
262
+ const store = new FileStore('test-store', { baseDir: TEST_DIR });
263
+
264
+ await store.set('1', { id: '1', name: 'Alice' });
265
+ await store.set('2', { id: '2', name: 'Bob' });
266
+
267
+ const dump = store.dump();
268
+ expect(dump).toHaveLength(2);
269
+ });
270
+
271
+ it('should return file path', () => {
272
+ const store = new FileStore('test-store', { baseDir: TEST_DIR });
273
+ expect(store.getFilePath()).toBe(join(TEST_DIR, 'test-store.json'));
274
+ });
275
+ });
276
+ });
@@ -0,0 +1,211 @@
1
+ import { writeFile } from 'node:fs/promises';
2
+ import { writeFileSync, existsSync } from 'node:fs';
3
+ import { join, dirname } from 'node:path';
4
+ import type { StoreAdapter, StoreFilter } from './types.js';
5
+ import { applyStoreFilter } from './types.js';
6
+ import {
7
+ ensureDirectory,
8
+ readJsonFile,
9
+ serialize,
10
+ } from '../utils/file.js';
11
+
12
+ export interface FileStoreOptions {
13
+ /** Base directory for data files (default: '.reqon-data') */
14
+ baseDir?: string;
15
+ /** Write mode: 'immediate' writes on every change, 'batch' only on flush/close, 'debounce' batches writes */
16
+ persist?: 'immediate' | 'batch' | 'debounce';
17
+ /** Pretty-print JSON for readability (default: true) */
18
+ pretty?: boolean;
19
+ /** Debounce delay in milliseconds (default: 100ms, only used with persist: 'debounce') */
20
+ debounceMs?: number;
21
+ }
22
+
23
+ const DEFAULT_OPTIONS: Required<FileStoreOptions> = {
24
+ baseDir: '.reqon-data',
25
+ persist: 'immediate',
26
+ pretty: true,
27
+ debounceMs: 100,
28
+ };
29
+
30
+ /**
31
+ * File-based JSON store for local development
32
+ * Persists data to .reqon-data/{name}.json
33
+ */
34
+ export class FileStore implements StoreAdapter {
35
+ private data: Map<string, Record<string, unknown>> = new Map();
36
+ private filePath: string;
37
+ private options: Required<FileStoreOptions>;
38
+ private dirty = false;
39
+ private initialized: Promise<void>;
40
+ private debounceTimer: ReturnType<typeof setTimeout> | null = null;
41
+ private pendingWrite: Promise<void> | null = null;
42
+
43
+ constructor(name: string, options: FileStoreOptions = {}) {
44
+ this.options = { ...DEFAULT_OPTIONS, ...options };
45
+ this.filePath = join(this.options.baseDir, `${name}.json`);
46
+ this.initialized = this.init();
47
+ }
48
+
49
+ private async init(): Promise<void> {
50
+ const dir = dirname(this.filePath);
51
+ await ensureDirectory(dir);
52
+ // Create .gitignore if it doesn't exist
53
+ const gitignorePath = join(dir, '.gitignore');
54
+ if (!existsSync(gitignorePath)) {
55
+ await writeFile(gitignorePath, '# Reqon local data\n*.json\n', 'utf-8');
56
+ }
57
+ await this.load();
58
+ }
59
+
60
+ private async load(): Promise<void> {
61
+ const parsed = await readJsonFile<Record<string, Record<string, unknown>>>(this.filePath);
62
+ if (parsed) {
63
+ this.data = new Map(Object.entries(parsed));
64
+ }
65
+ }
66
+
67
+ private async persist(): Promise<void> {
68
+ if (this.options.persist === 'batch') {
69
+ this.dirty = true;
70
+ return;
71
+ }
72
+ if (this.options.persist === 'debounce') {
73
+ this.dirty = true;
74
+ this.scheduleDebouncedWrite();
75
+ return;
76
+ }
77
+ await this.writeToDisk();
78
+ }
79
+
80
+ private scheduleDebouncedWrite(): void {
81
+ // Clear existing timer if any
82
+ if (this.debounceTimer) {
83
+ clearTimeout(this.debounceTimer);
84
+ }
85
+ // Schedule new write
86
+ this.debounceTimer = setTimeout(() => {
87
+ this.debounceTimer = null;
88
+ if (this.dirty && !this.pendingWrite) {
89
+ this.pendingWrite = this.writeToDisk().finally(() => {
90
+ this.pendingWrite = null;
91
+ });
92
+ }
93
+ }, this.options.debounceMs);
94
+ }
95
+
96
+ private async writeToDisk(): Promise<void> {
97
+ const obj = Object.fromEntries(this.data);
98
+ const content = serialize(obj, this.options.pretty);
99
+ await writeFile(this.filePath, content, 'utf-8');
100
+ this.dirty = false;
101
+ }
102
+
103
+ /** Synchronous write for flush/close operations */
104
+ private writeToDiskSync(): void {
105
+ const obj = Object.fromEntries(this.data);
106
+ const content = serialize(obj, this.options.pretty);
107
+ writeFileSync(this.filePath, content, 'utf-8');
108
+ this.dirty = false;
109
+ }
110
+
111
+ async get(key: string): Promise<Record<string, unknown> | null> {
112
+ await this.initialized;
113
+ return this.data.get(key) ?? null;
114
+ }
115
+
116
+ async set(key: string, value: Record<string, unknown>): Promise<void> {
117
+ await this.initialized;
118
+ this.data.set(key, { ...value });
119
+ await this.persist();
120
+ }
121
+
122
+ async bulkSet(records: Array<{ key: string; value: Record<string, unknown> }>): Promise<void> {
123
+ await this.initialized;
124
+ // Set all records in memory first (no disk I/O per record)
125
+ for (const { key, value } of records) {
126
+ this.data.set(key, { ...value });
127
+ }
128
+ // Single persist operation for all records
129
+ await this.persist();
130
+ }
131
+
132
+ async update(key: string, value: Partial<Record<string, unknown>>): Promise<void> {
133
+ await this.initialized;
134
+ const existing = this.data.get(key);
135
+ if (existing) {
136
+ this.data.set(key, { ...existing, ...value });
137
+ } else {
138
+ this.data.set(key, value as Record<string, unknown>);
139
+ }
140
+ await this.persist();
141
+ }
142
+
143
+ async delete(key: string): Promise<void> {
144
+ await this.initialized;
145
+ this.data.delete(key);
146
+ await this.persist();
147
+ }
148
+
149
+ async list(filter?: StoreFilter): Promise<Record<string, unknown>[]> {
150
+ await this.initialized;
151
+ return applyStoreFilter(Array.from(this.data.values()), filter);
152
+ }
153
+
154
+ async count(filter?: StoreFilter): Promise<number> {
155
+ await this.initialized;
156
+ // Apply only the where clause for counting (ignore limit/offset)
157
+ const filtered = applyStoreFilter(Array.from(this.data.values()), {
158
+ where: filter?.where,
159
+ });
160
+ return filtered.length;
161
+ }
162
+
163
+ async clear(): Promise<void> {
164
+ await this.initialized;
165
+ this.data.clear();
166
+ await this.persist();
167
+ }
168
+
169
+ /**
170
+ * Flush pending changes to disk (needed in 'batch' or 'debounce' mode)
171
+ * Uses synchronous I/O to ensure data is written before process exits
172
+ */
173
+ flush(): void {
174
+ // Cancel any pending debounce timer
175
+ if (this.debounceTimer) {
176
+ clearTimeout(this.debounceTimer);
177
+ this.debounceTimer = null;
178
+ }
179
+ if (this.dirty) {
180
+ this.writeToDiskSync();
181
+ }
182
+ }
183
+
184
+ /**
185
+ * Close the store, ensuring all pending changes are written to disk.
186
+ * Should be called before the process exits to prevent data loss in batch mode.
187
+ */
188
+ close(): void {
189
+ this.flush();
190
+ }
191
+
192
+ /**
193
+ * Reload data from disk (useful if file was modified externally)
194
+ */
195
+ async reload(): Promise<void> {
196
+ await this.load();
197
+ }
198
+
199
+ // For debugging
200
+ size(): number {
201
+ return this.data.size;
202
+ }
203
+
204
+ dump(): Record<string, unknown>[] {
205
+ return Array.from(this.data.values());
206
+ }
207
+
208
+ getFilePath(): string {
209
+ return this.filePath;
210
+ }
211
+ }
@@ -0,0 +1,6 @@
1
+ export type { StoreAdapter, StoreFilter, StoreConfig } from './types.js';
2
+ export { applyStoreFilter } from './types.js';
3
+ export { MemoryStore } from './memory.js';
4
+ export { FileStore, type FileStoreOptions } from './file.js';
5
+ export { PostgRESTStore, PostgRESTError, type PostgRESTOptions } from './postgrest.js';
6
+ export { createStore, resolveStoreType, type StoreType, type CreateStoreOptions } from './factory.js';