@fncts/io 0.0.49 → 0.0.50

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 (226) hide show
  1. package/IO/api/foreachExec.d.ts +18 -0
  2. package/Push/Bounds.d.ts +4 -0
  3. package/Push/FlattenStrategy.d.ts +47 -0
  4. package/Push/IndexedBuffer.d.ts +19 -0
  5. package/Push/MergeStrategy.d.ts +19 -0
  6. package/Push/Operator/IOLoopOperator.d.ts +61 -0
  7. package/Push/Operator/IOOperator.d.ts +63 -0
  8. package/Push/Operator/LoopOperator.d.ts +50 -0
  9. package/Push/Operator/SyncOperator.d.ts +41 -0
  10. package/Push/Operator.d.ts +4 -0
  11. package/Push/Producer/IOProducer.d.ts +41 -0
  12. package/Push/Producer/SyncProducer.d.ts +61 -0
  13. package/Push/Producer.d.ts +1 -0
  14. package/Push/Sink.d.ts +65 -0
  15. package/Push/api.d.ts +78 -161
  16. package/Push/definition.d.ts +92 -19
  17. package/Push/internal.d.ts +3 -0
  18. package/Push.d.ts +6 -0
  19. package/Ref/Synchronized/definition.d.ts +32 -9
  20. package/Ref/definition.d.ts +12 -0
  21. package/Scope/api.d.ts +16 -1
  22. package/Sink/definition.d.ts +5 -6
  23. package/Subject/Atomic.d.ts +19 -9
  24. package/Subject/DeferredRef.d.ts +17 -0
  25. package/Subject/Hold.d.ts +19 -0
  26. package/Subject/RefSubject/Atomic.d.ts +41 -0
  27. package/Subject/RefSubject/Derived.d.ts +43 -0
  28. package/Subject/RefSubject/RefSubject.d.ts +27 -0
  29. package/Subject/definition.d.ts +14 -2
  30. package/Subject.d.ts +2 -0
  31. package/SubscriptionRef.d.ts +0 -1
  32. package/_cjs/IO/api/foreachExec.cjs +29 -4
  33. package/_cjs/IO/api/foreachExec.cjs.map +1 -1
  34. package/_cjs/Push/Bounds.cjs +2 -0
  35. package/_cjs/Push/Bounds.cjs.map +1 -0
  36. package/_cjs/Push/FlattenStrategy.cjs +95 -0
  37. package/_cjs/Push/FlattenStrategy.cjs.map +1 -0
  38. package/_cjs/Push/IndexedBuffer.cjs +70 -0
  39. package/_cjs/Push/IndexedBuffer.cjs.map +1 -0
  40. package/_cjs/Push/MergeStrategy.cjs +45 -0
  41. package/_cjs/Push/MergeStrategy.cjs.map +1 -0
  42. package/_cjs/Push/Operator/IOLoopOperator.cjs +225 -0
  43. package/_cjs/Push/Operator/IOLoopOperator.cjs.map +1 -0
  44. package/_cjs/Push/Operator/IOOperator.cjs +104 -0
  45. package/_cjs/Push/Operator/IOOperator.cjs.map +1 -0
  46. package/_cjs/Push/Operator/LoopOperator.cjs +165 -0
  47. package/_cjs/Push/Operator/LoopOperator.cjs.map +1 -0
  48. package/_cjs/Push/Operator/SyncOperator.cjs +78 -0
  49. package/_cjs/Push/Operator/SyncOperator.cjs.map +1 -0
  50. package/_cjs/Push/Operator.cjs +2 -0
  51. package/_cjs/Push/Operator.cjs.map +1 -0
  52. package/_cjs/Push/Producer/IOProducer.cjs +67 -0
  53. package/_cjs/Push/Producer/IOProducer.cjs.map +1 -0
  54. package/_cjs/Push/Producer/SyncProducer.cjs +107 -0
  55. package/_cjs/Push/Producer/SyncProducer.cjs.map +1 -0
  56. package/_cjs/Push/Producer.cjs +2 -0
  57. package/_cjs/Push/Producer.cjs.map +1 -0
  58. package/_cjs/Push/Sink.cjs +219 -0
  59. package/_cjs/Push/Sink.cjs.map +1 -0
  60. package/_cjs/Push/api.cjs +339 -383
  61. package/_cjs/Push/api.cjs.map +1 -1
  62. package/_cjs/Push/definition.cjs +79 -22
  63. package/_cjs/Push/definition.cjs.map +1 -1
  64. package/_cjs/Push/internal.cjs +31 -26
  65. package/_cjs/Push/internal.cjs.map +1 -1
  66. package/_cjs/Push.cjs +66 -0
  67. package/_cjs/Push.cjs.map +1 -1
  68. package/_cjs/Ref/Synchronized/definition.cjs +12 -12
  69. package/_cjs/Ref/Synchronized/definition.cjs.map +1 -1
  70. package/_cjs/Ref/definition.cjs.map +1 -1
  71. package/_cjs/Scope/api.cjs +28 -7
  72. package/_cjs/Scope/api.cjs.map +1 -1
  73. package/_cjs/Sink/definition.cjs +2 -3
  74. package/_cjs/Sink/definition.cjs.map +1 -1
  75. package/_cjs/Stream/definition.cjs.map +1 -1
  76. package/_cjs/Subject/Atomic.cjs +45 -10
  77. package/_cjs/Subject/Atomic.cjs.map +1 -1
  78. package/_cjs/Subject/DeferredRef.cjs +53 -0
  79. package/_cjs/Subject/DeferredRef.cjs.map +1 -0
  80. package/_cjs/Subject/Hold.cjs +41 -0
  81. package/_cjs/Subject/Hold.cjs.map +1 -0
  82. package/_cjs/Subject/RefSubject/Atomic.cjs +130 -0
  83. package/_cjs/Subject/RefSubject/Atomic.cjs.map +1 -0
  84. package/_cjs/Subject/RefSubject/Derived.cjs +98 -0
  85. package/_cjs/Subject/RefSubject/Derived.cjs.map +1 -0
  86. package/_cjs/Subject/RefSubject/RefSubject.cjs +19 -0
  87. package/_cjs/Subject/RefSubject/RefSubject.cjs.map +1 -0
  88. package/_cjs/Subject/definition.cjs +16 -1
  89. package/_cjs/Subject/definition.cjs.map +1 -1
  90. package/_cjs/Subject.cjs +22 -0
  91. package/_cjs/Subject.cjs.map +1 -1
  92. package/_cjs/SubscriptionRef.cjs +0 -1
  93. package/_cjs/SubscriptionRef.cjs.map +1 -1
  94. package/_mjs/IO/api/foreachExec.mjs +27 -4
  95. package/_mjs/IO/api/foreachExec.mjs.map +1 -1
  96. package/_mjs/Push/Bounds.mjs +2 -0
  97. package/_mjs/Push/Bounds.mjs.map +1 -0
  98. package/_mjs/Push/FlattenStrategy.mjs +86 -0
  99. package/_mjs/Push/FlattenStrategy.mjs.map +1 -0
  100. package/_mjs/Push/IndexedBuffer.mjs +61 -0
  101. package/_mjs/Push/IndexedBuffer.mjs.map +1 -0
  102. package/_mjs/Push/MergeStrategy.mjs +34 -0
  103. package/_mjs/Push/MergeStrategy.mjs.map +1 -0
  104. package/_mjs/Push/Operator/IOLoopOperator.mjs +208 -0
  105. package/_mjs/Push/Operator/IOLoopOperator.mjs.map +1 -0
  106. package/_mjs/Push/Operator/IOOperator.mjs +91 -0
  107. package/_mjs/Push/Operator/IOOperator.mjs.map +1 -0
  108. package/_mjs/Push/Operator/LoopOperator.mjs +151 -0
  109. package/_mjs/Push/Operator/LoopOperator.mjs.map +1 -0
  110. package/_mjs/Push/Operator/SyncOperator.mjs +67 -0
  111. package/_mjs/Push/Operator/SyncOperator.mjs.map +1 -0
  112. package/_mjs/Push/Operator.mjs +2 -0
  113. package/_mjs/Push/Operator.mjs.map +1 -0
  114. package/_mjs/Push/Producer/IOProducer.mjs +55 -0
  115. package/_mjs/Push/Producer/IOProducer.mjs.map +1 -0
  116. package/_mjs/Push/Producer/SyncProducer.mjs +90 -0
  117. package/_mjs/Push/Producer/SyncProducer.mjs.map +1 -0
  118. package/_mjs/Push/Producer.mjs +2 -0
  119. package/_mjs/Push/Producer.mjs.map +1 -0
  120. package/_mjs/Push/Sink.mjs +206 -0
  121. package/_mjs/Push/Sink.mjs.map +1 -0
  122. package/_mjs/Push/api.mjs +311 -344
  123. package/_mjs/Push/api.mjs.map +1 -1
  124. package/_mjs/Push/definition.mjs +73 -18
  125. package/_mjs/Push/definition.mjs.map +1 -1
  126. package/_mjs/Push/internal.mjs +30 -26
  127. package/_mjs/Push/internal.mjs.map +1 -1
  128. package/_mjs/Push.mjs +7 -1
  129. package/_mjs/Push.mjs.map +1 -1
  130. package/_mjs/Ref/Synchronized/definition.mjs +12 -12
  131. package/_mjs/Ref/Synchronized/definition.mjs.map +1 -1
  132. package/_mjs/Ref/definition.mjs.map +1 -1
  133. package/_mjs/Scope/api.mjs +22 -4
  134. package/_mjs/Scope/api.mjs.map +1 -1
  135. package/_mjs/Sink/definition.mjs +2 -3
  136. package/_mjs/Sink/definition.mjs.map +1 -1
  137. package/_mjs/Stream/definition.mjs.map +1 -1
  138. package/_mjs/Subject/Atomic.mjs +46 -10
  139. package/_mjs/Subject/Atomic.mjs.map +1 -1
  140. package/_mjs/Subject/DeferredRef.mjs +45 -0
  141. package/_mjs/Subject/DeferredRef.mjs.map +1 -0
  142. package/_mjs/Subject/Hold.mjs +33 -0
  143. package/_mjs/Subject/Hold.mjs.map +1 -0
  144. package/_mjs/Subject/RefSubject/Atomic.mjs +122 -0
  145. package/_mjs/Subject/RefSubject/Atomic.mjs.map +1 -0
  146. package/_mjs/Subject/RefSubject/Derived.mjs +90 -0
  147. package/_mjs/Subject/RefSubject/Derived.mjs.map +1 -0
  148. package/_mjs/Subject/RefSubject/RefSubject.mjs +12 -0
  149. package/_mjs/Subject/RefSubject/RefSubject.mjs.map +1 -0
  150. package/_mjs/Subject/definition.mjs +14 -0
  151. package/_mjs/Subject/definition.mjs.map +1 -1
  152. package/_mjs/Subject.mjs +2 -0
  153. package/_mjs/Subject.mjs.map +1 -1
  154. package/_mjs/SubscriptionRef.mjs +0 -1
  155. package/_mjs/SubscriptionRef.mjs.map +1 -1
  156. package/_src/IO/api/foreachExec.ts +47 -0
  157. package/_src/Push/Bounds.ts +4 -0
  158. package/_src/Push/FlattenStrategy.ts +137 -0
  159. package/_src/Push/IndexedBuffer.ts +79 -0
  160. package/_src/Push/MergeStrategy.ts +59 -0
  161. package/_src/Push/Operator/IOLoopOperator.ts +413 -0
  162. package/_src/Push/Operator/IOOperator.ts +173 -0
  163. package/_src/Push/Operator/LoopOperator.ts +242 -0
  164. package/_src/Push/Operator/SyncOperator.ts +107 -0
  165. package/_src/Push/Operator.ts +7 -0
  166. package/_src/Push/Producer/IOProducer.ts +83 -0
  167. package/_src/Push/Producer/SyncProducer.ts +105 -0
  168. package/_src/Push/Producer.ts +0 -0
  169. package/_src/Push/Sink.ts +302 -0
  170. package/_src/Push/api.ts +387 -513
  171. package/_src/Push/definition.ts +216 -23
  172. package/_src/Push/internal.ts +11 -0
  173. package/_src/Push.ts +7 -1
  174. package/_src/Ref/Synchronized/definition.ts +81 -37
  175. package/_src/Ref/definition.ts +26 -0
  176. package/_src/Scope/api.ts +22 -0
  177. package/_src/Sink/definition.ts +4 -5
  178. package/_src/Stream/definition.ts +0 -1
  179. package/_src/Subject/Atomic.ts +68 -11
  180. package/_src/Subject/DeferredRef.ts +44 -0
  181. package/_src/Subject/Hold.ts +48 -0
  182. package/_src/Subject/RefSubject/Atomic.ts +193 -0
  183. package/_src/Subject/RefSubject/Derived.ts +179 -0
  184. package/_src/Subject/RefSubject/RefSubject.ts +90 -0
  185. package/_src/Subject/definition.ts +21 -3
  186. package/_src/Subject.ts +5 -3
  187. package/_src/SubscriptionRef.ts +1 -1
  188. package/_src/index.ts +49 -50
  189. package/index.d.ts +0 -1
  190. package/package.json +4 -4
  191. package/RefSubject/Atomic.d.ts +0 -34
  192. package/RefSubject/Synchronized/api.d.ts +0 -9
  193. package/RefSubject/Synchronized/definition.d.ts +0 -39
  194. package/RefSubject/api.d.ts +0 -120
  195. package/RefSubject/definition.d.ts +0 -52
  196. package/RefSubject.d.ts +0 -3
  197. package/_cjs/RefSubject/Atomic.cjs +0 -107
  198. package/_cjs/RefSubject/Atomic.cjs.map +0 -1
  199. package/_cjs/RefSubject/Synchronized/api.cjs +0 -22
  200. package/_cjs/RefSubject/Synchronized/api.cjs.map +0 -1
  201. package/_cjs/RefSubject/Synchronized/definition.cjs +0 -55
  202. package/_cjs/RefSubject/Synchronized/definition.cjs.map +0 -1
  203. package/_cjs/RefSubject/api.cjs +0 -251
  204. package/_cjs/RefSubject/api.cjs.map +0 -1
  205. package/_cjs/RefSubject/definition.cjs +0 -26
  206. package/_cjs/RefSubject/definition.cjs.map +0 -1
  207. package/_cjs/RefSubject.cjs +0 -39
  208. package/_cjs/RefSubject.cjs.map +0 -1
  209. package/_mjs/RefSubject/Atomic.mjs +0 -99
  210. package/_mjs/RefSubject/Atomic.mjs.map +0 -1
  211. package/_mjs/RefSubject/Synchronized/api.mjs +0 -15
  212. package/_mjs/RefSubject/Synchronized/api.mjs.map +0 -1
  213. package/_mjs/RefSubject/Synchronized/definition.mjs +0 -47
  214. package/_mjs/RefSubject/Synchronized/definition.mjs.map +0 -1
  215. package/_mjs/RefSubject/api.mjs +0 -229
  216. package/_mjs/RefSubject/api.mjs.map +0 -1
  217. package/_mjs/RefSubject/definition.mjs +0 -19
  218. package/_mjs/RefSubject/definition.mjs.map +0 -1
  219. package/_mjs/RefSubject.mjs +0 -5
  220. package/_mjs/RefSubject.mjs.map +0 -1
  221. package/_src/RefSubject/Atomic.ts +0 -129
  222. package/_src/RefSubject/Synchronized/api.ts +0 -14
  223. package/_src/RefSubject/Synchronized/definition.ts +0 -76
  224. package/_src/RefSubject/api.ts +0 -253
  225. package/_src/RefSubject/definition.ts +0 -70
  226. package/_src/RefSubject.ts +0 -5
@@ -0,0 +1,137 @@
1
+ import { withScope } from "@fncts/io/Push/internal";
2
+
3
+ type ScopedFork = <R, E, A>(io: IO<R, E, A>) => IO<R | Scope, never, Fiber.Runtime<E, A>>;
4
+
5
+ type Fork = <R>(io: IO<R, never, void>) => IO<R | Scope, never, void>;
6
+
7
+ /**
8
+ * @tsplus type fncts.io.Push.FlattenStrategy
9
+ * @tsplus companion fncts.io.Push.FlattenStrategyOps
10
+ */
11
+ export abstract class FlattenStrategy {
12
+ abstract withFork<R>(
13
+ f: (fork: Fork, scope: Scope) => IO<R, never, void>,
14
+ executionStrategy: ExecutionStrategy,
15
+ ): IO<R | Scope, never, void>;
16
+ }
17
+
18
+ function withScopedFork<R, E, A>(
19
+ f: (fork: ScopedFork, scope: Scope.Closeable) => IO<R, E, A>,
20
+ executionStrategy: ExecutionStrategy,
21
+ ): IO<R | Scope, E, A> {
22
+ return withScope((scope) => f((io) => io.forkIn(scope), scope), executionStrategy);
23
+ }
24
+
25
+ /**
26
+ * @tsplus static fncts.io.Push.FlattenStrategyOps Switch
27
+ */
28
+ export const SwitchStrategy = new (class SwitchStrategy extends FlattenStrategy {
29
+ withFork<R>(
30
+ f: (fork: Fork, scope: Scope) => IO<R, never, void>,
31
+ executionStrategy: ExecutionStrategy,
32
+ ): IO<R | Scope, never, void> {
33
+ return withScopedFork(
34
+ (fork, scope) =>
35
+ Ref.Synchronized.make<Fiber<never, void>>(Fiber.unit).flatMap(
36
+ (ref) =>
37
+ f((io) => ref.updateIO((fiber) => fiber.interrupt > fork(io)), scope) >
38
+ ref.get.flatMap((fiber) => fiber.join),
39
+ ),
40
+ executionStrategy,
41
+ );
42
+ }
43
+ })();
44
+
45
+ /**
46
+ * @tsplus static fncts.io.Push.FlattenStrategyOps Exhaust
47
+ */
48
+ export const ExhaustStrategy = new (class ExhaustStrategy extends FlattenStrategy {
49
+ withFork<R>(
50
+ f: (fork: Fork, scope: Scope) => IO<R, never, void>,
51
+ executionStrategy: ExecutionStrategy,
52
+ ): IO<R | Scope, never, void> {
53
+ return withScopedFork(
54
+ (fork, scope) =>
55
+ Ref.Synchronized.make<Fiber.Runtime<never, void> | null>(null).flatMap(
56
+ (ref) =>
57
+ f(
58
+ (io) =>
59
+ ref.updateIO((fiber) => {
60
+ if (fiber) {
61
+ return IO.succeedNow(fiber);
62
+ } else {
63
+ return fork(io.onExit(() => ref.set(null)));
64
+ }
65
+ }),
66
+ scope,
67
+ ) >
68
+ ref.get.flatMap((fiber) => {
69
+ if (fiber) {
70
+ return fiber.join;
71
+ } else {
72
+ return IO.unit;
73
+ }
74
+ }),
75
+ ),
76
+ executionStrategy,
77
+ );
78
+ }
79
+ })();
80
+
81
+ /**
82
+ * @tsplus static fncts.io.Push.FlattenStrategyOps Unbounded
83
+ */
84
+ export const UnboundedStrategy = new (class UnboundedStrategy extends FlattenStrategy {
85
+ withFork<R>(f: (fork: Fork, scope: Scope) => IO<R, never, void>): IO<R | Scope, never, void> {
86
+ return IO.scopeWith((scope) =>
87
+ FiberSet.make<never, void>().flatMap((fiberSet) => f((io) => fiberSet.run(io), scope) > Fiber.joinAll(fiberSet)),
88
+ );
89
+ }
90
+ })();
91
+
92
+ /**
93
+ * @tsplus static fncts.io.Push.FlattenStrategyOps Bounded
94
+ */
95
+ export function makeBounded(capacity: number): BoundedStrategy {
96
+ return new BoundedStrategy(capacity);
97
+ }
98
+
99
+ class BoundedStrategy extends FlattenStrategy {
100
+ constructor(readonly capacity: number) {
101
+ super();
102
+ }
103
+
104
+ withFork<R>(
105
+ f: (fork: Fork, scope: Scope) => IO<R, never, void>,
106
+ executionStrategy: ExecutionStrategy,
107
+ ): IO<R | Scope, never, void> {
108
+ return withScopedFork(
109
+ (fork, scope) =>
110
+ Do((Δ) => {
111
+ const ref = Δ(Ref.make<HashSet<Fiber.Runtime<never, void>>>(HashSet.empty()));
112
+ const semaphore = Δ(Semaphore.make(this.capacity));
113
+ Δ(
114
+ f(
115
+ (io) =>
116
+ Do((Δ) => {
117
+ const fiber = Δ(fork(semaphore.withPermit(io)));
118
+ Δ(ref.update((set) => set.add(fiber)));
119
+ Δ(scope.addFinalizer(ref.update((set) => set.remove(fiber))));
120
+ }),
121
+ scope,
122
+ ),
123
+ );
124
+ return ref;
125
+ }).flatMap((ref) =>
126
+ ref.get.flatMap((fibers) => {
127
+ if (fibers.size > 0) {
128
+ return Fiber.joinAll(fibers);
129
+ } else {
130
+ return IO.unit;
131
+ }
132
+ }),
133
+ ),
134
+ executionStrategy,
135
+ );
136
+ }
137
+ }
@@ -0,0 +1,79 @@
1
+ import type { Push } from "./definition.js";
2
+
3
+ export class IndexedBuffer<R, E, A> {
4
+ constructor(
5
+ public ready: boolean,
6
+ readonly future: Future<never, void>,
7
+ readonly sink: Push.UnsafeSink<R, E, A>,
8
+ readonly onDone: UIO<void>,
9
+ ) {}
10
+
11
+ buffer: Array<A> = [];
12
+
13
+ onSuccess(value: A) {
14
+ if (this.ready) {
15
+ if (this.buffer.length === 0) {
16
+ return this.sink.onSuccess(value);
17
+ }
18
+
19
+ this.buffer.push(value);
20
+ const b = this.buffer;
21
+ this.buffer = [];
22
+ return IO.foreach(b, this.sink.onSuccess);
23
+ } else {
24
+ this.buffer.push(value);
25
+ return IO.unit;
26
+ }
27
+ }
28
+
29
+ get onEnd() {
30
+ return (
31
+ this.future.await >
32
+ IO.defer(() => {
33
+ if (this.buffer.length === 0) {
34
+ return this.onDone;
35
+ }
36
+ const b = this.buffer;
37
+ this.buffer = [];
38
+ return IO.foreach(b, this.sink.onSuccess) > this.onDone;
39
+ })
40
+ );
41
+ }
42
+ }
43
+
44
+ export const withBuffers = <R, E, A>(
45
+ size: number,
46
+ sink: Push.UnsafeSink<R, E, A>,
47
+ id: FiberId,
48
+ ): {
49
+ onSuccess: (index: number, value: A) => IO<R, never, void>;
50
+ onEnd: (index: number) => IO<R, never, void>;
51
+ buffers: Map<number, IndexedBuffer<R, E, A>>;
52
+ } => {
53
+ const buffers = new Map<number, IndexedBuffer<R, E, A>>();
54
+ const last = size - 1;
55
+
56
+ for (let i = 0; i < size; i++) {
57
+ const future = Future.unsafeMake<never, void>(id);
58
+ const ready = i === 0;
59
+ if (i === 0) {
60
+ future.unsafeDone(IO.unit);
61
+ }
62
+ const onDone =
63
+ i === last
64
+ ? IO.unit
65
+ : IO.defer(() => {
66
+ const next = buffers.get(i + 1)!;
67
+ next.ready = true;
68
+ return next.future.done(Exit.unit);
69
+ });
70
+
71
+ buffers.set(i, new IndexedBuffer(ready, future, sink, onDone));
72
+ }
73
+
74
+ return {
75
+ buffers,
76
+ onSuccess: (index, value) => buffers.get(index)!.onSuccess(value),
77
+ onEnd: (index) => buffers.get(index)!.onEnd,
78
+ };
79
+ };
@@ -0,0 +1,59 @@
1
+ import type { Push } from "./definition.js";
2
+
3
+ import { withBuffers } from "./IndexedBuffer.js";
4
+ import { Sink } from "./Sink.js";
5
+
6
+ export abstract class MergeStrategy {
7
+ abstract runMerge<Ps extends ReadonlyArray<Push<any, any, any>>, R>(
8
+ ps: Ps,
9
+ sink: Push.UnsafeSink<R, any, any>,
10
+ ): IO<R | Push.EnvironmentOf<Ps[number]>, never, void>;
11
+ }
12
+
13
+ export class Unordered extends MergeStrategy {
14
+ constructor(readonly executionStrategy: ExecutionStrategy) {
15
+ super();
16
+ }
17
+
18
+ runMerge<Ps extends ReadonlyArray<Push<any, any, any>>, R>(
19
+ ps: Ps,
20
+ sink: Push.UnsafeSink<R, any, any>,
21
+ ): IO<R | Push.EnvironmentOf<Ps[number]>, never, void> {
22
+ return IO.foreachExecDiscard(ps, this.executionStrategy, (p) => p.run(sink));
23
+ }
24
+ }
25
+
26
+ export class Ordered extends MergeStrategy {
27
+ constructor(readonly executionStrategy: ExecutionStrategy) {
28
+ super();
29
+ }
30
+
31
+ runMerge<Ps extends ReadonlyArray<Push<any, any, any>>, R>(
32
+ ps: Ps,
33
+ sink: Push.UnsafeSink<R, any, any>,
34
+ ): IO<R | Push.EnvironmentOf<Ps[number]>, never, void> {
35
+ return IO.fiberIdWith((fiberId) => {
36
+ const buffers = withBuffers(ps.length, sink, fiberId);
37
+ return IO.foreachWithIndexDiscardExec(
38
+ ps,
39
+ this.executionStrategy,
40
+ (index, p) =>
41
+ p.run(
42
+ Sink.unsafeMake(
43
+ (value) => buffers.onSuccess(index, value),
44
+ (cause) => (cause.isInterruptedOnly ? buffers.onEnd(index) : sink.onFailure(cause)),
45
+ ),
46
+ ) > buffers.onEnd(index),
47
+ );
48
+ });
49
+ }
50
+ }
51
+
52
+ export class Switch extends MergeStrategy {
53
+ runMerge<Ps extends ReadonlyArray<Push<any, any, any>>, R>(
54
+ ps: Ps,
55
+ sink: Push.UnsafeSink<R, any, any>,
56
+ ): IO<R | Push.EnvironmentOf<Ps[number]>, never, void> {
57
+ return IO.foreachDiscard(ps, (p) => p.run(sink));
58
+ }
59
+ }
@@ -0,0 +1,413 @@
1
+ import type { IOOperator } from "./IOOperator.js";
2
+ import type { SyncLoopOperator } from "@fncts/io/Push/Operator/LoopOperator";
3
+ import type { SyncOperator } from "@fncts/io/Push/Operator/SyncOperator";
4
+
5
+ export const enum IOLoopOperatorTag {
6
+ LoopIO = "LoopIO",
7
+ FilterMapLoopIO = "FilterMapLoopIO",
8
+ }
9
+
10
+ /**
11
+ * @tsplus type fncts.io.Push.IOLoopOperator
12
+ */
13
+ export type IOLoopOperator<B = any, A = any, R = any, E = any, C = any> =
14
+ | LoopIOOperator<B, A, R, E, C>
15
+ | FilterMapLoopIOOperator<B, A, R, E, C>;
16
+
17
+ export interface LoopIOOperator<B, A, R, E, C> {
18
+ readonly _tag: IOLoopOperatorTag.LoopIO;
19
+ readonly seed: B;
20
+ readonly f: (acc: B, a: A) => IO<R, E, readonly [C, B]>;
21
+ }
22
+
23
+ export function LoopIOOperator<const B, A, R, E, C>(
24
+ seed: B,
25
+ f: (acc: B, a: A) => IO<R, E, readonly [C, B]>,
26
+ ): LoopIOOperator<B, A, R, E, C> {
27
+ return {
28
+ _tag: IOLoopOperatorTag.LoopIO,
29
+ seed,
30
+ f,
31
+ };
32
+ }
33
+
34
+ export interface FilterMapLoopIOOperator<B, A, R, E, C> {
35
+ readonly _tag: IOLoopOperatorTag.FilterMapLoopIO;
36
+ readonly seed: B;
37
+ readonly f: (acc: B, a: A) => IO<R, E, readonly [Maybe<C>, B]>;
38
+ }
39
+
40
+ export function FilterMapLoopIOOperator<const B, A, R, E, C>(
41
+ seed: B,
42
+ f: (acc: B, a: A) => IO<R, E, readonly [Maybe<C>, B]>,
43
+ ): FilterMapLoopIOOperator<B, A, R, E, C> {
44
+ return {
45
+ _tag: IOLoopOperatorTag.FilterMapLoopIO,
46
+ seed,
47
+ f,
48
+ };
49
+ }
50
+
51
+ /**
52
+ * @tsplus pipeable fncts.io.Push.IOLoopOperator match
53
+ */
54
+ export function match<B, A, R, E, C, D, F>(matchers: {
55
+ LoopIO: (op: LoopIOOperator<B, A, R, E, C>) => D;
56
+ FilterMapLoopIO: (op: FilterMapLoopIOOperator<B, A, R, E, C>) => F;
57
+ }) {
58
+ return (self: IOLoopOperator<B, A, R, E, C>): D | F => {
59
+ switch (self._tag) {
60
+ case IOLoopOperatorTag.LoopIO:
61
+ return matchers.LoopIO(self);
62
+ case IOLoopOperatorTag.FilterMapLoopIO:
63
+ return matchers.FilterMapLoopIO(self);
64
+ }
65
+ };
66
+ }
67
+
68
+ /**
69
+ * @tsplus pipeable fncts.io.Push.IOLoopOperator fuse
70
+ */
71
+ export function fuse<C, D, F, R2, E2>(that: IOLoopOperator<D, C, R2, E2, F>) {
72
+ return <A, B, R1, E1>(
73
+ self: IOLoopOperator<A, B, R1, E1, C>,
74
+ ): IOLoopOperator<readonly [A, D], B, R1 | R2, E1 | E2, F> =>
75
+ self.match({
76
+ LoopIO: (op1) =>
77
+ that.match({
78
+ LoopIO: (op2) =>
79
+ LoopIOOperator([op1.seed, op2.seed], ([a, d], b) =>
80
+ op1.f(a, b).flatMap(([c, a1]) => op2.f(d, c).map(([f, d1]) => [f, [a1, d1]])),
81
+ ),
82
+ FilterMapLoopIO: (op2) =>
83
+ FilterMapLoopIOOperator([op1.seed, op2.seed], ([a, d], b) =>
84
+ op1.f(a, b).flatMap(([c, a1]) => op2.f(d, c).map(([f, d1]) => [f, [a1, d1]])),
85
+ ),
86
+ }),
87
+ FilterMapLoopIO: (op1) =>
88
+ that.match({
89
+ LoopIO: (op2) =>
90
+ FilterMapLoopIOOperator([op1.seed, op2.seed], ([a, d], b) =>
91
+ op1.f(a, b).flatMap(([c, a1]) =>
92
+ c.match(
93
+ () => [Nothing(), [a1, d]] as const,
94
+ (c) => op2.f(d, c).map(([f, d1]) => [Just(f), [a1, d1]]),
95
+ ),
96
+ ),
97
+ ),
98
+ FilterMapLoopIO: (op2) =>
99
+ FilterMapLoopIOOperator([op1.seed, op2.seed], ([a, d], b) =>
100
+ op1.f(a, b).flatMap(([c, a1]) =>
101
+ c.match(
102
+ () => [Nothing(), [a1, d]] as const,
103
+ (c) => op2.f(d, c).map(([f, d1]) => [f, [a1, d1]]),
104
+ ),
105
+ ),
106
+ ),
107
+ }),
108
+ });
109
+ }
110
+
111
+ /**
112
+ * @tsplus pipeable fncts.io.Push.IOLoopOperator after
113
+ */
114
+ export function fuseSyncLoopOperatorBefore<A, B, C>(that: SyncLoopOperator<A, B, C>) {
115
+ return <D, R, E, F>(self: IOLoopOperator<D, C, R, E, F>): IOLoopOperator<readonly [A, D], B, R, E, F> =>
116
+ that.match({
117
+ Loop: (op1) =>
118
+ self.match({
119
+ LoopIO: (op2) =>
120
+ LoopIOOperator([op1.seed, op2.seed], ([a, d], b) => {
121
+ const [c, a1] = op1.f(a, b);
122
+ return op2.f(d, c).map(([f, d1]) => [f, [a1, d1]]);
123
+ }),
124
+ FilterMapLoopIO: (op2) =>
125
+ FilterMapLoopIOOperator([op1.seed, op2.seed], ([a, d], b) => {
126
+ const [c, a1] = op1.f(a, b);
127
+ return op2.f(d, c).map(([f, d1]) => [f, [a1, d1]]);
128
+ }),
129
+ }),
130
+ FilterMapLoop: (op1) =>
131
+ self.match({
132
+ LoopIO: (op2) =>
133
+ FilterMapLoopIOOperator([op1.seed, op2.seed], ([a, d], b) => {
134
+ const [c, a1] = op1.f(a, b);
135
+ return c.match(
136
+ () => IO.succeedNow([Nothing(), [a1, d]]),
137
+ (c) => op2.f(d, c).map(([f, d1]) => [Just(f), [a1, d1]]),
138
+ );
139
+ }),
140
+ FilterMapLoopIO: (op2) =>
141
+ FilterMapLoopIOOperator([op1.seed, op2.seed], ([a, d], b) => {
142
+ const [c, a1] = op1.f(a, b);
143
+ return c.match(
144
+ () => IO.succeedNow([Nothing(), [a1, d]]),
145
+ (c) => op2.f(d, c).map(([f, d1]) => [f, [a1, d1]]),
146
+ );
147
+ }),
148
+ }),
149
+ });
150
+ }
151
+
152
+ /**
153
+ * @tsplus pipeable fncts.io.Push.IOLoopOperator before
154
+ */
155
+ export function fuseSyncLoopOperatorAfter<D, C, F>(that: SyncLoopOperator<D, C, F>) {
156
+ return <A, B, R, E>(self: IOLoopOperator<A, B, R, E, C>): IOLoopOperator<readonly [A, D], B, R, E, F> =>
157
+ self.match({
158
+ LoopIO: (op1) =>
159
+ that.match({
160
+ Loop: (op2) =>
161
+ LoopIOOperator([op1.seed, op2.seed], ([a, d], b) =>
162
+ op1.f(a, b).map(([c, a1]) => {
163
+ const [f, d1] = op2.f(d, c);
164
+ return [f, [a1, d1]];
165
+ }),
166
+ ),
167
+ FilterMapLoop: (op2) =>
168
+ FilterMapLoopIOOperator([op1.seed, op2.seed], ([a, d], b) =>
169
+ op1.f(a, b).map(([c, a1]) => {
170
+ const [f, d1] = op2.f(d, c);
171
+ return [f, [a1, d1]];
172
+ }),
173
+ ),
174
+ }),
175
+ FilterMapLoopIO: (op1) =>
176
+ that.match({
177
+ Loop: (op2) =>
178
+ FilterMapLoopIOOperator([op1.seed, op2.seed], ([a, d], b) =>
179
+ op1.f(a, b).map(([c, a1]) =>
180
+ c.match(
181
+ () => [Nothing(), [a1, d]],
182
+ (c) => {
183
+ const [f, d1] = op2.f(d, c);
184
+ return [Just(f), [a1, d1]];
185
+ },
186
+ ),
187
+ ),
188
+ ),
189
+ FilterMapLoop: (op2) =>
190
+ FilterMapLoopIOOperator([op1.seed, op2.seed], ([a, d], b) =>
191
+ op1.f(a, b).map(([c, a1]) =>
192
+ c.match(
193
+ () => [Nothing(), [a1, d]],
194
+ (c) => {
195
+ const [f, d1] = op2.f(d, c);
196
+ return [f, [a1, d1]];
197
+ },
198
+ ),
199
+ ),
200
+ ),
201
+ }),
202
+ });
203
+ }
204
+
205
+ /**
206
+ * @tsplus pipeable fncts.io.Push.IOLoopOperator after
207
+ */
208
+ export function fuseSyncOperatorBefore(that: SyncOperator) {
209
+ return (self: IOLoopOperator): IOLoopOperator =>
210
+ that.match({
211
+ Map: (op) =>
212
+ self.match({
213
+ LoopIO: (op2) => LoopIOOperator(op2.seed, (acc, a) => op2.f(acc, op.f(a))),
214
+ FilterMapLoopIO: (op2) => FilterMapLoopIOOperator(op2.seed, (acc, a) => op2.f(acc, op.f(a))),
215
+ }),
216
+ Filter: (op) =>
217
+ self.match({
218
+ LoopIO: (op2) =>
219
+ LoopIOOperator(op2.seed, (acc, a) => {
220
+ if (op.f(a)) {
221
+ return op2.f(acc, a);
222
+ } else {
223
+ return IO.succeedNow([Nothing(), acc]);
224
+ }
225
+ }),
226
+ FilterMapLoopIO: (op2) =>
227
+ FilterMapLoopIOOperator(op2.seed, (acc, a) => {
228
+ if (op.f(a)) {
229
+ return op2.f(acc, a);
230
+ } else {
231
+ return IO.succeedNow([Nothing(), acc]);
232
+ }
233
+ }),
234
+ }),
235
+ FilterMap: (op) =>
236
+ self.match({
237
+ LoopIO: (op2) =>
238
+ LoopIOOperator(op2.seed, (acc, a) =>
239
+ op.f(a).match(
240
+ () => IO.succeedNow([Nothing(), acc] as const),
241
+ (b) => self.f(acc, b),
242
+ ),
243
+ ),
244
+ FilterMapLoopIO: (op2) =>
245
+ FilterMapLoopIOOperator(op2.seed, (acc, a) =>
246
+ op.f(a).match(
247
+ () => IO.succeedNow([Nothing(), acc] as const),
248
+ (b) => self.f(acc, b),
249
+ ),
250
+ ),
251
+ }),
252
+ });
253
+ }
254
+
255
+ /**
256
+ * @tsplus pipeable fncts.io.Push.IOLoopOperator before
257
+ */
258
+ export function fuseSyncOperatorAfter(that: SyncOperator) {
259
+ return (self: IOLoopOperator): IOLoopOperator =>
260
+ self.match({
261
+ LoopIO: (op1) =>
262
+ that.match({
263
+ Map: (op) => LoopIOOperator(op1.seed, (acc, a) => op1.f(acc, a).map(([b, c]) => [op.f(b), c] as const)),
264
+ Filter: (op) =>
265
+ FilterMapLoopIOOperator(op1.seed, (acc, a) =>
266
+ op1.f(acc, a).map(([b, c]) => [Maybe.fromPredicate(b, op.f), c] as const),
267
+ ),
268
+ FilterMap: (op) =>
269
+ FilterMapLoopIOOperator(op1.seed, (acc, a) => op1.f(acc, a).map(([b, c]) => [op.f(b), c] as const)),
270
+ }),
271
+ FilterMapLoopIO: (op1) =>
272
+ that.match({
273
+ Map: (op) =>
274
+ FilterMapLoopIOOperator(op1.seed, (acc, a) => op1.f(acc, a).map(([b, c]) => [b.map(op.f), c] as const)),
275
+ Filter: (op) =>
276
+ FilterMapLoopIOOperator(op1.seed, (acc, a) => op1.f(acc, a).map(([b, c]) => [b.filter(op.f), c] as const)),
277
+ FilterMap: (op) =>
278
+ FilterMapLoopIOOperator(op1.seed, (acc, a) => op1.f(acc, a).map(([b, c]) => [b.flatMap(op.f), c] as const)),
279
+ }),
280
+ });
281
+ }
282
+
283
+ export function fuseIOOperatorBefore(that: IOOperator) {
284
+ return (self: IOLoopOperator): IOLoopOperator =>
285
+ that.match({
286
+ MapIO: (op) =>
287
+ self.match({
288
+ LoopIO: (op2) => LoopIOOperator(op2.seed, (acc, a) => op.f(a).flatMap((b) => op2.f(acc, b))),
289
+ FilterMapLoopIO: (op2) =>
290
+ FilterMapLoopIOOperator(op2.seed, (acc, a) => op.f(a).flatMap((b) => op2.f(acc, b))),
291
+ }),
292
+ TapIO: (op) =>
293
+ self.match({
294
+ LoopIO: (op2) => LoopIOOperator(op2.seed, (acc, a) => op.f(a).flatMap(() => op2.f(acc, a))),
295
+ FilterMapLoopIO: (op2) => FilterMapLoopIOOperator(op2.seed, (acc, a) => op.f(a).flatMap(() => op2.f(acc, a))),
296
+ }),
297
+ FilterIO: (op) =>
298
+ self.match({
299
+ LoopIO: (op2) =>
300
+ LoopIOOperator(op2.seed, (acc, a) =>
301
+ op.f(a).flatMap((b) => {
302
+ if (b) {
303
+ return op2.f(acc, a);
304
+ } else {
305
+ return IO.succeedNow([Nothing(), acc] as const);
306
+ }
307
+ }),
308
+ ),
309
+ FilterMapLoopIO: (op2) =>
310
+ FilterMapLoopIOOperator(op2.seed, (acc, a) =>
311
+ op.f(a).flatMap((b) => {
312
+ if (b) {
313
+ return op2.f(acc, a);
314
+ } else {
315
+ return IO.succeedNow([Nothing(), acc] as const);
316
+ }
317
+ }),
318
+ ),
319
+ }),
320
+ FilterMapIO: (op) =>
321
+ self.match({
322
+ LoopIO: (op2) =>
323
+ LoopIOOperator(op2.seed, (acc, a) =>
324
+ op.f(a).flatMap((b) =>
325
+ b.match(
326
+ () => IO.succeedNow([Nothing(), acc] as const),
327
+ (c) => op2.f(acc, c),
328
+ ),
329
+ ),
330
+ ),
331
+ FilterMapLoopIO: (op2) =>
332
+ FilterMapLoopIOOperator(op2.seed, (acc, a) =>
333
+ op.f(a).flatMap((b) =>
334
+ b.match(
335
+ () => IO.succeedNow([Nothing(), acc] as const),
336
+ (c) => op2.f(acc, c),
337
+ ),
338
+ ),
339
+ ),
340
+ }),
341
+ });
342
+ }
343
+
344
+ export function fuseIOOperatorAfter(that: IOOperator) {
345
+ return (self: IOLoopOperator) =>
346
+ self.match({
347
+ LoopIO: (op1) =>
348
+ that.match({
349
+ MapIO: (op) =>
350
+ LoopIOOperator(op1.seed, (acc, a) =>
351
+ op1.f(acc, a).flatMap(([b, c]) => op.f(b).map((d) => [d, c] as const)),
352
+ ),
353
+ TapIO: (op) =>
354
+ LoopIOOperator(op1.seed, (acc, a) => op1.f(acc, a).flatMap(([b, c]) => op.f(b).map(() => [b, c] as const))),
355
+ FilterIO: (op) =>
356
+ FilterMapLoopIOOperator(op1.seed, (acc, a) =>
357
+ op1
358
+ .f(acc, a)
359
+ .flatMap(([b, c]) => op.f(b).map((d) => (d ? ([Just(b), c] as const) : ([Nothing(), c] as const)))),
360
+ ),
361
+ FilterMapIO: (op) =>
362
+ FilterMapLoopIOOperator(op1.seed, (acc, a) =>
363
+ op1.f(acc, a).flatMap(([b, c]) => op.f(b).map((d) => [d, c] as const)),
364
+ ),
365
+ }),
366
+ FilterMapLoopIO: (op1) =>
367
+ that.match({
368
+ MapIO: (op) =>
369
+ FilterMapLoopIOOperator(op1.seed, (acc, a) =>
370
+ op1.f(acc, a).flatMap(([b, c]) =>
371
+ b.match(
372
+ () => IO.succeedNow([Nothing(), c] as const),
373
+ (d) => op.f(d).map((e) => [Just(e), c] as const),
374
+ ),
375
+ ),
376
+ ),
377
+ TapIO: (op) =>
378
+ FilterMapLoopIOOperator(op1.seed, (acc, a) =>
379
+ op1.f(acc, a).flatMap(([b, c]) =>
380
+ b.match(
381
+ () => IO.succeedNow([Nothing(), c] as const),
382
+ (d) => op.f(d).map(() => [Just(d), c] as const),
383
+ ),
384
+ ),
385
+ ),
386
+ FilterIO: (op) =>
387
+ FilterMapLoopIOOperator(op1.seed, (acc, a) =>
388
+ op1.f(acc, a).flatMap(([b, c]) =>
389
+ b.match(
390
+ () => IO.succeedNow([Nothing(), c] as const),
391
+ (d) => op.f(d).map((e) => (e ? ([Just(d), c] as const) : ([Nothing(), c] as const))),
392
+ ),
393
+ ),
394
+ ),
395
+ FilterMapIO: (op) =>
396
+ FilterMapLoopIOOperator(op1.seed, (acc, a) =>
397
+ op1.f(acc, a).flatMap(([b, c]) =>
398
+ b.match(
399
+ () => IO.succeedNow([Nothing(), c] as const),
400
+ (d) => op.f(d).map((e) => [e, c] as const),
401
+ ),
402
+ ),
403
+ ),
404
+ }),
405
+ });
406
+ }
407
+
408
+ export function liftLoopOperator(self: SyncLoopOperator): IOLoopOperator {
409
+ return self.match({
410
+ Loop: (op) => LoopIOOperator(op.seed, (acc, a) => IO.succeedNow(op.f(acc, a))),
411
+ FilterMapLoop: (op) => FilterMapLoopIOOperator(op.seed, (acc, a) => IO.succeedNow(op.f(acc, a))),
412
+ });
413
+ }