const { cwd, chdir, makeTempDir, mkdir, open } = Deno; const { remove } = Deno; import { walk, walkSync, WalkOptions, WalkInfo } from "./walk.ts"; import { assertEquals, assertThrowsAsync } from "../testing/asserts.ts"; export async function testWalk( setup: (arg0: string) => void | Promise, t: Deno.TestFunction ): Promise { const name = t.name; async function fn(): Promise { const origCwd = cwd(); const d = await makeTempDir(); chdir(d); try { await setup(d); await t(); } finally { chdir(origCwd); remove(d, { recursive: true }); } } Deno.test({ name, fn }); } function normalize({ filename }: WalkInfo): string { return filename.replace(/\\/g, "/"); } export async function walkArray( root: string, options: WalkOptions = {} ): Promise { const arr: string[] = []; for await (const w of walk(root, { ...options })) { arr.push(normalize(w)); } arr.sort(); // TODO(ry) Remove sort. The order should be deterministic. const arrSync = Array.from(walkSync(root, options), normalize); arrSync.sort(); // TODO(ry) Remove sort. The order should be deterministic. assertEquals(arr, arrSync); return arr; } export async function touch(path: string): Promise { await open(path, "w"); } function assertReady(expectedLength: number): void { const arr = Array.from(walkSync("."), normalize); assertEquals(arr.length, expectedLength); } testWalk( async (d: string): Promise => { await mkdir(d + "/empty"); }, async function emptyDir(): Promise { const arr = await walkArray("."); assertEquals(arr, [".", "empty"]); } ); testWalk( async (d: string): Promise => { await touch(d + "/x"); }, async function singleFile(): Promise { const arr = await walkArray("."); assertEquals(arr, [".", "x"]); } ); testWalk( async (d: string): Promise => { await touch(d + "/x"); }, async function iteratable(): Promise { let count = 0; for (const _ of walkSync(".")) { count += 1; } assertEquals(count, 2); for await (const _ of walk(".")) { count += 1; } assertEquals(count, 4); } ); testWalk( async (d: string): Promise => { await mkdir(d + "/a"); await touch(d + "/a/x"); }, async function nestedSingleFile(): Promise { const arr = await walkArray("."); assertEquals(arr, [".", "a", "a/x"]); } ); testWalk( async (d: string): Promise => { await mkdir(d + "/a/b/c/d", { recursive: true }); await touch(d + "/a/b/c/d/x"); }, async function depth(): Promise { assertReady(6); const arr3 = await walkArray(".", { maxDepth: 3 }); assertEquals(arr3, [".", "a", "a/b", "a/b/c"]); const arr5 = await walkArray(".", { maxDepth: 5 }); assertEquals(arr5, [".", "a", "a/b", "a/b/c", "a/b/c/d", "a/b/c/d/x"]); } ); testWalk( async (d: string): Promise => { await touch(d + "/a"); await mkdir(d + "/b"); await touch(d + "/b/c"); }, async function includeDirs(): Promise { assertReady(4); const arr = await walkArray(".", { includeDirs: false }); assertEquals(arr, ["a", "b/c"]); } ); testWalk( async (d: string): Promise => { await touch(d + "/a"); await mkdir(d + "/b"); await touch(d + "/b/c"); }, async function includeFiles(): Promise { assertReady(4); const arr = await walkArray(".", { includeFiles: false }); assertEquals(arr, [".", "b"]); } ); testWalk( async (d: string): Promise => { await touch(d + "/x.ts"); await touch(d + "/y.rs"); }, async function ext(): Promise { assertReady(3); const arr = await walkArray(".", { exts: [".ts"] }); assertEquals(arr, ["x.ts"]); } ); testWalk( async (d: string): Promise => { await touch(d + "/x.ts"); await touch(d + "/y.rs"); await touch(d + "/z.py"); }, async function extAny(): Promise { assertReady(4); const arr = await walkArray(".", { exts: [".rs", ".ts"] }); assertEquals(arr, ["x.ts", "y.rs"]); } ); testWalk( async (d: string): Promise => { await touch(d + "/x"); await touch(d + "/y"); }, async function match(): Promise { assertReady(3); const arr = await walkArray(".", { match: [/x/] }); assertEquals(arr, ["x"]); } ); testWalk( async (d: string): Promise => { await touch(d + "/x"); await touch(d + "/y"); await touch(d + "/z"); }, async function matchAny(): Promise { assertReady(4); const arr = await walkArray(".", { match: [/x/, /y/] }); assertEquals(arr, ["x", "y"]); } ); testWalk( async (d: string): Promise => { await touch(d + "/x"); await touch(d + "/y"); }, async function skip(): Promise { assertReady(3); const arr = await walkArray(".", { skip: [/x/] }); assertEquals(arr, [".", "y"]); } ); testWalk( async (d: string): Promise => { await touch(d + "/x"); await touch(d + "/y"); await touch(d + "/z"); }, async function skipAny(): Promise { assertReady(4); const arr = await walkArray(".", { skip: [/x/, /y/] }); assertEquals(arr, [".", "z"]); } ); testWalk( async (d: string): Promise => { await mkdir(d + "/a"); await mkdir(d + "/b"); await touch(d + "/a/x"); await touch(d + "/a/y"); await touch(d + "/b/z"); }, async function subDir(): Promise { assertReady(6); const arr = await walkArray("b"); assertEquals(arr, ["b", "b/z"]); } ); testWalk( async (_d: string): Promise => {}, async function nonexistentRoot(): Promise { await assertThrowsAsync(async () => { await walkArray("nonexistent"); }, Deno.Err.NotFound); } ); /* TODO(ry) Re-enable followSymlinks testWalk( async (d: string): Promise => { await mkdir(d + "/a"); await mkdir(d + "/b"); await touch(d + "/a/x"); await touch(d + "/a/y"); await touch(d + "/b/z"); try { await symlink(d + "/b", d + "/a/bb"); } catch (err) { assert(isWindows); assert(err.message, "Not implemented"); } }, async function symlink(): Promise { // symlink is not yet implemented on Windows. if (isWindows) { return; } assertReady(6); const files = await walkArray("a"); assertEquals(files.length, 2); assert(!files.includes("a/bb/z")); const arr = await walkArray("a", { followSymlinks: true }); assertEquals(arr.length, 3); assert(arr.some((f): boolean => f.endsWith("/b/z"))); } ); */