@ersbeth/picoflow 1.1.2 → 2.0.1
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.
- package/.vscode/settings.json +3 -3
- package/CHANGELOG.md +43 -0
- package/README.md +2 -18
- package/biome.json +45 -35
- package/dist/picoflow.js +854 -1520
- package/dist/types/api/base/flowDisposable.d.ts +41 -0
- package/dist/types/api/base/flowDisposable.d.ts.map +1 -0
- package/dist/types/api/base/flowObservable.d.ts +27 -0
- package/dist/types/api/base/flowObservable.d.ts.map +1 -0
- package/dist/types/api/base/flowSubscribable.d.ts +79 -0
- package/dist/types/api/base/flowSubscribable.d.ts.map +1 -0
- package/dist/types/api/base/flowTracker.d.ts +8 -0
- package/dist/types/api/base/flowTracker.d.ts.map +1 -0
- package/dist/types/api/base/index.d.ts +5 -0
- package/dist/types/api/base/index.d.ts.map +1 -0
- package/dist/types/api/index.d.ts +3 -0
- package/dist/types/api/index.d.ts.map +1 -0
- package/dist/types/api/nodes/async/flowConstantAsync.d.ts +31 -0
- package/dist/types/api/nodes/async/flowConstantAsync.d.ts.map +1 -0
- package/dist/types/api/nodes/async/flowDerivationAsync.d.ts +37 -0
- package/dist/types/api/nodes/async/flowDerivationAsync.d.ts.map +1 -0
- package/dist/types/api/nodes/async/flowStateAsync.d.ts +41 -0
- package/dist/types/api/nodes/async/flowStateAsync.d.ts.map +1 -0
- package/dist/types/api/nodes/async/flowWritableDerivationAsync.d.ts +30 -0
- package/dist/types/api/nodes/async/flowWritableDerivationAsync.d.ts.map +1 -0
- package/dist/types/{flow → api}/nodes/async/index.d.ts +1 -2
- package/dist/types/api/nodes/async/index.d.ts.map +1 -0
- package/dist/types/api/nodes/collections/flowArray.d.ts +134 -0
- package/dist/types/api/nodes/collections/flowArray.d.ts.map +1 -0
- package/dist/types/api/nodes/collections/flowMap.d.ts +98 -0
- package/dist/types/api/nodes/collections/flowMap.d.ts.map +1 -0
- package/dist/types/api/nodes/collections/index.d.ts.map +1 -0
- package/dist/types/api/nodes/flowEffect.d.ts +29 -0
- package/dist/types/api/nodes/flowEffect.d.ts.map +1 -0
- package/dist/types/api/nodes/flowSignal.d.ts +25 -0
- package/dist/types/api/nodes/flowSignal.d.ts.map +1 -0
- package/dist/types/api/nodes/flowValue.d.ts +36 -0
- package/dist/types/api/nodes/flowValue.d.ts.map +1 -0
- package/dist/types/api/nodes/index.d.ts +8 -0
- package/dist/types/api/nodes/index.d.ts.map +1 -0
- package/dist/types/api/nodes/sync/flowConstant.d.ts +29 -0
- package/dist/types/api/nodes/sync/flowConstant.d.ts.map +1 -0
- package/dist/types/api/nodes/sync/flowDerivation.d.ts +36 -0
- package/dist/types/api/nodes/sync/flowDerivation.d.ts.map +1 -0
- package/dist/types/api/nodes/sync/flowState.d.ts +39 -0
- package/dist/types/api/nodes/sync/flowState.d.ts.map +1 -0
- package/dist/types/api/nodes/sync/flowWritableDerivation.d.ts +28 -0
- package/dist/types/api/nodes/sync/flowWritableDerivation.d.ts.map +1 -0
- package/dist/types/{flow → api}/nodes/sync/index.d.ts +1 -2
- package/dist/types/api/nodes/sync/index.d.ts.map +1 -0
- package/dist/types/api/nodes/utils.d.ts +23 -0
- package/dist/types/api/nodes/utils.d.ts.map +1 -0
- package/dist/types/base/disposable.d.ts +11 -0
- package/dist/types/base/disposable.d.ts.map +1 -0
- package/dist/types/base/executionStack.d.ts +14 -0
- package/dist/types/base/executionStack.d.ts.map +1 -0
- package/dist/types/base/index.d.ts +6 -0
- package/dist/types/base/index.d.ts.map +1 -0
- package/dist/types/base/node.d.ts +27 -0
- package/dist/types/base/node.d.ts.map +1 -0
- package/dist/types/base/observable.d.ts +35 -0
- package/dist/types/base/observable.d.ts.map +1 -0
- package/dist/types/base/observer.d.ts +25 -0
- package/dist/types/base/observer.d.ts.map +1 -0
- package/dist/types/converters/index.d.ts +2 -0
- package/dist/types/converters/index.d.ts.map +1 -0
- package/dist/types/converters/solid.d.ts +46 -0
- package/dist/types/converters/solid.d.ts.map +1 -0
- package/dist/types/index.d.ts +2 -63
- package/dist/types/index.d.ts.map +1 -1
- package/dist/types/nodes/arrayNode.d.ts +25 -0
- package/dist/types/nodes/arrayNode.d.ts.map +1 -0
- package/dist/types/nodes/effectNode.d.ts +16 -0
- package/dist/types/nodes/effectNode.d.ts.map +1 -0
- package/dist/types/nodes/index.d.ts +8 -0
- package/dist/types/nodes/index.d.ts.map +1 -0
- package/dist/types/nodes/mapNode.d.ts +19 -0
- package/dist/types/nodes/mapNode.d.ts.map +1 -0
- package/dist/types/nodes/signalNode.d.ts +10 -0
- package/dist/types/nodes/signalNode.d.ts.map +1 -0
- package/dist/types/nodes/valueAsyncNode.d.ts +23 -0
- package/dist/types/nodes/valueAsyncNode.d.ts.map +1 -0
- package/dist/types/nodes/valueNode.d.ts +20 -0
- package/dist/types/nodes/valueNode.d.ts.map +1 -0
- package/dist/types/nodes/valueSyncNode.d.ts +23 -0
- package/dist/types/nodes/valueSyncNode.d.ts.map +1 -0
- package/dist/types/schedulers/asyncResolver.d.ts +19 -0
- package/dist/types/schedulers/asyncResolver.d.ts.map +1 -0
- package/dist/types/schedulers/asyncScheduler.d.ts +20 -0
- package/dist/types/schedulers/asyncScheduler.d.ts.map +1 -0
- package/dist/types/schedulers/index.d.ts +5 -0
- package/dist/types/schedulers/index.d.ts.map +1 -0
- package/dist/types/schedulers/pendingError.d.ts +9 -0
- package/dist/types/schedulers/pendingError.d.ts.map +1 -0
- package/dist/types/schedulers/scheduler.d.ts +10 -0
- package/dist/types/schedulers/scheduler.d.ts.map +1 -0
- package/dist/types/schedulers/syncResolver.d.ts +19 -0
- package/dist/types/schedulers/syncResolver.d.ts.map +1 -0
- package/dist/types/schedulers/syncScheduler.d.ts +19 -0
- package/dist/types/schedulers/syncScheduler.d.ts.map +1 -0
- package/docs/.vitepress/config.mts +128 -93
- package/docs/api/functions/array.md +14 -37
- package/docs/api/functions/constant.md +13 -25
- package/docs/api/functions/constantAsync.md +69 -0
- package/docs/api/functions/derivation.md +14 -33
- package/docs/api/functions/derivationAsync.md +34 -0
- package/docs/api/functions/from.md +62 -153
- package/docs/api/functions/isDisposable.md +8 -30
- package/docs/api/functions/map.md +15 -36
- package/docs/api/functions/signal.md +8 -23
- package/docs/api/functions/state.md +43 -23
- package/docs/api/functions/stateAsync.md +69 -0
- package/docs/api/functions/subscribe.md +40 -0
- package/docs/api/functions/writableDerivation.md +33 -0
- package/docs/api/functions/writableDerivationAsync.md +34 -0
- package/docs/api/index.md +45 -102
- package/docs/api/interfaces/FlowArray.md +439 -0
- package/docs/api/interfaces/FlowConstant.md +220 -0
- package/docs/api/interfaces/FlowConstantAsync.md +221 -0
- package/docs/api/interfaces/FlowDerivation.md +241 -0
- package/docs/api/interfaces/FlowDerivationAsync.md +242 -0
- package/docs/api/interfaces/FlowDisposable.md +32 -38
- package/docs/api/interfaces/FlowEffect.md +64 -0
- package/docs/api/interfaces/FlowMap.md +374 -0
- package/docs/api/interfaces/FlowObservable.md +155 -0
- package/docs/api/interfaces/FlowSignal.md +156 -0
- package/docs/api/interfaces/FlowState.md +269 -0
- package/docs/api/interfaces/FlowStateAsync.md +268 -0
- package/docs/api/interfaces/FlowSubscribable.md +55 -0
- package/docs/api/interfaces/FlowTracker.md +61 -0
- package/docs/api/interfaces/FlowValue.md +222 -0
- package/docs/api/interfaces/FlowWritableDerivation.md +292 -0
- package/docs/api/interfaces/FlowWritableDerivationAsync.md +293 -0
- package/docs/api/type-aliases/DerivationFunction.md +28 -0
- package/docs/api/type-aliases/DerivationFunctionAsync.md +28 -0
- package/docs/api/type-aliases/FlowArrayAction.md +19 -8
- package/docs/api/type-aliases/FlowDataTracker.md +33 -0
- package/docs/api/type-aliases/FlowMapAction.md +48 -0
- package/docs/api/type-aliases/FlowOnDataListener.md +33 -0
- package/docs/api/type-aliases/FlowOnErrorListener.md +27 -0
- package/docs/api/type-aliases/FlowOnPendingListener.md +21 -0
- package/docs/api/type-aliases/FlowReadonly.md +22 -0
- package/docs/api/type-aliases/InitFunction.md +21 -0
- package/docs/api/type-aliases/InitFunctionAsync.md +21 -0
- package/docs/api/type-aliases/NotPromise.md +6 -3
- package/docs/api/type-aliases/UpdateFunction.md +27 -0
- package/docs/api/type-aliases/UpdateFunctionAsync.md +27 -0
- package/docs/api/typedoc-sidebar.json +63 -79
- package/docs/examples/examples.md +0 -2
- package/docs/guide/advanced/architecture.md +1234 -0
- package/docs/guide/advanced/migration-v2.md +204 -0
- package/docs/guide/advanced/solidjs.md +2 -88
- package/docs/guide/introduction/concepts.md +4 -3
- package/docs/guide/introduction/conventions.md +2 -33
- package/docs/guide/introduction/getting-started.md +28 -23
- package/docs/guide/introduction/lifecycle.md +16 -19
- package/docs/guide/primitives/array.md +102 -216
- package/docs/guide/primitives/constant.md +39 -212
- package/docs/guide/primitives/derivations.md +55 -122
- package/docs/guide/primitives/effects.md +155 -241
- package/docs/guide/primitives/map.md +64 -186
- package/docs/guide/primitives/overview.md +45 -128
- package/docs/guide/primitives/signal.md +51 -88
- package/docs/guide/primitives/state.md +34 -130
- package/package.json +56 -60
- package/src/api/base/flowDisposable.ts +44 -0
- package/src/api/base/flowObservable.ts +28 -0
- package/src/api/base/flowSubscribable.ts +87 -0
- package/src/api/base/flowTracker.ts +7 -0
- package/src/api/base/index.ts +4 -0
- package/src/{flow → api}/index.ts +0 -1
- package/src/api/nodes/async/flowConstantAsync.ts +36 -0
- package/src/api/nodes/async/flowDerivationAsync.ts +42 -0
- package/src/api/nodes/async/flowStateAsync.ts +47 -0
- package/src/api/nodes/async/flowWritableDerivationAsync.ts +33 -0
- package/src/{flow → api}/nodes/async/index.ts +1 -2
- package/src/api/nodes/collections/flowArray.ts +155 -0
- package/src/api/nodes/collections/flowMap.ts +115 -0
- package/src/api/nodes/flowEffect.ts +42 -0
- package/src/api/nodes/flowSignal.ts +28 -0
- package/src/api/nodes/flowValue.ts +37 -0
- package/src/api/nodes/index.ts +7 -0
- package/src/api/nodes/sync/flowConstant.ts +33 -0
- package/src/api/nodes/sync/flowDerivation.ts +41 -0
- package/src/api/nodes/sync/flowState.ts +45 -0
- package/src/api/nodes/sync/flowWritableDerivation.ts +31 -0
- package/src/{flow → api}/nodes/sync/index.ts +1 -2
- package/src/api/nodes/utils.ts +24 -0
- package/src/base/disposable.ts +18 -0
- package/src/base/executionStack.ts +42 -0
- package/src/base/index.ts +5 -0
- package/src/base/node.ts +98 -0
- package/src/base/observable.ts +92 -0
- package/src/base/observer.ts +51 -0
- package/src/converters/index.ts +1 -0
- package/src/converters/solid.ts +109 -0
- package/src/index.ts +2 -64
- package/src/nodes/arrayNode.ts +180 -0
- package/src/nodes/effectNode.ts +58 -0
- package/src/nodes/index.ts +7 -0
- package/src/nodes/mapNode.ts +125 -0
- package/src/nodes/signalNode.ts +19 -0
- package/src/nodes/valueAsyncNode.ts +85 -0
- package/src/nodes/valueNode.ts +148 -0
- package/src/nodes/valueSyncNode.ts +125 -0
- package/src/schedulers/asyncResolver.ts +78 -0
- package/src/schedulers/asyncScheduler.ts +66 -0
- package/src/schedulers/index.ts +4 -0
- package/src/schedulers/pendingError.ts +13 -0
- package/src/schedulers/scheduler.ts +9 -0
- package/src/schedulers/syncResolver.ts +69 -0
- package/src/schedulers/syncScheduler.ts +55 -0
- package/test/base/pendingError.test.ts +67 -0
- package/test/converters/solid.derivation.browser.test.tsx +69 -0
- package/test/converters/solid.node.test.ts +654 -0
- package/test/converters/solid.state.browser.test.tsx +1592 -0
- package/test/reactivity/flowSignal.test.ts +226 -0
- package/test/reactivity/nodes/async/asyncScheduler/asyncResolver.test.ts +593 -0
- package/test/reactivity/nodes/async/asyncScheduler/asyncScheduler.test.ts +317 -0
- package/test/reactivity/nodes/async/flowConstantAsync.test.ts +652 -0
- package/test/reactivity/nodes/async/flowDerivation.test.ts +898 -0
- package/test/reactivity/nodes/async/flowDerivationAsync.test.ts +1716 -0
- package/test/reactivity/nodes/async/flowStateAsync.test.ts +708 -0
- package/test/reactivity/nodes/async/flowWritableDerivationAsync.test.ts +614 -0
- package/test/reactivity/nodes/collections/flowArray.asyncStates.test.ts +1289 -0
- package/test/reactivity/nodes/collections/flowArray.scalars.test.ts +961 -0
- package/test/reactivity/nodes/collections/flowArray.states.test.ts +1035 -0
- package/test/reactivity/nodes/collections/flowMap.asyncStates.test.ts +960 -0
- package/test/reactivity/nodes/collections/flowMap.scalars.test.ts +775 -0
- package/test/reactivity/nodes/collections/flowMap.states.test.ts +958 -0
- package/test/reactivity/nodes/sync/flowConstant.test.ts +377 -0
- package/test/reactivity/nodes/sync/flowDerivation.test.ts +896 -0
- package/test/reactivity/nodes/sync/flowState.test.ts +341 -0
- package/test/reactivity/nodes/sync/flowWritableDerivation.test.ts +603 -0
- package/test/vitest.d.ts +10 -0
- package/tsconfig.json +31 -20
- package/typedoc.json +35 -35
- package/vite.config.ts +25 -23
- package/vitest.browser.config.ts +21 -0
- package/vitest.config.ts +12 -12
- package/.cursor/plans/unifier-flowresource-avec-flowderivation-c9506e24.plan.md +0 -372
- package/.cursor/plans/update-js-e795d61b.plan.md +0 -567
- package/dist/types/flow/base/flowDisposable.d.ts +0 -67
- package/dist/types/flow/base/flowDisposable.d.ts.map +0 -1
- package/dist/types/flow/base/flowEffect.d.ts +0 -127
- package/dist/types/flow/base/flowEffect.d.ts.map +0 -1
- package/dist/types/flow/base/flowGraph.d.ts +0 -97
- package/dist/types/flow/base/flowGraph.d.ts.map +0 -1
- package/dist/types/flow/base/flowSignal.d.ts +0 -134
- package/dist/types/flow/base/flowSignal.d.ts.map +0 -1
- package/dist/types/flow/base/flowTracker.d.ts +0 -15
- package/dist/types/flow/base/flowTracker.d.ts.map +0 -1
- package/dist/types/flow/base/index.d.ts +0 -7
- package/dist/types/flow/base/index.d.ts.map +0 -1
- package/dist/types/flow/base/utils.d.ts +0 -20
- package/dist/types/flow/base/utils.d.ts.map +0 -1
- package/dist/types/flow/collections/flowArray.d.ts +0 -148
- package/dist/types/flow/collections/flowArray.d.ts.map +0 -1
- package/dist/types/flow/collections/flowMap.d.ts +0 -224
- package/dist/types/flow/collections/flowMap.d.ts.map +0 -1
- package/dist/types/flow/collections/index.d.ts.map +0 -1
- package/dist/types/flow/index.d.ts +0 -4
- package/dist/types/flow/index.d.ts.map +0 -1
- package/dist/types/flow/nodes/async/flowConstantAsync.d.ts +0 -137
- package/dist/types/flow/nodes/async/flowConstantAsync.d.ts.map +0 -1
- package/dist/types/flow/nodes/async/flowDerivationAsync.d.ts +0 -137
- package/dist/types/flow/nodes/async/flowDerivationAsync.d.ts.map +0 -1
- package/dist/types/flow/nodes/async/flowNodeAsync.d.ts +0 -343
- package/dist/types/flow/nodes/async/flowNodeAsync.d.ts.map +0 -1
- package/dist/types/flow/nodes/async/flowReadonlyAsync.d.ts +0 -81
- package/dist/types/flow/nodes/async/flowReadonlyAsync.d.ts.map +0 -1
- package/dist/types/flow/nodes/async/flowStateAsync.d.ts +0 -111
- package/dist/types/flow/nodes/async/flowStateAsync.d.ts.map +0 -1
- package/dist/types/flow/nodes/async/index.d.ts.map +0 -1
- package/dist/types/flow/nodes/index.d.ts +0 -3
- package/dist/types/flow/nodes/index.d.ts.map +0 -1
- package/dist/types/flow/nodes/sync/flowConstant.d.ts +0 -108
- package/dist/types/flow/nodes/sync/flowConstant.d.ts.map +0 -1
- package/dist/types/flow/nodes/sync/flowDerivation.d.ts +0 -100
- package/dist/types/flow/nodes/sync/flowDerivation.d.ts.map +0 -1
- package/dist/types/flow/nodes/sync/flowNode.d.ts +0 -314
- package/dist/types/flow/nodes/sync/flowNode.d.ts.map +0 -1
- package/dist/types/flow/nodes/sync/flowReadonly.d.ts +0 -57
- package/dist/types/flow/nodes/sync/flowReadonly.d.ts.map +0 -1
- package/dist/types/flow/nodes/sync/flowState.d.ts +0 -96
- package/dist/types/flow/nodes/sync/flowState.d.ts.map +0 -1
- package/dist/types/flow/nodes/sync/index.d.ts.map +0 -1
- package/dist/types/solid/converters.d.ts +0 -57
- package/dist/types/solid/converters.d.ts.map +0 -1
- package/dist/types/solid/index.d.ts +0 -3
- package/dist/types/solid/index.d.ts.map +0 -1
- package/dist/types/solid/primitives.d.ts +0 -181
- package/dist/types/solid/primitives.d.ts.map +0 -1
- package/docs/api/classes/FlowArray.md +0 -489
- package/docs/api/classes/FlowConstant.md +0 -350
- package/docs/api/classes/FlowDerivation.md +0 -334
- package/docs/api/classes/FlowEffect.md +0 -100
- package/docs/api/classes/FlowMap.md +0 -512
- package/docs/api/classes/FlowObservable.md +0 -306
- package/docs/api/classes/FlowResource.md +0 -380
- package/docs/api/classes/FlowResourceAsync.md +0 -362
- package/docs/api/classes/FlowSignal.md +0 -160
- package/docs/api/classes/FlowState.md +0 -368
- package/docs/api/classes/FlowStream.md +0 -367
- package/docs/api/classes/FlowStreamAsync.md +0 -364
- package/docs/api/classes/SolidDerivation.md +0 -75
- package/docs/api/classes/SolidResource.md +0 -91
- package/docs/api/classes/SolidState.md +0 -71
- package/docs/api/classes/TrackingContext.md +0 -33
- package/docs/api/functions/effect.md +0 -49
- package/docs/api/functions/resource.md +0 -52
- package/docs/api/functions/resourceAsync.md +0 -50
- package/docs/api/functions/stream.md +0 -53
- package/docs/api/functions/streamAsync.md +0 -50
- package/docs/api/interfaces/SolidObservable.md +0 -19
- package/docs/api/type-aliases/FlowStreamDisposer.md +0 -15
- package/docs/api/type-aliases/FlowStreamSetter.md +0 -27
- package/docs/api/type-aliases/FlowStreamUpdater.md +0 -32
- package/docs/api/type-aliases/SolidGetter.md +0 -17
- package/docs/guide/primitives/resources.md +0 -858
- package/docs/guide/primitives/streams.md +0 -931
- package/src/flow/base/flowDisposable.ts +0 -71
- package/src/flow/base/flowEffect.ts +0 -171
- package/src/flow/base/flowGraph.ts +0 -288
- package/src/flow/base/flowSignal.ts +0 -207
- package/src/flow/base/flowTracker.ts +0 -17
- package/src/flow/base/index.ts +0 -6
- package/src/flow/base/utils.ts +0 -19
- package/src/flow/collections/flowArray.ts +0 -409
- package/src/flow/collections/flowMap.ts +0 -398
- package/src/flow/nodes/async/flowConstantAsync.ts +0 -142
- package/src/flow/nodes/async/flowDerivationAsync.ts +0 -143
- package/src/flow/nodes/async/flowNodeAsync.ts +0 -474
- package/src/flow/nodes/async/flowReadonlyAsync.ts +0 -81
- package/src/flow/nodes/async/flowStateAsync.ts +0 -116
- package/src/flow/nodes/await/advanced/index.ts +0 -5
- package/src/flow/nodes/await/advanced/resource.ts +0 -134
- package/src/flow/nodes/await/advanced/resourceAsync.ts +0 -109
- package/src/flow/nodes/await/advanced/stream.ts +0 -188
- package/src/flow/nodes/await/advanced/streamAsync.ts +0 -176
- package/src/flow/nodes/await/flowConstantAwait.ts +0 -154
- package/src/flow/nodes/await/flowDerivationAwait.ts +0 -154
- package/src/flow/nodes/await/flowNodeAwait.ts +0 -508
- package/src/flow/nodes/await/flowReadonlyAwait.ts +0 -89
- package/src/flow/nodes/await/flowStateAwait.ts +0 -130
- package/src/flow/nodes/await/index.ts +0 -5
- package/src/flow/nodes/index.ts +0 -3
- package/src/flow/nodes/sync/flowConstant.ts +0 -111
- package/src/flow/nodes/sync/flowDerivation.ts +0 -105
- package/src/flow/nodes/sync/flowNode.ts +0 -439
- package/src/flow/nodes/sync/flowReadonly.ts +0 -57
- package/src/flow/nodes/sync/flowState.ts +0 -101
- package/src/solid/converters.ts +0 -148
- package/src/solid/index.ts +0 -2
- package/src/solid/primitives.ts +0 -215
- package/test/base/flowEffect.test.ts +0 -108
- package/test/base/flowGraph.test.ts +0 -485
- package/test/base/flowSignal.test.ts +0 -372
- package/test/collections/flowArray.asyncStates.test.ts +0 -1553
- package/test/collections/flowArray.scalars.test.ts +0 -1129
- package/test/collections/flowArray.states.test.ts +0 -1365
- package/test/collections/flowMap.asyncStates.test.ts +0 -1105
- package/test/collections/flowMap.scalars.test.ts +0 -877
- package/test/collections/flowMap.states.test.ts +0 -1097
- package/test/nodes/async/flowConstantAsync.test.ts +0 -860
- package/test/nodes/async/flowDerivationAsync.test.ts +0 -1517
- package/test/nodes/async/flowStateAsync.test.ts +0 -1387
- package/test/nodes/await/advanced/resource.test.ts +0 -129
- package/test/nodes/await/advanced/resourceAsync.test.ts +0 -108
- package/test/nodes/await/advanced/stream.test.ts +0 -198
- package/test/nodes/await/advanced/streamAsync.test.ts +0 -196
- package/test/nodes/await/flowConstantAwait.test.ts +0 -643
- package/test/nodes/await/flowDerivationAwait.test.ts +0 -1583
- package/test/nodes/await/flowStateAwait.test.ts +0 -999
- package/test/nodes/mixed/derivation.test.ts +0 -1527
- package/test/nodes/sync/flowConstant.test.ts +0 -620
- package/test/nodes/sync/flowDerivation.test.ts +0 -1373
- package/test/nodes/sync/flowState.test.ts +0 -945
- package/test/solid/converters.test.ts +0 -721
- package/test/solid/primitives.test.ts +0 -1031
- /package/dist/types/{flow → api/nodes}/collections/index.d.ts +0 -0
- /package/docs/guide/advanced/{upgrading.md → migration-v1.md} +0 -0
- /package/src/{flow → api/nodes}/collections/index.ts +0 -0
|
@@ -0,0 +1,148 @@
|
|
|
1
|
+
import type {
|
|
2
|
+
FlowEffect,
|
|
3
|
+
FlowOnDataListener,
|
|
4
|
+
FlowOnErrorListener,
|
|
5
|
+
FlowOnPendingListener,
|
|
6
|
+
FlowTracker,
|
|
7
|
+
NotPromise,
|
|
8
|
+
} from "../api";
|
|
9
|
+
import { ExecutionStack, Node, type ObservableStatus, type Observer } from "../base";
|
|
10
|
+
import { PendingError, type Scheduler } from "../schedulers";
|
|
11
|
+
import { EffectNode } from "./effectNode";
|
|
12
|
+
|
|
13
|
+
/**
|
|
14
|
+
* Base class for reactive values that compute, cache, and propagate changes through the dependency graph.
|
|
15
|
+
* @internal
|
|
16
|
+
*/
|
|
17
|
+
export abstract class ValueNode<T> extends Node<T> {
|
|
18
|
+
protected abstract _scheduler: Scheduler;
|
|
19
|
+
protected _value?: NotPromise<T>;
|
|
20
|
+
protected _error?: unknown;
|
|
21
|
+
|
|
22
|
+
override watch(tracker: FlowTracker): void {
|
|
23
|
+
if (this.disposed) throw new Error("[PicoFlow] Primitive is disposed");
|
|
24
|
+
super.watch(tracker as unknown as Observer);
|
|
25
|
+
|
|
26
|
+
switch (this.status as ObservableStatus) {
|
|
27
|
+
case "resolved":
|
|
28
|
+
return;
|
|
29
|
+
case "error":
|
|
30
|
+
throw this._error;
|
|
31
|
+
case "pending": {
|
|
32
|
+
throw new PendingError(this._scheduler.settled);
|
|
33
|
+
}
|
|
34
|
+
case "dirty": {
|
|
35
|
+
this.execute();
|
|
36
|
+
switch (this.status as ObservableStatus) {
|
|
37
|
+
case "pending":
|
|
38
|
+
throw new PendingError(this._scheduler.settled);
|
|
39
|
+
case "resolved":
|
|
40
|
+
return;
|
|
41
|
+
case "error":
|
|
42
|
+
throw this._error;
|
|
43
|
+
case "dirty": {
|
|
44
|
+
throw new Error("[PicoFlow] Internal error");
|
|
45
|
+
}
|
|
46
|
+
}
|
|
47
|
+
}
|
|
48
|
+
}
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
override notify(): void {
|
|
52
|
+
if (this.disposed) throw new Error("[PicoFlow] Primitive is disposed");
|
|
53
|
+
if (this.status === "dirty") return;
|
|
54
|
+
if (this.status === "pending") {
|
|
55
|
+
ExecutionStack.pushPending(this);
|
|
56
|
+
}
|
|
57
|
+
this.status = "dirty";
|
|
58
|
+
this.notifyDependents();
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
override dispose(): void {
|
|
62
|
+
super.dispose();
|
|
63
|
+
this._scheduler.dispose();
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
get(tracker: FlowTracker): T {
|
|
67
|
+
if (this.disposed) throw new Error("[PicoFlow] Primitive is disposed");
|
|
68
|
+
super.watch(tracker);
|
|
69
|
+
|
|
70
|
+
switch (this.status as ObservableStatus) {
|
|
71
|
+
case "resolved":
|
|
72
|
+
return this._value as T;
|
|
73
|
+
case "error":
|
|
74
|
+
throw this._error;
|
|
75
|
+
case "pending": {
|
|
76
|
+
throw new PendingError(this._scheduler.settled);
|
|
77
|
+
}
|
|
78
|
+
case "dirty": {
|
|
79
|
+
this.execute();
|
|
80
|
+
switch (this.status as ObservableStatus) {
|
|
81
|
+
case "resolved":
|
|
82
|
+
return this._value as T;
|
|
83
|
+
case "error":
|
|
84
|
+
throw this._error;
|
|
85
|
+
case "pending": {
|
|
86
|
+
throw new PendingError(this._scheduler.settled);
|
|
87
|
+
}
|
|
88
|
+
case "dirty": {
|
|
89
|
+
throw new Error("[PicoFlow] Internal error");
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
}
|
|
95
|
+
|
|
96
|
+
async pick(): Promise<T> {
|
|
97
|
+
if (this.disposed) throw new Error("[PicoFlow] Primitive is disposed");
|
|
98
|
+
switch (this.status) {
|
|
99
|
+
case "resolved":
|
|
100
|
+
return this._value as T;
|
|
101
|
+
case "pending": {
|
|
102
|
+
await this._scheduler.settled;
|
|
103
|
+
if ((this.status as ObservableStatus) === "resolved") return this._value as T;
|
|
104
|
+
if ((this.status as ObservableStatus) === "error") throw this._error;
|
|
105
|
+
throw new Error("[PicoFlow] Internal error");
|
|
106
|
+
}
|
|
107
|
+
case "error":
|
|
108
|
+
throw this._error;
|
|
109
|
+
case "dirty": {
|
|
110
|
+
this.execute();
|
|
111
|
+
switch (this.status as ObservableStatus) {
|
|
112
|
+
case "resolved":
|
|
113
|
+
return this._value as T;
|
|
114
|
+
case "pending": {
|
|
115
|
+
await this._scheduler.settled;
|
|
116
|
+
if ((this.status as ObservableStatus) === "resolved") return this._value as T;
|
|
117
|
+
if ((this.status as ObservableStatus) === "error") throw this._error;
|
|
118
|
+
throw new Error("[PicoFlow] Internal error");
|
|
119
|
+
}
|
|
120
|
+
case "error":
|
|
121
|
+
throw this._error;
|
|
122
|
+
case "dirty": {
|
|
123
|
+
throw new Error("[PicoFlow] Internal error");
|
|
124
|
+
}
|
|
125
|
+
}
|
|
126
|
+
}
|
|
127
|
+
}
|
|
128
|
+
}
|
|
129
|
+
|
|
130
|
+
execute(): void {
|
|
131
|
+
// TODO: Should we throw instead ?
|
|
132
|
+
if (this.disposed) return;
|
|
133
|
+
|
|
134
|
+
this.clearDependencies();
|
|
135
|
+
this.status = "pending";
|
|
136
|
+
this._scheduler.compute();
|
|
137
|
+
}
|
|
138
|
+
|
|
139
|
+
subscribe(
|
|
140
|
+
onValue: FlowOnDataListener<T>,
|
|
141
|
+
onError?: FlowOnErrorListener,
|
|
142
|
+
onPending?: FlowOnPendingListener,
|
|
143
|
+
): FlowEffect {
|
|
144
|
+
if (this.disposed) throw new Error("[PicoFlow] Primitive is disposed");
|
|
145
|
+
const effect = new EffectNode((t) => this.get(t), onValue, onError, onPending);
|
|
146
|
+
return effect;
|
|
147
|
+
}
|
|
148
|
+
}
|
|
@@ -0,0 +1,125 @@
|
|
|
1
|
+
import type { DerivationFunction, InitFunction, NotPromise, UpdateFunction } from "../api";
|
|
2
|
+
import { SyncScheduler } from "../schedulers";
|
|
3
|
+
import { ValueNode } from "./valueNode";
|
|
4
|
+
|
|
5
|
+
/**
|
|
6
|
+
* Function that computes a value (either initialization or derivation).
|
|
7
|
+
* @internal
|
|
8
|
+
*/
|
|
9
|
+
export type ComputeFunction<T> = InitFunction<T> | DerivationFunction<T>;
|
|
10
|
+
|
|
11
|
+
/**
|
|
12
|
+
* Synchronous reactive value that computes immediately without async scheduling.
|
|
13
|
+
* @internal
|
|
14
|
+
*/
|
|
15
|
+
export class ValueSyncNode<T extends NotPromise<unknown>> extends ValueNode<T> {
|
|
16
|
+
protected _scheduler: SyncScheduler<T>;
|
|
17
|
+
private _compute: () => T;
|
|
18
|
+
|
|
19
|
+
constructor(valueOrCompute: NotPromise<T> | ComputeFunction<T>) {
|
|
20
|
+
super();
|
|
21
|
+
if (typeof valueOrCompute === "function") {
|
|
22
|
+
// compute is a function
|
|
23
|
+
// @ts-expect-error valueOrCompute is a function
|
|
24
|
+
this._compute = () => valueOrCompute(this, this._value);
|
|
25
|
+
} else {
|
|
26
|
+
// compute is a value
|
|
27
|
+
this._compute = () => valueOrCompute;
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
this.status = "dirty";
|
|
31
|
+
this._scheduler = new SyncScheduler<T>(
|
|
32
|
+
() => this._compute(),
|
|
33
|
+
(value) => this._onResolve(value as NotPromise<T>),
|
|
34
|
+
(error) => this._onReject(error),
|
|
35
|
+
);
|
|
36
|
+
|
|
37
|
+
if (typeof valueOrCompute !== "function") {
|
|
38
|
+
this._scheduler.compute();
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
private _onResolve(value: NotPromise<T>) {
|
|
43
|
+
this.status = "resolved";
|
|
44
|
+
this._value = value;
|
|
45
|
+
this._error = undefined;
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
private _onReject(error: unknown) {
|
|
49
|
+
this.status = "error";
|
|
50
|
+
this._error = error;
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
set(value: NotPromise<T>): void;
|
|
54
|
+
set(updater: UpdateFunction<T>): void;
|
|
55
|
+
set(valueOrUpdater: NotPromise<T> | UpdateFunction<T>): void {
|
|
56
|
+
if (this.disposed) throw new Error("[PicoFlow] Primitive is disposed");
|
|
57
|
+
|
|
58
|
+
if (typeof valueOrUpdater === "function") {
|
|
59
|
+
const updater = valueOrUpdater as UpdateFunction<T>;
|
|
60
|
+
switch (this.status) {
|
|
61
|
+
case "resolved": {
|
|
62
|
+
const nextValue = updater(this._value);
|
|
63
|
+
if (nextValue === this._value) return;
|
|
64
|
+
this.status = "pending";
|
|
65
|
+
this._scheduler.overwrite(nextValue);
|
|
66
|
+
this.notifyDependents();
|
|
67
|
+
return;
|
|
68
|
+
}
|
|
69
|
+
case "pending":
|
|
70
|
+
case "error":
|
|
71
|
+
case "dirty": {
|
|
72
|
+
// TODO: For now we use latest value to compute the next value
|
|
73
|
+
// Maybe we should throw instead ?
|
|
74
|
+
const nextValue = updater(this._value);
|
|
75
|
+
this.status = "pending";
|
|
76
|
+
this._scheduler.overwrite(nextValue);
|
|
77
|
+
this.notifyDependents();
|
|
78
|
+
return;
|
|
79
|
+
}
|
|
80
|
+
}
|
|
81
|
+
} else {
|
|
82
|
+
const nextValue = valueOrUpdater;
|
|
83
|
+
switch (this.status) {
|
|
84
|
+
case "resolved": {
|
|
85
|
+
if (nextValue === this._value) return;
|
|
86
|
+
this.status = "pending";
|
|
87
|
+
this._scheduler.overwrite(nextValue);
|
|
88
|
+
this.notifyDependents();
|
|
89
|
+
return;
|
|
90
|
+
}
|
|
91
|
+
case "pending":
|
|
92
|
+
case "error":
|
|
93
|
+
case "dirty": {
|
|
94
|
+
this.status = "pending";
|
|
95
|
+
this._scheduler.overwrite(nextValue);
|
|
96
|
+
this.notifyDependents();
|
|
97
|
+
return;
|
|
98
|
+
}
|
|
99
|
+
}
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
|
|
103
|
+
refresh(): void {
|
|
104
|
+
if (this.disposed) throw new Error("[PicoFlow] Primitive is disposed");
|
|
105
|
+
|
|
106
|
+
const currentValue = this._value;
|
|
107
|
+
const currentStatus = this.status;
|
|
108
|
+
this.execute();
|
|
109
|
+
|
|
110
|
+
switch (this.status) {
|
|
111
|
+
case "resolved": {
|
|
112
|
+
const nextValue = this._value;
|
|
113
|
+
if (nextValue === currentValue && currentStatus === this.status) return;
|
|
114
|
+
this.notifyDependents();
|
|
115
|
+
return;
|
|
116
|
+
}
|
|
117
|
+
case "error":
|
|
118
|
+
case "pending":
|
|
119
|
+
case "dirty": {
|
|
120
|
+
this.notifyDependents();
|
|
121
|
+
return;
|
|
122
|
+
}
|
|
123
|
+
}
|
|
124
|
+
}
|
|
125
|
+
}
|
|
@@ -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,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,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 "../../src/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
|
+
});
|