166 lines
5.6 KiB
JavaScript
166 lines
5.6 KiB
JavaScript
import * as Effect from "../../Effect.js";
|
|
import { dual, pipe } from "../../Function.js";
|
|
import * as Option from "../../Option.js";
|
|
import { pipeArguments } from "../../Pipeable.js";
|
|
import * as STM from "../../STM.js";
|
|
import * as TPubSub from "../../TPubSub.js";
|
|
import * as TQueue from "../../TQueue.js";
|
|
import * as TRef from "../../TRef.js";
|
|
import * as stream from "../stream.js";
|
|
import { tDequeueVariance } from "./tQueue.js";
|
|
import { tRefVariance } from "./tRef.js";
|
|
/** @internal */
|
|
const TSubscriptionRefSymbolKey = "effect/TSubscriptionRef";
|
|
/** @internal */
|
|
export const TSubscriptionRefTypeId = /*#__PURE__*/Symbol.for(TSubscriptionRefSymbolKey);
|
|
const TSubscriptionRefVariance = {
|
|
/* c8 ignore next */
|
|
_A: _ => _
|
|
};
|
|
class TDequeueMerge {
|
|
first;
|
|
second;
|
|
[TQueue.TDequeueTypeId] = tDequeueVariance;
|
|
constructor(first, second) {
|
|
this.first = first;
|
|
this.second = second;
|
|
}
|
|
peek = /*#__PURE__*/STM.gen(this, function* () {
|
|
const first = yield* this.peekOption;
|
|
if (first._tag === "Some") {
|
|
return first.value;
|
|
}
|
|
return yield* STM.retry;
|
|
});
|
|
peekOption = /*#__PURE__*/STM.gen(this, function* () {
|
|
const first = yield* this.first.peekOption;
|
|
if (first._tag === "Some") {
|
|
return first;
|
|
}
|
|
const second = yield* this.second.peekOption;
|
|
if (second._tag === "Some") {
|
|
return second;
|
|
}
|
|
return Option.none();
|
|
});
|
|
take = /*#__PURE__*/STM.gen(this, function* () {
|
|
if (!(yield* this.first.isEmpty)) {
|
|
return yield* this.first.take;
|
|
}
|
|
if (!(yield* this.second.isEmpty)) {
|
|
return yield* this.second.take;
|
|
}
|
|
return yield* STM.retry;
|
|
});
|
|
takeAll = /*#__PURE__*/STM.gen(this, function* () {
|
|
return [...(yield* this.first.takeAll), ...(yield* this.second.takeAll)];
|
|
});
|
|
takeUpTo(max) {
|
|
return STM.gen(this, function* () {
|
|
const first = yield* this.first.takeUpTo(max);
|
|
if (first.length >= max) {
|
|
return first;
|
|
}
|
|
return [...first, ...(yield* this.second.takeUpTo(max - first.length))];
|
|
});
|
|
}
|
|
capacity() {
|
|
return this.first.capacity() + this.second.capacity();
|
|
}
|
|
size = /*#__PURE__*/STM.gen(this, function* () {
|
|
return (yield* this.first.size) + (yield* this.second.size);
|
|
});
|
|
isFull = /*#__PURE__*/STM.gen(this, function* () {
|
|
return (yield* this.first.isFull) && (yield* this.second.isFull);
|
|
});
|
|
isEmpty = /*#__PURE__*/STM.gen(this, function* () {
|
|
return (yield* this.first.isEmpty) && (yield* this.second.isEmpty);
|
|
});
|
|
shutdown = /*#__PURE__*/STM.gen(this, function* () {
|
|
yield* this.first.shutdown;
|
|
yield* this.second.shutdown;
|
|
});
|
|
isShutdown = /*#__PURE__*/STM.gen(this, function* () {
|
|
return (yield* this.first.isShutdown) && (yield* this.second.isShutdown);
|
|
});
|
|
awaitShutdown = /*#__PURE__*/STM.gen(this, function* () {
|
|
yield* this.first.awaitShutdown;
|
|
yield* this.second.awaitShutdown;
|
|
});
|
|
}
|
|
/** @internal */
|
|
class TSubscriptionRefImpl {
|
|
ref;
|
|
pubsub;
|
|
[TSubscriptionRefTypeId] = TSubscriptionRefVariance;
|
|
[TRef.TRefTypeId] = tRefVariance;
|
|
constructor(ref, pubsub) {
|
|
this.ref = ref;
|
|
this.pubsub = pubsub;
|
|
}
|
|
get todos() {
|
|
return this.ref.todos;
|
|
}
|
|
get versioned() {
|
|
return this.ref.versioned;
|
|
}
|
|
pipe() {
|
|
return pipeArguments(this, arguments);
|
|
}
|
|
get changes() {
|
|
return STM.gen(this, function* () {
|
|
const first = yield* TQueue.unbounded();
|
|
yield* TQueue.offer(first, yield* TRef.get(this.ref));
|
|
return new TDequeueMerge(first, yield* TPubSub.subscribe(this.pubsub));
|
|
});
|
|
}
|
|
modify(f) {
|
|
return pipe(TRef.get(this.ref), STM.map(f), STM.flatMap(([b, a]) => pipe(TRef.set(this.ref, a), STM.as(b), STM.zipLeft(TPubSub.publish(this.pubsub, a)))));
|
|
}
|
|
}
|
|
/** @internal */
|
|
export const make = value => pipe(STM.all([TPubSub.unbounded(), TRef.make(value)]), STM.map(([pubsub, ref]) => new TSubscriptionRefImpl(ref, pubsub)));
|
|
/** @internal */
|
|
export const get = self => TRef.get(self.ref);
|
|
/** @internal */
|
|
export const set = /*#__PURE__*/dual(2, (self, value) => self.modify(() => [void 0, value]));
|
|
/** @internal */
|
|
export const getAndSet = /*#__PURE__*/dual(2, (self, value) => self.modify(a => [a, value]));
|
|
/** @internal */
|
|
export const getAndUpdate = /*#__PURE__*/dual(2, (self, f) => self.modify(a => [a, f(a)]));
|
|
/** @internal */
|
|
export const getAndUpdateSome = /*#__PURE__*/dual(2, (self, f) => self.modify(a => Option.match(f(a), {
|
|
onNone: () => [a, a],
|
|
onSome: b => [a, b]
|
|
})));
|
|
/** @internal */
|
|
export const setAndGet = /*#__PURE__*/dual(2, (self, value) => self.modify(() => [value, value]));
|
|
/** @internal */
|
|
export const modify = /*#__PURE__*/dual(2, (self, f) => self.modify(f));
|
|
/** @internal */
|
|
export const modifySome = /*#__PURE__*/dual(3, (self, fallback, f) => self.modify(a => Option.match(f(a), {
|
|
onNone: () => [fallback, a],
|
|
onSome: b => b
|
|
})));
|
|
/** @internal */
|
|
export const update = /*#__PURE__*/dual(2, (self, f) => self.modify(a => [void 0, f(a)]));
|
|
/** @internal */
|
|
export const updateAndGet = /*#__PURE__*/dual(2, (self, f) => self.modify(a => {
|
|
const b = f(a);
|
|
return [b, b];
|
|
}));
|
|
/** @internal */
|
|
export const updateSome = /*#__PURE__*/dual(2, (self, f) => self.modify(a => [void 0, Option.match(f(a), {
|
|
onNone: () => a,
|
|
onSome: b => b
|
|
})]));
|
|
/** @internal */
|
|
export const updateSomeAndGet = /*#__PURE__*/dual(2, (self, f) => self.modify(a => Option.match(f(a), {
|
|
onNone: () => [a, a],
|
|
onSome: b => [b, b]
|
|
})));
|
|
/** @internal */
|
|
export const changesScoped = self => Effect.acquireRelease(self.changes, TQueue.shutdown);
|
|
/** @internal */
|
|
export const changesStream = self => stream.unwrap(Effect.map(self.changes, stream.fromTQueue));
|
|
//# sourceMappingURL=tSubscriptionRef.js.map
|