2019-05-14 17:14:08 -04:00
|
|
|
// Documentation and interface for walk were adapted from Go
|
|
|
|
// https://golang.org/pkg/path/filepath/#Walk
|
|
|
|
// Copyright 2009 The Go Authors. All rights reserved. BSD license.
|
2020-02-07 16:23:38 +09:00
|
|
|
import { unimplemented, assert } from "../testing/asserts.ts";
|
2020-04-29 22:00:31 +02:00
|
|
|
import { basename, join, normalize } from "../path/mod.ts";
|
2020-03-06 08:34:02 -05:00
|
|
|
const { readdir, readdirSync, stat, statSync } = Deno;
|
2019-02-15 08:20:59 -08:00
|
|
|
|
2020-04-29 22:00:31 +02:00
|
|
|
export function createWalkEntrySync(path: string): WalkEntry {
|
|
|
|
path = normalize(path);
|
|
|
|
const name = basename(path);
|
|
|
|
const info = statSync(path);
|
|
|
|
return {
|
|
|
|
path,
|
|
|
|
name,
|
|
|
|
isFile: info.isFile,
|
|
|
|
isDirectory: info.isDirectory,
|
|
|
|
isSymlink: info.isSymlink,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
export async function createWalkEntry(path: string): Promise<WalkEntry> {
|
|
|
|
path = normalize(path);
|
|
|
|
const name = basename(path);
|
|
|
|
const info = await stat(path);
|
|
|
|
return {
|
|
|
|
path,
|
|
|
|
name,
|
|
|
|
isFile: info.isFile,
|
|
|
|
isDirectory: info.isDirectory,
|
|
|
|
isSymlink: info.isSymlink,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2019-02-15 08:20:59 -08:00
|
|
|
export interface WalkOptions {
|
|
|
|
maxDepth?: number;
|
2019-10-02 18:59:27 +01:00
|
|
|
includeFiles?: boolean;
|
2019-09-18 16:37:37 +01:00
|
|
|
includeDirs?: boolean;
|
2019-10-02 18:59:27 +01:00
|
|
|
followSymlinks?: boolean;
|
2019-02-15 08:20:59 -08:00
|
|
|
exts?: string[];
|
|
|
|
match?: RegExp[];
|
|
|
|
skip?: RegExp[];
|
2019-03-12 14:51:51 +09:00
|
|
|
}
|
|
|
|
|
2019-11-15 03:22:33 +00:00
|
|
|
function include(
|
2020-04-29 22:00:31 +02:00
|
|
|
path: string,
|
2019-11-15 03:22:33 +00:00
|
|
|
exts?: string[],
|
|
|
|
match?: RegExp[],
|
|
|
|
skip?: RegExp[]
|
|
|
|
): boolean {
|
2020-04-29 22:00:31 +02:00
|
|
|
if (exts && !exts.some((ext): boolean => path.endsWith(ext))) {
|
2019-03-12 14:51:51 +09:00
|
|
|
return false;
|
|
|
|
}
|
2020-04-29 22:00:31 +02:00
|
|
|
if (match && !match.some((pattern): boolean => !!path.match(pattern))) {
|
2019-03-12 14:51:51 +09:00
|
|
|
return false;
|
|
|
|
}
|
2020-04-29 22:00:31 +02:00
|
|
|
if (skip && skip.some((pattern): boolean => !!path.match(pattern))) {
|
2019-03-12 14:51:51 +09:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2020-04-29 22:00:31 +02:00
|
|
|
export interface WalkEntry extends Deno.DirEntry {
|
|
|
|
path: string;
|
2019-03-12 14:51:51 +09:00
|
|
|
}
|
|
|
|
|
2019-09-18 16:37:37 +01:00
|
|
|
/** Walks the file tree rooted at root, yielding each file or directory in the
|
|
|
|
* tree filtered according to the given options. The files are walked in lexical
|
2019-05-14 17:14:08 -04:00
|
|
|
* order, which makes the output deterministic but means that for very large
|
|
|
|
* directories walk() can be inefficient.
|
|
|
|
*
|
|
|
|
* Options:
|
2019-10-02 18:59:27 +01:00
|
|
|
* - maxDepth?: number = Infinity;
|
|
|
|
* - includeFiles?: boolean = true;
|
|
|
|
* - includeDirs?: boolean = true;
|
|
|
|
* - followSymlinks?: boolean = false;
|
2019-05-14 17:14:08 -04:00
|
|
|
* - exts?: string[];
|
|
|
|
* - match?: RegExp[];
|
|
|
|
* - skip?: RegExp[];
|
2019-02-15 08:20:59 -08:00
|
|
|
*
|
2020-04-29 22:00:31 +02:00
|
|
|
* for await (const { name, info } of walk(".")) {
|
|
|
|
* console.log(name);
|
2020-04-16 06:40:30 +01:00
|
|
|
* assert(info.isFile);
|
2019-02-15 08:20:59 -08:00
|
|
|
* };
|
|
|
|
*/
|
|
|
|
export async function* walk(
|
2019-05-14 17:14:08 -04:00
|
|
|
root: string,
|
2019-11-15 03:22:33 +00:00
|
|
|
{
|
|
|
|
maxDepth = Infinity,
|
|
|
|
includeFiles = true,
|
|
|
|
includeDirs = true,
|
|
|
|
followSymlinks = false,
|
2020-02-19 21:36:18 +01:00
|
|
|
exts = undefined,
|
|
|
|
match = undefined,
|
2020-03-29 04:03:49 +11:00
|
|
|
skip = undefined,
|
2019-11-15 03:22:33 +00:00
|
|
|
}: WalkOptions = {}
|
2020-04-16 06:40:30 +01:00
|
|
|
): AsyncIterableIterator<WalkEntry> {
|
2019-10-02 18:59:27 +01:00
|
|
|
if (maxDepth < 0) {
|
|
|
|
return;
|
|
|
|
}
|
2019-11-15 03:22:33 +00:00
|
|
|
if (includeDirs && include(root, exts, match, skip)) {
|
2020-04-29 22:00:31 +02:00
|
|
|
yield await createWalkEntry(root);
|
2019-10-02 18:59:27 +01:00
|
|
|
}
|
2020-02-19 21:36:18 +01:00
|
|
|
if (maxDepth < 1 || !include(root, undefined, undefined, skip)) {
|
2019-10-02 18:59:27 +01:00
|
|
|
return;
|
2019-09-18 16:37:37 +01:00
|
|
|
}
|
2020-04-29 22:00:31 +02:00
|
|
|
for await (const entry of readdir(root)) {
|
|
|
|
if (entry.isSymlink) {
|
2019-11-15 03:22:33 +00:00
|
|
|
if (followSymlinks) {
|
2019-05-14 17:14:08 -04:00
|
|
|
// TODO(ry) Re-enable followSymlinks.
|
|
|
|
unimplemented();
|
2019-02-15 08:20:59 -08:00
|
|
|
} else {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
2019-05-14 17:14:08 -04:00
|
|
|
|
2020-04-29 22:00:31 +02:00
|
|
|
assert(entry.name != null);
|
|
|
|
const path = join(root, entry.name);
|
2019-05-14 17:14:08 -04:00
|
|
|
|
2020-04-29 22:00:31 +02:00
|
|
|
if (entry.isFile) {
|
|
|
|
if (includeFiles && include(path, exts, match, skip)) {
|
|
|
|
yield { path, ...entry };
|
2019-02-15 08:20:59 -08:00
|
|
|
}
|
|
|
|
} else {
|
2020-04-29 22:00:31 +02:00
|
|
|
yield* walk(path, {
|
2019-11-15 03:22:33 +00:00
|
|
|
maxDepth: maxDepth - 1,
|
|
|
|
includeFiles,
|
|
|
|
includeDirs,
|
|
|
|
followSymlinks,
|
|
|
|
exts,
|
|
|
|
match,
|
2020-03-29 04:03:49 +11:00
|
|
|
skip,
|
2019-11-15 03:22:33 +00:00
|
|
|
});
|
2019-02-15 08:20:59 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-14 17:14:08 -04:00
|
|
|
/** Same as walk() but uses synchronous ops */
|
2019-02-15 08:20:59 -08:00
|
|
|
export function* walkSync(
|
2019-10-02 18:59:27 +01:00
|
|
|
root: string,
|
2019-11-15 03:22:33 +00:00
|
|
|
{
|
|
|
|
maxDepth = Infinity,
|
|
|
|
includeFiles = true,
|
|
|
|
includeDirs = true,
|
|
|
|
followSymlinks = false,
|
2020-02-19 21:36:18 +01:00
|
|
|
exts = undefined,
|
|
|
|
match = undefined,
|
2020-03-29 04:03:49 +11:00
|
|
|
skip = undefined,
|
2019-11-15 03:22:33 +00:00
|
|
|
}: WalkOptions = {}
|
2020-04-16 06:40:30 +01:00
|
|
|
): IterableIterator<WalkEntry> {
|
2019-10-02 18:59:27 +01:00
|
|
|
if (maxDepth < 0) {
|
|
|
|
return;
|
|
|
|
}
|
2019-11-15 03:22:33 +00:00
|
|
|
if (includeDirs && include(root, exts, match, skip)) {
|
2020-04-29 22:00:31 +02:00
|
|
|
yield createWalkEntrySync(root);
|
2019-10-02 18:59:27 +01:00
|
|
|
}
|
2020-02-19 21:36:18 +01:00
|
|
|
if (maxDepth < 1 || !include(root, undefined, undefined, skip)) {
|
2019-10-02 18:59:27 +01:00
|
|
|
return;
|
2019-09-18 16:37:37 +01:00
|
|
|
}
|
2020-04-29 22:00:31 +02:00
|
|
|
for (const entry of readdirSync(root)) {
|
|
|
|
if (entry.isSymlink) {
|
2019-11-15 03:22:33 +00:00
|
|
|
if (followSymlinks) {
|
2019-05-14 17:14:08 -04:00
|
|
|
unimplemented();
|
2019-02-15 08:20:59 -08:00
|
|
|
} else {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
2019-05-14 17:14:08 -04:00
|
|
|
|
2020-04-29 22:00:31 +02:00
|
|
|
assert(entry.name != null);
|
|
|
|
const path = join(root, entry.name);
|
2019-05-14 17:14:08 -04:00
|
|
|
|
2020-04-29 22:00:31 +02:00
|
|
|
if (entry.isFile) {
|
|
|
|
if (includeFiles && include(path, exts, match, skip)) {
|
|
|
|
yield { path, ...entry };
|
2019-02-15 08:20:59 -08:00
|
|
|
}
|
|
|
|
} else {
|
2020-04-29 22:00:31 +02:00
|
|
|
yield* walkSync(path, {
|
2019-11-15 03:22:33 +00:00
|
|
|
maxDepth: maxDepth - 1,
|
|
|
|
includeFiles,
|
|
|
|
includeDirs,
|
|
|
|
followSymlinks,
|
|
|
|
exts,
|
|
|
|
match,
|
2020-03-29 04:03:49 +11:00
|
|
|
skip,
|
2019-11-15 03:22:33 +00:00
|
|
|
});
|
2019-02-15 08:20:59 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|