// Copyright 2021 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only /* eslint-disable max-classes-per-file */ export function isIterable(value: unknown): value is Iterable { return ( (typeof value === 'object' && value !== null && Symbol.iterator in value) || typeof value === 'string' ); } export function size(iterable: Iterable): number { // We check for common types as an optimization. if (typeof iterable === 'string' || Array.isArray(iterable)) { return iterable.length; } if (iterable instanceof Set || iterable instanceof Map) { return iterable.size; } const iterator = iterable[Symbol.iterator](); let result = -1; for (let done = false; !done; result += 1) { done = Boolean(iterator.next().done); } return result; } export function filter( iterable: Iterable, predicate: (value: T) => value is S ): Iterable; export function filter( iterable: Iterable, predicate: (value: T) => unknown ): Iterable; export function filter( iterable: Iterable, predicate: (value: T) => unknown ): Iterable { return new FilterIterable(iterable, predicate); } class FilterIterable implements Iterable { constructor( private readonly iterable: Iterable, private readonly predicate: (value: T) => unknown ) {} [Symbol.iterator](): Iterator { return new FilterIterator(this.iterable[Symbol.iterator](), this.predicate); } } class FilterIterator implements Iterator { constructor( private readonly iterator: Iterator, private readonly predicate: (value: T) => unknown ) {} next(): IteratorResult { // eslint-disable-next-line no-constant-condition while (true) { const nextIteration = this.iterator.next(); if (nextIteration.done || this.predicate(nextIteration.value)) { return nextIteration; } } } } export function map( iterable: Iterable, fn: (value: T) => ResultT ): Iterable { return new MapIterable(iterable, fn); } class MapIterable implements Iterable { constructor( private readonly iterable: Iterable, private readonly fn: (value: T) => ResultT ) {} [Symbol.iterator](): Iterator { return new MapIterator(this.iterable[Symbol.iterator](), this.fn); } } class MapIterator implements Iterator { constructor( private readonly iterator: Iterator, private readonly fn: (value: T) => ResultT ) {} next(): IteratorResult { const nextIteration = this.iterator.next(); if (nextIteration.done) { return nextIteration; } return { done: false, value: this.fn(nextIteration.value), }; } } export function take(iterable: Iterable, amount: number): Iterable { return new TakeIterable(iterable, amount); } class TakeIterable implements Iterable { constructor( private readonly iterable: Iterable, private readonly amount: number ) {} [Symbol.iterator](): Iterator { return new TakeIterator(this.iterable[Symbol.iterator](), this.amount); } } class TakeIterator implements Iterator { constructor(private readonly iterator: Iterator, private amount: number) {} next(): IteratorResult { const nextIteration = this.iterator.next(); if (nextIteration.done || this.amount === 0) { return { done: true, value: undefined }; } this.amount -= 1; return nextIteration; } }