@ersbeth/picoflow 1.1.2 → 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 +856 -1530
  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 -57
  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 -148
  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
@@ -0,0 +1,128 @@
1
+ import type { InitFunction } from "~/api/nodes/sync/flowConstant";
2
+ import type { DerivationFunction } from "~/api/nodes/sync/flowDerivation";
3
+ import type { UpdateFunction } from "~/api/nodes/sync/flowState";
4
+ import type { NotPromise } from "~/api/nodes/utils";
5
+ import { SyncScheduler } from "~/schedulers";
6
+ import { ValueNode } from "./valueNode";
7
+
8
+ /**
9
+ * Function that computes a value (either initialization or derivation).
10
+ * @internal
11
+ */
12
+ export type ComputeFunction<T> = InitFunction<T> | DerivationFunction<T>;
13
+
14
+ /**
15
+ * Synchronous reactive value that computes immediately without async scheduling.
16
+ * @internal
17
+ */
18
+ export class ValueSyncNode<T extends NotPromise<unknown>> extends ValueNode<T> {
19
+ protected _scheduler: SyncScheduler<T>;
20
+ private _compute: () => T;
21
+
22
+ constructor(valueOrCompute: NotPromise<T> | ComputeFunction<T>) {
23
+ super();
24
+ if (typeof valueOrCompute === "function") {
25
+ // compute is a function
26
+ // @ts-expect-error valueOrCompute is a function
27
+ this._compute = () => valueOrCompute(this, this._value);
28
+ } else {
29
+ // compute is a value
30
+ this._compute = () => valueOrCompute;
31
+ }
32
+
33
+ this.status = "dirty";
34
+ this._scheduler = new SyncScheduler<T>(
35
+ () => this._compute(),
36
+ (value) => this._onResolve(value as NotPromise<T>),
37
+ (error) => this._onReject(error),
38
+ );
39
+
40
+ if (typeof valueOrCompute !== "function") {
41
+ this._scheduler.compute();
42
+ }
43
+ }
44
+
45
+ private _onResolve(value: NotPromise<T>) {
46
+ this.status = "resolved";
47
+ this._value = value;
48
+ this._error = undefined;
49
+ }
50
+
51
+ private _onReject(error: unknown) {
52
+ this.status = "error";
53
+ this._error = error;
54
+ }
55
+
56
+ set(value: NotPromise<T>): void;
57
+ set(updater: UpdateFunction<T>): void;
58
+ set(valueOrUpdater: NotPromise<T> | UpdateFunction<T>): void {
59
+ if (this.disposed) throw new Error("[PicoFlow] Primitive is disposed");
60
+
61
+ if (typeof valueOrUpdater === "function") {
62
+ const updater = valueOrUpdater as UpdateFunction<T>;
63
+ switch (this.status) {
64
+ case "resolved": {
65
+ const nextValue = updater(this._value);
66
+ if (nextValue === this._value) return;
67
+ this.status = "pending";
68
+ this._scheduler.overwrite(nextValue);
69
+ this.notifyDependents();
70
+ return;
71
+ }
72
+ case "pending":
73
+ case "error":
74
+ case "dirty": {
75
+ // TODO: For now we use latest value to compute the next value
76
+ // Maybe we should throw instead ?
77
+ const nextValue = updater(this._value);
78
+ this.status = "pending";
79
+ this._scheduler.overwrite(nextValue);
80
+ this.notifyDependents();
81
+ return;
82
+ }
83
+ }
84
+ } else {
85
+ const nextValue = valueOrUpdater;
86
+ switch (this.status) {
87
+ case "resolved": {
88
+ if (nextValue === this._value) return;
89
+ this.status = "pending";
90
+ this._scheduler.overwrite(nextValue);
91
+ this.notifyDependents();
92
+ return;
93
+ }
94
+ case "pending":
95
+ case "error":
96
+ case "dirty": {
97
+ this.status = "pending";
98
+ this._scheduler.overwrite(nextValue);
99
+ this.notifyDependents();
100
+ return;
101
+ }
102
+ }
103
+ }
104
+ }
105
+
106
+ refresh(): void {
107
+ if (this.disposed) throw new Error("[PicoFlow] Primitive is disposed");
108
+
109
+ const currentValue = this._value;
110
+ const currentStatus = this.status;
111
+ this.execute();
112
+
113
+ switch (this.status) {
114
+ case "resolved": {
115
+ const nextValue = this._value;
116
+ if (nextValue === currentValue && currentStatus === this.status) return;
117
+ this.notifyDependents();
118
+ return;
119
+ }
120
+ case "error":
121
+ case "pending":
122
+ case "dirty": {
123
+ this.notifyDependents();
124
+ return;
125
+ }
126
+ }
127
+ }
128
+ }
@@ -0,0 +1,78 @@
1
+ import { PendingError } from "./pendingError";
2
+
3
+ /**
4
+ * Manages asynchronous computation lifecycle with iteration tracking to prevent race conditions.
5
+ * @internal
6
+ */
7
+ export class AsyncResolver<T> {
8
+ private readonly _compute: () => Promise<T>;
9
+ private readonly _computed = Promise.withResolvers<T>();
10
+ private _iteration = 0;
11
+ private _aborted = false;
12
+ private _finished = false;
13
+
14
+ constructor(compute: () => Promise<T>) {
15
+ this._compute = compute;
16
+
17
+ // Prevent unhandled rejections before finally block is executed: catch all errors so they don't propagate.
18
+ // Actual error handling occurs elsewhere in ComputationScheduler through proper promise chaining.
19
+ this._computed.promise
20
+ .catch(() => {})
21
+ .finally(() => {
22
+ this._finished = true;
23
+ });
24
+ }
25
+
26
+ get computed(): Promise<T> {
27
+ return this._computed.promise;
28
+ }
29
+
30
+ get aborted(): boolean {
31
+ return this._aborted;
32
+ }
33
+
34
+ get finished(): boolean {
35
+ return this._finished;
36
+ }
37
+
38
+ compute(): void {
39
+ if (this._finished) throw new Error("[Picoflow] PendingResolver: Can't restart a settled resolver");
40
+ if (this._aborted) throw new Error("[Picoflow] PendingResolver: Can't restart an aborted resolver");
41
+
42
+ this._iteration++;
43
+ const currentIteration = this._iteration;
44
+ this._compute()
45
+ .then((value) => {
46
+ if (this._iteration === currentIteration && !this._aborted) this._computed.resolve(value);
47
+ })
48
+ .catch((error) => {
49
+ if (this._iteration === currentIteration && !this._aborted) {
50
+ if (!(error instanceof PendingError)) {
51
+ this._computed.reject(error);
52
+ }
53
+ }
54
+ });
55
+ }
56
+
57
+ overwrite(promise: Promise<T>): void {
58
+ if (this._finished) throw new Error("[Picoflow] PendingResolver: Can't overwrite a settled resolver");
59
+ if (this._aborted) throw new Error("[Picoflow] PendingResolver: Can't overwrite an aborted resolver");
60
+
61
+ this._iteration++;
62
+ const currentIteration = this._iteration;
63
+ promise
64
+ .then((value) => {
65
+ if (this._iteration === currentIteration && !this._aborted) this._computed.resolve(value);
66
+ })
67
+ .catch((error) => {
68
+ if (this._iteration === currentIteration && !this._aborted) {
69
+ this._computed.reject(error);
70
+ }
71
+ });
72
+ }
73
+
74
+ abort(): void {
75
+ this._iteration++;
76
+ this._aborted = true;
77
+ }
78
+ }
@@ -0,0 +1,66 @@
1
+ import { AsyncResolver } from "./asyncResolver";
2
+ import type { Scheduler } from "./scheduler";
3
+
4
+ /**
5
+ * Scheduler that manages asynchronous computation lifecycle with resolver instances.
6
+ * @internal
7
+ */
8
+ export class AsyncScheduler<T> implements Scheduler {
9
+ private readonly _compute: () => Promise<T>;
10
+ private _resolver: AsyncResolver<T>;
11
+ private _onResolve: (value: T) => void;
12
+ private _onReject: (error: unknown) => void;
13
+ private _settled = Promise.withResolvers<void>();
14
+ private _disposed = false;
15
+
16
+ constructor(compute: () => Promise<T>, onResolve: (value: T) => void, onReject: (error: unknown) => void) {
17
+ this._compute = compute;
18
+ this._onResolve = onResolve;
19
+ this._onReject = onReject;
20
+ this._resolver = new AsyncResolver(compute);
21
+ this._resolver.computed
22
+ .then(this._onResolve)
23
+ .catch(this._onReject)
24
+ .finally(() => this._settled.resolve());
25
+ }
26
+
27
+ get settled(): Promise<void> {
28
+ if (this._disposed) throw new Error("[PicoFlow] ComputationScheduler is disposed");
29
+ return this._settled.promise;
30
+ }
31
+
32
+ get disposed(): boolean {
33
+ return this._disposed;
34
+ }
35
+
36
+ overwrite(promise: Promise<T>): void {
37
+ if (this._disposed) throw new Error("[PicoFlow] ComputationScheduler is disposed");
38
+ if (this._resolver.finished) {
39
+ this._resolver = new AsyncResolver(this._compute);
40
+ this._settled = Promise.withResolvers<void>();
41
+ this._resolver.computed
42
+ .then(this._onResolve)
43
+ .catch(this._onReject)
44
+ .finally(() => this._settled.resolve());
45
+ }
46
+ this._resolver.overwrite(promise);
47
+ }
48
+
49
+ compute(): void {
50
+ if (this._disposed) throw new Error("[PicoFlow] ComputationScheduler is disposed");
51
+ if (this._resolver.finished) {
52
+ this._resolver = new AsyncResolver(this._compute);
53
+ this._settled = Promise.withResolvers<void>();
54
+ this._resolver.computed
55
+ .then(this._onResolve)
56
+ .catch(this._onReject)
57
+ .finally(() => this._settled.resolve());
58
+ }
59
+ this._resolver.compute();
60
+ }
61
+
62
+ dispose(): void {
63
+ this._resolver.abort();
64
+ this._disposed = true;
65
+ }
66
+ }
@@ -0,0 +1,4 @@
1
+ export * from "./asyncScheduler";
2
+ export * from "./pendingError";
3
+ export * from "./scheduler";
4
+ export * from "./syncScheduler";
@@ -0,0 +1,13 @@
1
+ /**
2
+ * Error thrown when an async computation is pending, encapsulating its promise for cascading dependencies.
3
+ * @internal
4
+ */
5
+ export class PendingError extends Error {
6
+ readonly pendingPromise: Promise<unknown>;
7
+
8
+ constructor(promise: Promise<unknown>) {
9
+ super("[PicoFlow] PendingResolver pending");
10
+ this.name = "PendingError";
11
+ this.pendingPromise = promise;
12
+ }
13
+ }
@@ -0,0 +1,9 @@
1
+ /**
2
+ * Contract for schedulers that coordinate reactive computations and their lifecycle.
3
+ * @internal
4
+ */
5
+ export interface Scheduler {
6
+ compute(): void;
7
+ dispose(): void;
8
+ settled: Promise<void>;
9
+ }
@@ -0,0 +1,69 @@
1
+ import { PendingError } from "./pendingError";
2
+
3
+ /**
4
+ * Manages synchronous computation lifecycle with iteration tracking to prevent race conditions.
5
+ * @internal
6
+ */
7
+ export class SyncResolver<T> {
8
+ private readonly _compute: () => T;
9
+ private _iteration = 0;
10
+ private _aborted = false;
11
+ private _finished = false;
12
+
13
+ private _onValue: (value: T) => void;
14
+ private _onError: (error: unknown) => void;
15
+
16
+ constructor(compute: () => T, onValue: (value: T) => void, onError: (error: unknown) => void) {
17
+ this._compute = compute;
18
+ this._onValue = onValue;
19
+ this._onError = onError;
20
+ }
21
+
22
+ get aborted(): boolean {
23
+ return this._aborted;
24
+ }
25
+
26
+ get finished(): boolean {
27
+ return this._finished;
28
+ }
29
+
30
+ compute(): void {
31
+ if (this._finished) throw new Error("[Picoflow] PendingResolver: Can't restart a settled resolver");
32
+ if (this._aborted) throw new Error("[Picoflow] PendingResolver: Can't restart an aborted resolver");
33
+
34
+ this._iteration++;
35
+ const currentIteration = this._iteration;
36
+ try {
37
+ const value = this._compute();
38
+ if (this._iteration === currentIteration && !this._aborted) {
39
+ this._finished = true;
40
+ this._onValue(value);
41
+ }
42
+ } catch (error) {
43
+ if (this._iteration === currentIteration && !this._aborted) {
44
+ if (!(error instanceof PendingError)) {
45
+ this._finished = true;
46
+ this._onError(error);
47
+ }
48
+ }
49
+ }
50
+ }
51
+
52
+ overwrite(value: T): void {
53
+ if (this._finished) throw new Error("[Picoflow] PendingResolver: Can't overwrite a settled resolver");
54
+ if (this._aborted) throw new Error("[Picoflow] PendingResolver: Can't overwrite an aborted resolver");
55
+
56
+ this._iteration++;
57
+ const currentIteration = this._iteration;
58
+
59
+ if (this._iteration === currentIteration && !this._aborted) {
60
+ this._finished = true;
61
+ this._onValue(value);
62
+ }
63
+ }
64
+
65
+ abort(): void {
66
+ this._iteration++;
67
+ this._aborted = true;
68
+ }
69
+ }
@@ -0,0 +1,55 @@
1
+ import type { Scheduler } from "./scheduler";
2
+ import { SyncResolver } from "./syncResolver";
3
+
4
+ /**
5
+ * Scheduler that manages synchronous computation lifecycle with resolver instances.
6
+ * @internal
7
+ */
8
+ export class SyncScheduler<T> implements Scheduler {
9
+ private readonly _compute: () => T;
10
+ private _resolver: SyncResolver<T>;
11
+ private _onResolve: (value: T) => void;
12
+ private _onReject: (error: unknown) => void;
13
+ private _settled = Promise.withResolvers<void>();
14
+ private _disposed = false;
15
+
16
+ constructor(compute: () => T, onResolve: (value: T) => void, onReject: (error: unknown) => void) {
17
+ this._compute = compute;
18
+ this._onResolve = (value) => {
19
+ onResolve(value);
20
+ this._settled.resolve();
21
+ };
22
+ this._onReject = (error) => {
23
+ onReject(error);
24
+ this._settled.resolve();
25
+ };
26
+ this._resolver = new SyncResolver(compute, this._onResolve, this._onReject);
27
+ }
28
+
29
+ get settled(): Promise<void> {
30
+ return this._settled.promise;
31
+ }
32
+
33
+ overwrite(value: T): void {
34
+ if (this._disposed) throw new Error("[PicoFlow] ComputationScheduler is disposed");
35
+ if (this._resolver.finished) {
36
+ this._settled = Promise.withResolvers<void>();
37
+ this._resolver = new SyncResolver(this._compute, this._onResolve, this._onReject);
38
+ }
39
+ this._resolver.overwrite(value);
40
+ }
41
+
42
+ compute(): void {
43
+ if (this._disposed) throw new Error("[PicoFlow] ComputationScheduler is disposed");
44
+ if (this._resolver.finished) {
45
+ this._settled = Promise.withResolvers<void>();
46
+ this._resolver = new SyncResolver(this._compute, this._onResolve, this._onReject);
47
+ }
48
+ this._resolver.compute();
49
+ }
50
+
51
+ dispose(): void {
52
+ this._resolver.abort();
53
+ this._disposed = true;
54
+ }
55
+ }
@@ -0,0 +1,67 @@
1
+ import { describe, expect, it } from "vitest";
2
+ import { PendingError } from "~/schedulers";
3
+
4
+ describe("PendingError", () => {
5
+ describe("constructor and properties", () => {
6
+ it("should create instance with pending promise", () => {
7
+ const promise = Promise.resolve(42);
8
+ const error = new PendingError(promise);
9
+
10
+ expect(error).toBeInstanceOf(PendingError);
11
+ expect(error.pendingPromise).toBe(promise);
12
+ });
13
+
14
+ it("should have correct name", () => {
15
+ const promise = Promise.resolve(42);
16
+ const error = new PendingError(promise);
17
+
18
+ expect(error.name).toBe("PendingError");
19
+ });
20
+
21
+ it("should have correct message", () => {
22
+ const promise = Promise.resolve(42);
23
+ const error = new PendingError(promise);
24
+
25
+ expect(error.message).toBe("[PicoFlow] PendingResolver pending");
26
+ });
27
+
28
+ it("should extend Error", () => {
29
+ const promise = Promise.resolve(42);
30
+ const error = new PendingError(promise);
31
+
32
+ expect(error).toBeInstanceOf(Error);
33
+ });
34
+
35
+ it("should store pendingPromise property correctly", async () => {
36
+ const promise = Promise.resolve("test-value");
37
+ const error = new PendingError(promise);
38
+
39
+ expect(error.pendingPromise).toBe(promise);
40
+ expect(await error.pendingPromise).toBe("test-value");
41
+ });
42
+ });
43
+
44
+ describe("pendingPromise behavior", () => {
45
+ it("should expose the pending promise", () => {
46
+ const promise = Promise.resolve(100);
47
+ const error = new PendingError(promise);
48
+
49
+ expect(error.pendingPromise).toBe(promise);
50
+ });
51
+
52
+ it("should allow awaiting the pending promise", async () => {
53
+ const promise = Promise.resolve("resolved-value");
54
+ const error = new PendingError(promise);
55
+
56
+ const result = await error.pendingPromise;
57
+ expect(result).toBe("resolved-value");
58
+ });
59
+
60
+ it("should allow awaiting rejected promise", async () => {
61
+ const promise = Promise.reject(new Error("test-error"));
62
+ const error = new PendingError(promise);
63
+
64
+ await expect(error.pendingPromise).rejects.toThrow("test-error");
65
+ });
66
+ });
67
+ });
@@ -0,0 +1,69 @@
1
+ import { cleanup, render } from "@solidjs/testing-library";
2
+ import { type Component, Match, Switch } from "solid-js";
3
+ import { beforeEach, describe, expect, it, vi } from "vitest";
4
+ import { userEvent } from "vitest/browser";
5
+ import { derivation, from } from "~";
6
+
7
+ describe("solid converter", () => {
8
+ beforeEach(() => {
9
+ cleanup();
10
+ });
11
+
12
+ describe("updates", () => {
13
+ describe("error", () => {
14
+ describe("flag", () => {
15
+ it("update-error-flag: should handle error in FlowDerivation", async () => {
16
+ const Comp: Component = () => {
17
+ let condition = false;
18
+ const $derivation = derivation(() => {
19
+ if (condition) {
20
+ throw new Error("Test error");
21
+ }
22
+ return 42;
23
+ });
24
+
25
+ const $$derivation = from($derivation);
26
+
27
+ return (
28
+ <div>
29
+ <Switch>
30
+ <Match when={$$derivation.error}>
31
+ <div>Error</div>
32
+ </Match>
33
+ <Match when={$$derivation()}>
34
+ <div>{$$derivation()}</div>
35
+ </Match>
36
+ </Switch>
37
+ <button
38
+ type="button"
39
+ onClick={() => {
40
+ condition = true;
41
+ $derivation.refresh();
42
+ }}
43
+ >
44
+ Update Error
45
+ </button>
46
+ <button
47
+ type="button"
48
+ onClick={() => {
49
+ condition = false;
50
+ $derivation.refresh();
51
+ }}
52
+ >
53
+ Update Value
54
+ </button>
55
+ </div>
56
+ );
57
+ };
58
+
59
+ const { getByText } = render(() => <Comp />);
60
+ await vi.waitFor(() => expect.element(getByText("42")).toBeInTheDocument());
61
+ await userEvent.click(getByText("Update Error"));
62
+ await vi.waitFor(() => expect.element(getByText("Error")).toBeInTheDocument());
63
+ await userEvent.click(getByText("Update Value"));
64
+ await vi.waitFor(() => expect.element(getByText("42")).toBeInTheDocument());
65
+ });
66
+ });
67
+ });
68
+ });
69
+ });