Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(collections/unstable): add cycle iterator utility #6386

Open
wants to merge 10 commits into
base: main
Choose a base branch
from
3 changes: 2 additions & 1 deletion collections/deno.json
Original file line number Diff line number Diff line change
Expand Up @@ -58,6 +58,7 @@
"./unstable-without-all": "./unstable_without_all.ts",
"./unzip": "./unzip.ts",
"./without-all": "./without_all.ts",
"./zip": "./zip.ts"
"./zip": "./zip.ts",
"./unstable-cycle": "./cycle.ts"
}
}
1 change: 1 addition & 0 deletions collections/mod.ts
Original file line number Diff line number Diff line change
Expand Up @@ -74,3 +74,4 @@ export * from "./union.ts";
export * from "./unzip.ts";
export * from "./without_all.ts";
export * from "./zip.ts";
export * from "./unstable_cycle.ts";
42 changes: 42 additions & 0 deletions collections/unstable_cycle.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,42 @@
// Copyright 2018-2025 the Deno authors. MIT license.

/**
* Creates an iterator that cycles indefinitely over the provided iterable.
*
* Each time the iterable is exhausted, a new iterator is obtained to start the cycle again.
* This generator will yield the values from the iterable continuously.
*
* > **Note:** If the iterable is empty, the generator will keep restarting and yield nothing.
*
* @typeParam T The type of the elements in the iterable.
* @param iterable The iterable whose values are to be cycled.
* @returns A generator that yields values from the iterable in an endless cycle.
*
* @example Basic usage
* ```ts
* import { cycle } from "@std/collections/cycle";
* import { assertEquals } from "@std/assert";
*
* const cyclic = cycle([1, 2, 3]);
* const result: number[] = [];
*
* for (const num of cyclic) {
* result.push(num);
* if (result.length === 7) break;
* }
*
* assertEquals(result, [1, 2, 3, 1, 2, 3, 1]);
* ```
*/
export function* cycle<T>(iterable: Iterable<T>): Generator<T> {
let iterator = iterable[Symbol.iterator]();

while (true) {
const result = iterator.next();
if (result.done) {
iterator = iterable[Symbol.iterator]();
} else {
yield result.value;
}
}
}
62 changes: 62 additions & 0 deletions collections/unstable_cycle_test.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,62 @@
// Copyright 2018-2025 the Deno authors. MIT license.

import { assertEquals } from "@std/assert";
import { cycle } from "./unstable_cycle.ts";

Deno.test({
name: "cycle() basic usage with an array",
fn() {
const cyclic = cycle([1, 2, 3]);
const result = [];

for (const num of cyclic) {
result.push(num);
if (result.length === 7) break;
}
assertEquals(result, [1, 2, 3, 1, 2, 3, 1]);
},
});

Deno.test({
name: "cycle() works with a string iterable",
fn() {
const cyclic = cycle("ab");
const result = [];

for (const ch of cyclic) {
result.push(ch);
if (result.length === 5) break;
}
assertEquals(result, ["a", "b", "a", "b", "a"]);
},
});

Deno.test({
name: "cycle() works with a single element iterable",
fn() {
const cyclic = cycle([42]);
const result = [];

for (const num of cyclic) {
result.push(num);
if (result.length === 4) break;
}
assertEquals(result, [42, 42, 42, 42]);
},
});

Deno.test({
name: "cycle() does not mutate the input iterable",
fn() {
const input = [1, 2, 3];
const cyclic = cycle(input);

const result = [];
for (const num of cyclic) {
result.push(num);
if (result.length === 5) break;
}

assertEquals(input, [1, 2, 3]);
},
});
Loading