doddle
Version:
Tiny yet feature-packed (async) iteration toolkit.
1,026 lines • 35.9 kB
JavaScript
import { doddle, lazyOperator, pull } from "../doddle/index.js";
import { chk, DoddleError, invalidRecursionError, loadCheckers, reduceOnEmptyError } from "../errors/error.js";
import { _iter, createCompareKey, orderedStages, shuffleArray } from "../utils.js";
import { seq } from "./seq.ctor.js";
const SPECIAL = Symbol("S");
const SPECIAL2 = Symbol("S2");
/**
* The Seq class, which wraps a synchronous iterable.
*
* @category Use
*/
export class Seq {
/** @internal */
constructor() {
this.flatMap = this.concatMap;
// Class name is used for various checks
// Need to make sure it's accessible even while minified
loadCheckers(Seq.prototype);
}
/** @internal */
get _qr() {
return this.toArray().pull();
}
/**
* Calls a side-effect function after all elements have been yielded, but before iteration
* finishes.
*
* ⚠️ If the client stops iterating early, the action won't be executed.
*
* @param action A function to invoke after iteration completes.
* @returns A new sequence that acts like `this` but invokes `action` before it's finished.
*/
after(action) {
chk(this.after).action(action);
return SeqOperator(this, function* after(input) {
yield* input;
pull(action());
});
}
/**
* Reinterprets the declared element type of `this` as another, arbitrary type.
*
* ℹ️ This is only useful in TypeScript and has no runtime effects.
*
* @template S The new element type.
* @returns The same sequence, but with a different declared type.
*/
as() {
return this;
}
/**
* 🦥**Lazily** gets the element at the given index in `this` sequence, or undefined if the
* index is out of bounds.
*
* ℹ️ Negative indexes count from the end of the sequence.\
* ⚠️ Requires iterating over the sequence up to the given index.
*
* @param index The index of the item to retrieve.
* @returns A 🦥{@link Doddle} that resolves to the item at the given index.
*/
at(index) {
chk(this.at).index(index);
return lazyOperator(this, function at(input) {
if (index < 0) {
return input.take(index).first().pull();
}
return input.skip(index).first().pull();
});
}
/**
* Executes a side effect action once before any elements are yielded, but after iteration has
* begun.
*
* @param action Invokes before any elements are yielded.
* @returns A new async sequence that performs `action` before yielding elements.
*/
before(action) {
chk(this.before).action(action);
return SeqOperator(this, function* before(input) {
pull(action());
yield* input;
});
}
/**
* Caches the elements of `this` sequence as they're iterated over, so that it's evaluated only
* once.
*
* @returns A new sequence with the same elements as the original sequence.
*/
cache() {
const self = this;
const _cache = [];
let alreadyDone = false;
let iterator;
let isCallingNext = false;
return SeqOperator(this, function* cache() {
let i = 0;
if (isCallingNext) {
throw new DoddleError(invalidRecursionError("cache"));
}
for (;;) {
if (i < _cache.length) {
const cur = _cache[i];
yield cur;
i++;
}
else if (!alreadyDone) {
iterator ??= _iter(self);
try {
isCallingNext = true;
var { done, value } = iterator.next();
}
finally {
isCallingNext = false;
}
if (done) {
alreadyDone = true;
return;
}
_cache.push(value);
yield value;
i++;
}
else {
return;
}
}
});
}
/**
* Handles errors thrown while iterating over `this` sequence.
*
* @param handler A handler that will be called with the error and the index of the element that
* caused it. Should return a new sequence or `undefined`, which stops iteration.
* @returns A new sequence that handles errors.
*/
catch(handler) {
chk(this.catch).handler(handler);
return SeqOperator(this, function* catch_(input) {
let i = 0;
const iterator = _iter(input);
for (;;) {
try {
const result = iterator.next();
var value = result.value;
if (result.done) {
return;
}
yield value;
}
catch (err) {
const error = err;
const result = pull(handler(error, i));
if (!result || result == null) {
return;
}
yield* seq(result);
return;
}
i++;
}
});
}
/**
* Splits `this` sequence into chunks of the given size, optionally applying a projection to
* each chunk.
*
* ℹ️ The last chunk may be smaller than the given size.
*
* @param size The size of each chunk.
* @param projection Optionally, an N-ary projection to apply to each chunk. Defaults to
* collecting the elements into an array.
* @returns A new sequence.
*/
chunk(size, projection) {
const c = chk(this.chunk);
c.size(size);
projection ??= (...chunk) => chunk;
c.projection(projection);
return SeqOperator(this, function* chunk(input) {
let chunk = [];
for (const item of input) {
chunk.push(item);
if (chunk.length === size) {
yield pull(projection(...chunk));
chunk = [];
}
}
if (chunk.length) {
yield pull(projection(...chunk));
}
});
}
/**
* Returns a new sequence. When iterated, before yielding its first element, it will iterate
* over all the elements of `this` and store them in memory. Then it will yield all of them one
* by one.
*
* ℹ️ Used to control side-effects. Makes sure all side-effects execute before continuing to
* apply other operators.
*
* @returns A new sequence with the same elements as this one, but where iteration has already
* completed.
*/
collect() {
return SeqOperator(this, function* collect(input) {
yield* [...input];
});
}
/**
* Concatenates one or more sequences to the end of `this`, so that their elements appear in
* order.
*
* @param inputs The sequential inputs to concatenate to the end of `this`.
* @returns A new sequence with the concatenated elements.
*/
concat(...inputs) {
const iterables = inputs.map(seq);
return SeqOperator(this, function* concat(input) {
yield* input;
for (const iterable of iterables) {
yield* iterable;
}
});
}
/**
* Applies a sequence projection on each element of `this` sequence and concatenates the
* resulting sequences.
*
* @param projection The sequence projection to apply to each element.
* @returns A new sequence with the flattened results.
*/
concatMap(projection) {
chk(this.concatMap).projection(projection);
return SeqOperator(this, function* concatMap(input) {
let index = 0;
for (const element of input) {
for (const projected of seq(pull(projection(element, index++)))) {
yield projected;
}
}
});
}
/**
* Concatenates `this` sequence to the end of one or more other sequences.
*
* ℹ️ Input sequences are concatenated in the order that they appear.
*
* @param inputs One or more other sequences.
* @returns A new sequence with the concatenated elements.
* @see {@link Seq.concat}
*/
concatTo(...inputs) {
return seq([]).concat(...inputs, this);
}
count(predicate) {
predicate ??= () => true;
predicate = chk(this.count).predicate(predicate);
return lazyOperator(this, function count(input) {
let index = 0;
let count = 0;
for (const element of input) {
if (pull(predicate(element, index++))) {
count++;
}
}
return count;
});
}
/**
* Calls an action function as each element in `this` is iterated over. Calls the function
* before or after yielding the element, or both.
*
* @param action The action function to invoke for each element.
* @param stage The **stage** at which to invoke the function. Can be `"before"`, `"after"`, or
* `"both"`.
* @returns A new sequence that invokes the action function while being iterated.
*/
each(action, stage = orderedStages[1 /* Stage.Before */]) {
const c = chk(this.each);
c.action(action);
c.stage(stage);
const myStage = orderedStages.indexOf(stage);
return SeqOperator(this, function* each(input) {
let index = 0;
for (const element of input) {
if (myStage & 1 /* Stage.Before */) {
pull(action(element, index, orderedStages[1 /* Stage.Before */]));
}
yield element;
if (myStage & 2 /* Stage.After */) {
pull(action(element, index, orderedStages[2 /* Stage.After */]));
}
index++;
}
});
}
/**
* 🦥**Lazily** checks if all elements in `this` sequence match the given predicate.
*
* ⚠️ May iterate over the entire sequence.
*
* @param predicate The predicate.
* @returns A 🦥{@link Doddle} that yields `true` if all elements match, or `false` otherwise.
*/
every(predicate) {
predicate = chk(this.every).predicate(predicate);
return lazyOperator(this, function every(input) {
let index = 0;
for (const element of input) {
if (!pull(predicate(element, index++))) {
return false;
}
}
return true;
});
}
filter(predicate) {
predicate = chk(this.filter).predicate(predicate);
return SeqOperator(this, function* filter(input) {
let index = 0;
for (const x of input) {
if (pull(predicate(x, index++))) {
yield x;
}
}
});
}
first(predicate, alt) {
predicate ??= () => true;
return lazyOperator(this, function first(input) {
let index = 0;
for (const element of input) {
if (pull(predicate(element, index++))) {
return element;
}
}
return alt;
});
}
/**
* Groups the elements of `this` sequence by key, resulting in a sequence of pairs where the
* first element is the key and the second is a sequence of values.
*
* @param keyProjection The projection used to determine the key for each element.
* @returns A sequence of pairs.
*/
groupBy(keyProjection) {
chk(this.groupBy).keyProjection(keyProjection);
return SeqOperator(this, function* groupBy(input) {
const map = new Map();
const keys = [];
const shared = input
.map(v => {
const key = pull(keyProjection(v));
const group = map.get(key);
if (group) {
group.push(v);
}
else {
keys.push(key);
map.set(key, [v]);
}
return undefined;
})
.share();
function* getGroupIterable(key) {
const group = map.get(key);
for (let i = 0;; i++) {
if (i < group.length) {
yield group[i];
continue;
}
for (const _ of shared) {
if (i < group.length) {
break;
}
}
if (i >= group.length) {
// must've completed
return;
}
i--;
}
}
for (let i = 0;; i++) {
if (i < keys.length) {
const key = keys[i];
yield [key, seq(() => getGroupIterable(key))];
continue;
}
for (const _ of shared) {
if (i < keys.length) {
break;
}
}
if (i >= keys.length) {
// must've completed
return;
}
i--;
}
});
}
includes(..._values) {
const values = new Set(_values);
return lazyOperator(this, function includes(input) {
for (const element of input) {
values.delete(element);
if (values.size === 0) {
return true;
}
}
return false;
});
}
/**
* 🦥**Lazily** joins the elements of `this` sequence into a single string, separated by the
* given separator.
*
* ⚠️ Requires iterating over the entire sequence.
*
* @param separator The string to use as a separator between elements.
* @returns A 🦥{@link Doddle} that resolves to the joined string.
*/
join(separator = ",") {
chk(this.join).separator(separator);
return lazyOperator(this, function join(input) {
const results = [];
for (const x of input) {
results.push(x);
}
return results.join(separator);
});
}
last(predicate, alt) {
predicate ??= () => true;
chk(this.last).predicate(predicate);
return lazyOperator(this, function last(input) {
let lastOrAlt = alt;
let index = 0;
for (const element of input) {
if (!pull(predicate(element, index++))) {
continue;
}
lastOrAlt = element;
}
return lastOrAlt;
});
}
/**
* Applies a projection to each element of `this` sequence.
*
* @param projection The projection to apply to each element.
* @returns A new sequence with the projected elements.
*/
map(projection) {
chk(this.map).projection(projection);
return SeqOperator(this, function* map(input) {
let index = 0;
for (const element of input) {
yield pull(projection(element, index++));
}
});
}
/**
* 🦥**Lazily** finds the maximum element in `this` sequence by key, or the given alternative
* value if the sequence is empty.
*
* @param projection Projects each element into a key so it can be compared.
* @param alt The value to return if the sequence is empty. Defaults to `undefined`.
*/
maxBy(projection, alt) {
chk(this.maxBy).projection(projection);
return lazyOperator(this, function maxBy(input) {
let curMax = alt;
let curMaxKey = undefined;
let index = 0;
for (const element of input) {
const curKey = pull(projection(element, index++));
if (index === 1 || curKey > curMaxKey) {
curMax = element;
curMaxKey = curKey;
continue;
}
}
return curMax;
});
}
/**
* 🦥**Lazily** finds the minimum element in `this` sequence by key, or the given alternative
* value if the sequence is empty.
*
* @param projection Projects each element into a key so it can be compared.
* @param alt The value to return if the sequence is empty. Defaults to `undefined`.
* @returns A 🦥{@link Doddle} that resolves to the element with the minimum key, or `alt` if the
* sequence is empty.
*/
minBy(projection, alt) {
chk(this.minBy).projection(projection);
return lazyOperator(this, function minBy(input) {
let curMin = alt;
let curMinKey = undefined;
let index = 0;
for (const element of input) {
const curKey = pull(projection(element, index++));
if (index === 1 || curKey < curMinKey) {
curMin = element;
curMinKey = curKey;
continue;
}
}
return curMin;
});
}
orderBy(projection, descending = false) {
chk(this.orderBy).projection(projection);
chk(this.orderBy).descending(descending);
const compareKey = createCompareKey(descending);
return SeqOperator(this, function* orderBy(input) {
const kvps = [];
for (const element of input) {
const key = pull(projection(element));
kvps.push([key, element]);
}
kvps.sort(compareKey);
for (const [_, value] of kvps) {
yield value;
}
});
}
/**
* Returns a cartesian product of `this` sequence with one or more other sequences, optionally
* applying an N-ary projection to each combination of elements.
*
* The product of `N` sequences is the collection of all possible sets of elements from each
* sequence.
*
* For example, the product of `[1, 2]` and `[3, 4]` is:
*
* ```ts
* ;[
* [1, 3],
* [1, 4],
* [2, 3],
* [2, 4]
* ]
* ```
*
* @example
* seq([1, 2]).product([3, 4])
* // => [[1, 3], [1, 4], [2, 3], [2, 4]]
* seq([]).product([3, 4])
* // => []
* seq([1, 2]).product([3, 4], (a, b) => a + b)
* // => [4, 5, 5, 6]
*
* @param others One or more sequence-like inputs for the product.
* @param projection Optionally, an N-ary projection to apply to each combination of elements.
* If not given, each combination is yielded as an array.
* @returns A new sequence.
*/
product(_others, projection) {
const others = _others.map(seq).map(x => x.cache());
projection ??= (...args) => args;
chk(this.product).projection(projection);
return SeqOperator(this, function* product(input) {
let partialProducts = [[]];
for (const iterable of [input, ...others].reverse()) {
const oldPartialProducts = partialProducts;
partialProducts = [];
for (const item of iterable) {
partialProducts = partialProducts.concat(oldPartialProducts.map(x => [item, ...x]));
}
}
yield* partialProducts.map(x => pull(projection.apply(null, x)));
});
}
reduce(reducer, initial) {
chk(this.reduce).reducer(reducer);
return lazyOperator(this, function reduce(input) {
let acc = initial ?? SPECIAL;
let index = 0;
for (const element of input) {
if (acc === SPECIAL) {
acc = element;
continue;
}
acc = pull(reducer(acc, element, index++));
}
if (acc === SPECIAL) {
throw new DoddleError(reduceOnEmptyError);
}
return acc;
});
}
/**
* Reverses `this` sequence.
*
* ⚠️ Requires iterating over the entire sequence.
*
* @returns A new sequence with the elements in reverse order.
*/
reverse() {
return SeqOperator(this, function* reverse(input) {
const elements = [];
for (const element of input) {
elements.push(element);
}
yield* elements.reverse();
});
}
scan(reduction, initial) {
chk(this.scan).reducer(reduction);
return SeqOperator(this, function* scan(input) {
let hasAcc = initial !== undefined;
let acc = initial;
let index = 0;
if (hasAcc) {
yield acc;
}
for (const element of input) {
if (!hasAcc) {
acc = element;
hasAcc = true;
}
else {
acc = pull(reduction(acc, element, index++));
}
yield acc;
}
});
}
seqEquals(_input, projection = x => x) {
const other = seq(_input);
return lazyOperator(this, function seqEqualsBy(input) {
const otherIterator = _iter(other);
try {
for (const element of input) {
const otherElement = otherIterator.next();
const keyThis = pull(projection(element));
const keyOther = pull(projection(otherElement.value));
if (otherElement.done || keyThis !== keyOther) {
return false;
}
}
return !!otherIterator.next().done;
}
finally {
otherIterator.return?.();
}
});
}
setEquals(_other, projection = x => x) {
const other = seq(_other);
return lazyOperator(this, function setEqualsBy(input) {
const set = new Set();
for (const element of other) {
set.add(pull(projection(element)));
}
for (const element of input) {
if (!set.delete(pull(projection(element)))) {
return false;
}
}
return set.size === 0;
});
}
/**
* Returns a new sequence that shares its iterator state. This allows different loops to iterate
* over it, sharing progress.
*
* ⚠️ Can be iterated over exactly once, and will be empty afterwards.
*
* @returns A new, shared iterable sequence that can be iterated over exactly once.
*/
share() {
const iter = doddle(() => _iter(this));
let isCallingNext = false;
return SeqOperator(this, function* share() {
if (isCallingNext) {
throw new DoddleError(invalidRecursionError("share"));
}
while (true) {
try {
isCallingNext = true;
var { done, value } = iter.pull().next();
}
finally {
isCallingNext = false;
}
if (done) {
return;
}
yield value;
}
});
}
/**
* Shuffles the elements of `this` sequence randomly.
*
* ⚠️ Requires iterating over the entire sequence.
*
* @returns A new sequence with the shuffled elements.
*/
shuffle() {
return SeqOperator(this, function* shuffle(input) {
const array = seq(input).toArray().pull();
shuffleArray(array);
yield* array;
});
}
/**
* Skips the first `count` elements of `this` sequence, yielding the rest.
*
* ℹ️ If `count` is negative, skips the final elements instead (e.g. `skipLast`)
*
* @param count The number of elements to skip.
* @returns A new sequence without the skipped elements.
*/
skip(count) {
chk(this.skip).count(count);
return SeqOperator(this, function* skip(input) {
let myCount = count;
if (myCount < 0) {
myCount = -myCount;
yield* seq(input)
.window(myCount + 1, (...window) => {
if (window.length === myCount + 1) {
return window[0];
}
return SPECIAL2;
})
.filter(x => x !== SPECIAL2);
}
else {
for (const x of input) {
if (myCount > 0) {
myCount--;
continue;
}
yield x;
}
}
});
}
/**
* Skips elements from `this` sequence while the given predicate is true, and yields the rest.
*
* ℹ️ You can use the `options` argument to skip the first element that returns `false`.
*
* @param predicate The predicate to determine whether to continue skipping.
* @param options Options for skipping behavior.
* @returns A new sequence without the skipped elements.
*/
skipWhile(predicate, options) {
predicate = chk(this.skipWhile).predicate(predicate);
return SeqOperator(this, function* skipWhile(input) {
let prevMode = 0 /* SkippingMode.None */;
let index = 0;
for (const element of input) {
if (prevMode === 2 /* SkippingMode.NotSkipping */) {
yield element;
continue;
}
const newSkipping = pull(predicate(element, index++));
if (!newSkipping) {
if (prevMode !== 1 /* SkippingMode.Skipping */ || !options?.skipFinal) {
yield element;
}
}
prevMode = newSkipping ? 1 /* SkippingMode.Skipping */ : 2 /* SkippingMode.NotSkipping */;
}
});
}
/**
* 🦥**Lazily** checks if any element in `this` sequence matches the given predicate, by
* iterating over it until a match is found.
*
* @param predicate The predicate to match the element.
* @returns A 🦥{@link Doddle} that resolves to `true` if any element matches, or `false`
* otherwise.
*/
some(predicate) {
predicate = chk(this.some).predicate(predicate);
return lazyOperator(this, function some(input) {
let index = 0;
for (const element of input) {
if (pull(predicate(element, index++))) {
return true;
}
}
return false;
});
}
/**
* 🦥**Lazily** sums the elements of `this` sequence by iterating over it, applying the given
* projection to each element.
*
* @param projection The projection function to apply to each element.
* @returns A 🦥{@link Doddle} that resolves to the sum of the projected elements.
*/
sumBy(projection) {
chk(this.sumBy).projection(projection);
return lazyOperator(this, function sumBy(input) {
let cur = 0;
let index = 0;
for (const element of input) {
cur += pull(projection(element, index++));
}
return cur;
});
}
/**
* Yields the first `count` elements of `this` sequence.
*
* ℹ️ If `count` is negative, yields the last `-count` elements instead.\
* ℹ️ If the sequence is smaller than `count`, it yields all elements.
*
* @param count The number of elements to yield.
* @returns A new sequence with the yielded elements.
*/
take(count) {
chk(this.take).count(count);
return SeqOperator(this, function* take(input) {
let myCount = count;
if (myCount === 0) {
return;
}
if (myCount < 0) {
myCount = -myCount;
const results = seq(input)
.concat([SPECIAL2])
.window(myCount + 1, (...window) => {
if (window[window.length - 1] === SPECIAL2) {
window.pop();
return window;
}
return undefined;
})
.filter(x => x !== undefined)
.first()
.pull();
yield* results;
}
else {
for (const element of input) {
yield element;
myCount--;
if (myCount <= 0) {
return;
}
}
}
});
}
/**
* Yields the first elements of `this` sequence while the given predicate is true and skips the
* rest.
*
* ℹ️ If the sequence is too small, the result will be empty.\
* ℹ️ The `options` argument lets you keep the first element for which the predicate returns
* `false`.
*
* @param predicate The predicate to determine whether to continue yielding.
* @param options Extra options.
* @returns A new sequence with the yielded elements.
*/
takeWhile(predicate, options) {
chk(this.takeWhile).predicate(predicate);
return SeqOperator(this, function* takeWhile(input) {
let index = 0;
for (const element of input) {
if (pull(predicate(element, index++))) {
yield element;
}
else {
if (options?.takeFinal) {
yield element;
}
return;
}
}
});
}
/**
* 🦥**Lazily** converts `this` sequence into an array.
*
* ⚠️ Has to iterate over the entire sequence.
*
* @returns A 🦥{@link Doddle} that resolves to an array.
*/
toArray() {
return lazyOperator(this, function toArray(input) {
return [...input];
});
}
/**
* 🦥**Lazily** converts `this` sequence into a Map.
*
* ⚠️ Has to iterate over the entire sequence.
*
* @param kvpProjection A function that takes an element and returns a key-value pair.
* @returns A 🦥{@link Doddle} that resolves to a Map of the elements in the sequence.
*/
toMap(kvpProjection) {
kvpProjection = chk(this.toMap).kvpProjection(kvpProjection);
return lazyOperator(this, function toMap(input) {
const m = new Map();
let index = 0;
for (const element of input) {
const [key, value] = pull(kvpProjection(element, index++));
m.set(key, value);
}
return m;
});
}
/**
* 🦥**Lazily** converts `this` sequence into a plain JS object. Uses the given `kvpProjection`
* to determine each key-value pair.
*
* @param kvpProjection A function that takes an element and returns a key-value pair. Each key
* must be a valid PropertyKey.
* @returns A 🦥{@link Doddle} that resolves to a plain JS object.
*/
toRecord(kvpProjection) {
chk(this.toRecord).kvpProjection(kvpProjection);
return lazyOperator(this, function toObject(input) {
const o = {};
let index = 0;
for (const element of input) {
const [key, value] = pull(kvpProjection(element, index++));
o[key] = value;
}
return o;
});
}
/**
* 🦥**Lazily** converts `this` sequence into a Set.
*
* ⚠️ Has to iterate over the entire sequence.
*
* @returns A 🦥{@link Doddle} that resolves to a Set of the elements in the sequence.
*/
toSet() {
return lazyOperator(this, function toSet(input) {
return new Set(input);
});
}
/**
* Filters out duplicate elements from `this` sequence, optionally using a key projection.
*
* ℹ️ **Doesn't** need to iterate over the entire sequence.\
* ⚠️ Needs to cache the sequence as it's iterated over.
*
* @param keyProjection A function that takes an element and returns a key used to check for
* uniqueness.
* @returns A sequence of unique elements.
*/
uniq(keyProjection = x => x) {
chk(this.uniq).keyProjection(keyProjection);
return SeqOperator(this, function* uniq(input) {
const seen = new Set();
for (const element of input) {
const key = pull(keyProjection(element));
if (!seen.has(key)) {
seen.add(key);
yield element;
}
}
});
}
window(size, projection) {
const c = chk(this.window);
c.size(size);
projection ??= (...window) => window;
c.projection(projection);
return SeqOperator(this, function* window(input) {
const buffer = Array(size);
let i = 0;
for (const item of input) {
buffer[i++ % size] = item;
if (i >= size) {
yield pull(projection(...buffer.slice(i % size), ...buffer.slice(0, i % size)));
}
}
if (i > 0 && i < size) {
yield pull(projection(...buffer.slice(0, i)));
}
});
}
zip(_others, projection) {
const others = _others.map(seq);
projection ??= (...args) => args;
chk(this.zip).projection(projection);
return SeqOperator(this, function* zip(input) {
const iterators = [input, ...others].map(_iter);
while (true) {
const results = iterators.map((iter, i) => {
if (!iter) {
return undefined;
}
const result = iter.next();
if (result.done) {
iterators[i]?.return?.();
iterators[i] = undefined;
return undefined;
}
return result;
});
if (results.every(r => !r)) {
break;
}
yield pull(projection.apply(undefined, results.map(r => r?.value)));
}
});
}
}
/** @internal */
export const SeqOperator = function seq(operand, impl) {
const mySeq = Object.assign(new Seq(), [impl.name, operand]);
return Object.defineProperty(mySeq, Symbol.iterator, {
get: () => impl.bind(mySeq, mySeq[1])
});
};
//# sourceMappingURL=seq.class.js.map