Files
ospab.host/node_modules/effect/dist/esm/internal/stm/tSubscriptionRef.js
2025-09-15 18:10:26 +03:00

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