@ersbeth/picoflow 1.1.1 → 2.0.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 (383) hide show
  1. package/.vscode/settings.json +3 -3
  2. package/CHANGELOG.md +43 -0
  3. package/README.md +2 -18
  4. package/biome.json +45 -35
  5. package/dist/picoflow.js +857 -1528
  6. package/dist/types/api/base/flowDisposable.d.ts +41 -0
  7. package/dist/types/api/base/flowDisposable.d.ts.map +1 -0
  8. package/dist/types/api/base/flowObservable.d.ts +27 -0
  9. package/dist/types/api/base/flowObservable.d.ts.map +1 -0
  10. package/dist/types/api/base/flowSubscribable.d.ts +79 -0
  11. package/dist/types/api/base/flowSubscribable.d.ts.map +1 -0
  12. package/dist/types/api/base/flowTracker.d.ts +8 -0
  13. package/dist/types/api/base/flowTracker.d.ts.map +1 -0
  14. package/dist/types/api/base/index.d.ts +5 -0
  15. package/dist/types/api/base/index.d.ts.map +1 -0
  16. package/dist/types/api/index.d.ts +3 -0
  17. package/dist/types/api/index.d.ts.map +1 -0
  18. package/dist/types/api/nodes/async/flowConstantAsync.d.ts +31 -0
  19. package/dist/types/api/nodes/async/flowConstantAsync.d.ts.map +1 -0
  20. package/dist/types/api/nodes/async/flowDerivationAsync.d.ts +37 -0
  21. package/dist/types/api/nodes/async/flowDerivationAsync.d.ts.map +1 -0
  22. package/dist/types/api/nodes/async/flowStateAsync.d.ts +41 -0
  23. package/dist/types/api/nodes/async/flowStateAsync.d.ts.map +1 -0
  24. package/dist/types/api/nodes/async/flowWritableDerivationAsync.d.ts +30 -0
  25. package/dist/types/api/nodes/async/flowWritableDerivationAsync.d.ts.map +1 -0
  26. package/dist/types/{flow → api}/nodes/async/index.d.ts +1 -2
  27. package/dist/types/api/nodes/async/index.d.ts.map +1 -0
  28. package/dist/types/api/nodes/collections/flowArray.d.ts +134 -0
  29. package/dist/types/api/nodes/collections/flowArray.d.ts.map +1 -0
  30. package/dist/types/api/nodes/collections/flowMap.d.ts +98 -0
  31. package/dist/types/api/nodes/collections/flowMap.d.ts.map +1 -0
  32. package/dist/types/api/nodes/collections/index.d.ts.map +1 -0
  33. package/dist/types/api/nodes/flowEffect.d.ts +28 -0
  34. package/dist/types/api/nodes/flowEffect.d.ts.map +1 -0
  35. package/dist/types/api/nodes/flowSignal.d.ts +25 -0
  36. package/dist/types/api/nodes/flowSignal.d.ts.map +1 -0
  37. package/dist/types/api/nodes/flowValue.d.ts +35 -0
  38. package/dist/types/api/nodes/flowValue.d.ts.map +1 -0
  39. package/dist/types/api/nodes/index.d.ts +8 -0
  40. package/dist/types/api/nodes/index.d.ts.map +1 -0
  41. package/dist/types/api/nodes/sync/flowConstant.d.ts +29 -0
  42. package/dist/types/api/nodes/sync/flowConstant.d.ts.map +1 -0
  43. package/dist/types/api/nodes/sync/flowDerivation.d.ts +36 -0
  44. package/dist/types/api/nodes/sync/flowDerivation.d.ts.map +1 -0
  45. package/dist/types/api/nodes/sync/flowState.d.ts +39 -0
  46. package/dist/types/api/nodes/sync/flowState.d.ts.map +1 -0
  47. package/dist/types/api/nodes/sync/flowWritableDerivation.d.ts +28 -0
  48. package/dist/types/api/nodes/sync/flowWritableDerivation.d.ts.map +1 -0
  49. package/dist/types/{flow → api}/nodes/sync/index.d.ts +1 -2
  50. package/dist/types/api/nodes/sync/index.d.ts.map +1 -0
  51. package/dist/types/api/nodes/utils.d.ts +22 -0
  52. package/dist/types/api/nodes/utils.d.ts.map +1 -0
  53. package/dist/types/base/disposable.d.ts +11 -0
  54. package/dist/types/base/disposable.d.ts.map +1 -0
  55. package/dist/types/base/executionStack.d.ts +14 -0
  56. package/dist/types/base/executionStack.d.ts.map +1 -0
  57. package/dist/types/base/index.d.ts +6 -0
  58. package/dist/types/base/index.d.ts.map +1 -0
  59. package/dist/types/base/node.d.ts +27 -0
  60. package/dist/types/base/node.d.ts.map +1 -0
  61. package/dist/types/base/observable.d.ts +37 -0
  62. package/dist/types/base/observable.d.ts.map +1 -0
  63. package/dist/types/base/observer.d.ts +25 -0
  64. package/dist/types/base/observer.d.ts.map +1 -0
  65. package/dist/types/converters/index.d.ts +2 -0
  66. package/dist/types/converters/index.d.ts.map +1 -0
  67. package/dist/types/converters/solid.d.ts +46 -0
  68. package/dist/types/converters/solid.d.ts.map +1 -0
  69. package/dist/types/index.d.ts +2 -63
  70. package/dist/types/index.d.ts.map +1 -1
  71. package/dist/types/nodes/arrayNode.d.ts +2 -0
  72. package/dist/types/nodes/arrayNode.d.ts.map +1 -0
  73. package/dist/types/nodes/effectNode.d.ts +2 -0
  74. package/dist/types/nodes/effectNode.d.ts.map +1 -0
  75. package/dist/types/nodes/index.d.ts +9 -0
  76. package/dist/types/nodes/index.d.ts.map +1 -0
  77. package/dist/types/nodes/mapNode.d.ts +2 -0
  78. package/dist/types/nodes/mapNode.d.ts.map +1 -0
  79. package/dist/types/nodes/signalNode.d.ts +2 -0
  80. package/dist/types/nodes/signalNode.d.ts.map +1 -0
  81. package/dist/types/nodes/valueAsyncNode.d.ts +2 -0
  82. package/dist/types/nodes/valueAsyncNode.d.ts.map +1 -0
  83. package/dist/types/nodes/valueNode.d.ts +2 -0
  84. package/dist/types/nodes/valueNode.d.ts.map +1 -0
  85. package/dist/types/nodes/valueSyncNode.d.ts +2 -0
  86. package/dist/types/nodes/valueSyncNode.d.ts.map +1 -0
  87. package/dist/types/schedulers/asyncResolver.d.ts +2 -0
  88. package/dist/types/schedulers/asyncResolver.d.ts.map +1 -0
  89. package/dist/types/schedulers/asyncScheduler.d.ts +2 -0
  90. package/dist/types/schedulers/asyncScheduler.d.ts.map +1 -0
  91. package/dist/types/schedulers/index.d.ts +5 -0
  92. package/dist/types/schedulers/index.d.ts.map +1 -0
  93. package/dist/types/schedulers/pendingError.d.ts +2 -0
  94. package/dist/types/schedulers/pendingError.d.ts.map +1 -0
  95. package/dist/types/schedulers/scheduler.d.ts +2 -0
  96. package/dist/types/schedulers/scheduler.d.ts.map +1 -0
  97. package/dist/types/schedulers/syncResolver.d.ts +2 -0
  98. package/dist/types/schedulers/syncResolver.d.ts.map +1 -0
  99. package/dist/types/schedulers/syncScheduler.d.ts +2 -0
  100. package/dist/types/schedulers/syncScheduler.d.ts.map +1 -0
  101. package/docs/.vitepress/config.mts +128 -93
  102. package/docs/api/functions/array.md +14 -37
  103. package/docs/api/functions/constant.md +13 -25
  104. package/docs/api/functions/constantAsync.md +69 -0
  105. package/docs/api/functions/derivation.md +14 -33
  106. package/docs/api/functions/derivationAsync.md +34 -0
  107. package/docs/api/functions/from.md +62 -153
  108. package/docs/api/functions/isDisposable.md +8 -30
  109. package/docs/api/functions/map.md +15 -36
  110. package/docs/api/functions/signal.md +8 -23
  111. package/docs/api/functions/state.md +43 -23
  112. package/docs/api/functions/stateAsync.md +69 -0
  113. package/docs/api/functions/subscribe.md +40 -0
  114. package/docs/api/functions/writableDerivation.md +33 -0
  115. package/docs/api/functions/writableDerivationAsync.md +34 -0
  116. package/docs/api/index.md +45 -102
  117. package/docs/api/interfaces/FlowArray.md +439 -0
  118. package/docs/api/interfaces/FlowConstant.md +220 -0
  119. package/docs/api/interfaces/FlowConstantAsync.md +221 -0
  120. package/docs/api/interfaces/FlowDerivation.md +241 -0
  121. package/docs/api/interfaces/FlowDerivationAsync.md +242 -0
  122. package/docs/api/interfaces/FlowDisposable.md +32 -38
  123. package/docs/api/interfaces/FlowEffect.md +64 -0
  124. package/docs/api/interfaces/FlowMap.md +374 -0
  125. package/docs/api/interfaces/FlowObservable.md +155 -0
  126. package/docs/api/interfaces/FlowSignal.md +156 -0
  127. package/docs/api/interfaces/FlowState.md +269 -0
  128. package/docs/api/interfaces/FlowStateAsync.md +268 -0
  129. package/docs/api/interfaces/FlowSubscribable.md +55 -0
  130. package/docs/api/interfaces/FlowTracker.md +61 -0
  131. package/docs/api/interfaces/FlowValue.md +222 -0
  132. package/docs/api/interfaces/FlowWritableDerivation.md +292 -0
  133. package/docs/api/interfaces/FlowWritableDerivationAsync.md +293 -0
  134. package/docs/api/type-aliases/DerivationFunction.md +28 -0
  135. package/docs/api/type-aliases/DerivationFunctionAsync.md +28 -0
  136. package/docs/api/type-aliases/FlowArrayAction.md +19 -8
  137. package/docs/api/type-aliases/FlowDataTracker.md +33 -0
  138. package/docs/api/type-aliases/FlowMapAction.md +48 -0
  139. package/docs/api/type-aliases/FlowOnDataListener.md +33 -0
  140. package/docs/api/type-aliases/FlowOnErrorListener.md +27 -0
  141. package/docs/api/type-aliases/FlowOnPendingListener.md +21 -0
  142. package/docs/api/type-aliases/FlowReadonly.md +22 -0
  143. package/docs/api/type-aliases/InitFunction.md +21 -0
  144. package/docs/api/type-aliases/InitFunctionAsync.md +21 -0
  145. package/docs/api/type-aliases/NotPromise.md +6 -3
  146. package/docs/api/type-aliases/UpdateFunction.md +27 -0
  147. package/docs/api/type-aliases/UpdateFunctionAsync.md +27 -0
  148. package/docs/api/typedoc-sidebar.json +1 -81
  149. package/docs/examples/examples.md +0 -2
  150. package/docs/guide/advanced/architecture.md +1234 -0
  151. package/docs/guide/advanced/migration-v2.md +204 -0
  152. package/docs/guide/advanced/solidjs.md +2 -88
  153. package/docs/guide/introduction/concepts.md +4 -3
  154. package/docs/guide/introduction/conventions.md +2 -33
  155. package/docs/guide/introduction/getting-started.md +28 -23
  156. package/docs/guide/introduction/lifecycle.md +16 -19
  157. package/docs/guide/primitives/array.md +102 -216
  158. package/docs/guide/primitives/constant.md +39 -212
  159. package/docs/guide/primitives/derivations.md +55 -122
  160. package/docs/guide/primitives/effects.md +155 -241
  161. package/docs/guide/primitives/map.md +64 -186
  162. package/docs/guide/primitives/overview.md +45 -128
  163. package/docs/guide/primitives/signal.md +51 -88
  164. package/docs/guide/primitives/state.md +34 -130
  165. package/package.json +56 -60
  166. package/src/api/base/flowDisposable.ts +44 -0
  167. package/src/api/base/flowObservable.ts +28 -0
  168. package/src/api/base/flowSubscribable.ts +87 -0
  169. package/src/api/base/flowTracker.ts +7 -0
  170. package/src/api/base/index.ts +4 -0
  171. package/src/{flow → api}/index.ts +0 -1
  172. package/src/api/nodes/async/flowConstantAsync.ts +36 -0
  173. package/src/api/nodes/async/flowDerivationAsync.ts +42 -0
  174. package/src/api/nodes/async/flowStateAsync.ts +47 -0
  175. package/src/api/nodes/async/flowWritableDerivationAsync.ts +33 -0
  176. package/src/{flow → api}/nodes/async/index.ts +1 -2
  177. package/src/api/nodes/collections/flowArray.ts +155 -0
  178. package/src/api/nodes/collections/flowMap.ts +115 -0
  179. package/src/api/nodes/flowEffect.ts +42 -0
  180. package/src/api/nodes/flowSignal.ts +28 -0
  181. package/src/api/nodes/flowValue.ts +36 -0
  182. package/src/api/nodes/index.ts +7 -0
  183. package/src/api/nodes/sync/flowConstant.ts +33 -0
  184. package/src/api/nodes/sync/flowDerivation.ts +41 -0
  185. package/src/api/nodes/sync/flowState.ts +45 -0
  186. package/src/api/nodes/sync/flowWritableDerivation.ts +31 -0
  187. package/src/{flow → api}/nodes/sync/index.ts +1 -2
  188. package/src/api/nodes/utils.ts +22 -0
  189. package/src/base/disposable.ts +18 -0
  190. package/src/base/executionStack.ts +42 -0
  191. package/src/base/index.ts +5 -0
  192. package/src/base/node.ts +98 -0
  193. package/src/base/observable.ts +87 -0
  194. package/src/base/observer.ts +51 -0
  195. package/src/converters/index.ts +1 -0
  196. package/src/converters/solid.ts +109 -0
  197. package/src/index.ts +2 -64
  198. package/src/nodes/arrayNode.ts +172 -0
  199. package/src/nodes/effectNode.ts +59 -0
  200. package/src/nodes/index.ts +8 -0
  201. package/src/nodes/mapNode.ts +127 -0
  202. package/src/nodes/signalNode.ts +21 -0
  203. package/src/nodes/valueAsyncNode.ts +88 -0
  204. package/src/nodes/valueNode.ts +144 -0
  205. package/src/nodes/valueSyncNode.ts +128 -0
  206. package/src/schedulers/asyncResolver.ts +78 -0
  207. package/src/schedulers/asyncScheduler.ts +66 -0
  208. package/src/schedulers/index.ts +4 -0
  209. package/src/schedulers/pendingError.ts +13 -0
  210. package/src/schedulers/scheduler.ts +9 -0
  211. package/src/schedulers/syncResolver.ts +69 -0
  212. package/src/schedulers/syncScheduler.ts +55 -0
  213. package/test/base/pendingError.test.ts +67 -0
  214. package/test/converters/solid.derivation.browser.test.tsx +69 -0
  215. package/test/converters/solid.node.test.ts +654 -0
  216. package/test/converters/solid.state.browser.test.tsx +1592 -0
  217. package/test/reactivity/flowSignal.test.ts +226 -0
  218. package/test/reactivity/nodes/async/asyncScheduler/asyncResolver.test.ts +593 -0
  219. package/test/reactivity/nodes/async/asyncScheduler/asyncScheduler.test.ts +317 -0
  220. package/test/reactivity/nodes/async/flowConstantAsync.test.ts +652 -0
  221. package/test/reactivity/nodes/async/flowDerivation.test.ts +898 -0
  222. package/test/reactivity/nodes/async/flowDerivationAsync.test.ts +1716 -0
  223. package/test/reactivity/nodes/async/flowStateAsync.test.ts +708 -0
  224. package/test/reactivity/nodes/async/flowWritableDerivationAsync.test.ts +614 -0
  225. package/test/reactivity/nodes/collections/flowArray.asyncStates.test.ts +1289 -0
  226. package/test/reactivity/nodes/collections/flowArray.scalars.test.ts +961 -0
  227. package/test/reactivity/nodes/collections/flowArray.states.test.ts +1035 -0
  228. package/test/reactivity/nodes/collections/flowMap.asyncStates.test.ts +960 -0
  229. package/test/reactivity/nodes/collections/flowMap.scalars.test.ts +775 -0
  230. package/test/reactivity/nodes/collections/flowMap.states.test.ts +958 -0
  231. package/test/reactivity/nodes/sync/flowConstant.test.ts +377 -0
  232. package/test/reactivity/nodes/sync/flowDerivation.test.ts +896 -0
  233. package/test/reactivity/nodes/sync/flowState.test.ts +341 -0
  234. package/test/reactivity/nodes/sync/flowWritableDerivation.test.ts +603 -0
  235. package/test/vitest.d.ts +10 -0
  236. package/tsconfig.json +31 -20
  237. package/typedoc.json +35 -35
  238. package/vite.config.ts +25 -23
  239. package/vitest.browser.config.ts +21 -0
  240. package/vitest.config.ts +12 -12
  241. package/.cursor/plans/unifier-flowresource-avec-flowderivation-c9506e24.plan.md +0 -372
  242. package/.cursor/plans/update-js-e795d61b.plan.md +0 -567
  243. package/dist/types/flow/base/flowDisposable.d.ts +0 -67
  244. package/dist/types/flow/base/flowDisposable.d.ts.map +0 -1
  245. package/dist/types/flow/base/flowEffect.d.ts +0 -127
  246. package/dist/types/flow/base/flowEffect.d.ts.map +0 -1
  247. package/dist/types/flow/base/flowGraph.d.ts +0 -97
  248. package/dist/types/flow/base/flowGraph.d.ts.map +0 -1
  249. package/dist/types/flow/base/flowSignal.d.ts +0 -134
  250. package/dist/types/flow/base/flowSignal.d.ts.map +0 -1
  251. package/dist/types/flow/base/flowTracker.d.ts +0 -15
  252. package/dist/types/flow/base/flowTracker.d.ts.map +0 -1
  253. package/dist/types/flow/base/index.d.ts +0 -7
  254. package/dist/types/flow/base/index.d.ts.map +0 -1
  255. package/dist/types/flow/base/utils.d.ts +0 -20
  256. package/dist/types/flow/base/utils.d.ts.map +0 -1
  257. package/dist/types/flow/collections/flowArray.d.ts +0 -148
  258. package/dist/types/flow/collections/flowArray.d.ts.map +0 -1
  259. package/dist/types/flow/collections/flowMap.d.ts +0 -224
  260. package/dist/types/flow/collections/flowMap.d.ts.map +0 -1
  261. package/dist/types/flow/collections/index.d.ts.map +0 -1
  262. package/dist/types/flow/index.d.ts +0 -4
  263. package/dist/types/flow/index.d.ts.map +0 -1
  264. package/dist/types/flow/nodes/async/flowConstantAsync.d.ts +0 -137
  265. package/dist/types/flow/nodes/async/flowConstantAsync.d.ts.map +0 -1
  266. package/dist/types/flow/nodes/async/flowDerivationAsync.d.ts +0 -137
  267. package/dist/types/flow/nodes/async/flowDerivationAsync.d.ts.map +0 -1
  268. package/dist/types/flow/nodes/async/flowNodeAsync.d.ts +0 -343
  269. package/dist/types/flow/nodes/async/flowNodeAsync.d.ts.map +0 -1
  270. package/dist/types/flow/nodes/async/flowReadonlyAsync.d.ts +0 -81
  271. package/dist/types/flow/nodes/async/flowReadonlyAsync.d.ts.map +0 -1
  272. package/dist/types/flow/nodes/async/flowStateAsync.d.ts +0 -111
  273. package/dist/types/flow/nodes/async/flowStateAsync.d.ts.map +0 -1
  274. package/dist/types/flow/nodes/async/index.d.ts.map +0 -1
  275. package/dist/types/flow/nodes/index.d.ts +0 -3
  276. package/dist/types/flow/nodes/index.d.ts.map +0 -1
  277. package/dist/types/flow/nodes/sync/flowConstant.d.ts +0 -108
  278. package/dist/types/flow/nodes/sync/flowConstant.d.ts.map +0 -1
  279. package/dist/types/flow/nodes/sync/flowDerivation.d.ts +0 -100
  280. package/dist/types/flow/nodes/sync/flowDerivation.d.ts.map +0 -1
  281. package/dist/types/flow/nodes/sync/flowNode.d.ts +0 -314
  282. package/dist/types/flow/nodes/sync/flowNode.d.ts.map +0 -1
  283. package/dist/types/flow/nodes/sync/flowReadonly.d.ts +0 -57
  284. package/dist/types/flow/nodes/sync/flowReadonly.d.ts.map +0 -1
  285. package/dist/types/flow/nodes/sync/flowState.d.ts +0 -96
  286. package/dist/types/flow/nodes/sync/flowState.d.ts.map +0 -1
  287. package/dist/types/flow/nodes/sync/index.d.ts.map +0 -1
  288. package/dist/types/solid/converters.d.ts +0 -53
  289. package/dist/types/solid/converters.d.ts.map +0 -1
  290. package/dist/types/solid/index.d.ts +0 -3
  291. package/dist/types/solid/index.d.ts.map +0 -1
  292. package/dist/types/solid/primitives.d.ts +0 -181
  293. package/dist/types/solid/primitives.d.ts.map +0 -1
  294. package/docs/api/classes/FlowArray.md +0 -489
  295. package/docs/api/classes/FlowConstant.md +0 -350
  296. package/docs/api/classes/FlowDerivation.md +0 -334
  297. package/docs/api/classes/FlowEffect.md +0 -100
  298. package/docs/api/classes/FlowMap.md +0 -512
  299. package/docs/api/classes/FlowObservable.md +0 -306
  300. package/docs/api/classes/FlowResource.md +0 -380
  301. package/docs/api/classes/FlowResourceAsync.md +0 -362
  302. package/docs/api/classes/FlowSignal.md +0 -160
  303. package/docs/api/classes/FlowState.md +0 -368
  304. package/docs/api/classes/FlowStream.md +0 -367
  305. package/docs/api/classes/FlowStreamAsync.md +0 -364
  306. package/docs/api/classes/SolidDerivation.md +0 -75
  307. package/docs/api/classes/SolidResource.md +0 -91
  308. package/docs/api/classes/SolidState.md +0 -71
  309. package/docs/api/classes/TrackingContext.md +0 -33
  310. package/docs/api/functions/effect.md +0 -49
  311. package/docs/api/functions/resource.md +0 -52
  312. package/docs/api/functions/resourceAsync.md +0 -50
  313. package/docs/api/functions/stream.md +0 -53
  314. package/docs/api/functions/streamAsync.md +0 -50
  315. package/docs/api/interfaces/SolidObservable.md +0 -19
  316. package/docs/api/type-aliases/FlowStreamDisposer.md +0 -15
  317. package/docs/api/type-aliases/FlowStreamSetter.md +0 -27
  318. package/docs/api/type-aliases/FlowStreamUpdater.md +0 -32
  319. package/docs/api/type-aliases/SolidGetter.md +0 -17
  320. package/docs/guide/primitives/resources.md +0 -858
  321. package/docs/guide/primitives/streams.md +0 -931
  322. package/src/flow/base/flowDisposable.ts +0 -71
  323. package/src/flow/base/flowEffect.ts +0 -171
  324. package/src/flow/base/flowGraph.ts +0 -288
  325. package/src/flow/base/flowSignal.ts +0 -207
  326. package/src/flow/base/flowTracker.ts +0 -17
  327. package/src/flow/base/index.ts +0 -6
  328. package/src/flow/base/utils.ts +0 -19
  329. package/src/flow/collections/flowArray.ts +0 -409
  330. package/src/flow/collections/flowMap.ts +0 -398
  331. package/src/flow/nodes/async/flowConstantAsync.ts +0 -142
  332. package/src/flow/nodes/async/flowDerivationAsync.ts +0 -143
  333. package/src/flow/nodes/async/flowNodeAsync.ts +0 -474
  334. package/src/flow/nodes/async/flowReadonlyAsync.ts +0 -81
  335. package/src/flow/nodes/async/flowStateAsync.ts +0 -116
  336. package/src/flow/nodes/await/advanced/index.ts +0 -5
  337. package/src/flow/nodes/await/advanced/resource.ts +0 -134
  338. package/src/flow/nodes/await/advanced/resourceAsync.ts +0 -109
  339. package/src/flow/nodes/await/advanced/stream.ts +0 -188
  340. package/src/flow/nodes/await/advanced/streamAsync.ts +0 -176
  341. package/src/flow/nodes/await/flowConstantAwait.ts +0 -154
  342. package/src/flow/nodes/await/flowDerivationAwait.ts +0 -154
  343. package/src/flow/nodes/await/flowNodeAwait.ts +0 -508
  344. package/src/flow/nodes/await/flowReadonlyAwait.ts +0 -89
  345. package/src/flow/nodes/await/flowStateAwait.ts +0 -130
  346. package/src/flow/nodes/await/index.ts +0 -5
  347. package/src/flow/nodes/index.ts +0 -3
  348. package/src/flow/nodes/sync/flowConstant.ts +0 -111
  349. package/src/flow/nodes/sync/flowDerivation.ts +0 -105
  350. package/src/flow/nodes/sync/flowNode.ts +0 -439
  351. package/src/flow/nodes/sync/flowReadonly.ts +0 -57
  352. package/src/flow/nodes/sync/flowState.ts +0 -101
  353. package/src/solid/converters.ts +0 -139
  354. package/src/solid/index.ts +0 -2
  355. package/src/solid/primitives.ts +0 -215
  356. package/test/base/flowEffect.test.ts +0 -108
  357. package/test/base/flowGraph.test.ts +0 -485
  358. package/test/base/flowSignal.test.ts +0 -372
  359. package/test/collections/flowArray.asyncStates.test.ts +0 -1553
  360. package/test/collections/flowArray.scalars.test.ts +0 -1129
  361. package/test/collections/flowArray.states.test.ts +0 -1365
  362. package/test/collections/flowMap.asyncStates.test.ts +0 -1105
  363. package/test/collections/flowMap.scalars.test.ts +0 -877
  364. package/test/collections/flowMap.states.test.ts +0 -1097
  365. package/test/nodes/async/flowConstantAsync.test.ts +0 -860
  366. package/test/nodes/async/flowDerivationAsync.test.ts +0 -1517
  367. package/test/nodes/async/flowStateAsync.test.ts +0 -1387
  368. package/test/nodes/await/advanced/resource.test.ts +0 -129
  369. package/test/nodes/await/advanced/resourceAsync.test.ts +0 -108
  370. package/test/nodes/await/advanced/stream.test.ts +0 -198
  371. package/test/nodes/await/advanced/streamAsync.test.ts +0 -196
  372. package/test/nodes/await/flowConstantAwait.test.ts +0 -643
  373. package/test/nodes/await/flowDerivationAwait.test.ts +0 -1583
  374. package/test/nodes/await/flowStateAwait.test.ts +0 -999
  375. package/test/nodes/mixed/derivation.test.ts +0 -1527
  376. package/test/nodes/sync/flowConstant.test.ts +0 -620
  377. package/test/nodes/sync/flowDerivation.test.ts +0 -1373
  378. package/test/nodes/sync/flowState.test.ts +0 -945
  379. package/test/solid/converters.test.ts +0 -721
  380. package/test/solid/primitives.test.ts +0 -1031
  381. /package/dist/types/{flow → api/nodes}/collections/index.d.ts +0 -0
  382. /package/docs/guide/advanced/{upgrading.md → migration-v1.md} +0 -0
  383. /package/src/{flow → api/nodes}/collections/index.ts +0 -0
@@ -1,999 +0,0 @@
1
- import { describe, expect, it, vi } from "vitest";
2
- import { effect, state, stateAwait } from "#package";
3
-
4
- describe("flowStateAwait", () => {
5
- describe("unit", () => {
6
- describe("initialization", () => {
7
- it("should initialize with direct Promise", async () => {
8
- const $state = stateAwait(Promise.resolve(1));
9
- expect(await $state.pick()).toBe(1);
10
- });
11
-
12
- it("should initialize lazily with function", async () => {
13
- const initFn = vi.fn(() => Promise.resolve(42));
14
- const $state = stateAwait(initFn);
15
-
16
- // Function should not be called yet
17
- expect(initFn).not.toHaveBeenCalled();
18
-
19
- // First access triggers initialization
20
- const value = await $state.pick();
21
- expect(value).toBe(42);
22
- expect(initFn).toHaveBeenCalledTimes(1);
23
- });
24
-
25
- it("should not call lazy function on creation", () => {
26
- const initFn = vi.fn(() => Promise.resolve(100));
27
- stateAwait(initFn);
28
-
29
- expect(initFn).not.toHaveBeenCalled();
30
- });
31
-
32
- it("should call lazy function on first access", async () => {
33
- const initFn = vi.fn(() => Promise.resolve(200));
34
- const $state = stateAwait(initFn);
35
-
36
- expect(initFn).not.toHaveBeenCalled();
37
- await $state.pick();
38
- expect(initFn).toHaveBeenCalledTimes(1);
39
- });
40
-
41
- it("should call lazy function only once", async () => {
42
- const initFn = vi.fn(() => Promise.resolve(300));
43
- const $state = stateAwait(initFn);
44
-
45
- // Multiple accesses should only call init once
46
- await $state.pick();
47
- await $state.pick();
48
- await $state.pick();
49
-
50
- expect(initFn).toHaveBeenCalledTimes(1);
51
- });
52
-
53
- it("should cache value after lazy initialization", async () => {
54
- const initFn = vi.fn(() => Promise.resolve("cached"));
55
- const $state = stateAwait(initFn);
56
-
57
- const value1 = await $state.pick();
58
- const value2 = await $state.pick();
59
- const value3 = await $state.pick();
60
-
61
- expect(value1).toBe("cached");
62
- expect(value2).toBe("cached");
63
- expect(value3).toBe("cached");
64
- expect(initFn).toHaveBeenCalledTimes(1);
65
- });
66
-
67
- it("should handle number values", async () => {
68
- const $state = stateAwait(Promise.resolve(42));
69
- expect(await $state.pick()).toBe(42);
70
- });
71
-
72
- it("should handle string values", async () => {
73
- const $state = stateAwait(Promise.resolve("hello"));
74
- expect(await $state.pick()).toBe("hello");
75
- });
76
-
77
- it("should handle object values", async () => {
78
- const obj = { a: 1, b: 2 };
79
- const $state = stateAwait(Promise.resolve(obj));
80
- const value = await $state.pick();
81
- expect(value).toBe(obj);
82
- expect(value).toEqual({ a: 1, b: 2 });
83
- });
84
-
85
- it("should handle array values", async () => {
86
- const arr = [1, 2, 3];
87
- const $state = stateAwait(Promise.resolve(arr));
88
- const value = await $state.pick();
89
- expect(value).toBe(arr);
90
- expect(value).toEqual([1, 2, 3]);
91
- });
92
-
93
- it("should handle null values", async () => {
94
- const $state = stateAwait(Promise.resolve(null));
95
- const value = await $state.pick();
96
- expect(value).toBeNull();
97
- });
98
-
99
- it("should handle undefined values", async () => {
100
- const $state = stateAwait(Promise.resolve(undefined));
101
- const value = await $state.pick();
102
- expect(value).toBeUndefined();
103
- });
104
- });
105
-
106
- describe("get", () => {
107
- it("should return resolved value with tracking context", async () => {
108
- const $state = stateAwait(Promise.resolve(5));
109
- const $tracker = state(0); // Use state as FlowTracker
110
-
111
- // Wait for Promise to resolve first
112
- await $state.pick();
113
-
114
- const value = $state.get($tracker);
115
- expect(value).toBe(5);
116
- });
117
-
118
- it("should return undefined if Promise not yet resolved", () => {
119
- // Create a Promise that resolves after a delay
120
- const $state = stateAwait(
121
- new Promise<number>((resolve) => setTimeout(() => resolve(10), 100)),
122
- );
123
- const $tracker = state(0);
124
-
125
- // Before Promise resolves, get() should return undefined
126
- const value = $state.get($tracker);
127
- expect(value).toBeUndefined();
128
- });
129
-
130
- it("should compute lazy value on first get call", async () => {
131
- const initFn = vi.fn(() => Promise.resolve(10));
132
- const $state = stateAwait(initFn);
133
- const $tracker = state(0);
134
-
135
- expect(initFn).not.toHaveBeenCalled();
136
-
137
- // First get() may return undefined, but triggers computation
138
- $state.get($tracker);
139
- // Wait for Promise to resolve
140
- await $state.pick();
141
-
142
- const value2 = $state.get($tracker);
143
- expect(value2).toBe(10);
144
- expect(initFn).toHaveBeenCalledTimes(1);
145
- });
146
-
147
- it("should return cached value on subsequent get calls", async () => {
148
- const initFn = vi.fn(() => Promise.resolve(20));
149
- const $state = stateAwait(initFn);
150
- const $tracker = state(0);
151
-
152
- // Wait for initial resolution
153
- await $state.pick();
154
-
155
- const value1 = $state.get($tracker);
156
- const value2 = $state.get($tracker);
157
- const value3 = $state.get($tracker);
158
-
159
- expect(value1).toBe(20);
160
- expect(value2).toBe(20);
161
- expect(value3).toBe(20);
162
- expect(initFn).toHaveBeenCalledTimes(1);
163
- });
164
-
165
- it("should throw error when get is called after disposal", async () => {
166
- const $state = stateAwait(Promise.resolve(1));
167
- const $tracker = state(0);
168
-
169
- await $state.pick();
170
- $state.get($tracker);
171
- $state.dispose();
172
-
173
- expect(() => $state.get($tracker)).toThrow(
174
- "[PicoFlow] Primitive is disposed",
175
- );
176
- });
177
- });
178
-
179
- describe("pick", () => {
180
- it("should return Promise without tracking", async () => {
181
- const $state = stateAwait(Promise.resolve(15));
182
- const promise = $state.pick();
183
-
184
- expect(promise).toBeInstanceOf(Promise);
185
- const value = await promise;
186
- expect(value).toBe(15);
187
- });
188
-
189
- it("should compute lazy value on first pick call", async () => {
190
- const initFn = vi.fn(() => Promise.resolve(25));
191
- const $state = stateAwait(initFn);
192
-
193
- expect(initFn).not.toHaveBeenCalled();
194
- const promise = $state.pick();
195
- expect(promise).toBeInstanceOf(Promise);
196
- const value = await promise;
197
- expect(value).toBe(25);
198
- expect(initFn).toHaveBeenCalledTimes(1);
199
- });
200
-
201
- it("should return cached Promise on subsequent pick calls", async () => {
202
- const initFn = vi.fn(() => Promise.resolve(30));
203
- const $state = stateAwait(initFn);
204
-
205
- const promise1 = $state.pick();
206
- const promise2 = $state.pick();
207
- const promise3 = $state.pick();
208
-
209
- const value1 = await promise1;
210
- const value2 = await promise2;
211
- const value3 = await promise3;
212
-
213
- expect(value1).toBe(30);
214
- expect(value2).toBe(30);
215
- expect(value3).toBe(30);
216
- expect(initFn).toHaveBeenCalledTimes(1);
217
- });
218
-
219
- it("should await Promise to get value", async () => {
220
- const $state = stateAwait(Promise.resolve(35));
221
- const value = await $state.pick();
222
- expect(value).toBe(35);
223
- });
224
-
225
- it("should throw error when pick is called after disposal", async () => {
226
- const $state = stateAwait(Promise.resolve(1));
227
-
228
- expect(await $state.pick()).toBe(1);
229
-
230
- $state.dispose();
231
-
232
- await expect($state.pick()).rejects.toThrow(
233
- "[PicoFlow] Primitive is disposed",
234
- );
235
- });
236
- });
237
-
238
- describe("set", () => {
239
- it("should update value with Promise", async () => {
240
- const $state = stateAwait(Promise.resolve(1));
241
- expect(await $state.pick()).toBe(1);
242
-
243
- await $state.set(Promise.resolve(2));
244
- expect(await $state.pick()).toBe(2);
245
- });
246
-
247
- it("should update value with updater function", async () => {
248
- const $state = stateAwait(Promise.resolve(1));
249
- expect(await $state.pick()).toBe(1);
250
-
251
- await $state.set(async (current) => Promise.resolve(current + 1));
252
- expect(await $state.pick()).toBe(2);
253
- });
254
-
255
- it("should return Promise<void>", async () => {
256
- const $state = stateAwait(Promise.resolve(1));
257
- const result = await $state.set(Promise.resolve(2));
258
- expect(result).toBeUndefined();
259
- });
260
-
261
- it("should not notify when set with same value", async () => {
262
- const $state = stateAwait(Promise.resolve(1));
263
- const listener = vi.fn();
264
- $state.subscribe(listener);
265
-
266
- await vi.waitFor(() => expect(listener).toHaveBeenCalledTimes(1));
267
-
268
- await $state.set(Promise.resolve(1));
269
-
270
- // Wait to ensure no additional calls
271
- await new Promise((resolve) => setTimeout(resolve, 50));
272
- expect(listener).toHaveBeenCalledTimes(1);
273
- });
274
-
275
- it("should notify when set with different value", async () => {
276
- const $state = stateAwait(Promise.resolve(1));
277
- const listener = vi.fn();
278
- $state.subscribe(listener);
279
-
280
- await vi.waitFor(() => expect(listener).toHaveBeenCalledTimes(1));
281
-
282
- await $state.set(Promise.resolve(2));
283
- await vi.waitFor(() => expect(listener).toHaveBeenCalledTimes(2));
284
- expect(listener).toHaveBeenLastCalledWith(2);
285
- });
286
-
287
- it("should not notify when updater returns same value", async () => {
288
- const $state = stateAwait(Promise.resolve(5));
289
- const listener = vi.fn();
290
- $state.subscribe(listener);
291
-
292
- await vi.waitFor(() => expect(listener).toHaveBeenCalledTimes(1));
293
-
294
- await $state.set(async (current) => Promise.resolve(current));
295
-
296
- // Wait to ensure no additional calls
297
- await new Promise((resolve) => setTimeout(resolve, 50));
298
- expect(listener).toHaveBeenCalledTimes(1);
299
- });
300
-
301
- it("should notify when updater returns different value", async () => {
302
- const $state = stateAwait(Promise.resolve(5));
303
- const listener = vi.fn();
304
- $state.subscribe(listener);
305
-
306
- await vi.waitFor(() => expect(listener).toHaveBeenCalledTimes(1));
307
-
308
- await $state.set(async (current) => Promise.resolve(current + 1));
309
- await vi.waitFor(() => expect(listener).toHaveBeenCalledTimes(2));
310
- expect(listener).toHaveBeenLastCalledWith(6);
311
- });
312
-
313
- it("should handle async updater function", async () => {
314
- const $state = stateAwait(Promise.resolve(1));
315
-
316
- await $state.set(async (current) => {
317
- await new Promise((resolve) => setTimeout(resolve, 10));
318
- return Promise.resolve(current + 10);
319
- });
320
-
321
- expect(await $state.pick()).toBe(11);
322
- });
323
-
324
- it("should throw error when set is called after disposal", async () => {
325
- const $state = stateAwait(Promise.resolve(1));
326
-
327
- expect(await $state.pick()).toBe(1);
328
-
329
- await $state.set(Promise.resolve(2));
330
- expect(await $state.pick()).toBe(2);
331
-
332
- $state.dispose();
333
-
334
- await expect($state.set(Promise.resolve(3))).rejects.toThrow(
335
- "[PicoFlow] Primitive is disposed",
336
- );
337
- });
338
-
339
- // TODO: Fix this test
340
- it.skip("should handle Promise rejection in set", async () => {
341
- const $state = stateAwait(Promise.resolve(1));
342
-
343
- await expect(
344
- $state.set(Promise.reject(new Error("Set error"))),
345
- ).rejects.toThrow("Set error");
346
- });
347
- });
348
-
349
- describe("watch", () => {
350
- it("should register dependency when watch is called", () => {
351
- const $state = stateAwait(Promise.resolve(35));
352
- const $tracker = state(0);
353
-
354
- expect(() => $state.watch($tracker)).not.toThrow();
355
- });
356
-
357
- it("should compute lazy value when watch is called", async () => {
358
- const initFn = vi.fn(() => Promise.resolve(40));
359
- const $state = stateAwait(initFn);
360
- const $tracker = state(0);
361
-
362
- expect(initFn).not.toHaveBeenCalled();
363
- $state.watch($tracker);
364
- // Wait a bit for the computation to complete
365
- await new Promise((resolve) => setTimeout(resolve, 10));
366
- expect(initFn).toHaveBeenCalledTimes(1);
367
- });
368
-
369
- it("should throw error when watch is called after disposal", () => {
370
- const $state = stateAwait(Promise.resolve(1));
371
- const $tracker = state(0);
372
-
373
- $state.watch($tracker);
374
- $state.dispose();
375
-
376
- expect(() => $state.watch($tracker)).toThrow(
377
- "[PicoFlow] Primitive is disposed",
378
- );
379
- });
380
- });
381
-
382
- describe("subscribe", () => {
383
- it("should return a disposer function", () => {
384
- const $state = stateAwait(Promise.resolve(50));
385
- const listener = vi.fn();
386
- const disposer = $state.subscribe(listener);
387
-
388
- expect(typeof disposer).toBe("function");
389
- });
390
-
391
- it("should call listener immediately with resolved value (T | undefined)", async () => {
392
- const $state = stateAwait(Promise.resolve(55));
393
- const listener = vi.fn();
394
-
395
- $state.subscribe(listener);
396
-
397
- await vi.waitFor(() => expect(listener).toHaveBeenCalledTimes(1));
398
- expect(listener).toHaveBeenLastCalledWith(55);
399
- });
400
-
401
- it("should call listener with lazy init value", async () => {
402
- const $state = stateAwait(() => Promise.resolve(60));
403
- const listener = vi.fn();
404
-
405
- $state.subscribe(listener);
406
-
407
- await vi.waitFor(() => expect(listener).toHaveBeenCalledTimes(1));
408
- expect(listener).toHaveBeenLastCalledWith(60);
409
- });
410
-
411
- it("should call listener when value is updated", async () => {
412
- const $state = stateAwait(Promise.resolve(65));
413
- const listener = vi.fn();
414
-
415
- $state.subscribe(listener);
416
- await vi.waitFor(() => expect(listener).toHaveBeenCalledTimes(1));
417
-
418
- await $state.set(Promise.resolve(70));
419
- await vi.waitFor(() => expect(listener).toHaveBeenCalledTimes(2));
420
- expect(listener).toHaveBeenLastCalledWith(70);
421
- });
422
-
423
- it("should not call listener when updated with same value", async () => {
424
- const $state = stateAwait(Promise.resolve(75));
425
- const listener = vi.fn();
426
-
427
- $state.subscribe(listener);
428
- await vi.waitFor(() => expect(listener).toHaveBeenCalledTimes(1));
429
-
430
- await $state.set(Promise.resolve(80));
431
- await vi.waitFor(() => expect(listener).toHaveBeenCalledTimes(2));
432
-
433
- await $state.set(Promise.resolve(80));
434
- await new Promise((resolve) => setTimeout(resolve, 50));
435
- expect(listener).toHaveBeenCalledTimes(2);
436
- });
437
-
438
- it("should not call listener after disposer is called", async () => {
439
- const $state = stateAwait(Promise.resolve(85));
440
- const listener = vi.fn();
441
-
442
- const unsubscribe = $state.subscribe(listener);
443
- await vi.waitFor(() => expect(listener).toHaveBeenCalledTimes(1));
444
-
445
- unsubscribe();
446
-
447
- await $state.set(Promise.resolve(90));
448
- await new Promise((resolve) => setTimeout(resolve, 50));
449
- expect(listener).toHaveBeenCalledTimes(1);
450
- });
451
-
452
- it("should support multiple subscriptions", async () => {
453
- const $state = stateAwait(Promise.resolve(95));
454
- const listener1 = vi.fn();
455
- const listener2 = vi.fn();
456
- const listener3 = vi.fn();
457
-
458
- $state.subscribe(listener1);
459
- $state.subscribe(listener2);
460
- $state.subscribe(listener3);
461
-
462
- await vi.waitFor(() => {
463
- expect(listener1).toHaveBeenCalledTimes(1);
464
- expect(listener2).toHaveBeenCalledTimes(1);
465
- expect(listener3).toHaveBeenCalledTimes(1);
466
- });
467
-
468
- expect(listener1).toHaveBeenLastCalledWith(95);
469
- expect(listener2).toHaveBeenLastCalledWith(95);
470
- expect(listener3).toHaveBeenLastCalledWith(95);
471
- });
472
-
473
- it("should throw error when subscribe is called after disposal", () => {
474
- const $state = stateAwait(Promise.resolve(1));
475
- const listener = vi.fn();
476
-
477
- $state.dispose();
478
-
479
- expect(() => $state.subscribe(listener)).toThrow(
480
- "[PicoFlow] Primitive is disposed",
481
- );
482
- });
483
- });
484
-
485
- describe("trigger", () => {
486
- it("should return a Promise", () => {
487
- const $state = stateAwait(Promise.resolve(1));
488
- const result = $state.trigger();
489
- expect(result).toBeInstanceOf(Promise);
490
- });
491
-
492
- it("should throw error when trigger is called after disposal", async () => {
493
- const $state = stateAwait(Promise.resolve(1));
494
- $state.dispose();
495
- await expect($state.trigger()).rejects.toThrow(
496
- "[PicoFlow] Primitive is disposed",
497
- );
498
- });
499
-
500
- it("should allow multiple triggers", async () => {
501
- const $state = stateAwait(Promise.resolve(1));
502
- const promise1 = $state.trigger();
503
- const promise2 = $state.trigger();
504
- const promise3 = $state.trigger();
505
-
506
- expect(promise1).toBeInstanceOf(Promise);
507
- expect(promise2).toBeInstanceOf(Promise);
508
- expect(promise3).toBeInstanceOf(Promise);
509
-
510
- await Promise.all([promise1, promise2, promise3]);
511
- });
512
- });
513
-
514
- describe("disposal", () => {
515
- it("should have disposed property set to false initially", () => {
516
- const $state = stateAwait(Promise.resolve(1));
517
- expect($state.disposed).toBe(false);
518
- });
519
-
520
- it("should have disposed property set to true after disposal", () => {
521
- const $state = stateAwait(Promise.resolve(1));
522
- $state.dispose();
523
- expect($state.disposed).toBe(true);
524
- });
525
-
526
- it("should throw error when disposed twice", () => {
527
- const $state = stateAwait(Promise.resolve(1));
528
- $state.dispose();
529
- expect(() => $state.dispose()).toThrow(
530
- "[PicoFlow] Primitive is disposed",
531
- );
532
- });
533
-
534
- it("should accept self option without throwing", () => {
535
- const $state = stateAwait(Promise.resolve(1));
536
- expect(() => $state.dispose({ self: true })).not.toThrow();
537
- expect($state.disposed).toBe(true);
538
- });
539
- });
540
-
541
- describe("special cases", () => {
542
- it("should handle lazy initialization with function returning different values", async () => {
543
- let counter = 0;
544
- const $state = stateAwait(() => {
545
- counter++;
546
- return Promise.resolve(counter);
547
- });
548
-
549
- // First access should return 1
550
- await $state.pick();
551
- expect(await $state.pick()).toBe(1);
552
-
553
- // Subsequent accesses should return cached value (1), not 2
554
- expect(await $state.pick()).toBe(1);
555
- expect(await $state.pick()).toBe(1);
556
- expect(counter).toBe(1);
557
- });
558
-
559
- // TODO: Fix this test
560
- it.skip("should handle Promise that rejects", async () => {
561
- const error = new Error("Promise rejected");
562
- const $state = stateAwait(Promise.reject(error));
563
-
564
- await expect($state.pick()).rejects.toThrow("Promise rejected");
565
- });
566
-
567
- // TODO: Fix this test
568
- it.skip("should handle lazy function that rejects", async () => {
569
- const error = new Error("Lazy function rejected");
570
- const $state = stateAwait(() => Promise.reject(error));
571
-
572
- await expect($state.pick()).rejects.toThrow("Lazy function rejected");
573
- });
574
-
575
- it("should handle Promise that resolves to null", async () => {
576
- const $state = stateAwait(Promise.resolve(null));
577
- const value = await $state.pick();
578
- expect(value).toBeNull();
579
- });
580
-
581
- it("should handle Promise that resolves to undefined", async () => {
582
- const $state = stateAwait(Promise.resolve(undefined));
583
- const value = await $state.pick();
584
- expect(value).toBeUndefined();
585
- });
586
-
587
- it("should handle complex object values in Promise", async () => {
588
- const complexObj = {
589
- nested: { deep: { value: 42 } },
590
- array: [1, 2, 3],
591
- };
592
- const $state = stateAwait(Promise.resolve(complexObj));
593
- const value = await $state.pick();
594
- expect(value).toBe(complexObj);
595
- expect(value.nested.deep.value).toBe(42);
596
- expect(value.array).toEqual([1, 2, 3]);
597
- });
598
- });
599
- });
600
-
601
- describe("integration", () => {
602
- describe("with effects - get", () => {
603
- it("should create reactive dependency when get is used in effect", async () => {
604
- const $state = stateAwait(Promise.resolve(100));
605
- const effectFn = vi.fn();
606
-
607
- effect((t) => {
608
- $state.get(t);
609
- effectFn();
610
- });
611
-
612
- await vi.waitFor(() => expect(effectFn).toHaveBeenCalledTimes(1));
613
- });
614
-
615
- it("should call effect with initial value", async () => {
616
- const $state = stateAwait(Promise.resolve(1));
617
- const effectFn = vi.fn();
618
-
619
- effect((t) => {
620
- const value = $state.get(t);
621
- effectFn(value);
622
- });
623
-
624
- await vi.waitFor(() => expect(effectFn).toHaveBeenCalledTimes(1));
625
- // Value might be undefined on first call if Promise hasn't resolved
626
- // Wait a bit more for Promise to resolve
627
- await new Promise((resolve) => setTimeout(resolve, 10));
628
- // Check that eventually the value is correct
629
- const lastCall = effectFn.mock.calls[effectFn.mock.calls.length - 1];
630
- expect(lastCall[0]).toBe(1);
631
- });
632
-
633
- it("should call effect with lazy init value", async () => {
634
- const $state = stateAwait(() => Promise.resolve(1));
635
- const effectFn = vi.fn();
636
-
637
- effect((t) => {
638
- effectFn($state.get(t));
639
- });
640
-
641
- await vi.waitFor(() => expect(effectFn).toHaveBeenCalledTimes(1));
642
- expect(effectFn).toHaveBeenLastCalledWith(1);
643
- });
644
-
645
- it("should call effect when value is updated", async () => {
646
- const $state = stateAwait(Promise.resolve(1));
647
- const effectFn = vi.fn();
648
-
649
- effect((t) => {
650
- effectFn($state.get(t));
651
- });
652
-
653
- await vi.waitFor(() => expect(effectFn).toHaveBeenCalledTimes(1));
654
- expect(effectFn).toHaveBeenLastCalledWith(1);
655
-
656
- await $state.set(Promise.resolve(2));
657
- await vi.waitFor(() => expect(effectFn).toHaveBeenCalledTimes(2));
658
- expect(effectFn).toHaveBeenLastCalledWith(2);
659
- });
660
-
661
- it("should not call effect when updated with same value", async () => {
662
- const $state = stateAwait(Promise.resolve(1));
663
- const effectFn = vi.fn();
664
-
665
- effect((t) => {
666
- effectFn($state.get(t));
667
- });
668
-
669
- await vi.waitFor(() => expect(effectFn).toHaveBeenCalledTimes(1));
670
-
671
- await $state.set(Promise.resolve(2));
672
- await vi.waitFor(() => expect(effectFn).toHaveBeenCalledTimes(2));
673
- expect(effectFn).toHaveBeenLastCalledWith(2);
674
-
675
- await $state.set(Promise.resolve(2));
676
- await new Promise((resolve) => setTimeout(resolve, 50));
677
- expect(effectFn).toHaveBeenCalledTimes(2);
678
- });
679
-
680
- it("should not call effect after effect is disposed", async () => {
681
- const $state = stateAwait(Promise.resolve(1));
682
- const effectFn = vi.fn();
683
- const $effect = effect((t) => {
684
- effectFn($state.get(t));
685
- });
686
-
687
- await vi.waitFor(() => expect(effectFn).toHaveBeenCalledTimes(1));
688
- expect(effectFn).toHaveBeenLastCalledWith(1);
689
-
690
- await $state.set(Promise.resolve(2));
691
- await vi.waitFor(() => expect(effectFn).toHaveBeenCalledTimes(2));
692
- expect(effectFn).toHaveBeenLastCalledWith(2);
693
-
694
- $effect.dispose();
695
-
696
- await $state.set(Promise.resolve(3));
697
- await new Promise((resolve) => setTimeout(resolve, 50));
698
- expect(effectFn).toHaveBeenCalledTimes(2);
699
- });
700
-
701
- it("should support multiple effects depending on same state", async () => {
702
- const $state = stateAwait(Promise.resolve(200));
703
- const effectFn1 = vi.fn();
704
- const effectFn2 = vi.fn();
705
- const effectFn3 = vi.fn();
706
-
707
- effect((t) => {
708
- $state.get(t);
709
- effectFn1();
710
- });
711
-
712
- effect((t) => {
713
- $state.get(t);
714
- effectFn2();
715
- });
716
-
717
- effect((t) => {
718
- $state.get(t);
719
- effectFn3();
720
- });
721
-
722
- await vi.waitFor(() => {
723
- expect(effectFn1).toHaveBeenCalledTimes(1);
724
- expect(effectFn2).toHaveBeenCalledTimes(1);
725
- expect(effectFn3).toHaveBeenCalledTimes(1);
726
- });
727
-
728
- await $state.set(Promise.resolve(300));
729
- await vi.waitFor(() => {
730
- expect(effectFn1).toHaveBeenCalledTimes(2);
731
- expect(effectFn2).toHaveBeenCalledTimes(2);
732
- expect(effectFn3).toHaveBeenCalledTimes(2);
733
- });
734
- });
735
-
736
- it("should support effect depending on multiple states", async () => {
737
- const $stateA = stateAwait(Promise.resolve(5));
738
- const $stateB = stateAwait(Promise.resolve(10));
739
- const effectFn = vi.fn();
740
-
741
- effect((t) => {
742
- const a = $stateA.get(t);
743
- const b = $stateB.get(t);
744
- effectFn(a, b);
745
- });
746
-
747
- await vi.waitFor(() => expect(effectFn).toHaveBeenCalledTimes(1));
748
- // Wait for Promises to resolve
749
- await new Promise((resolve) => setTimeout(resolve, 10));
750
- const lastCall = effectFn.mock.calls[effectFn.mock.calls.length - 1];
751
- expect(lastCall[0]).toBe(5);
752
- expect(lastCall[1]).toBe(10);
753
-
754
- await $stateA.set(Promise.resolve(6));
755
- await vi.waitFor(() => expect(effectFn).toHaveBeenCalledTimes(2));
756
- expect(effectFn).toHaveBeenLastCalledWith(6, 10);
757
-
758
- await $stateB.set(Promise.resolve(11));
759
- await vi.waitFor(() => expect(effectFn).toHaveBeenCalledTimes(3));
760
- expect(effectFn).toHaveBeenLastCalledWith(6, 11);
761
- });
762
- });
763
-
764
- describe("with effects - watch", () => {
765
- it("should register dependency when watch is used in effect", async () => {
766
- const $state = stateAwait(Promise.resolve(400));
767
- const effectFn = vi.fn();
768
-
769
- effect((t) => {
770
- $state.watch(t);
771
- effectFn();
772
- });
773
-
774
- await vi.waitFor(() => expect(effectFn).toHaveBeenCalledTimes(1));
775
- });
776
-
777
- it("should trigger re-runs when state changes after watch", async () => {
778
- const $state = stateAwait(Promise.resolve(500));
779
- const effectFn = vi.fn();
780
-
781
- effect((t) => {
782
- $state.watch(t);
783
- effectFn();
784
- });
785
-
786
- await vi.waitFor(() => expect(effectFn).toHaveBeenCalledTimes(1));
787
-
788
- await $state.set(Promise.resolve(600));
789
- await vi.waitFor(() => expect(effectFn).toHaveBeenCalledTimes(2));
790
- });
791
- });
792
-
793
- describe("with effects - subscribe", () => {
794
- it("should call listener immediately when subscribe is used", async () => {
795
- const $state = stateAwait(Promise.resolve(700));
796
- const listener = vi.fn();
797
-
798
- $state.subscribe(listener);
799
-
800
- await vi.waitFor(() => expect(listener).toHaveBeenCalledTimes(1));
801
- expect(listener).toHaveBeenLastCalledWith(700);
802
- });
803
-
804
- it("should support multiple subscriptions with effects", async () => {
805
- const $state = stateAwait(Promise.resolve(800));
806
- const listener1 = vi.fn();
807
- const listener2 = vi.fn();
808
-
809
- $state.subscribe(listener1);
810
- $state.subscribe(listener2);
811
-
812
- await vi.waitFor(() => {
813
- expect(listener1).toHaveBeenCalledTimes(1);
814
- expect(listener2).toHaveBeenCalledTimes(1);
815
- });
816
-
817
- await $state.set(Promise.resolve(900));
818
- await vi.waitFor(() => {
819
- expect(listener1).toHaveBeenCalledTimes(2);
820
- expect(listener2).toHaveBeenCalledTimes(2);
821
- });
822
- });
823
-
824
- it("should dispose effect when disposer is called", async () => {
825
- const $state = stateAwait(Promise.resolve(1000));
826
- const listener = vi.fn();
827
-
828
- const unsubscribe = $state.subscribe(listener);
829
- await vi.waitFor(() => expect(listener).toHaveBeenCalledTimes(1));
830
-
831
- await $state.set(Promise.resolve(1100));
832
- await vi.waitFor(() => expect(listener).toHaveBeenCalledTimes(2));
833
-
834
- unsubscribe();
835
-
836
- await $state.set(Promise.resolve(1200));
837
- await new Promise((resolve) => setTimeout(resolve, 50));
838
- expect(listener).toHaveBeenCalledTimes(2);
839
- });
840
- });
841
-
842
- describe("with effects - disposal", () => {
843
- it("should dispose effects when state is disposed", async () => {
844
- const $state = stateAwait(Promise.resolve(1300));
845
- const effectFn = vi.fn();
846
- const $effect = effect((t) => {
847
- $state.get(t);
848
- effectFn();
849
- });
850
-
851
- await vi.waitFor(() => expect(effectFn).toHaveBeenCalledTimes(1));
852
- expect($effect.disposed).toBe(false);
853
-
854
- $state.dispose();
855
-
856
- expect($effect.disposed).toBe(true);
857
- });
858
-
859
- it("should not dispose effects when state is disposed with self option", async () => {
860
- const $state = stateAwait(Promise.resolve(1400));
861
- const effectFn = vi.fn();
862
- const $effect = effect((t) => {
863
- $state.get(t);
864
- effectFn();
865
- });
866
-
867
- await vi.waitFor(() => expect(effectFn).toHaveBeenCalledTimes(1));
868
- expect($effect.disposed).toBe(false);
869
-
870
- $state.dispose({ self: true });
871
-
872
- expect($effect.disposed).toBe(false);
873
- expect($state.disposed).toBe(true);
874
- });
875
-
876
- it("should unregister effects when state is disposed with self option", async () => {
877
- const $state = stateAwait(Promise.resolve(1500));
878
- const effectFn = vi.fn();
879
- const $effect = effect((t) => {
880
- $state.get(t);
881
- effectFn();
882
- });
883
-
884
- await vi.waitFor(() => expect(effectFn).toHaveBeenCalledTimes(1));
885
-
886
- $state.dispose({ self: true });
887
-
888
- // Effect should still be active but not notified
889
- expect($effect.disposed).toBe(false);
890
-
891
- // But state is disposed so operations should fail
892
- const $tracker = state(0);
893
- expect(() => $state.get($tracker)).toThrow(
894
- "[PicoFlow] Primitive is disposed",
895
- );
896
- });
897
- });
898
-
899
- describe("complex cases", () => {
900
- it("should handle nested states in effects", async () => {
901
- const $stateA = stateAwait(Promise.resolve(1));
902
- const $stateB = stateAwait(Promise.resolve(2));
903
- const $stateC = stateAwait(
904
- Promise.resolve({
905
- a: $stateA,
906
- b: $stateB,
907
- }),
908
- );
909
-
910
- const effectFn = vi.fn();
911
- effect((t) => {
912
- const c = $stateC.get(t);
913
- if (c) {
914
- const a = c.a.get(t);
915
- if (a !== undefined) {
916
- effectFn(a);
917
- }
918
- }
919
- });
920
-
921
- await vi.waitFor(() => expect(effectFn).toHaveBeenCalledTimes(1));
922
- // Wait for Promises to resolve
923
- await new Promise((resolve) => setTimeout(resolve, 10));
924
- const lastCall = effectFn.mock.calls[effectFn.mock.calls.length - 1];
925
- expect(lastCall[0]).toBe(1);
926
-
927
- await $stateA.set(Promise.resolve(10));
928
- await vi.waitFor(() => expect(effectFn).toHaveBeenCalledTimes(2));
929
- expect(effectFn).toHaveBeenLastCalledWith(10);
930
- });
931
-
932
- it("should handle effect depending on multiple states", async () => {
933
- const $stateA = stateAwait(Promise.resolve(10));
934
- const $stateB = stateAwait(Promise.resolve(20));
935
- const $stateC = stateAwait(Promise.resolve(30));
936
- const effectFn = vi.fn();
937
-
938
- effect((t) => {
939
- const a = $stateA.get(t);
940
- const b = $stateB.get(t);
941
- const c = $stateC.get(t);
942
- if (a !== undefined && b !== undefined && c !== undefined) {
943
- effectFn(a, b, c);
944
- }
945
- });
946
-
947
- await vi.waitFor(() => expect(effectFn).toHaveBeenCalledTimes(1));
948
- // Wait for Promises to resolve
949
- await new Promise((resolve) => setTimeout(resolve, 10));
950
- const lastCall = effectFn.mock.calls[effectFn.mock.calls.length - 1];
951
- expect(lastCall[0]).toBe(10);
952
- expect(lastCall[1]).toBe(20);
953
- expect(lastCall[2]).toBe(30);
954
-
955
- await $stateA.set(Promise.resolve(11));
956
- await vi.waitFor(() => expect(effectFn).toHaveBeenCalledTimes(2));
957
- expect(effectFn).toHaveBeenLastCalledWith(11, 20, 30);
958
-
959
- await $stateB.set(Promise.resolve(21));
960
- await vi.waitFor(() => expect(effectFn).toHaveBeenCalledTimes(3));
961
- expect(effectFn).toHaveBeenLastCalledWith(11, 21, 30);
962
- });
963
-
964
- it("should handle state chaining (state A -> state B -> state C)", async () => {
965
- const $stateA = stateAwait(Promise.resolve(100));
966
- const $stateB = stateAwait(Promise.resolve(200));
967
- const $stateC = stateAwait(Promise.resolve(300));
968
- const effectFn = vi.fn();
969
-
970
- // Effect watches A, which triggers B, which triggers C
971
- effect((t) => {
972
- const a = $stateA.get(t);
973
- if (a !== undefined && a > 150) {
974
- $stateB.set(Promise.resolve(250));
975
- }
976
- });
977
-
978
- effect((t) => {
979
- const b = $stateB.get(t);
980
- if (b !== undefined && b > 220) {
981
- $stateC.set(Promise.resolve(350));
982
- }
983
- });
984
-
985
- effect((t) => {
986
- const c = $stateC.get(t);
987
- if (c !== undefined) {
988
- effectFn(c);
989
- }
990
- });
991
-
992
- await vi.waitFor(() => expect(effectFn).toHaveBeenCalledTimes(1));
993
-
994
- await $stateA.set(Promise.resolve(200));
995
- await vi.waitFor(() => expect(effectFn).toHaveBeenCalledTimes(2));
996
- });
997
- });
998
- });
999
- });