forked from vercel/next.js
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathrecursive-delete.ts
91 lines (80 loc) · 2.26 KB
/
recursive-delete.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
import { Dirent, promises } from 'fs'
import { join, isAbsolute, dirname } from 'path'
import isError from './is-error'
const sleep = (timeout: number) =>
new Promise((resolve) => setTimeout(resolve, timeout))
const unlinkPath = async (p: string, isDir = false, t = 1): Promise<void> => {
try {
if (isDir) {
await promises.rmdir(p)
} else {
await promises.unlink(p)
}
} catch (e) {
const code = isError(e) && e.code
if (
(code === 'EBUSY' ||
code === 'ENOTEMPTY' ||
code === 'EPERM' ||
code === 'EMFILE') &&
t < 3
) {
await sleep(t * 100)
return unlinkPath(p, isDir, t++)
}
if (code === 'ENOENT') {
return
}
throw e
}
}
/**
* Recursively delete directory contents
* @param {string} dir Directory to delete the contents of
* @param {RegExp} [exclude] Exclude based on relative file path
* @param {string} [previousPath] Ensures that parameter dir exists, this is not passed recursively
* @returns Promise void
*/
export async function recursiveDelete(
dir: string,
exclude?: RegExp,
previousPath: string = ''
): Promise<void> {
let result
try {
result = await promises.readdir(dir, { withFileTypes: true })
} catch (e) {
if (isError(e) && e.code === 'ENOENT') {
return
}
throw e
}
await Promise.all(
result.map(async (part: Dirent) => {
const absolutePath = join(dir, part.name)
// readdir does not follow symbolic links
// if part is a symbolic link, follow it using stat
let isDirectory = part.isDirectory()
const isSymlink = part.isSymbolicLink()
if (isSymlink) {
const linkPath = await promises.readlink(absolutePath)
try {
const stats = await promises.stat(
isAbsolute(linkPath)
? linkPath
: join(dirname(absolutePath), linkPath)
)
isDirectory = stats.isDirectory()
} catch (_) {}
}
const pp = join(previousPath, part.name)
const isNotExcluded = !exclude || !exclude.test(pp)
if (isNotExcluded) {
if (isDirectory) {
await recursiveDelete(absolutePath, exclude, pp)
}
return unlinkPath(absolutePath, !isSymlink && isDirectory)
}
})
)
}