@ersbeth/picoflow 1.0.0 → 1.1.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 (180) hide show
  1. package/.cursor/plans/unifier-flowresource-avec-flowderivation-c9506e24.plan.md +372 -0
  2. package/README.md +25 -171
  3. package/biome.json +4 -1
  4. package/dist/picoflow.js +1129 -661
  5. package/dist/types/flow/base/flowDisposable.d.ts +67 -0
  6. package/dist/types/flow/base/flowDisposable.d.ts.map +1 -0
  7. package/dist/types/flow/base/flowEffect.d.ts +127 -0
  8. package/dist/types/flow/base/flowEffect.d.ts.map +1 -0
  9. package/dist/types/flow/base/flowGraph.d.ts +97 -0
  10. package/dist/types/flow/base/flowGraph.d.ts.map +1 -0
  11. package/dist/types/flow/base/flowSignal.d.ts +134 -0
  12. package/dist/types/flow/base/flowSignal.d.ts.map +1 -0
  13. package/dist/types/flow/base/flowTracker.d.ts +15 -0
  14. package/dist/types/flow/base/flowTracker.d.ts.map +1 -0
  15. package/dist/types/flow/base/index.d.ts +7 -0
  16. package/dist/types/flow/base/index.d.ts.map +1 -0
  17. package/dist/types/flow/base/utils.d.ts +20 -0
  18. package/dist/types/flow/base/utils.d.ts.map +1 -0
  19. package/dist/types/{advanced/array.d.ts → flow/collections/flowArray.d.ts} +50 -12
  20. package/dist/types/flow/collections/flowArray.d.ts.map +1 -0
  21. package/dist/types/flow/collections/flowMap.d.ts +224 -0
  22. package/dist/types/flow/collections/flowMap.d.ts.map +1 -0
  23. package/dist/types/flow/collections/index.d.ts +3 -0
  24. package/dist/types/flow/collections/index.d.ts.map +1 -0
  25. package/dist/types/flow/index.d.ts +4 -0
  26. package/dist/types/flow/index.d.ts.map +1 -0
  27. package/dist/types/flow/nodes/async/flowConstantAsync.d.ts +137 -0
  28. package/dist/types/flow/nodes/async/flowConstantAsync.d.ts.map +1 -0
  29. package/dist/types/flow/nodes/async/flowDerivationAsync.d.ts +137 -0
  30. package/dist/types/flow/nodes/async/flowDerivationAsync.d.ts.map +1 -0
  31. package/dist/types/flow/nodes/async/flowNodeAsync.d.ts +343 -0
  32. package/dist/types/flow/nodes/async/flowNodeAsync.d.ts.map +1 -0
  33. package/dist/types/flow/nodes/async/flowReadonlyAsync.d.ts +81 -0
  34. package/dist/types/flow/nodes/async/flowReadonlyAsync.d.ts.map +1 -0
  35. package/dist/types/flow/nodes/async/flowStateAsync.d.ts +111 -0
  36. package/dist/types/flow/nodes/async/flowStateAsync.d.ts.map +1 -0
  37. package/dist/types/flow/nodes/async/index.d.ts +6 -0
  38. package/dist/types/flow/nodes/async/index.d.ts.map +1 -0
  39. package/dist/types/flow/nodes/index.d.ts +3 -0
  40. package/dist/types/flow/nodes/index.d.ts.map +1 -0
  41. package/dist/types/flow/nodes/sync/flowConstant.d.ts +108 -0
  42. package/dist/types/flow/nodes/sync/flowConstant.d.ts.map +1 -0
  43. package/dist/types/flow/nodes/sync/flowDerivation.d.ts +100 -0
  44. package/dist/types/flow/nodes/sync/flowDerivation.d.ts.map +1 -0
  45. package/dist/types/flow/nodes/sync/flowNode.d.ts +314 -0
  46. package/dist/types/flow/nodes/sync/flowNode.d.ts.map +1 -0
  47. package/dist/types/flow/nodes/sync/flowReadonly.d.ts +57 -0
  48. package/dist/types/flow/nodes/sync/flowReadonly.d.ts.map +1 -0
  49. package/dist/types/flow/nodes/sync/flowState.d.ts +96 -0
  50. package/dist/types/flow/nodes/sync/flowState.d.ts.map +1 -0
  51. package/dist/types/flow/nodes/sync/index.d.ts +6 -0
  52. package/dist/types/flow/nodes/sync/index.d.ts.map +1 -0
  53. package/dist/types/index.d.ts +1 -4
  54. package/dist/types/index.d.ts.map +1 -1
  55. package/dist/types/solid/converters.d.ts +34 -44
  56. package/dist/types/solid/converters.d.ts.map +1 -1
  57. package/dist/types/solid/primitives.d.ts +1 -0
  58. package/dist/types/solid/primitives.d.ts.map +1 -1
  59. package/docs/.vitepress/config.mts +1 -1
  60. package/docs/api/typedoc-sidebar.json +81 -1
  61. package/package.json +60 -58
  62. package/src/flow/base/flowDisposable.ts +71 -0
  63. package/src/flow/base/flowEffect.ts +171 -0
  64. package/src/flow/base/flowGraph.ts +288 -0
  65. package/src/flow/base/flowSignal.ts +207 -0
  66. package/src/flow/base/flowTracker.ts +17 -0
  67. package/src/flow/base/index.ts +6 -0
  68. package/src/flow/base/utils.ts +19 -0
  69. package/src/flow/collections/flowArray.ts +409 -0
  70. package/src/flow/collections/flowMap.ts +398 -0
  71. package/src/flow/collections/index.ts +2 -0
  72. package/src/flow/index.ts +3 -0
  73. package/src/flow/nodes/async/flowConstantAsync.ts +142 -0
  74. package/src/flow/nodes/async/flowDerivationAsync.ts +143 -0
  75. package/src/flow/nodes/async/flowNodeAsync.ts +474 -0
  76. package/src/flow/nodes/async/flowReadonlyAsync.ts +81 -0
  77. package/src/flow/nodes/async/flowStateAsync.ts +116 -0
  78. package/src/flow/nodes/async/index.ts +5 -0
  79. package/src/flow/nodes/await/advanced/index.ts +5 -0
  80. package/src/{advanced → flow/nodes/await/advanced}/resource.ts +37 -3
  81. package/src/{advanced → flow/nodes/await/advanced}/resourceAsync.ts +35 -3
  82. package/src/{advanced → flow/nodes/await/advanced}/stream.ts +40 -2
  83. package/src/{advanced → flow/nodes/await/advanced}/streamAsync.ts +38 -3
  84. package/src/flow/nodes/await/flowConstantAwait.ts +154 -0
  85. package/src/flow/nodes/await/flowDerivationAwait.ts +154 -0
  86. package/src/flow/nodes/await/flowNodeAwait.ts +508 -0
  87. package/src/flow/nodes/await/flowReadonlyAwait.ts +89 -0
  88. package/src/flow/nodes/await/flowStateAwait.ts +130 -0
  89. package/src/flow/nodes/await/index.ts +5 -0
  90. package/src/flow/nodes/index.ts +3 -0
  91. package/src/flow/nodes/sync/flowConstant.ts +111 -0
  92. package/src/flow/nodes/sync/flowDerivation.ts +105 -0
  93. package/src/flow/nodes/sync/flowNode.ts +439 -0
  94. package/src/flow/nodes/sync/flowReadonly.ts +57 -0
  95. package/src/flow/nodes/sync/flowState.ts +101 -0
  96. package/src/flow/nodes/sync/index.ts +5 -0
  97. package/src/index.ts +1 -47
  98. package/src/solid/converters.ts +59 -198
  99. package/src/solid/primitives.ts +4 -0
  100. package/test/base/flowEffect.test.ts +108 -0
  101. package/test/base/flowGraph.test.ts +485 -0
  102. package/test/base/flowSignal.test.ts +372 -0
  103. package/test/collections/flowArray.asyncStates.test.ts +1553 -0
  104. package/test/collections/flowArray.scalars.test.ts +1129 -0
  105. package/test/collections/flowArray.states.test.ts +1365 -0
  106. package/test/collections/flowMap.asyncStates.test.ts +1105 -0
  107. package/test/collections/flowMap.scalars.test.ts +877 -0
  108. package/test/collections/flowMap.states.test.ts +1097 -0
  109. package/test/nodes/async/flowConstantAsync.test.ts +860 -0
  110. package/test/nodes/async/flowDerivationAsync.test.ts +1517 -0
  111. package/test/nodes/async/flowStateAsync.test.ts +1387 -0
  112. package/test/{resource.test.ts → nodes/await/advanced/resource.test.ts} +21 -19
  113. package/test/{resourceAsync.test.ts → nodes/await/advanced/resourceAsync.test.ts} +3 -1
  114. package/test/{stream.test.ts → nodes/await/advanced/stream.test.ts} +30 -28
  115. package/test/{streamAsync.test.ts → nodes/await/advanced/streamAsync.test.ts} +16 -14
  116. package/test/nodes/await/flowConstantAwait.test.ts +643 -0
  117. package/test/nodes/await/flowDerivationAwait.test.ts +1583 -0
  118. package/test/nodes/await/flowStateAwait.test.ts +999 -0
  119. package/test/nodes/mixed/derivation.test.ts +1527 -0
  120. package/test/nodes/sync/flowConstant.test.ts +620 -0
  121. package/test/nodes/sync/flowDerivation.test.ts +1373 -0
  122. package/test/nodes/sync/flowState.test.ts +945 -0
  123. package/test/solid/converters.test.ts +721 -0
  124. package/test/solid/primitives.test.ts +1031 -0
  125. package/tsconfig.json +2 -1
  126. package/vitest.config.ts +7 -1
  127. package/IMPLEMENTATION_GUIDE.md +0 -1578
  128. package/dist/types/advanced/array.d.ts.map +0 -1
  129. package/dist/types/advanced/index.d.ts +0 -9
  130. package/dist/types/advanced/index.d.ts.map +0 -1
  131. package/dist/types/advanced/map.d.ts +0 -166
  132. package/dist/types/advanced/map.d.ts.map +0 -1
  133. package/dist/types/advanced/resource.d.ts +0 -78
  134. package/dist/types/advanced/resource.d.ts.map +0 -1
  135. package/dist/types/advanced/resourceAsync.d.ts +0 -56
  136. package/dist/types/advanced/resourceAsync.d.ts.map +0 -1
  137. package/dist/types/advanced/stream.d.ts +0 -117
  138. package/dist/types/advanced/stream.d.ts.map +0 -1
  139. package/dist/types/advanced/streamAsync.d.ts +0 -97
  140. package/dist/types/advanced/streamAsync.d.ts.map +0 -1
  141. package/dist/types/basic/constant.d.ts +0 -60
  142. package/dist/types/basic/constant.d.ts.map +0 -1
  143. package/dist/types/basic/derivation.d.ts +0 -89
  144. package/dist/types/basic/derivation.d.ts.map +0 -1
  145. package/dist/types/basic/disposable.d.ts +0 -82
  146. package/dist/types/basic/disposable.d.ts.map +0 -1
  147. package/dist/types/basic/effect.d.ts +0 -67
  148. package/dist/types/basic/effect.d.ts.map +0 -1
  149. package/dist/types/basic/index.d.ts +0 -10
  150. package/dist/types/basic/index.d.ts.map +0 -1
  151. package/dist/types/basic/observable.d.ts +0 -83
  152. package/dist/types/basic/observable.d.ts.map +0 -1
  153. package/dist/types/basic/signal.d.ts +0 -69
  154. package/dist/types/basic/signal.d.ts.map +0 -1
  155. package/dist/types/basic/state.d.ts +0 -47
  156. package/dist/types/basic/state.d.ts.map +0 -1
  157. package/dist/types/basic/trackingContext.d.ts +0 -33
  158. package/dist/types/basic/trackingContext.d.ts.map +0 -1
  159. package/dist/types/creators.d.ts +0 -340
  160. package/dist/types/creators.d.ts.map +0 -1
  161. package/src/advanced/array.ts +0 -222
  162. package/src/advanced/index.ts +0 -12
  163. package/src/advanced/map.ts +0 -193
  164. package/src/basic/constant.ts +0 -97
  165. package/src/basic/derivation.ts +0 -147
  166. package/src/basic/disposable.ts +0 -86
  167. package/src/basic/effect.ts +0 -104
  168. package/src/basic/index.ts +0 -9
  169. package/src/basic/observable.ts +0 -109
  170. package/src/basic/signal.ts +0 -145
  171. package/src/basic/state.ts +0 -60
  172. package/src/basic/trackingContext.ts +0 -45
  173. package/src/creators.ts +0 -395
  174. package/test/array.test.ts +0 -600
  175. package/test/constant.test.ts +0 -44
  176. package/test/derivation.test.ts +0 -539
  177. package/test/effect.test.ts +0 -29
  178. package/test/map.test.ts +0 -240
  179. package/test/signal.test.ts +0 -72
  180. package/test/state.test.ts +0 -212
@@ -0,0 +1,721 @@
1
+ import { createEffect, createRoot } from "solid-js";
2
+ import { describe, expect, it, vi } from "vitest";
3
+ import {
4
+ derivation,
5
+ derivationAsync,
6
+ type FlowTracker,
7
+ state,
8
+ stateAsync,
9
+ } from "#package";
10
+ import { from } from "../../src/solid/converters";
11
+ import { SolidResource } from "../../src/solid/primitives";
12
+
13
+ describe("converters", () => {
14
+ describe("from", () => {
15
+ describe("unit", () => {
16
+ describe("initialization", () => {
17
+ it("should convert synchronous FlowNode to SolidResource", async () => {
18
+ await createRoot(async (dispose) => {
19
+ const $state = state(42);
20
+ const solidResource = from($state);
21
+
22
+ expect(solidResource).toBeInstanceOf(SolidResource);
23
+
24
+ await vi.waitFor(() => expect(solidResource.get()).toBe(42));
25
+ dispose();
26
+ });
27
+ });
28
+
29
+ it("should convert FlowNodeAsync to SolidResource", async () => {
30
+ await createRoot(async (dispose) => {
31
+ const $stateAsync = stateAsync(Promise.resolve(42));
32
+ const solidResource = from($stateAsync);
33
+
34
+ expect(solidResource).toBeInstanceOf(SolidResource);
35
+
36
+ await vi.waitFor(() => expect(solidResource.get()).toBe(42));
37
+ dispose();
38
+ });
39
+ });
40
+
41
+ it("should convert synchronous getter function to SolidResource", async () => {
42
+ await createRoot(async (dispose) => {
43
+ const $state = state(42);
44
+ const getter = (t: FlowTracker) => $state.get(t);
45
+ const solidResource = from(getter);
46
+
47
+ expect(solidResource).toBeInstanceOf(SolidResource);
48
+
49
+ await vi.waitFor(() => expect(solidResource.get()).toBe(42));
50
+ dispose();
51
+ });
52
+ });
53
+
54
+ it("should convert asynchronous getter function to SolidResource", async () => {
55
+ await createRoot(async (dispose) => {
56
+ const $stateAsync = stateAsync(Promise.resolve(42));
57
+ const getter = async (t: FlowTracker) => await $stateAsync.get(t);
58
+ const solidResource = from(getter);
59
+
60
+ expect(solidResource).toBeInstanceOf(SolidResource);
61
+
62
+ await vi.waitFor(() => expect(solidResource.get()).toBe(42));
63
+ dispose();
64
+ });
65
+ });
66
+
67
+ it("should return a SolidResource instance", async () => {
68
+ await createRoot(async (dispose) => {
69
+ const $state = state(42);
70
+ const solidResource = from($state);
71
+
72
+ expect(solidResource).toBeInstanceOf(SolidResource);
73
+
74
+ await vi.waitFor(() => expect(solidResource.get()).toBe(42));
75
+ dispose();
76
+ });
77
+ });
78
+ });
79
+
80
+ describe("conversion from FlowNode (synchronous)", () => {
81
+ it("should convert a FlowState to SolidResource", async () => {
82
+ await createRoot(async (dispose) => {
83
+ const $state = state(42);
84
+ const solidResource = from($state);
85
+
86
+ expect(solidResource).toBeInstanceOf(SolidResource);
87
+
88
+ await vi.waitFor(() => expect(solidResource.get()).toBe(42));
89
+ dispose();
90
+ });
91
+ });
92
+
93
+ it("should convert a FlowDerivation to SolidResource", async () => {
94
+ await createRoot(async (dispose) => {
95
+ const $state = state(5);
96
+ const $derivation = derivation((t) => $state.get(t) * 2);
97
+ const solidResource = from($derivation);
98
+
99
+ expect(solidResource).toBeInstanceOf(SolidResource);
100
+
101
+ await vi.waitFor(() => expect(solidResource.get()).toBe(10));
102
+ dispose();
103
+ });
104
+ });
105
+
106
+ it("should initialize with FlowNode value", async () => {
107
+ await createRoot(async (dispose) => {
108
+ const $state = state(42);
109
+ const solidResource = from($state);
110
+
111
+ // Wait for resource to load
112
+ await vi.waitFor(() => expect(solidResource.get()).toBe(42));
113
+ dispose();
114
+ });
115
+ });
116
+
117
+ it("should handle number values", async () => {
118
+ await createRoot(async (dispose) => {
119
+ const $state = state(42);
120
+ const solidResource = from($state);
121
+
122
+ await vi.waitFor(() => expect(solidResource.get()).toBe(42));
123
+ dispose();
124
+ });
125
+ });
126
+
127
+ it("should handle string values", async () => {
128
+ await createRoot(async (dispose) => {
129
+ const $state = state("hello");
130
+ const solidResource = from($state);
131
+
132
+ await vi.waitFor(() => expect(solidResource.get()).toBe("hello"));
133
+ dispose();
134
+ });
135
+ });
136
+
137
+ it("should handle object values", async () => {
138
+ await createRoot(async (dispose) => {
139
+ const obj = { a: 1, b: 2 };
140
+ const $state = state(obj);
141
+ const solidResource = from($state);
142
+
143
+ await vi.waitFor(() =>
144
+ expect(solidResource.get()).toEqual({ a: 1, b: 2 }),
145
+ );
146
+ dispose();
147
+ });
148
+ });
149
+
150
+ it("should handle array values", async () => {
151
+ await createRoot(async (dispose) => {
152
+ const arr = [1, 2, 3];
153
+ const $state = state(arr);
154
+ const solidResource = from($state);
155
+
156
+ await vi.waitFor(() =>
157
+ expect(solidResource.get()).toEqual([1, 2, 3]),
158
+ );
159
+ dispose();
160
+ });
161
+ });
162
+
163
+ it("should handle null values", async () => {
164
+ await createRoot(async (dispose) => {
165
+ const $stateNull = state(null);
166
+ const solidResourceNull = from($stateNull);
167
+
168
+ await vi.waitFor(() => expect(solidResourceNull.get()).toBeNull());
169
+ dispose();
170
+ });
171
+ });
172
+
173
+ it("should handle undefined values", async () => {
174
+ await createRoot(async (dispose) => {
175
+ const $stateUndefined = state(undefined);
176
+ const solidResourceUndefined = from($stateUndefined);
177
+
178
+ await vi.waitFor(() =>
179
+ expect(solidResourceUndefined.get()).toBeUndefined(),
180
+ );
181
+ dispose();
182
+ });
183
+ });
184
+ });
185
+
186
+ describe("conversion from FlowNodeAsync (asynchronous)", () => {
187
+ it("should convert a FlowStateAsync to SolidResource", async () => {
188
+ await createRoot(async (dispose) => {
189
+ const $stateAsync = stateAsync(Promise.resolve(42));
190
+ const solidResource = from($stateAsync);
191
+
192
+ expect(solidResource).toBeInstanceOf(SolidResource);
193
+
194
+ await vi.waitFor(() => expect(solidResource.get()).toBe(42));
195
+ dispose();
196
+ });
197
+ });
198
+
199
+ it("should convert a FlowDerivationAsync to SolidResource", async () => {
200
+ await createRoot(async (dispose) => {
201
+ const $stateAsync = stateAsync(Promise.resolve(5));
202
+ const $derivationAsync = derivationAsync(
203
+ async (t) => (await $stateAsync.get(t)) * 2,
204
+ );
205
+ const solidResource = from($derivationAsync);
206
+
207
+ expect(solidResource).toBeInstanceOf(SolidResource);
208
+
209
+ await vi.waitFor(() => expect(solidResource.get()).toBe(10));
210
+ dispose();
211
+ });
212
+ });
213
+
214
+ it("should initialize with FlowNodeAsync value", async () => {
215
+ await createRoot(async (dispose) => {
216
+ const $stateAsync = stateAsync(Promise.resolve(42));
217
+ const solidResource = from($stateAsync);
218
+
219
+ await vi.waitFor(() => expect(solidResource.get()).toBe(42));
220
+ dispose();
221
+ });
222
+ });
223
+
224
+ it("should handle resolved Promises", async () => {
225
+ await createRoot(async (dispose) => {
226
+ const $stateAsync = stateAsync(Promise.resolve(42));
227
+ const solidResource = from($stateAsync);
228
+
229
+ await vi.waitFor(() => expect(solidResource.get()).toBe(42));
230
+ dispose();
231
+ });
232
+ });
233
+
234
+ it("should handle rejected Promises", async () => {
235
+ await createRoot(async (dispose) => {
236
+ const $stateAsync = stateAsync(
237
+ Promise.reject(new Error("Test error")),
238
+ );
239
+
240
+ const solidResource = from($stateAsync);
241
+
242
+ solidResource.get();
243
+
244
+ await vi.waitFor(() =>
245
+ expect(solidResource.state()).toBe("errored"),
246
+ );
247
+ dispose();
248
+ });
249
+ });
250
+ });
251
+
252
+ describe("conversion from getter function", () => {
253
+ it("should convert a synchronous getter to SolidResource", async () => {
254
+ await createRoot(async (dispose) => {
255
+ const $state = state(42);
256
+ const getter = (t: FlowTracker) => $state.get(t);
257
+ const solidResource = from(getter);
258
+
259
+ expect(solidResource).toBeInstanceOf(SolidResource);
260
+
261
+ await vi.waitFor(() => expect(solidResource.get()).toBe(42));
262
+ dispose();
263
+ });
264
+ });
265
+
266
+ it("should convert an asynchronous getter to SolidResource", async () => {
267
+ await createRoot(async (dispose) => {
268
+ const $stateAsync = stateAsync(Promise.resolve(42));
269
+ const getter = async (t: FlowTracker) => await $stateAsync.get(t);
270
+ const solidResource = from(getter);
271
+
272
+ expect(solidResource).toBeInstanceOf(SolidResource);
273
+
274
+ await vi.waitFor(() => expect(solidResource.get()).toBe(42));
275
+ dispose();
276
+ });
277
+ });
278
+
279
+ it("should track dependencies in the getter", async () => {
280
+ await createRoot(async (dispose) => {
281
+ const $state = state(5);
282
+ const getter = (t: FlowTracker) => $state.get(t) * 2;
283
+ const solidResource = from(getter);
284
+
285
+ await vi.waitFor(() => expect(solidResource.get()).toBe(10));
286
+
287
+ $state.set(6);
288
+ await vi.waitFor(() => expect(solidResource.get()).toBe(12));
289
+ dispose();
290
+ });
291
+ });
292
+
293
+ it("should create a FlowNodeAsync from the getter", async () => {
294
+ await createRoot(async (dispose) => {
295
+ const $stateAsync = stateAsync(Promise.resolve(42));
296
+ const getter = async (t: FlowTracker) => await $stateAsync.get(t);
297
+ const solidResource = from(getter);
298
+
299
+ await vi.waitFor(() => expect(solidResource.get()).toBe(42));
300
+ dispose();
301
+ });
302
+ });
303
+ });
304
+
305
+ describe("reactivity", () => {
306
+ it("should update SolidResource when FlowState changes", async () => {
307
+ await createRoot(async (dispose) => {
308
+ const $state = state(0);
309
+ const solidResource = from($state);
310
+
311
+ // Wait for initial load
312
+ await vi.waitFor(() => expect(solidResource.get()).toBe(0));
313
+ const fn = vi.fn();
314
+
315
+ createEffect(() => {
316
+ fn(solidResource.get());
317
+ });
318
+
319
+ // Wait for effect to run
320
+ await vi.waitFor(() => {
321
+ expect(fn).toHaveBeenCalledTimes(1);
322
+ });
323
+ expect(fn).toHaveBeenLastCalledWith(0);
324
+
325
+ // Update FlowState
326
+ $state.set(1);
327
+
328
+ // Wait for update
329
+ await vi.waitFor(() => {
330
+ expect(fn).toHaveBeenCalledTimes(2);
331
+ });
332
+ expect(fn).toHaveBeenLastCalledWith(1);
333
+ dispose();
334
+ });
335
+ });
336
+
337
+ it("should update SolidResource when FlowDerivation changes", async () => {
338
+ await createRoot(async (dispose) => {
339
+ const $state = state(5);
340
+ const $derivation = derivation((t) => $state.get(t) * 2);
341
+ const solidResource = from($derivation);
342
+
343
+ // Wait for initial load
344
+ await vi.waitFor(() => expect(solidResource.get()).toBe(10));
345
+
346
+ const fn = vi.fn();
347
+ createEffect(() => {
348
+ fn(solidResource.get());
349
+ });
350
+
351
+ // Wait for effect to run
352
+ await vi.waitFor(() => expect(fn).toHaveBeenCalledTimes(1));
353
+ expect(fn).toHaveBeenLastCalledWith(10);
354
+
355
+ // Update dependency
356
+ $state.set(6);
357
+
358
+ // Wait for update
359
+ await vi.waitFor(() => {
360
+ expect(fn).toHaveBeenCalledTimes(2);
361
+ });
362
+ expect(fn).toHaveBeenLastCalledWith(12);
363
+ dispose();
364
+ });
365
+ });
366
+
367
+ it("should update SolidResource when FlowStateAsync changes", async () => {
368
+ await createRoot(async (dispose) => {
369
+ const $stateAsync = stateAsync(Promise.resolve(0));
370
+ const solidResource = from($stateAsync);
371
+
372
+ // Wait for initial load
373
+ await vi.waitFor(() => expect(solidResource.get()).toBe(0));
374
+
375
+ const fn = vi.fn();
376
+ createEffect(() => {
377
+ fn(solidResource.get());
378
+ });
379
+
380
+ // Wait for effect to run
381
+ await vi.waitFor(() => expect(fn).toHaveBeenCalledTimes(1));
382
+ expect(fn).toHaveBeenLastCalledWith(0);
383
+
384
+ // Update FlowStateAsync
385
+ await $stateAsync.set(Promise.resolve(1));
386
+
387
+ // Wait for update
388
+ await vi.waitFor(() => {
389
+ expect(fn).toHaveBeenCalledTimes(2);
390
+ });
391
+ expect(fn).toHaveBeenLastCalledWith(1);
392
+
393
+ dispose();
394
+ });
395
+ });
396
+
397
+ it("should update SolidResource when FlowDerivationAsync changes", async () => {
398
+ await createRoot(async (dispose) => {
399
+ const $stateAsync = stateAsync(Promise.resolve(5));
400
+ const $derivationAsync = derivationAsync(
401
+ async (t) => (await $stateAsync.get(t)) * 2,
402
+ );
403
+ const solidResource = from($derivationAsync);
404
+
405
+ // Wait for initial load
406
+ await vi.waitFor(() => expect(solidResource.get()).toBe(10));
407
+
408
+ const fn = vi.fn();
409
+ createEffect(() => {
410
+ fn(solidResource.get());
411
+ });
412
+
413
+ // Wait for effect to run
414
+ await vi.waitFor(() => expect(fn).toHaveBeenCalledTimes(1));
415
+ expect(fn).toHaveBeenLastCalledWith(10);
416
+
417
+ // Update async dependency
418
+ await $stateAsync.set(Promise.resolve(6));
419
+
420
+ // Wait for update
421
+ await vi.waitFor(() => {
422
+ expect(fn).toHaveBeenCalledTimes(2);
423
+ });
424
+ expect(fn).toHaveBeenLastCalledWith(12);
425
+ dispose();
426
+ });
427
+ });
428
+
429
+ it("should update SolidResource when getter dependency changes", async () => {
430
+ await createRoot(async (dispose) => {
431
+ const $state = state(5);
432
+ const getter = (t: FlowTracker) => $state.get(t) * 2;
433
+ const solidResource = from(getter);
434
+
435
+ // Wait for initial load
436
+ await vi.waitFor(() => expect(solidResource.get()).toBe(10));
437
+
438
+ const fn = vi.fn();
439
+ createEffect(() => {
440
+ fn(solidResource.get());
441
+ });
442
+
443
+ // Wait for effect to run
444
+ await vi.waitFor(() => expect(fn).toHaveBeenCalledTimes(1));
445
+ expect(fn).toHaveBeenLastCalledWith(10);
446
+
447
+ // Update the tracked FlowNode
448
+ $state.set(6);
449
+
450
+ // Wait for update
451
+ await vi.waitFor(() => {
452
+ expect(fn).toHaveBeenCalledTimes(2);
453
+ });
454
+ expect(fn).toHaveBeenLastCalledWith(12);
455
+
456
+ dispose();
457
+ });
458
+ });
459
+
460
+ it("should propagate multiple sequential changes", async () => {
461
+ await createRoot(async (dispose) => {
462
+ const $state = state(0);
463
+ const solidResource = from($state);
464
+
465
+ // Wait for initial load
466
+ await vi.waitFor(() => expect(solidResource.get()).toBe(0));
467
+
468
+ const fn = vi.fn();
469
+ createEffect(() => {
470
+ fn(solidResource.get());
471
+ });
472
+
473
+ // Wait for initial effect
474
+ await vi.waitFor(() => expect(fn).toHaveBeenCalledTimes(1));
475
+ expect(fn).toHaveBeenLastCalledWith(0);
476
+
477
+ // Make multiple changes
478
+ $state.set(1);
479
+ await vi.waitFor(() => expect(solidResource.get()).toBe(1));
480
+
481
+ $state.set(2);
482
+ await vi.waitFor(() => expect(solidResource.get()).toBe(2));
483
+
484
+ $state.set(3);
485
+ await vi.waitFor(() => {
486
+ expect(solidResource.get()).toBe(3);
487
+ });
488
+ dispose();
489
+ });
490
+ });
491
+
492
+ it("should be reactive within createRoot", async () => {
493
+ await createRoot(async (dispose) => {
494
+ const $state = state(0);
495
+ const solidResource = from($state);
496
+
497
+ // Wait for initial load
498
+ await vi.waitFor(() => expect(solidResource.get()).toBe(0));
499
+
500
+ $state.set(1);
501
+ await vi.waitFor(() => expect(solidResource.get()).toBe(1));
502
+
503
+ $state.set(2);
504
+ await vi.waitFor(() => expect(solidResource.get()).toBe(2));
505
+ dispose();
506
+ });
507
+ });
508
+
509
+ it("should update SolidResource.get() value when FlowNode changes", async () => {
510
+ await createRoot(async (dispose) => {
511
+ const $state = state(0);
512
+ const solidResource = from($state);
513
+
514
+ // Wait for initial load
515
+ await vi.waitFor(() => expect(solidResource.get()).toBe(0));
516
+
517
+ $state.set(1);
518
+ await vi.waitFor(() => expect(solidResource.get()).toBe(1));
519
+
520
+ // Test async updates for FlowNodeAsync
521
+ const $stateAsync = stateAsync(Promise.resolve(10));
522
+ const solidResourceAsync = from($stateAsync);
523
+
524
+ await vi.waitFor(() => expect(solidResourceAsync.get()).toBe(10));
525
+
526
+ await $stateAsync.set(Promise.resolve(20));
527
+ await vi.waitFor(() => expect(solidResourceAsync.get()).toBe(20));
528
+ dispose();
529
+ });
530
+ });
531
+
532
+ it("should trigger createEffect when FlowNode value changes", async () => {
533
+ await createRoot(async (dispose) => {
534
+ const $state = state(0);
535
+ const solidResource = from($state);
536
+
537
+ // Wait for initial load
538
+ await vi.waitFor(() => expect(solidResource.get()).toBe(0));
539
+
540
+ const fn = vi.fn();
541
+ createEffect(() => {
542
+ fn(solidResource.get());
543
+ });
544
+
545
+ // Wait for initial effect
546
+ await vi.waitFor(() => expect(fn).toHaveBeenCalledTimes(1));
547
+ expect(fn).toHaveBeenLastCalledWith(0);
548
+
549
+ // Change FlowNode
550
+ $state.set(1);
551
+ await vi.waitFor(() => expect(fn).toHaveBeenCalledTimes(2));
552
+ expect(fn).toHaveBeenLastCalledWith(1);
553
+
554
+ // Test with async FlowNode
555
+ const $stateAsync = stateAsync(Promise.resolve(10));
556
+ const solidResourceAsync = from($stateAsync);
557
+
558
+ await vi.waitFor(() => expect(solidResourceAsync.get()).toBe(10));
559
+
560
+ const fn2 = vi.fn();
561
+ createEffect(() => {
562
+ fn2(solidResourceAsync.get());
563
+ });
564
+
565
+ await vi.waitFor(() => expect(fn2).toHaveBeenCalledTimes(1));
566
+ expect(fn2).toHaveBeenLastCalledWith(10);
567
+
568
+ await $stateAsync.set(Promise.resolve(20));
569
+ await vi.waitFor(() => expect(fn2).toHaveBeenCalledTimes(2));
570
+ expect(fn2).toHaveBeenLastCalledWith(20);
571
+ dispose();
572
+ });
573
+ });
574
+ });
575
+
576
+ describe("cleanup", () => {
577
+ it("should dispose FlowEffect on cleanup", async () => {
578
+ await createRoot(async (dispose) => {
579
+ const $state = state(0);
580
+ const solidResource = from($state);
581
+
582
+ // Wait for initial load
583
+ await vi.waitFor(() => expect(solidResource.get()).toBe(0));
584
+
585
+ const fn = vi.fn();
586
+ createEffect(() => {
587
+ fn(solidResource.get());
588
+ });
589
+
590
+ await vi.waitFor(() => expect(fn).toHaveBeenCalledTimes(1));
591
+ expect(fn).toHaveBeenLastCalledWith(0);
592
+
593
+ // Dispose
594
+ dispose();
595
+
596
+ // Try to update - should not trigger effects after dispose
597
+ $state.set(1);
598
+ await new Promise((resolve) => setTimeout(resolve, 20));
599
+
600
+ // Effect count should not have increased after dispose
601
+ expect(fn).toHaveBeenCalledTimes(1);
602
+ });
603
+ });
604
+
605
+ it("should clean up subscriptions on unmount", async () => {
606
+ await createRoot(async (dispose) => {
607
+ const $state = state(0);
608
+ const solidResource = from($state);
609
+
610
+ // Wait for initial load
611
+ await vi.waitFor(() => expect(solidResource.get()).toBe(0));
612
+
613
+ const fn = vi.fn();
614
+ createEffect(() => {
615
+ fn(solidResource.get());
616
+ });
617
+
618
+ await vi.waitFor(() => expect(fn).toHaveBeenCalledTimes(1));
619
+ expect(fn).toHaveBeenLastCalledWith(0);
620
+
621
+ // Dispose (unmount)
622
+ dispose();
623
+
624
+ // Update should not trigger after unmount
625
+ $state.set(1);
626
+ await new Promise((resolve) => setTimeout(resolve, 20));
627
+
628
+ expect(fn).toHaveBeenCalledTimes(1);
629
+ });
630
+ });
631
+
632
+ it("should not update after dispose", async () => {
633
+ await createRoot(async (dispose) => {
634
+ const $state = state(0);
635
+ const solidResource = from($state);
636
+
637
+ // Wait for initial load
638
+ await vi.waitFor(() => expect(solidResource.get()).toBe(0));
639
+
640
+ const initialValue = solidResource.get();
641
+ expect(initialValue).toBe(0);
642
+
643
+ // Dispose
644
+ dispose();
645
+
646
+ // Update FlowNode
647
+ $state.set(1);
648
+
649
+ // Wait a bit
650
+ await new Promise((resolve) => setTimeout(resolve, 20));
651
+
652
+ // Value should not have changed after dispose
653
+ // Note: The resource might still have the old value cached, but it shouldn't update
654
+ // This test verifies that updates stop after dispose
655
+ expect(solidResource.get()).toBe(initialValue);
656
+ });
657
+ });
658
+ });
659
+
660
+ describe("integration with SolidJS", () => {
661
+ it("should work with createEffect", async () => {
662
+ await createRoot(async (dispose) => {
663
+ const $state = state(0);
664
+ const solidResource = from($state);
665
+
666
+ // Wait for initial load
667
+ await vi.waitFor(() => expect(solidResource.get()).toBe(0));
668
+
669
+ const values: number[] = [];
670
+ createEffect(() => {
671
+ values.push(solidResource.get() ?? 0);
672
+ });
673
+
674
+ await vi.waitFor(() => {
675
+ expect(values.length).toBeGreaterThan(0);
676
+ expect(values[0]).toBe(0);
677
+ });
678
+
679
+ $state.set(1);
680
+ await vi.waitFor(() => {
681
+ expect(solidResource.get()).toBe(1);
682
+ expect(values.length).toBeGreaterThan(1);
683
+ });
684
+ dispose();
685
+ });
686
+ });
687
+
688
+ it("should handle multiple simultaneous conversions", async () => {
689
+ await createRoot(async (dispose) => {
690
+ const $state1 = state(1);
691
+ const $state2 = state(2);
692
+ const $state3 = state(3);
693
+
694
+ const solidResource1 = from($state1);
695
+ const solidResource2 = from($state2);
696
+ const solidResource3 = from($state3);
697
+
698
+ // Wait for all to load
699
+ await Promise.all([
700
+ vi.waitFor(() => expect(solidResource1.get()).toBe(1)),
701
+ vi.waitFor(() => expect(solidResource2.get()).toBe(2)),
702
+ vi.waitFor(() => expect(solidResource3.get()).toBe(3)),
703
+ ]);
704
+
705
+ // Update all
706
+ $state1.set(10);
707
+ $state2.set(20);
708
+ $state3.set(30);
709
+
710
+ await Promise.all([
711
+ vi.waitFor(() => expect(solidResource1.get()).toBe(10)),
712
+ vi.waitFor(() => expect(solidResource2.get()).toBe(20)),
713
+ vi.waitFor(() => expect(solidResource3.get()).toBe(30)),
714
+ ]);
715
+ dispose();
716
+ });
717
+ });
718
+ });
719
+ });
720
+ });
721
+ });