UNPKG

rimraf

Version:

A deep deletion module for node (like `rm -rf`)

159 lines 6.48 kB
"use strict"; // This is the same as rimrafPosix, with the following changes: // // 1. EBUSY, ENFILE, EMFILE trigger retries and/or exponential backoff // 2. All non-directories are removed first and then all directories are // removed in a second sweep. // 3. If we hit ENOTEMPTY in the second sweep, fall back to move-remove on // the that folder. // // Note: "move then remove" is 2-10 times slower, and just as unreliable. Object.defineProperty(exports, "__esModule", { value: true }); exports.rimrafWindowsSync = exports.rimrafWindows = void 0; const path_1 = require("path"); const fix_eperm_js_1 = require("./fix-eperm.js"); const fs_js_1 = require("./fs.js"); const ignore_enoent_js_1 = require("./ignore-enoent.js"); const readdir_or_error_js_1 = require("./readdir-or-error.js"); const retry_busy_js_1 = require("./retry-busy.js"); const rimraf_move_remove_js_1 = require("./rimraf-move-remove.js"); const error_js_1 = require("./error.js"); const { unlink, rmdir, lstat } = fs_js_1.promises; const rimrafWindowsFile = (0, retry_busy_js_1.retryBusy)((0, fix_eperm_js_1.fixEPERM)(unlink)); const rimrafWindowsFileSync = (0, retry_busy_js_1.retryBusySync)((0, fix_eperm_js_1.fixEPERMSync)(fs_js_1.unlinkSync)); const rimrafWindowsDirRetry = (0, retry_busy_js_1.retryBusy)((0, fix_eperm_js_1.fixEPERM)(rmdir)); const rimrafWindowsDirRetrySync = (0, retry_busy_js_1.retryBusySync)((0, fix_eperm_js_1.fixEPERMSync)(fs_js_1.rmdirSync)); const rimrafWindowsDirMoveRemoveFallback = async (path, // already filtered, remove from options so we don't call unnecessarily // eslint-disable-next-line @typescript-eslint/no-unused-vars { filter, ...opt }) => { /* c8 ignore next */ opt?.signal?.throwIfAborted(); try { await rimrafWindowsDirRetry(path, opt); return true; } catch (er) { if ((0, error_js_1.errorCode)(er) === 'ENOTEMPTY') { return (0, rimraf_move_remove_js_1.rimrafMoveRemove)(path, opt); } throw er; } }; const rimrafWindowsDirMoveRemoveFallbackSync = (path, // already filtered, remove from options so we don't call unnecessarily // eslint-disable-next-line @typescript-eslint/no-unused-vars { filter, ...opt }) => { opt?.signal?.throwIfAborted(); try { rimrafWindowsDirRetrySync(path, opt); return true; } catch (er) { if ((0, error_js_1.errorCode)(er) === 'ENOTEMPTY') { return (0, rimraf_move_remove_js_1.rimrafMoveRemoveSync)(path, opt); } throw er; } }; const START = Symbol('start'); const CHILD = Symbol('child'); const FINISH = Symbol('finish'); const rimrafWindows = async (path, opt) => { opt?.signal?.throwIfAborted(); return ((await (0, ignore_enoent_js_1.ignoreENOENT)(lstat(path).then(stat => rimrafWindowsDir(path, opt, stat, START)))) ?? true); }; exports.rimrafWindows = rimrafWindows; const rimrafWindowsSync = (path, opt) => { opt?.signal?.throwIfAborted(); return ((0, ignore_enoent_js_1.ignoreENOENTSync)(() => rimrafWindowsDirSync(path, opt, (0, fs_js_1.lstatSync)(path), START)) ?? true); }; exports.rimrafWindowsSync = rimrafWindowsSync; const rimrafWindowsDir = async (path, opt, ent, state = START) => { opt?.signal?.throwIfAborted(); const entries = ent.isDirectory() ? await (0, readdir_or_error_js_1.readdirOrError)(path) : null; if (!Array.isArray(entries)) { // this can only happen if lstat/readdir lied, or if the dir was // swapped out with a file at just the right moment. /* c8 ignore start */ if (entries) { if ((0, error_js_1.errorCode)(entries) === 'ENOENT') { return true; } if ((0, error_js_1.errorCode)(entries) !== 'ENOTDIR') { throw entries; } } /* c8 ignore stop */ if (opt.filter && !(await opt.filter(path, ent))) { return false; } // is a file await (0, ignore_enoent_js_1.ignoreENOENT)(rimrafWindowsFile(path, opt)); return true; } const s = state === START ? CHILD : state; const removedAll = (await Promise.all(entries.map(ent => rimrafWindowsDir((0, path_1.resolve)(path, ent.name), opt, ent, s)))).every(v => v === true); if (state === START) { return rimrafWindowsDir(path, opt, ent, FINISH); } else if (state === FINISH) { if (opt.preserveRoot === false && path === (0, path_1.parse)(path).root) { return false; } if (!removedAll) { return false; } if (opt.filter && !(await opt.filter(path, ent))) { return false; } await (0, ignore_enoent_js_1.ignoreENOENT)(rimrafWindowsDirMoveRemoveFallback(path, opt)); } return true; }; const rimrafWindowsDirSync = (path, opt, ent, state = START) => { const entries = ent.isDirectory() ? (0, readdir_or_error_js_1.readdirOrErrorSync)(path) : null; if (!Array.isArray(entries)) { // this can only happen if lstat/readdir lied, or if the dir was // swapped out with a file at just the right moment. /* c8 ignore start */ if (entries) { if ((0, error_js_1.errorCode)(entries) === 'ENOENT') { return true; } if ((0, error_js_1.errorCode)(entries) !== 'ENOTDIR') { throw entries; } } /* c8 ignore stop */ if (opt.filter && !opt.filter(path, ent)) { return false; } // is a file (0, ignore_enoent_js_1.ignoreENOENTSync)(() => rimrafWindowsFileSync(path, opt)); return true; } let removedAll = true; for (const ent of entries) { const s = state === START ? CHILD : state; const p = (0, path_1.resolve)(path, ent.name); removedAll = rimrafWindowsDirSync(p, opt, ent, s) && removedAll; } if (state === START) { return rimrafWindowsDirSync(path, opt, ent, FINISH); } else if (state === FINISH) { if (opt.preserveRoot === false && path === (0, path_1.parse)(path).root) { return false; } if (!removedAll) { return false; } if (opt.filter && !opt.filter(path, ent)) { return false; } (0, ignore_enoent_js_1.ignoreENOENTSync)(() => rimrafWindowsDirMoveRemoveFallbackSync(path, opt)); } return true; }; //# sourceMappingURL=rimraf-windows.js.map