const { cwd, chdir, makeTempDir, mkdir, open, build, remove, symlink } = Deno; type FileInfo = Deno.FileInfo; import { walk, walkSync, WalkOptions } from "./walk.ts"; import { test, TestFunction } from "../testing/mod.ts"; import { assert, assertEquals } from "../testing/asserts.ts"; const isWindows = build.os === "win"; export async function testWalk( setup: (string) => void | Promise, t: 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 }); } } test({ name, fn }); } async function walkArray( dirname: string = ".", options: WalkOptions = {} ): Promise { const arr: string[] = []; for await (const f of walk(dirname, { ...options })) { arr.push(f.path.replace(/\\/g, "/")); } arr.sort(); const arrSync = Array.from(walkSync(dirname, options), (f: FileInfo) => f.path.replace(/\\/g, "/") ).sort(); assertEquals(arr, arrSync); return arr; } async function touch(path: string): Promise { await open(path, "w"); } function assertReady(expectedLength: number): void { const arr = Array.from(walkSync(), (f: FileInfo) => f.path); assertEquals(arr.length, expectedLength); } testWalk( async (d: string) => { await mkdir(d + "/empty"); }, async function emptyDir() { const arr = await walkArray(); assertEquals(arr.length, 0); } ); testWalk( async (d: string) => { await touch(d + "/x"); }, async function singleFile() { const arr = await walkArray(); assertEquals(arr.length, 1); assertEquals(arr[0], "./x"); } ); testWalk( async (d: string) => { await touch(d + "/x"); }, async function iteratable() { let count = 0; for (const _ of walkSync()) { count += 1; } assertEquals(count, 1); for await (const _ of walk()) { count += 1; } assertEquals(count, 2); } ); testWalk( async (d: string) => { await mkdir(d + "/a"); await touch(d + "/a/x"); }, async function nestedSingleFile() { const arr = await walkArray(); assertEquals(arr.length, 1); assertEquals(arr[0], "./a/x"); } ); testWalk( async (d: string) => { await mkdir(d + "/a/b/c/d", true); await touch(d + "/a/b/c/d/x"); }, async function depth() { assertReady(1); const arr3 = await walkArray(".", { maxDepth: 3 }); assertEquals(arr3.length, 0); const arr5 = await walkArray(".", { maxDepth: 5 }); assertEquals(arr5.length, 1); assertEquals(arr5[0], "./a/b/c/d/x"); } ); testWalk( async (d: string) => { await touch(d + "/x.ts"); await touch(d + "/y.rs"); }, async function ext() { assertReady(2); const arr = await walkArray(".", { exts: [".ts"] }); assertEquals(arr.length, 1); assertEquals(arr[0], "./x.ts"); } ); testWalk( async (d: string) => { await touch(d + "/x.ts"); await touch(d + "/y.rs"); await touch(d + "/z.py"); }, async function extAny() { assertReady(3); const arr = await walkArray(".", { exts: [".rs", ".ts"] }); assertEquals(arr.length, 2); assertEquals(arr[0], "./x.ts"); assertEquals(arr[1], "./y.rs"); } ); testWalk( async (d: string) => { await touch(d + "/x"); await touch(d + "/y"); }, async function match() { assertReady(2); const arr = await walkArray(".", { match: [/x/] }); assertEquals(arr.length, 1); assertEquals(arr[0], "./x"); } ); testWalk( async (d: string) => { await touch(d + "/x"); await touch(d + "/y"); await touch(d + "/z"); }, async function matchAny() { assertReady(3); const arr = await walkArray(".", { match: [/x/, /y/] }); assertEquals(arr.length, 2); assertEquals(arr[0], "./x"); assertEquals(arr[1], "./y"); } ); testWalk( async (d: string) => { await touch(d + "/x"); await touch(d + "/y"); }, async function skip() { assertReady(2); const arr = await walkArray(".", { skip: [/x/] }); assertEquals(arr.length, 1); assertEquals(arr[0], "./y"); } ); testWalk( async (d: string) => { await touch(d + "/x"); await touch(d + "/y"); await touch(d + "/z"); }, async function skipAny() { assertReady(3); const arr = await walkArray(".", { skip: [/x/, /y/] }); assertEquals(arr.length, 1); assertEquals(arr[0], "./z"); } ); testWalk( async (d: string) => { 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() { assertReady(3); const arr = await walkArray("b"); assertEquals(arr.length, 1); assertEquals(arr[0], "b/z"); } ); testWalk(async (_d: string) => {}, async function onError() { assertReady(0); const ignored = await walkArray("missing"); assertEquals(ignored.length, 0); let errors = 0; await walkArray("missing", { onError: _e => (errors += 1) }); // It's 2 since walkArray iterates over both sync and async. assertEquals(errors, 2); }); testWalk( async (d: string) => { 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() { // symlink is not yet implemented on Windows. if (isWindows) { return; } assertReady(3); 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 => f.endsWith("/b/z"))); } );