@effect-app/infra 4.0.0-beta.26 → 4.0.0-beta.261

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 (505) hide show
  1. package/CHANGELOG.md +1973 -0
  2. package/_check.sh +1 -1
  3. package/dist/CUPS.d.ts +30 -11
  4. package/dist/CUPS.d.ts.map +1 -1
  5. package/dist/CUPS.js +35 -14
  6. package/dist/ClusterCosmos.d.ts +64 -0
  7. package/dist/ClusterCosmos.d.ts.map +1 -0
  8. package/dist/ClusterCosmos.js +501 -0
  9. package/dist/ClusterServiceBus.d.ts +67 -0
  10. package/dist/ClusterServiceBus.d.ts.map +1 -0
  11. package/dist/ClusterServiceBus.js +82 -0
  12. package/dist/ContextProvider.d.ts +34 -0
  13. package/dist/ContextProvider.d.ts.map +1 -0
  14. package/dist/ContextProvider.js +40 -0
  15. package/dist/Emailer/Sendgrid.d.ts +111 -147
  16. package/dist/Emailer/Sendgrid.d.ts.map +1 -1
  17. package/dist/Emailer/Sendgrid.js +24 -19
  18. package/dist/Emailer/fake.d.ts +2 -2
  19. package/dist/Emailer/fake.d.ts.map +1 -1
  20. package/dist/Emailer/fake.js +4 -4
  21. package/dist/MainFiberSet.d.ts +12 -9
  22. package/dist/MainFiberSet.d.ts.map +1 -1
  23. package/dist/MainFiberSet.js +10 -6
  24. package/dist/QueueMaker/SQLQueue.d.ts +8 -9
  25. package/dist/QueueMaker/SQLQueue.d.ts.map +1 -1
  26. package/dist/QueueMaker/SQLQueue.js +138 -120
  27. package/dist/QueueMaker/errors.d.ts +5 -3
  28. package/dist/QueueMaker/errors.d.ts.map +1 -1
  29. package/dist/QueueMaker/errors.js +4 -2
  30. package/dist/QueueMaker/memQueue.d.ts +10 -6
  31. package/dist/QueueMaker/memQueue.d.ts.map +1 -1
  32. package/dist/QueueMaker/memQueue.js +84 -68
  33. package/dist/QueueMaker/sbqueue.d.ts +9 -5
  34. package/dist/QueueMaker/sbqueue.d.ts.map +1 -1
  35. package/dist/QueueMaker/sbqueue.js +60 -58
  36. package/dist/RequestFiberSet.d.ts +10 -7
  37. package/dist/RequestFiberSet.d.ts.map +1 -1
  38. package/dist/RequestFiberSet.js +13 -8
  39. package/dist/SQL/Model.d.ts +468 -0
  40. package/dist/SQL/Model.d.ts.map +1 -0
  41. package/dist/SQL/Model.js +469 -0
  42. package/dist/SQL.d.ts +2 -0
  43. package/dist/SQL.d.ts.map +1 -0
  44. package/dist/{adapters/SQL.js → SQL.js} +1 -1
  45. package/dist/ServiceBus.d.ts +61 -0
  46. package/dist/ServiceBus.d.ts.map +1 -0
  47. package/dist/ServiceBus.js +108 -0
  48. package/dist/Store/Cosmos/query.d.ts +15 -4
  49. package/dist/Store/Cosmos/query.d.ts.map +1 -1
  50. package/dist/Store/Cosmos/query.js +179 -41
  51. package/dist/Store/Cosmos.d.ts +3 -3
  52. package/dist/Store/Cosmos.d.ts.map +1 -1
  53. package/dist/Store/Cosmos.js +344 -246
  54. package/dist/Store/Disk.d.ts +5 -5
  55. package/dist/Store/Disk.d.ts.map +1 -1
  56. package/dist/Store/Disk.js +78 -38
  57. package/dist/Store/Memory.d.ts +7 -10
  58. package/dist/Store/Memory.d.ts.map +1 -1
  59. package/dist/Store/Memory.js +326 -66
  60. package/dist/Store/SQL/Pg.d.ts +4 -0
  61. package/dist/Store/SQL/Pg.d.ts.map +1 -0
  62. package/dist/Store/SQL/Pg.js +232 -0
  63. package/dist/Store/SQL/query.d.ts +49 -0
  64. package/dist/Store/SQL/query.d.ts.map +1 -0
  65. package/dist/Store/SQL/query.js +527 -0
  66. package/dist/Store/SQL.d.ts +21 -0
  67. package/dist/Store/SQL.d.ts.map +1 -0
  68. package/dist/Store/SQL.js +449 -0
  69. package/dist/Store/codeFilter.d.ts +5 -5
  70. package/dist/Store/codeFilter.d.ts.map +1 -1
  71. package/dist/Store/codeFilter.js +6 -3
  72. package/dist/Store/index.d.ts +7 -5
  73. package/dist/Store/index.d.ts.map +1 -1
  74. package/dist/Store/index.js +18 -5
  75. package/dist/Store/utils.d.ts +4 -3
  76. package/dist/Store/utils.d.ts.map +1 -1
  77. package/dist/Store/utils.js +5 -5
  78. package/dist/WorkflowEngineCosmos.d.ts +29 -0
  79. package/dist/WorkflowEngineCosmos.d.ts.map +1 -0
  80. package/dist/WorkflowEngineCosmos.js +521 -0
  81. package/dist/WorkflowEngineSqlite.d.ts +24 -0
  82. package/dist/WorkflowEngineSqlite.d.ts.map +1 -0
  83. package/dist/WorkflowEngineSqlite.js +550 -0
  84. package/dist/arbs.d.ts +2 -2
  85. package/dist/arbs.d.ts.map +1 -1
  86. package/dist/arbs.js +5 -3
  87. package/dist/codec.d.ts +5 -0
  88. package/dist/codec.d.ts.map +1 -0
  89. package/dist/codec.js +5 -0
  90. package/dist/cosmos-client.d.ts +16 -0
  91. package/dist/cosmos-client.d.ts.map +1 -0
  92. package/dist/cosmos-client.js +11 -0
  93. package/dist/errorReporter.d.ts +7 -5
  94. package/dist/errorReporter.d.ts.map +1 -1
  95. package/dist/errorReporter.js +23 -27
  96. package/dist/errors.d.ts +1 -1
  97. package/dist/fileUtil.d.ts +2 -2
  98. package/dist/fileUtil.d.ts.map +1 -1
  99. package/dist/fileUtil.js +2 -2
  100. package/dist/index.d.ts +3 -2
  101. package/dist/index.d.ts.map +1 -1
  102. package/dist/index.js +3 -2
  103. package/dist/internal/RequestContextMiddleware.d.ts +5 -0
  104. package/dist/internal/RequestContextMiddleware.d.ts.map +1 -0
  105. package/dist/internal/RequestContextMiddleware.js +45 -0
  106. package/dist/internal/auth.d.ts +53 -0
  107. package/dist/internal/auth.d.ts.map +1 -0
  108. package/dist/internal/auth.js +180 -0
  109. package/dist/internal/events.d.ts +11 -0
  110. package/dist/internal/events.d.ts.map +1 -0
  111. package/dist/internal/events.js +49 -0
  112. package/dist/internal/health.d.ts +3 -0
  113. package/dist/internal/health.d.ts.map +1 -0
  114. package/dist/internal/health.js +5 -0
  115. package/dist/layerUtils.d.ts +32 -0
  116. package/dist/layerUtils.d.ts.map +1 -0
  117. package/dist/layerUtils.js +17 -0
  118. package/dist/logger/jsonLogger.d.ts +2 -2
  119. package/dist/logger/jsonLogger.d.ts.map +1 -1
  120. package/dist/logger/jsonLogger.js +5 -3
  121. package/dist/logger/logFmtLogger.d.ts +2 -2
  122. package/dist/logger/logFmtLogger.d.ts.map +1 -1
  123. package/dist/logger/logFmtLogger.js +3 -3
  124. package/dist/logger/shared.d.ts +3 -3
  125. package/dist/logger/shared.d.ts.map +1 -1
  126. package/dist/logger/shared.js +5 -5
  127. package/dist/logger.d.ts +1 -1
  128. package/dist/logger.d.ts.map +1 -1
  129. package/dist/memQueue.d.ts +15 -0
  130. package/dist/memQueue.d.ts.map +1 -0
  131. package/dist/memQueue.js +21 -0
  132. package/dist/middlewares.d.ts +10 -0
  133. package/dist/middlewares.d.ts.map +1 -0
  134. package/dist/{api/middlewares.js → middlewares.js} +1 -1
  135. package/dist/mongo-client.d.ts +11 -0
  136. package/dist/mongo-client.d.ts.map +1 -0
  137. package/dist/mongo-client.js +15 -0
  138. package/dist/otel.d.ts +75 -0
  139. package/dist/otel.d.ts.map +1 -0
  140. package/dist/otel.js +65 -0
  141. package/dist/rateLimit.d.ts +12 -4
  142. package/dist/rateLimit.d.ts.map +1 -1
  143. package/dist/rateLimit.js +7 -12
  144. package/dist/redis-client.d.ts +42 -0
  145. package/dist/redis-client.d.ts.map +1 -0
  146. package/dist/redis-client.js +98 -0
  147. package/dist/reportError.d.ts +4 -0
  148. package/dist/reportError.d.ts.map +1 -0
  149. package/dist/reportError.js +28 -0
  150. package/dist/routing/middleware/RouterMiddleware.d.ts +16 -0
  151. package/dist/routing/middleware/RouterMiddleware.d.ts.map +1 -0
  152. package/dist/{api/routing → routing}/middleware/RouterMiddleware.js +1 -1
  153. package/dist/routing/middleware/middleware.d.ts +48 -0
  154. package/dist/routing/middleware/middleware.d.ts.map +1 -0
  155. package/dist/routing/middleware/middleware.js +128 -0
  156. package/dist/routing/middleware.d.ts +3 -0
  157. package/dist/routing/middleware.d.ts.map +1 -0
  158. package/dist/{api/routing → routing}/middleware.js +1 -2
  159. package/dist/routing/schema/jwt.d.ts +4 -0
  160. package/dist/routing/schema/jwt.d.ts.map +1 -0
  161. package/dist/routing/schema/jwt.js +13 -0
  162. package/dist/routing/tsort.d.ts +8 -0
  163. package/dist/routing/tsort.d.ts.map +1 -0
  164. package/dist/routing/tsort.js +51 -0
  165. package/dist/routing/utils.d.ts +19 -0
  166. package/dist/routing/utils.d.ts.map +1 -0
  167. package/dist/routing/utils.js +45 -0
  168. package/dist/routing.d.ts +184 -0
  169. package/dist/routing.d.ts.map +1 -0
  170. package/dist/routing.js +236 -0
  171. package/dist/test.d.ts +3 -3
  172. package/dist/test.d.ts.map +1 -1
  173. package/dist/test.js +2 -2
  174. package/dist/util.d.ts +3 -0
  175. package/dist/util.d.ts.map +1 -0
  176. package/dist/util.js +14 -0
  177. package/dist/vitest.d.ts +1 -1
  178. package/docs/cluster-storage.md +26 -0
  179. package/docs/workflow-engine.md +262 -0
  180. package/examples/query.ts +47 -39
  181. package/package.json +31 -345
  182. package/run.sh +1 -0
  183. package/src/CUPS.ts +52 -13
  184. package/src/ClusterCosmos.ts +984 -0
  185. package/src/ClusterServiceBus.ts +242 -0
  186. package/src/{api/ContextProvider.ts → ContextProvider.ts} +19 -16
  187. package/src/Emailer/Sendgrid.ts +82 -59
  188. package/src/Emailer/fake.ts +3 -3
  189. package/src/MainFiberSet.ts +12 -10
  190. package/src/QueueMaker/SQLQueue.ts +153 -156
  191. package/src/QueueMaker/errors.ts +3 -1
  192. package/src/QueueMaker/memQueue.ts +113 -107
  193. package/src/QueueMaker/sbqueue.ts +78 -90
  194. package/src/RequestFiberSet.ts +13 -8
  195. package/src/{adapters/SQL → SQL}/Model.ts +42 -41
  196. package/src/ServiceBus.ts +219 -0
  197. package/src/Store/Cosmos/query.ts +216 -52
  198. package/src/Store/Cosmos.ts +493 -353
  199. package/src/Store/Disk.ts +109 -69
  200. package/src/Store/Memory.ts +365 -96
  201. package/src/Store/SQL/Pg.ts +363 -0
  202. package/src/Store/SQL/query.ts +603 -0
  203. package/src/Store/SQL.ts +735 -0
  204. package/src/Store/codeFilter.ts +8 -5
  205. package/src/Store/index.ts +21 -6
  206. package/src/Store/utils.ts +26 -24
  207. package/src/WorkflowEngineCosmos.ts +719 -0
  208. package/src/WorkflowEngineSqlite.ts +813 -0
  209. package/src/arbs.ts +5 -3
  210. package/src/{adapters/cosmos-client.ts → cosmos-client.ts} +5 -3
  211. package/src/errorReporter.ts +66 -76
  212. package/src/fileUtil.ts +1 -1
  213. package/src/index.ts +2 -1
  214. package/src/{api/internal → internal}/RequestContextMiddleware.ts +23 -6
  215. package/src/internal/auth.ts +272 -0
  216. package/src/{api/internal → internal}/events.ts +22 -13
  217. package/src/{api/layerUtils.ts → layerUtils.ts} +14 -10
  218. package/src/logger/jsonLogger.ts +4 -2
  219. package/src/logger/logFmtLogger.ts +2 -2
  220. package/src/logger/shared.ts +5 -4
  221. package/src/{adapters/memQueue.ts → memQueue.ts} +5 -4
  222. package/src/{adapters/mongo-client.ts → mongo-client.ts} +4 -2
  223. package/src/otel.ts +152 -0
  224. package/src/rateLimit.ts +34 -23
  225. package/src/{adapters/redis-client.ts → redis-client.ts} +7 -3
  226. package/src/{api/reportError.ts → reportError.ts} +3 -2
  227. package/src/{api/routing → routing}/middleware/RouterMiddleware.ts +5 -4
  228. package/src/{api/routing → routing}/middleware/middleware.ts +62 -17
  229. package/src/routing/middleware.ts +4 -0
  230. package/src/{api/routing → routing}/schema/jwt.ts +2 -1
  231. package/src/{api/routing → routing}/utils.ts +2 -1
  232. package/src/routing.ts +768 -0
  233. package/src/test.ts +2 -2
  234. package/test/auth.test.ts +101 -0
  235. package/test/cluster-cosmos.test.ts +590 -0
  236. package/test/cluster-servicebus.test.ts +180 -0
  237. package/test/cluster-sqlite.test.ts +207 -0
  238. package/test/contextProvider.test.ts +15 -12
  239. package/test/controller.test.ts +28 -32
  240. package/test/cosmos-query.test.ts +159 -0
  241. package/test/dist/_check-agg-infer.test-d.d.ts +2 -0
  242. package/test/dist/_check-agg-infer.test-d.d.ts.map +1 -0
  243. package/test/dist/_check-agg-infer.test-d.js +19 -0
  244. package/test/dist/_check-proj-infer.test-d.d.ts +2 -0
  245. package/test/dist/_check-proj-infer.test-d.d.ts.map +1 -0
  246. package/test/dist/_check-proj-infer.test-d.js +16 -0
  247. package/test/dist/_check-tighten.test-d.d.ts +2 -0
  248. package/test/dist/_check-tighten.test-d.d.ts.map +1 -0
  249. package/test/dist/_check-tighten.test-d.js +21 -0
  250. package/test/dist/auth.test.d.ts.map +1 -0
  251. package/test/dist/cluster-cosmos.test.d.ts.map +1 -0
  252. package/test/dist/cluster-servicebus.test.d.ts.map +1 -0
  253. package/test/dist/cluster-sqlite.test.d.ts.map +1 -0
  254. package/test/dist/contextProvider.test.d.ts.map +1 -1
  255. package/test/dist/controller.test.d.ts.map +1 -1
  256. package/test/dist/cosmos-query.test.d.ts.map +1 -0
  257. package/test/dist/date-query.test.d.ts.map +1 -0
  258. package/test/dist/fixtures.d.ts +30 -12
  259. package/test/dist/fixtures.d.ts.map +1 -1
  260. package/test/dist/fixtures.js +17 -10
  261. package/test/dist/query.test.d.ts.map +1 -1
  262. package/test/dist/rawQuery.test.d.ts.map +1 -1
  263. package/test/dist/repository-ext.test.d.ts.map +1 -0
  264. package/test/dist/requires.test.d.ts.map +1 -1
  265. package/test/dist/router-generator.test.d.ts.map +1 -0
  266. package/test/dist/routing-interruptibility.test.d.ts.map +1 -0
  267. package/test/dist/rpc-context-map-streaming.test.d.ts.map +1 -0
  268. package/test/dist/rpc-e2e-invalidation.test.d.ts.map +1 -0
  269. package/test/dist/rpc-multi-middleware.test.d.ts.map +1 -1
  270. package/test/dist/rpc-stream-fullstack.test.d.ts.map +1 -0
  271. package/test/dist/sql-store.test.d.ts.map +1 -0
  272. package/test/dist/workflow-engine-cosmos.test.d.ts.map +1 -0
  273. package/test/dist/workflow-engine-sqlite.test.d.ts.map +1 -0
  274. package/test/fixtures.ts +16 -9
  275. package/test/layerUtils.test.ts +2 -2
  276. package/test/query.test.ts +905 -40
  277. package/test/rawQuery.test.ts +340 -22
  278. package/test/repository-ext.test.ts +62 -0
  279. package/test/requires.test.ts +10 -5
  280. package/test/router-generator.test.ts +187 -0
  281. package/test/routing-interruptibility.test.ts +66 -0
  282. package/test/rpc-context-map-streaming.test.ts +262 -0
  283. package/test/rpc-e2e-invalidation.test.ts +256 -0
  284. package/test/rpc-multi-middleware.test.ts +85 -10
  285. package/test/rpc-stream-fullstack.test.ts +304 -0
  286. package/test/sql-store.test.ts +1711 -0
  287. package/test/validateSample.test.ts +26 -14
  288. package/test/workflow-engine-cosmos.test.ts +354 -0
  289. package/test/workflow-engine-sqlite.test.ts +299 -0
  290. package/tsconfig.examples.json +1 -1
  291. package/tsconfig.json +2 -1
  292. package/tsconfig.json.bak +2 -2
  293. package/tsconfig.src.json +35 -35
  294. package/tsconfig.test.json +2 -2
  295. package/dist/Emailer/service.d.ts +0 -55
  296. package/dist/Emailer/service.d.ts.map +0 -1
  297. package/dist/Emailer/service.js +0 -6
  298. package/dist/Emailer.d.ts +0 -2
  299. package/dist/Emailer.d.ts.map +0 -1
  300. package/dist/Emailer.js +0 -2
  301. package/dist/Model/Repository/ext.d.ts +0 -41
  302. package/dist/Model/Repository/ext.d.ts.map +0 -1
  303. package/dist/Model/Repository/ext.js +0 -65
  304. package/dist/Model/Repository/internal/internal.d.ts +0 -59
  305. package/dist/Model/Repository/internal/internal.d.ts.map +0 -1
  306. package/dist/Model/Repository/internal/internal.js +0 -316
  307. package/dist/Model/Repository/legacy.d.ts +0 -19
  308. package/dist/Model/Repository/legacy.d.ts.map +0 -1
  309. package/dist/Model/Repository/legacy.js +0 -2
  310. package/dist/Model/Repository/makeRepo.d.ts +0 -49
  311. package/dist/Model/Repository/makeRepo.d.ts.map +0 -1
  312. package/dist/Model/Repository/makeRepo.js +0 -24
  313. package/dist/Model/Repository/service.d.ts +0 -89
  314. package/dist/Model/Repository/service.d.ts.map +0 -1
  315. package/dist/Model/Repository/service.js +0 -2
  316. package/dist/Model/Repository/validation.d.ts +0 -42
  317. package/dist/Model/Repository/validation.d.ts.map +0 -1
  318. package/dist/Model/Repository/validation.js +0 -32
  319. package/dist/Model/Repository.d.ts +0 -6
  320. package/dist/Model/Repository.d.ts.map +0 -1
  321. package/dist/Model/Repository.js +0 -6
  322. package/dist/Model/dsl.d.ts +0 -32
  323. package/dist/Model/dsl.d.ts.map +0 -1
  324. package/dist/Model/dsl.js +0 -44
  325. package/dist/Model/filter/filterApi.d.ts +0 -30
  326. package/dist/Model/filter/filterApi.d.ts.map +0 -1
  327. package/dist/Model/filter/filterApi.js +0 -2
  328. package/dist/Model/filter/types/errors.d.ts +0 -29
  329. package/dist/Model/filter/types/errors.d.ts.map +0 -1
  330. package/dist/Model/filter/types/errors.js +0 -2
  331. package/dist/Model/filter/types/fields.d.ts +0 -15
  332. package/dist/Model/filter/types/fields.d.ts.map +0 -1
  333. package/dist/Model/filter/types/fields.js +0 -2
  334. package/dist/Model/filter/types/path/common.d.ts +0 -316
  335. package/dist/Model/filter/types/path/common.d.ts.map +0 -1
  336. package/dist/Model/filter/types/path/common.js +0 -2
  337. package/dist/Model/filter/types/path/eager.d.ts +0 -95
  338. package/dist/Model/filter/types/path/eager.d.ts.map +0 -1
  339. package/dist/Model/filter/types/path/eager.js +0 -31
  340. package/dist/Model/filter/types/path/index.d.ts +0 -4
  341. package/dist/Model/filter/types/path/index.d.ts.map +0 -1
  342. package/dist/Model/filter/types/path/index.js +0 -3
  343. package/dist/Model/filter/types/utils.d.ts +0 -79
  344. package/dist/Model/filter/types/utils.d.ts.map +0 -1
  345. package/dist/Model/filter/types/utils.js +0 -2
  346. package/dist/Model/filter/types/validator.d.ts +0 -30
  347. package/dist/Model/filter/types/validator.d.ts.map +0 -1
  348. package/dist/Model/filter/types/validator.js +0 -2
  349. package/dist/Model/filter/types.d.ts +0 -5
  350. package/dist/Model/filter/types.d.ts.map +0 -1
  351. package/dist/Model/filter/types.js +0 -7
  352. package/dist/Model/query/dsl.d.ts +0 -248
  353. package/dist/Model/query/dsl.d.ts.map +0 -1
  354. package/dist/Model/query/dsl.js +0 -104
  355. package/dist/Model/query/new-kid-interpreter.d.ts +0 -28
  356. package/dist/Model/query/new-kid-interpreter.d.ts.map +0 -1
  357. package/dist/Model/query/new-kid-interpreter.js +0 -165
  358. package/dist/Model/query.d.ts +0 -15
  359. package/dist/Model/query.d.ts.map +0 -1
  360. package/dist/Model/query.js +0 -3
  361. package/dist/Model.d.ts +0 -4
  362. package/dist/Model.d.ts.map +0 -1
  363. package/dist/Model.js +0 -4
  364. package/dist/Operations.d.ts +0 -55
  365. package/dist/Operations.d.ts.map +0 -1
  366. package/dist/Operations.js +0 -102
  367. package/dist/OperationsRepo.d.ts +0 -41
  368. package/dist/OperationsRepo.d.ts.map +0 -1
  369. package/dist/OperationsRepo.js +0 -14
  370. package/dist/QueueMaker/service.d.ts +0 -11
  371. package/dist/QueueMaker/service.d.ts.map +0 -1
  372. package/dist/QueueMaker/service.js +0 -4
  373. package/dist/RequestContext.d.ts +0 -63
  374. package/dist/RequestContext.d.ts.map +0 -1
  375. package/dist/RequestContext.js +0 -49
  376. package/dist/Store/ContextMapContainer.d.ts +0 -14
  377. package/dist/Store/ContextMapContainer.d.ts.map +0 -1
  378. package/dist/Store/ContextMapContainer.js +0 -16
  379. package/dist/Store/service.d.ts +0 -108
  380. package/dist/Store/service.d.ts.map +0 -1
  381. package/dist/Store/service.js +0 -71
  382. package/dist/Store.d.ts +0 -2
  383. package/dist/Store.d.ts.map +0 -1
  384. package/dist/Store.js +0 -2
  385. package/dist/adapters/SQL/Model.d.ts +0 -479
  386. package/dist/adapters/SQL/Model.d.ts.map +0 -1
  387. package/dist/adapters/SQL/Model.js +0 -478
  388. package/dist/adapters/SQL.d.ts +0 -2
  389. package/dist/adapters/SQL.d.ts.map +0 -1
  390. package/dist/adapters/ServiceBus.d.ts +0 -58
  391. package/dist/adapters/ServiceBus.d.ts.map +0 -1
  392. package/dist/adapters/ServiceBus.js +0 -99
  393. package/dist/adapters/cosmos-client.d.ts +0 -14
  394. package/dist/adapters/cosmos-client.d.ts.map +0 -1
  395. package/dist/adapters/cosmos-client.js +0 -9
  396. package/dist/adapters/index.d.ts +0 -2
  397. package/dist/adapters/index.d.ts.map +0 -1
  398. package/dist/adapters/index.js +0 -2
  399. package/dist/adapters/logger.d.ts +0 -9
  400. package/dist/adapters/logger.d.ts.map +0 -1
  401. package/dist/adapters/logger.js +0 -3
  402. package/dist/adapters/memQueue.d.ts +0 -13
  403. package/dist/adapters/memQueue.d.ts.map +0 -1
  404. package/dist/adapters/memQueue.js +0 -20
  405. package/dist/adapters/mongo-client.d.ts +0 -10
  406. package/dist/adapters/mongo-client.d.ts.map +0 -1
  407. package/dist/adapters/mongo-client.js +0 -13
  408. package/dist/adapters/redis-client.d.ts +0 -39
  409. package/dist/adapters/redis-client.d.ts.map +0 -1
  410. package/dist/adapters/redis-client.js +0 -94
  411. package/dist/api/ContextProvider.d.ts +0 -31
  412. package/dist/api/ContextProvider.d.ts.map +0 -1
  413. package/dist/api/ContextProvider.js +0 -38
  414. package/dist/api/codec.d.ts +0 -5
  415. package/dist/api/codec.d.ts.map +0 -1
  416. package/dist/api/codec.js +0 -5
  417. package/dist/api/internal/RequestContextMiddleware.d.ts +0 -5
  418. package/dist/api/internal/RequestContextMiddleware.d.ts.map +0 -1
  419. package/dist/api/internal/RequestContextMiddleware.js +0 -35
  420. package/dist/api/internal/auth.d.ts +0 -15
  421. package/dist/api/internal/auth.d.ts.map +0 -1
  422. package/dist/api/internal/auth.js +0 -47
  423. package/dist/api/internal/events.d.ts +0 -9
  424. package/dist/api/internal/events.d.ts.map +0 -1
  425. package/dist/api/internal/events.js +0 -42
  426. package/dist/api/internal/health.d.ts +0 -3
  427. package/dist/api/internal/health.d.ts.map +0 -1
  428. package/dist/api/internal/health.js +0 -5
  429. package/dist/api/layerUtils.d.ts +0 -24
  430. package/dist/api/layerUtils.d.ts.map +0 -1
  431. package/dist/api/layerUtils.js +0 -16
  432. package/dist/api/middlewares.d.ts +0 -10
  433. package/dist/api/middlewares.d.ts.map +0 -1
  434. package/dist/api/reportError.d.ts +0 -4
  435. package/dist/api/reportError.d.ts.map +0 -1
  436. package/dist/api/reportError.js +0 -27
  437. package/dist/api/routing/middleware/RouterMiddleware.d.ts +0 -15
  438. package/dist/api/routing/middleware/RouterMiddleware.d.ts.map +0 -1
  439. package/dist/api/routing/middleware/middleware.d.ts +0 -9
  440. package/dist/api/routing/middleware/middleware.d.ts.map +0 -1
  441. package/dist/api/routing/middleware/middleware.js +0 -92
  442. package/dist/api/routing/middleware.d.ts +0 -4
  443. package/dist/api/routing/middleware.d.ts.map +0 -1
  444. package/dist/api/routing/schema/jwt.d.ts +0 -4
  445. package/dist/api/routing/schema/jwt.d.ts.map +0 -1
  446. package/dist/api/routing/schema/jwt.js +0 -12
  447. package/dist/api/routing/tsort.d.ts +0 -8
  448. package/dist/api/routing/tsort.d.ts.map +0 -1
  449. package/dist/api/routing/tsort.js +0 -51
  450. package/dist/api/routing/utils.d.ts +0 -19
  451. package/dist/api/routing/utils.d.ts.map +0 -1
  452. package/dist/api/routing/utils.js +0 -44
  453. package/dist/api/routing.d.ts +0 -138
  454. package/dist/api/routing.d.ts.map +0 -1
  455. package/dist/api/routing.js +0 -166
  456. package/dist/api/setupRequest.d.ts +0 -12
  457. package/dist/api/setupRequest.d.ts.map +0 -1
  458. package/dist/api/setupRequest.js +0 -44
  459. package/dist/api/util.d.ts +0 -3
  460. package/dist/api/util.d.ts.map +0 -1
  461. package/dist/api/util.js +0 -14
  462. package/eslint.config.mjs +0 -24
  463. package/src/Emailer/service.ts +0 -52
  464. package/src/Emailer.ts +0 -1
  465. package/src/Model/Repository/ext.ts +0 -283
  466. package/src/Model/Repository/internal/internal.ts +0 -577
  467. package/src/Model/Repository/legacy.ts +0 -27
  468. package/src/Model/Repository/makeRepo.ts +0 -139
  469. package/src/Model/Repository/service.ts +0 -627
  470. package/src/Model/Repository/validation.ts +0 -31
  471. package/src/Model/Repository.ts +0 -5
  472. package/src/Model/dsl.ts +0 -128
  473. package/src/Model/filter/filterApi.ts +0 -60
  474. package/src/Model/filter/types/errors.ts +0 -47
  475. package/src/Model/filter/types/fields.ts +0 -50
  476. package/src/Model/filter/types/path/common.ts +0 -404
  477. package/src/Model/filter/types/path/eager.ts +0 -298
  478. package/src/Model/filter/types/path/index.ts +0 -4
  479. package/src/Model/filter/types/utils.ts +0 -128
  480. package/src/Model/filter/types/validator.ts +0 -46
  481. package/src/Model/filter/types.ts +0 -6
  482. package/src/Model/query/dsl.ts +0 -2110
  483. package/src/Model/query/new-kid-interpreter.ts +0 -210
  484. package/src/Model/query.ts +0 -13
  485. package/src/Model.ts +0 -3
  486. package/src/Operations.ts +0 -235
  487. package/src/OperationsRepo.ts +0 -16
  488. package/src/QueueMaker/service.ts +0 -17
  489. package/src/RequestContext.ts +0 -63
  490. package/src/Store/ContextMapContainer.ts +0 -20
  491. package/src/Store/service.ts +0 -184
  492. package/src/Store.ts +0 -1
  493. package/src/adapters/ServiceBus.ts +0 -209
  494. package/src/adapters/index.ts +0 -0
  495. package/src/adapters/logger.ts +0 -3
  496. package/src/api/internal/auth.ts +0 -68
  497. package/src/api/routing/middleware.ts +0 -6
  498. package/src/api/routing.ts +0 -598
  499. package/src/api/setupRequest.ts +0 -84
  500. /package/src/{adapters/SQL.ts → SQL.ts} +0 -0
  501. /package/src/{api/codec.ts → codec.ts} +0 -0
  502. /package/src/{api/internal → internal}/health.ts +0 -0
  503. /package/src/{api/middlewares.ts → middlewares.ts} +0 -0
  504. /package/src/{api/routing → routing}/tsort.ts +0 -0
  505. /package/src/{api/util.ts → util.ts} +0 -0
@@ -0,0 +1,590 @@
1
+ import { assert, describe, expect, it } from "@effect/vitest"
2
+ import { Context, Duration, Effect, Exit, Fiber, Latch, Layer, Option, Redacted, Schema } from "effect"
3
+ import { TestClock } from "effect/testing"
4
+ import { ClusterSchema, ClusterWorkflowEngine, Entity, EntityAddress, EntityId, EntityType, Envelope, Message, MessageStorage, Reply, Runner, RunnerAddress, RunnerHealth, Runners, RunnerStorage, ShardId, Sharding, ShardingConfig, Snowflake } from "effect/unstable/cluster"
5
+ import { Headers } from "effect/unstable/http"
6
+ import { Rpc, RpcSchema } from "effect/unstable/rpc"
7
+ import { DurableDeferred, Workflow } from "effect/unstable/workflow"
8
+ import { layerCosmos } from "../src/ClusterCosmos.js"
9
+
10
+ const cosmosUrl = process.env["COSMOS_TEST_URL"]
11
+ const cosmosDb = process.env["COSMOS_TEST_DB"] ?? "cluster-test"
12
+ const testRunId = `${Date.now()}-${process.pid}-${Math.random().toString(16).slice(2)}`
13
+ const runnerPortBase = 10000 + Date.now() % 40000
14
+
15
+ const layerFor = () =>
16
+ layerCosmos({
17
+ url: Redacted.make(cosmosUrl ?? ""),
18
+ dbName: cosmosDb,
19
+ prefix: "test-cluster-"
20
+ })
21
+ .pipe(
22
+ Layer.provideMerge(Snowflake.layerGenerator),
23
+ Layer.provide(ShardingConfig.layerDefaults)
24
+ )
25
+
26
+ describe.skipIf(!cosmosUrl)("ClusterCosmos MessageStorage", () => {
27
+ it.effect("deduplicates keyed requests and returns the last reply", () =>
28
+ Effect
29
+ .gen(function*() {
30
+ const storage = yield* MessageStorage.MessageStorage
31
+ const shardId = testShardId("message-duplicate")
32
+ const primaryKey = `primary/${testRunId}\\with?illegal#chars`
33
+ const request = yield* makeStreamRequest(primaryKey, shardId)
34
+
35
+ const saved = yield* storage.saveRequest(request)
36
+ assert.strictEqual(saved._tag, "Success")
37
+
38
+ const chunk = yield* makeChunkReply(request, 0)
39
+ yield* storage.saveReply(chunk)
40
+
41
+ const duplicateWithChunk = yield* storage.saveRequest(
42
+ yield* makeStreamRequest(primaryKey, shardId)
43
+ )
44
+ assert(duplicateWithChunk._tag === "Duplicate" && Option.isSome(duplicateWithChunk.lastReceivedReply))
45
+ assert.strictEqual(duplicateWithChunk.lastReceivedReply.value._tag, "Chunk")
46
+
47
+ const ackChunk = yield* makeAckChunk(request, chunk)
48
+ yield* storage.saveEnvelope(ackChunk)
49
+ const repliesAfterAck = yield* storage.repliesFor([request])
50
+ assert.strictEqual(repliesAfterAck.length, 0)
51
+
52
+ yield* storage.saveReply(yield* makeStreamReply(request))
53
+ const duplicateWithExit = yield* storage.saveRequest(
54
+ yield* makeStreamRequest(primaryKey, shardId)
55
+ )
56
+ assert(duplicateWithExit._tag === "Duplicate" && Option.isSome(duplicateWithExit.lastReceivedReply))
57
+ assert.strictEqual(duplicateWithExit.lastReceivedReply.value._tag, "WithExit")
58
+ })
59
+ .pipe(Effect.provide(layerFor())))
60
+
61
+ it.effect("marks reads, resets shards, and excludes completed requests", () =>
62
+ Effect
63
+ .gen(function*() {
64
+ const storage = yield* MessageStorage.MessageStorage
65
+ const shardId = testShardId("message-unprocessed")
66
+ const request1 = yield* makeRequest({ payload: { id: 1 }, shardId })
67
+ const request2 = yield* makeRequest({ payload: { id: 2 }, shardId })
68
+ assert.strictEqual((yield* storage.saveRequest(request1))._tag, "Success")
69
+ assert.strictEqual((yield* storage.saveRequest(request2))._tag, "Success")
70
+
71
+ let messages = yield* storage.unprocessedMessages([request1.envelope.address.shardId])
72
+ assert.deepStrictEqual(messages.map((message) => requestPayloadId(message)).sort(), [1, 2])
73
+
74
+ messages = yield* storage.unprocessedMessages([request1.envelope.address.shardId])
75
+ assert.strictEqual(messages.length, 0)
76
+
77
+ yield* storage.resetShards([request1.envelope.address.shardId])
78
+ messages = yield* storage.unprocessedMessages([request1.envelope.address.shardId])
79
+ assert.deepStrictEqual(messages.map((message) => requestPayloadId(message)).sort(), [1, 2])
80
+
81
+ yield* storage.saveReply(yield* makeReply(request1))
82
+ yield* storage.resetShards([request1.envelope.address.shardId])
83
+ messages = yield* storage.unprocessedMessages([request1.envelope.address.shardId])
84
+ assert.deepStrictEqual(messages.map((message) => requestPayloadId(message)), [2])
85
+ })
86
+ .pipe(Effect.provide(layerFor())))
87
+
88
+ it.effect("returns each unprocessed message to only one concurrent node poll", () =>
89
+ Effect
90
+ .gen(function*() {
91
+ const storage = yield* MessageStorage.MessageStorage
92
+ const shardId = testShardId("message-concurrent-poll")
93
+ const expectedIds = Array.from({ length: 20 }, (_, index) => index + 1)
94
+ const requests = yield* Effect.forEach(expectedIds, (id) => makeRequest({ payload: { id }, shardId }))
95
+ yield* Effect.forEach(requests, (request) => storage.saveRequest(request), { discard: true })
96
+
97
+ const polls = yield* Effect.forEach(
98
+ Array.from({ length: 8 }, () => void 0),
99
+ () => storage.unprocessedMessages([shardId]),
100
+ { concurrency: "unbounded" }
101
+ )
102
+ const receivedIds = polls
103
+ .flatMap((messages) => messages.map(requestPayloadId))
104
+ .sort((a, b) => a - b)
105
+
106
+ assert.deepStrictEqual(receivedIds, expectedIds)
107
+ assert.strictEqual((yield* storage.unprocessedMessages([shardId])).length, 0)
108
+ })
109
+ .pipe(Effect.provide(layerFor())), 20000)
110
+
111
+ it.effect("preserves terminal reply state after concurrent node polling", () =>
112
+ Effect
113
+ .gen(function*() {
114
+ const storage = yield* MessageStorage.MessageStorage
115
+ const shardId = testShardId("message-concurrent-reply")
116
+ const primaryKey = `concurrent-reply/${testRunId}`
117
+ const request = yield* makeStreamRequest(primaryKey, shardId)
118
+ assert.strictEqual((yield* storage.saveRequest(request))._tag, "Success")
119
+
120
+ const polls = yield* Effect.forEach(
121
+ Array.from({ length: 8 }, () => void 0),
122
+ () => storage.unprocessedMessages([shardId]),
123
+ { concurrency: "unbounded" }
124
+ )
125
+ assert.strictEqual(polls.flat().length, 1)
126
+
127
+ yield* storage.saveReply(yield* makeStreamReply(request))
128
+
129
+ const duplicate = yield* storage.saveRequest(yield* makeStreamRequest(primaryKey, shardId))
130
+ assert(duplicate._tag === "Duplicate" && Option.isSome(duplicate.lastReceivedReply))
131
+ assert.strictEqual(duplicate.lastReceivedReply.value._tag, "WithExit")
132
+ })
133
+ .pipe(Effect.provide(layerFor())), 20000)
134
+
135
+ it.effect("notifies registered reply handlers", () =>
136
+ Effect
137
+ .gen(function*() {
138
+ const storage = yield* MessageStorage.MessageStorage
139
+ const latch = yield* Latch.make()
140
+ const request = yield* makeRequest({ shardId: testShardId("message-handler") })
141
+ yield* storage.saveRequest(request)
142
+
143
+ const fiber = yield* storage
144
+ .registerReplyHandler(
145
+ new Message.OutgoingRequest({
146
+ ...request,
147
+ respond: () => latch.open
148
+ })
149
+ )
150
+ .pipe(Effect.forkChild)
151
+
152
+ yield* TestClock.adjust(1)
153
+ yield* storage.saveReply(yield* makeReply(request))
154
+ yield* latch.await
155
+ yield* Fiber.await(fiber)
156
+ })
157
+ .pipe(Effect.provide(layerFor())))
158
+ })
159
+
160
+ describe.skipIf(!cosmosUrl)("ClusterCosmos RunnerStorage", () => {
161
+ it.effect("registers runners and tracks health", () =>
162
+ Effect
163
+ .gen(function*() {
164
+ const storage = yield* RunnerStorage.RunnerStorage
165
+ const runnerAddress = testRunnerAddress(1)
166
+ const runner = Runner.make({
167
+ address: runnerAddress,
168
+ groups: ["default"],
169
+ weight: 1
170
+ })
171
+
172
+ const machineId1 = yield* storage.register(runner, true)
173
+ const machineId2 = yield* storage.register(runner, true)
174
+ assert.deepStrictEqual(machineId2, machineId1)
175
+ expect(runnerStatus(yield* storage.getRunners, runnerAddress)).toEqual([runner, true])
176
+
177
+ yield* storage.setRunnerHealth(runnerAddress, false)
178
+ expect(runnerStatus(yield* storage.getRunners, runnerAddress)).toEqual([runner, false])
179
+
180
+ yield* storage.unregister(runnerAddress)
181
+ expect(runnerStatus(yield* storage.getRunners, runnerAddress)).toBeUndefined()
182
+ })
183
+ .pipe(Effect.provide(layerFor())))
184
+
185
+ it.effect("preserves shard lock ownership when two runners acquire concurrently", () =>
186
+ Effect
187
+ .gen(function*() {
188
+ const storage = yield* RunnerStorage.RunnerStorage
189
+ const runnerAddress1 = testRunnerAddress(4)
190
+ const runnerAddress2 = testRunnerAddress(5)
191
+ const runner1 = Runner.make({
192
+ address: runnerAddress1,
193
+ groups: ["default"],
194
+ weight: 1
195
+ })
196
+ const runner2 = Runner.make({
197
+ address: runnerAddress2,
198
+ groups: ["default"],
199
+ weight: 1
200
+ })
201
+ const shards = Array.from({ length: 16 }, (_, index) => testShardId("runner-occ", index + 1))
202
+
203
+ yield* storage.register(runner1, true)
204
+ yield* storage.register(runner2, true)
205
+
206
+ const [acquired1, acquired2] = yield* Effect.all([
207
+ storage.acquire(runnerAddress1, shards),
208
+ storage.acquire(runnerAddress2, shards)
209
+ ], { concurrency: "unbounded" })
210
+ const acquiredIds = [...acquired1, ...acquired2].map((shard) => shard.id).sort((a, b) => a - b)
211
+
212
+ assert.deepStrictEqual(acquiredIds, shards.map((shard) => shard.id))
213
+ assert.strictEqual(new Set(acquiredIds).size, shards.length)
214
+ assert.deepStrictEqual(
215
+ (yield* storage.refresh(runnerAddress1, shards)).map((shard) => shard.id).sort((a, b) => a - b),
216
+ acquired1.map((shard) => shard.id).sort((a, b) => a - b)
217
+ )
218
+ assert.deepStrictEqual(
219
+ (yield* storage.refresh(runnerAddress2, shards)).map((shard) => shard.id).sort((a, b) => a - b),
220
+ acquired2.map((shard) => shard.id).sort((a, b) => a - b)
221
+ )
222
+ })
223
+ .pipe(Effect.provide(layerFor())), 20000)
224
+
225
+ it.effect("acquires, refreshes, releases, and re-acquires shard locks", () =>
226
+ Effect
227
+ .gen(function*() {
228
+ const storage = yield* RunnerStorage.RunnerStorage
229
+ const runnerAddress1 = testRunnerAddress(2)
230
+ const runnerAddress2 = testRunnerAddress(3)
231
+ const shards = [
232
+ testShardId("runner-locks", 1),
233
+ testShardId("runner-locks", 2),
234
+ testShardId("runner-locks", 3)
235
+ ]
236
+
237
+ let acquired = yield* storage.acquire(runnerAddress1, shards)
238
+ assert.deepStrictEqual(acquired.map((shard) => shard.id), [1, 2, 3])
239
+
240
+ acquired = yield* storage.acquire(runnerAddress2, shards)
241
+ assert.deepStrictEqual(acquired.map((shard) => shard.id), [])
242
+
243
+ const refreshed = yield* storage.refresh(runnerAddress1, shards)
244
+ assert.deepStrictEqual(refreshed.map((shard) => shard.id), [1, 2, 3])
245
+
246
+ yield* storage.release(runnerAddress1, testShardId("runner-locks", 2))
247
+ acquired = yield* storage.acquire(runnerAddress2, shards)
248
+ assert.deepStrictEqual(acquired.map((shard) => shard.id), [2])
249
+
250
+ yield* storage.releaseAll(runnerAddress1)
251
+ acquired = yield* storage.acquire(runnerAddress2, shards)
252
+ assert.deepStrictEqual(acquired.map((shard) => shard.id), [1, 2, 3])
253
+ })
254
+ .pipe(Effect.provide(layerFor())))
255
+ })
256
+
257
+ describe.skipIf(!cosmosUrl)("ClusterCosmos Sharding RPC", () => {
258
+ it.effect("runs persisted entity RPCs through Cosmos-backed cluster storage", () =>
259
+ Effect
260
+ .gen(function*() {
261
+ yield* TestClock.adjust(1)
262
+ const sharding = yield* Sharding.Sharding
263
+ const makeClient = yield* CosmosRpcEntity.client
264
+ const entityId = `entity/${testRunId}\\with?illegal#chars`
265
+ const shardId = sharding.getShardId(EntityId.make(entityId), testShardGroup("rpc"))
266
+ yield* waitForShard(sharding, shardId)
267
+ assert.isTrue(sharding.hasShardId(shardId))
268
+ const client = makeClient(entityId)
269
+
270
+ const user = yield* client.GetCosmosUser({ id: 42 })
271
+ expect(user).toEqual(new CosmosRpcUser({ id: 42, name: "User 42" }))
272
+
273
+ const primaryKey = `rpc/${testRunId}\\with?illegal#chars`
274
+ const first = yield* client.CosmosRequestWithKey({ key: primaryKey })
275
+ const duplicate = yield* client.CosmosRequestWithKey({ key: primaryKey })
276
+
277
+ assert.strictEqual(first, primaryKey)
278
+ assert.strictEqual(duplicate, primaryKey)
279
+ })
280
+ .pipe(Effect.provide(clusterRpcLayer("rpc"))), 20000)
281
+ })
282
+
283
+ describe.skipIf(!cosmosUrl)("ClusterCosmos Workflow", () => {
284
+ it.live("resumes a running workflow suspended on a durable deferred", () =>
285
+ Effect
286
+ .gen(function*() {
287
+ const sharding = yield* Sharding.Sharding
288
+ const payload = { id: `deferred/${testRunId}\\with?illegal#chars` }
289
+ const executionId = yield* CosmosDeferredWorkflow.executionId(payload)
290
+
291
+ const fiber = yield* CosmosDeferredWorkflow.execute(payload).pipe(Effect.forkScoped)
292
+ yield* waitForDeferredWorkflowSuspended(executionId)
293
+
294
+ const token = yield* DurableDeferred.tokenFromPayload(CosmosDeferred, {
295
+ workflow: CosmosDeferredWorkflow,
296
+ payload
297
+ })
298
+ yield* DurableDeferred.done(CosmosDeferred, { token, exit: Exit.succeed("resolved") })
299
+ yield* sharding.pollStorage
300
+
301
+ const value = yield* Fiber.join(fiber).pipe(Effect.timeout(Duration.seconds(15)))
302
+ assert.strictEqual(value, `${payload.id}:resolved`)
303
+ assert.strictEqual(yield* waitForDeferredWorkflowComplete(executionId), `${payload.id}:resolved`)
304
+ })
305
+ .pipe(Effect.provide(clusterWorkflowLayer())), 30000)
306
+ })
307
+
308
+ const GetUserRpc = Rpc.make("GetUser", {
309
+ payload: { id: Schema.Number }
310
+ })
311
+
312
+ class CosmosRpcUser extends Schema.Class<CosmosRpcUser>("CosmosRpcUser")({
313
+ id: Schema.Number,
314
+ name: Schema.String
315
+ }) {}
316
+
317
+ const CosmosRpcEntity = Entity
318
+ .make("CosmosRpcEntity", [
319
+ Rpc.make("GetCosmosUser", {
320
+ success: CosmosRpcUser,
321
+ payload: { id: Schema.Number }
322
+ }),
323
+ Rpc.make("CosmosRequestWithKey", {
324
+ success: Schema.String,
325
+ payload: { key: Schema.String },
326
+ primaryKey: ({ key }) => key
327
+ })
328
+ ])
329
+ .annotate(ClusterSchema.ShardGroup, () => testShardGroup("rpc"))
330
+ .annotateRpcs(ClusterSchema.Persisted, true)
331
+
332
+ const CosmosRpcEntityLayer = CosmosRpcEntity.toLayer(
333
+ Effect.succeed(
334
+ CosmosRpcEntity.of({
335
+ GetCosmosUser: (envelope) =>
336
+ Effect.succeed(new CosmosRpcUser({ id: envelope.payload.id, name: `User ${envelope.payload.id}` })),
337
+ CosmosRequestWithKey: (envelope) => Effect.succeed(envelope.payload.key)
338
+ })
339
+ )
340
+ )
341
+
342
+ const CosmosDeferred = DurableDeferred.make("ClusterCosmos/Deferred", { success: Schema.String })
343
+
344
+ const CosmosDeferredWorkflow = Workflow
345
+ .make({
346
+ name: "ClusterCosmos/DeferredWorkflow",
347
+ payload: { id: Schema.String },
348
+ success: Schema.String,
349
+ idempotencyKey: ({ id }) => id
350
+ })
351
+ .annotate(ClusterSchema.ShardGroup, () => testShardGroup("workflow"))
352
+
353
+ const CosmosDeferredWorkflowLayer = CosmosDeferredWorkflow.toLayer(Effect.fnUntraced(function*({ id }) {
354
+ const value = yield* DurableDeferred.await(CosmosDeferred)
355
+ return `${id}:${value}`
356
+ }))
357
+
358
+ class StreamRpc extends Rpc.make("StreamTest", {
359
+ success: RpcSchema.Stream(Schema.Void, Schema.Never),
360
+ payload: {
361
+ id: Schema.String
362
+ },
363
+ primaryKey: (value) => value.id.toString()
364
+ }) {}
365
+
366
+ const makeRequest = Effect.fnUntraced(function*(options?: {
367
+ readonly payload?: { readonly id: number }
368
+ readonly shardId?: ShardId.ShardId
369
+ }) {
370
+ const snowflake = yield* Snowflake.Generator
371
+ return new Message.OutgoingRequest({
372
+ envelope: Envelope.makeRequest<typeof GetUserRpc>({
373
+ requestId: snowflake.nextUnsafe(),
374
+ address: EntityAddress.make({
375
+ shardId: options?.shardId ?? testShardId("default"),
376
+ entityType: EntityType.make("test"),
377
+ entityId: EntityId.make("1")
378
+ }),
379
+ tag: GetUserRpc._tag,
380
+ payload: options?.payload ?? { id: 123 },
381
+ traceId: "noop",
382
+ spanId: "noop",
383
+ sampled: false,
384
+ headers: Headers.empty
385
+ }),
386
+ annotations: GetUserRpc.annotations,
387
+ context: Context.empty(),
388
+ rpc: GetUserRpc,
389
+ lastReceivedReply: Option.none(),
390
+ respond() {
391
+ return Effect.void
392
+ }
393
+ })
394
+ })
395
+
396
+ const makeStreamRequest = Effect.fnUntraced(function*(id: string, shardId = testShardId("stream")) {
397
+ const snowflake = yield* Snowflake.Generator
398
+ return new Message.OutgoingRequest({
399
+ envelope: Envelope.makeRequest<typeof StreamRpc>({
400
+ requestId: snowflake.nextUnsafe(),
401
+ address: EntityAddress.make({
402
+ shardId,
403
+ entityType: EntityType.make("test"),
404
+ entityId: EntityId.make("1")
405
+ }),
406
+ tag: StreamRpc._tag,
407
+ payload: StreamRpc.payloadSchema.make({ id }),
408
+ traceId: "noop",
409
+ spanId: "noop",
410
+ sampled: false,
411
+ headers: Headers.empty
412
+ }),
413
+ annotations: StreamRpc.annotations,
414
+ context: Context.empty(),
415
+ rpc: StreamRpc,
416
+ lastReceivedReply: Option.none(),
417
+ respond() {
418
+ return Effect.void
419
+ }
420
+ })
421
+ })
422
+
423
+ const makeReply = Effect.fnUntraced(function*(request: Message.OutgoingRequest<typeof GetUserRpc>) {
424
+ const snowflake = yield* Snowflake.Generator
425
+ return new Reply.ReplyWithContext({
426
+ reply: new Reply.WithExit<typeof GetUserRpc>({
427
+ id: snowflake.nextUnsafe(),
428
+ requestId: request.envelope.requestId,
429
+ exit: Exit.void
430
+ }),
431
+ context: request.context,
432
+ rpc: request.rpc
433
+ })
434
+ })
435
+
436
+ const makeStreamReply = Effect.fnUntraced(function*(request: Message.OutgoingRequest<typeof StreamRpc>) {
437
+ const snowflake = yield* Snowflake.Generator
438
+ return new Reply.ReplyWithContext({
439
+ reply: new Reply.WithExit<typeof StreamRpc>({
440
+ id: snowflake.nextUnsafe(),
441
+ requestId: request.envelope.requestId,
442
+ exit: Exit.void
443
+ }),
444
+ context: request.context,
445
+ rpc: request.rpc
446
+ })
447
+ })
448
+
449
+ const makeAckChunk = Effect.fnUntraced(function*(
450
+ request: Message.OutgoingRequest<typeof StreamRpc>,
451
+ chunk: Reply.ReplyWithContext<typeof StreamRpc>
452
+ ) {
453
+ const snowflake = yield* Snowflake.Generator
454
+ return new Message.OutgoingEnvelope({
455
+ envelope: new Envelope.AckChunk({
456
+ id: snowflake.nextUnsafe(),
457
+ address: request.envelope.address,
458
+ requestId: chunk.reply.requestId,
459
+ replyId: chunk.reply.id
460
+ }),
461
+ rpc: request.rpc
462
+ })
463
+ })
464
+
465
+ const makeChunkReply = Effect.fnUntraced(function*(
466
+ request: Message.OutgoingRequest<typeof StreamRpc>,
467
+ sequence: number
468
+ ) {
469
+ const snowflake = yield* Snowflake.Generator
470
+ return new Reply.ReplyWithContext({
471
+ reply: new Reply.Chunk<typeof StreamRpc>({
472
+ id: snowflake.nextUnsafe(),
473
+ requestId: request.envelope.requestId,
474
+ sequence,
475
+ values: [undefined]
476
+ }),
477
+ context: request.context,
478
+ rpc: request.rpc
479
+ })
480
+ })
481
+
482
+ const requestPayloadId = (message: Message.Incoming<never>) => {
483
+ if (message.envelope._tag !== "Request") {
484
+ throw new Error(`Expected Request envelope`)
485
+ }
486
+ const envelope = message.envelope
487
+ assert(typeof envelope.payload === "object" && envelope.payload !== null)
488
+ assert("id" in envelope.payload)
489
+ assert.strictEqual(typeof envelope.payload.id, "number")
490
+ return envelope.payload.id
491
+ }
492
+
493
+ const testShardId = (label: string, id = 1) => ShardId.make(`cluster-cosmos-${testRunId}-${label}`, id)
494
+
495
+ const testShardGroup = (label: string) => `cluster-cosmos-${testRunId}-${label}`
496
+
497
+ const testRunnerAddress = (offset: number) => RunnerAddress.make("localhost", runnerPortBase + offset)
498
+
499
+ const runnerStatus = (
500
+ runners: ReadonlyArray<readonly [Runner.Runner, boolean]>,
501
+ address: RunnerAddress.RunnerAddress
502
+ ) => runners.find(([runner]) => runner.address.host === address.host && runner.address.port === address.port)
503
+
504
+ const clusterRpcLayer = (label: string) => {
505
+ const shardGroup = testShardGroup(label)
506
+ return CosmosRpcEntityLayer.pipe(
507
+ Layer.provideMerge(Sharding.layer),
508
+ Layer.provide(Runners.layerNoop),
509
+ Layer.provide(RunnerHealth.layerNoop),
510
+ Layer.provide(layerCosmos({
511
+ url: Redacted.make(cosmosUrl ?? ""),
512
+ dbName: cosmosDb,
513
+ prefix: "test-cluster-"
514
+ })),
515
+ Layer.provide(ShardingConfig.layer({
516
+ runnerAddress: Option.some(testRunnerAddress(10)),
517
+ shardsPerGroup: 1,
518
+ availableShardGroups: [shardGroup],
519
+ assignedShardGroups: [shardGroup],
520
+ entityTerminationTimeout: 0,
521
+ entityMessagePollInterval: 50,
522
+ entityReplyPollInterval: 50,
523
+ refreshAssignmentsInterval: 0,
524
+ sendRetryInterval: 50
525
+ }))
526
+ )
527
+ }
528
+
529
+ const clusterWorkflowLayer = () =>
530
+ CosmosDeferredWorkflowLayer.pipe(
531
+ Layer.provideMerge(
532
+ ClusterWorkflowEngine.layer.pipe(
533
+ Layer.provideMerge(Sharding.layer),
534
+ Layer.provide(Runners.layerNoop),
535
+ Layer.provide(RunnerHealth.layerNoop),
536
+ Layer.provide(layerCosmos({
537
+ url: Redacted.make(cosmosUrl ?? ""),
538
+ dbName: cosmosDb,
539
+ prefix: "test-cluster-"
540
+ })),
541
+ Layer.provide(ShardingConfig.layer({
542
+ runnerAddress: Option.some(testRunnerAddress(20)),
543
+ shardsPerGroup: 1,
544
+ availableShardGroups: [testShardGroup("workflow")],
545
+ assignedShardGroups: [testShardGroup("workflow")],
546
+ entityTerminationTimeout: 0,
547
+ entityMessagePollInterval: 50,
548
+ entityReplyPollInterval: 50,
549
+ refreshAssignmentsInterval: 0,
550
+ sendRetryInterval: 50
551
+ }))
552
+ )
553
+ )
554
+ )
555
+
556
+ const waitForDeferredWorkflowSuspended = (executionId: string) =>
557
+ Effect.gen(function*() {
558
+ const sharding = yield* Sharding.Sharding
559
+ for (let i = 0; i < 100; i++) {
560
+ yield* sharding.pollStorage
561
+ const polled = yield* CosmosDeferredWorkflow.poll(executionId)
562
+ if (Option.isSome(polled) && polled.value._tag === "Suspended") return
563
+ yield* Effect.sleep(Duration.millis(100))
564
+ }
565
+ return yield* Effect.fail(new Error(`Workflow ${executionId} did not suspend`))
566
+ })
567
+
568
+ const waitForDeferredWorkflowComplete = (executionId: string) =>
569
+ Effect.gen(function*() {
570
+ const sharding = yield* Sharding.Sharding
571
+ for (let i = 0; i < 100; i++) {
572
+ yield* sharding.pollStorage
573
+ const polled = yield* CosmosDeferredWorkflow.poll(executionId)
574
+ if (Option.isSome(polled) && polled.value._tag === "Complete") {
575
+ const exit = polled.value.exit
576
+ assert(Exit.isSuccess(exit))
577
+ return exit.value
578
+ }
579
+ yield* Effect.sleep(Duration.millis(100))
580
+ }
581
+ return yield* Effect.fail(new Error(`Workflow ${executionId} did not complete`))
582
+ })
583
+
584
+ const waitForShard = (sharding: Sharding.Sharding["Service"], shardId: ShardId.ShardId) =>
585
+ Effect.gen(function*() {
586
+ for (let i = 0; i < 30; i++) {
587
+ if (sharding.hasShardId(shardId)) return
588
+ yield* Effect.promise<void>(() => new Promise((resolve) => setTimeout(resolve, 100)))
589
+ }
590
+ })