123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651 |
- /*
- MIT License http://www.opensource.org/licenses/mit-license.php
- Author Tobias Koppers @sokra
- */
- "use strict";
- const path = require("path");
- /** @typedef {import("../../declarations/WebpackOptions").WatchOptions} WatchOptions */
- /** @typedef {import("../FileSystemInfo").FileSystemInfoEntry} FileSystemInfoEntry */
- /**
- * @template T
- * @typedef {object} IStatsBase
- * @property {() => boolean} isFile
- * @property {() => boolean} isDirectory
- * @property {() => boolean} isBlockDevice
- * @property {() => boolean} isCharacterDevice
- * @property {() => boolean} isSymbolicLink
- * @property {() => boolean} isFIFO
- * @property {() => boolean} isSocket
- * @property {T} dev
- * @property {T} ino
- * @property {T} mode
- * @property {T} nlink
- * @property {T} uid
- * @property {T} gid
- * @property {T} rdev
- * @property {T} size
- * @property {T} blksize
- * @property {T} blocks
- * @property {T} atimeMs
- * @property {T} mtimeMs
- * @property {T} ctimeMs
- * @property {T} birthtimeMs
- * @property {Date} atime
- * @property {Date} mtime
- * @property {Date} ctime
- * @property {Date} birthtime
- */
- /**
- * @typedef {IStatsBase<number>} IStats
- */
- /**
- * @typedef {IStatsBase<bigint> & { atimeNs: bigint, mtimeNs: bigint, ctimeNs: bigint, birthtimeNs: bigint }} IBigIntStats
- */
- /**
- * @typedef {object} Dirent
- * @property {() => boolean} isFile
- * @property {() => boolean} isDirectory
- * @property {() => boolean} isBlockDevice
- * @property {() => boolean} isCharacterDevice
- * @property {() => boolean} isSymbolicLink
- * @property {() => boolean} isFIFO
- * @property {() => boolean} isSocket
- * @property {string} name
- * @property {string} path
- */
- /** @typedef {string | number | boolean | null} JsonPrimitive */
- /** @typedef {JsonValue[]} JsonArray */
- /** @typedef {JsonPrimitive | JsonObject | JsonArray} JsonValue */
- /** @typedef {{[Key in string]: JsonValue} & {[Key in string]?: JsonValue | undefined}} JsonObject */
- /** @typedef {function(NodeJS.ErrnoException | null): void} NoParamCallback */
- /** @typedef {function(NodeJS.ErrnoException | null, string=): void} StringCallback */
- /** @typedef {function(NodeJS.ErrnoException | null, Buffer=): void} BufferCallback */
- /** @typedef {function(NodeJS.ErrnoException | null, (string | Buffer)=): void} StringOrBufferCallback */
- /** @typedef {function(NodeJS.ErrnoException | null, (string[])=): void} ReaddirStringCallback */
- /** @typedef {function(NodeJS.ErrnoException | null, (Buffer[])=): void} ReaddirBufferCallback */
- /** @typedef {function(NodeJS.ErrnoException | null, (string[] | Buffer[])=): void} ReaddirStringOrBufferCallback */
- /** @typedef {function(NodeJS.ErrnoException | null, (Dirent[])=): void} ReaddirDirentCallback */
- /** @typedef {function(NodeJS.ErrnoException | null, IStats=): void} StatsCallback */
- /** @typedef {function(NodeJS.ErrnoException | null, IBigIntStats=): void} BigIntStatsCallback */
- /** @typedef {function(NodeJS.ErrnoException | null, (IStats | IBigIntStats)=): void} StatsOrBigIntStatsCallback */
- /** @typedef {function(NodeJS.ErrnoException | null, number=): void} NumberCallback */
- /** @typedef {function(NodeJS.ErrnoException | Error | null, JsonObject=): void} ReadJsonCallback */
- /**
- * @typedef {object} WatcherInfo
- * @property {Set<string>} changes get current aggregated changes that have not yet send to callback
- * @property {Set<string>} removals get current aggregated removals that have not yet send to callback
- * @property {Map<string, FileSystemInfoEntry | "ignore">} fileTimeInfoEntries get info about files
- * @property {Map<string, FileSystemInfoEntry | "ignore">} contextTimeInfoEntries get info about directories
- */
- // TODO webpack 6 deprecate missing getInfo
- /**
- * @typedef {object} Watcher
- * @property {function(): void} close closes the watcher and all underlying file watchers
- * @property {function(): void} pause closes the watcher, but keeps underlying file watchers alive until the next watch call
- * @property {function(): Set<string>=} getAggregatedChanges get current aggregated changes that have not yet send to callback
- * @property {function(): Set<string>=} getAggregatedRemovals get current aggregated removals that have not yet send to callback
- * @property {function(): Map<string, FileSystemInfoEntry | "ignore">} getFileTimeInfoEntries get info about files
- * @property {function(): Map<string, FileSystemInfoEntry | "ignore">} getContextTimeInfoEntries get info about directories
- * @property {function(): WatcherInfo=} getInfo get info about timestamps and changes
- */
- /**
- * @callback WatchMethod
- * @param {Iterable<string>} files watched files
- * @param {Iterable<string>} directories watched directories
- * @param {Iterable<string>} missing watched existence entries
- * @param {number} startTime timestamp of start time
- * @param {WatchOptions} options options object
- * @param {function(Error | null, Map<string, FileSystemInfoEntry | "ignore">, Map<string, FileSystemInfoEntry | "ignore">, Set<string>, Set<string>): void} callback aggregated callback
- * @param {function(string, number): void} callbackUndelayed callback when the first change was detected
- * @returns {Watcher} a watcher
- */
- // TODO webpack 6 make optional methods required and avoid using non standard methods like `join`, `relative`, `dirname`, move IntermediateFileSystemExtras methods to InputFilesystem or OutputFilesystem
- /**
- * @typedef {string | Buffer | URL} PathLike
- */
- /**
- * @typedef {PathLike | number} PathOrFileDescriptor
- */
- /**
- * @typedef {object} ObjectEncodingOptions
- * @property {BufferEncoding | null | undefined} [encoding]
- */
- /**
- * @typedef {{
- * (path: PathOrFileDescriptor, options: ({ encoding?: null | undefined, flag?: string | undefined } & import("events").Abortable) | undefined | null, callback: BufferCallback): void;
- * (path: PathOrFileDescriptor, options: ({ encoding: BufferEncoding, flag?: string | undefined } & import("events").Abortable) | BufferEncoding, callback: StringCallback): void;
- * (path: PathOrFileDescriptor, options: (ObjectEncodingOptions & { flag?: string | undefined } & import("events").Abortable) | BufferEncoding | undefined | null, callback: StringOrBufferCallback): void;
- * (path: PathOrFileDescriptor, callback: BufferCallback): void;
- * }} ReadFile
- */
- /**
- * @typedef {{
- * (path: PathOrFileDescriptor, options?: { encoding?: null | undefined, flag?: string | undefined } | null): Buffer;
- * (path: PathOrFileDescriptor, options: { encoding: BufferEncoding, flag?: string | undefined } | BufferEncoding): string;
- * (path: PathOrFileDescriptor, options?: (ObjectEncodingOptions & { flag?: string | undefined }) | BufferEncoding | null): string | Buffer;
- * }} ReadFileSync
- */
- /**
- * @typedef {ObjectEncodingOptions | BufferEncoding | undefined | null} EncodingOption
- */
- /**
- * @typedef {'buffer'| { encoding: 'buffer' }} BufferEncodingOption
- */
- /**
- * @typedef {object} StatOptions
- * @property {(boolean | undefined)=} bigint
- */
- /**
- * @typedef {object} StatSyncOptions
- * @property {(boolean | undefined)=} bigint
- * @property {(boolean | undefined)=} throwIfNoEntry
- */
- /**
- * @typedef {{
- * (path: PathLike, options: EncodingOption, callback: StringCallback): void;
- * (path: PathLike, options: BufferEncodingOption, callback: BufferCallback): void;
- * (path: PathLike, options: EncodingOption, callback: StringOrBufferCallback): void;
- * (path: PathLike, callback: StringCallback): void;
- * }} Readlink
- */
- /**
- * @typedef {{
- * (path: PathLike, options?: EncodingOption): string;
- * (path: PathLike, options: BufferEncodingOption): Buffer;
- * (path: PathLike, options?: EncodingOption): string | Buffer;
- * }} ReadlinkSync
- */
- /**
- * @typedef {{
- * (path: PathLike, options: { encoding: BufferEncoding | null, withFileTypes?: false | undefined, recursive?: boolean | undefined } | BufferEncoding | undefined | null, callback: ReaddirStringCallback): void;
- * (path: PathLike, options: { encoding: 'buffer', withFileTypes?: false | undefined, recursive?: boolean | undefined } | 'buffer', callback: ReaddirBufferCallback): void;
- * (path: PathLike, callback: ReaddirStringCallback): void;
- * (path: PathLike, options: (ObjectEncodingOptions & { withFileTypes?: false | undefined, recursive?: boolean | undefined }) | BufferEncoding | undefined | null, callback: ReaddirStringOrBufferCallback): void;
- * (path: PathLike, options: ObjectEncodingOptions & { withFileTypes: true, recursive?: boolean | undefined }, callback: ReaddirDirentCallback): void;
- * }} Readdir
- */
- /**
- * @typedef {{
- * (path: PathLike, options?: { encoding: BufferEncoding | null, withFileTypes?: false | undefined, recursive?: boolean | undefined } | BufferEncoding | null): string[];
- * (path: PathLike, options: { encoding: 'buffer', withFileTypes?: false | undefined, recursive?: boolean | undefined } | 'buffer'): Buffer[];
- * (path: PathLike, options?: (ObjectEncodingOptions & { withFileTypes?: false | undefined, recursive?: boolean | undefined }) | BufferEncoding | null): string[] | Buffer[];
- * (path: PathLike, options: ObjectEncodingOptions & { withFileTypes: true, recursive?: boolean | undefined }): Dirent[];
- * }} ReaddirSync
- */
- /**
- * @typedef {{
- * (path: PathLike, callback: StatsCallback): void;
- * (path: PathLike, options: (StatOptions & { bigint?: false | undefined }) | undefined, callback: StatsCallback): void;
- * (path: PathLike, options: StatOptions & { bigint: true }, callback: BigIntStatsCallback): void;
- * (path: PathLike, options: StatOptions | undefined, callback: StatsOrBigIntStatsCallback): void;
- * }} Stat
- */
- /**
- * @typedef {{
- * (path: PathLike, options?: undefined): IStats;
- * (path: PathLike, options?: StatSyncOptions & { bigint?: false | undefined, throwIfNoEntry: false }): IStats | undefined;
- * (path: PathLike, options: StatSyncOptions & { bigint: true, throwIfNoEntry: false }): IBigIntStats | undefined;
- * (path: PathLike, options?: StatSyncOptions & { bigint?: false | undefined }): IStats;
- * (path: PathLike, options: StatSyncOptions & { bigint: true }): IBigIntStats;
- * (path: PathLike, options: StatSyncOptions & { bigint: boolean, throwIfNoEntry?: false | undefined }): IStats | IBigIntStats;
- * (path: PathLike, options?: StatSyncOptions): IStats | IBigIntStats | undefined;
- * }} StatSync
- */
- /**
- * @typedef {{
- * (path: PathLike, callback: StatsCallback): void;
- * (path: PathLike, options: (StatOptions & { bigint?: false | undefined }) | undefined, callback: StatsCallback): void;
- * (path: PathLike, options: StatOptions & { bigint: true }, callback: BigIntStatsCallback): void;
- * (path: PathLike, options: StatOptions | undefined, callback: StatsOrBigIntStatsCallback): void;
- * }} LStat
- */
- /**
- * @typedef {{
- * (path: PathLike, options?: undefined): IStats;
- * (path: PathLike, options?: StatSyncOptions & { bigint?: false | undefined, throwIfNoEntry: false }): IStats | undefined;
- * (path: PathLike, options: StatSyncOptions & { bigint: true, throwIfNoEntry: false }): IBigIntStats | undefined;
- * (path: PathLike, options?: StatSyncOptions & { bigint?: false | undefined }): IStats;
- * (path: PathLike, options: StatSyncOptions & { bigint: true }): IBigIntStats;
- * (path: PathLike, options: StatSyncOptions & { bigint: boolean, throwIfNoEntry?: false | undefined }): IStats | IBigIntStats;
- * (path: PathLike, options?: StatSyncOptions): IStats | IBigIntStats | undefined;
- * }} LStatSync
- */
- /**
- * @typedef {{
- * (path: PathLike, options: EncodingOption, callback: StringCallback): void;
- * (path: PathLike, options: BufferEncodingOption, callback: BufferCallback): void;
- * (path: PathLike, options: EncodingOption, callback: StringOrBufferCallback): void;
- * (path: PathLike, callback: StringCallback): void;
- * }} RealPath
- */
- /**
- * @typedef {{
- * (path: PathLike, options?: EncodingOption): string;
- * (path: PathLike, options: BufferEncodingOption): Buffer;
- * (path: PathLike, options?: EncodingOption): string | Buffer;
- * }} RealPathSync
- */
- /**
- * @typedef {function(PathOrFileDescriptor, ReadJsonCallback): void} ReadJson
- */
- /**
- * @typedef {function(PathOrFileDescriptor): JsonObject} ReadJsonSync
- */
- /**
- * @typedef {function((string | string[] | Set<string>)=): void} Purge
- */
- /**
- * @typedef {object} InputFileSystem
- * @property {ReadFile} readFile
- * @property {ReadFileSync=} readFileSync
- * @property {Readlink} readlink
- * @property {ReadlinkSync=} readlinkSync
- * @property {Readdir} readdir
- * @property {ReaddirSync=} readdirSync
- * @property {Stat} stat
- * @property {StatSync=} statSync
- * @property {LStat=} lstat
- * @property {LStatSync=} lstatSync
- * @property {RealPath=} realpath
- * @property {RealPathSync=} realpathSync
- * @property {ReadJson=} readJson
- * @property {ReadJsonSync=} readJsonSync
- * @property {Purge=} purge
- * @property {(function(string, string): string)=} join
- * @property {(function(string, string): string)=} relative
- * @property {(function(string): string)=} dirname
- */
- /**
- * @typedef {number | string} Mode
- */
- /**
- * @typedef {(ObjectEncodingOptions & import("events").Abortable & { mode?: Mode | undefined, flag?: string | undefined, flush?: boolean | undefined }) | BufferEncoding | null} WriteFileOptions
- */
- /**
- * @typedef {{
- * (file: PathOrFileDescriptor, data: string | NodeJS.ArrayBufferView, options: WriteFileOptions, callback: NoParamCallback): void;
- * (file: PathOrFileDescriptor, data: string | NodeJS.ArrayBufferView, callback: NoParamCallback): void;
- * }} WriteFile
- */
- /**
- * @typedef {{ recursive?: boolean | undefined, mode?: Mode | undefined }} MakeDirectoryOptions
- */
- /**
- * @typedef {{
- * (file: PathLike, options: MakeDirectoryOptions & { recursive: true }, callback: StringCallback): void;
- * (file: PathLike, options: Mode | (MakeDirectoryOptions & { recursive?: false | undefined; }) | null | undefined, callback: NoParamCallback): void;
- * (file: PathLike, options: Mode | MakeDirectoryOptions | null | undefined, callback: StringCallback): void;
- * (file: PathLike, callback: NoParamCallback): void;
- * }} Mkdir
- */
- /**
- * @typedef {{ maxRetries?: number | undefined, recursive?: boolean | undefined, retryDelay?: number | undefined }} RmDirOptions
- */
- /**
- * @typedef {{
- * (file: PathLike, callback: NoParamCallback): void;
- * (file: PathLike, options: RmDirOptions, callback: NoParamCallback): void;
- * }} Rmdir
- */
- /**
- * @typedef {function(PathLike, NoParamCallback): void} Unlink
- */
- /**
- * @typedef {object} OutputFileSystem
- * @property {WriteFile} writeFile
- * @property {Mkdir} mkdir
- * @property {Readdir=} readdir
- * @property {Rmdir=} rmdir
- * @property {Unlink=} unlink
- * @property {Stat} stat
- * @property {LStat=} lstat
- * @property {ReadFile} readFile
- * @property {(function(string, string): string)=} join
- * @property {(function(string, string): string)=} relative
- * @property {(function(string): string)=} dirname
- */
- /**
- * @typedef {object} WatchFileSystem
- * @property {WatchMethod} watch
- */
- /**
- * @typedef {{
- * (path: PathLike, options: MakeDirectoryOptions & { recursive: true }): string | undefined;
- * (path: PathLike, options?: Mode | (MakeDirectoryOptions & { recursive?: false | undefined }) | null): void;
- * (path: PathLike, options?: Mode | MakeDirectoryOptions | null): string | undefined;
- * }} MkdirSync
- */
- /**
- * @typedef {object} StreamOptions
- * @property {(string | undefined)=} flags
- * @property {(BufferEncoding | undefined)} encoding
- * @property {(number | any | undefined)=} fd
- * @property {(number | undefined)=} mode
- * @property {(boolean | undefined)=} autoClose
- * @property {(boolean | undefined)=} emitClose
- * @property {(number | undefined)=} start
- * @property {(AbortSignal | null | undefined)=} signal
- */
- /**
- * @typedef {object} FSImplementation
- * @property {((...args: any[]) => any)=} open
- * @property {((...args: any[]) => any)=} close
- */
- /**
- * @typedef {FSImplementation & { write: (...args: any[]) => any; close?: (...args: any[]) => any }} CreateWriteStreamFSImplementation
- */
- /**
- * @typedef {StreamOptions & { fs?: CreateWriteStreamFSImplementation | null | undefined }} WriteStreamOptions
- */
- /**
- * @typedef {function(PathLike, (BufferEncoding | WriteStreamOptions)=): NodeJS.WritableStream} CreateWriteStream
- */
- /**
- * @typedef {number | string} OpenMode
- */
- /**
- * @typedef {{
- * (file: PathLike, flags: OpenMode | undefined, mode: Mode | undefined | null, callback: NumberCallback): void;
- * (file: PathLike, flags: OpenMode | undefined, callback: NumberCallback): void;
- * (file: PathLike, callback: NumberCallback): void;
- * }} Open
- */
- /**
- * @typedef {number | bigint} ReadPosition
- */
- /**
- * @typedef {object} ReadSyncOptions
- * @property {(number | undefined)=} offset
- * @property {(number | undefined)=} length
- * @property {(ReadPosition | null | undefined)=} position
- */
- /**
- * @template {NodeJS.ArrayBufferView} TBuffer
- * @typedef {object} ReadAsyncOptions
- * @property {(number | undefined)=} offset
- * @property {(number | undefined)=} length
- * @property {(ReadPosition | null | undefined)=} position
- * @property {TBuffer=} buffer
- */
- /**
- * @template {NodeJS.ArrayBufferView} [TBuffer=Buffer]
- * @typedef {{
- * (fd: number, buffer: TBuffer, offset: number, length: number, position: ReadPosition | null, callback: (err: NodeJS.ErrnoException | null, bytesRead: number, buffer: TBuffer) => void): void;
- * (fd: number, options: ReadAsyncOptions<TBuffer>, callback: (err: NodeJS.ErrnoException | null, bytesRead: number, buffer: TBuffer) => void): void;
- * (fd: number, callback: (err: NodeJS.ErrnoException | null, bytesRead: number, buffer: NodeJS.ArrayBufferView) => void): void;
- * }} Read
- */
- /** @typedef {function(number, NoParamCallback): void} Close */
- /** @typedef {function(PathLike, PathLike, NoParamCallback): void} Rename */
- /**
- * @typedef {object} IntermediateFileSystemExtras
- * @property {MkdirSync} mkdirSync
- * @property {CreateWriteStream} createWriteStream
- * @property {Open} open
- * @property {Read} read
- * @property {Close} close
- * @property {Rename} rename
- */
- /** @typedef {InputFileSystem & OutputFileSystem & IntermediateFileSystemExtras} IntermediateFileSystem */
- /**
- *
- * @param {InputFileSystem|OutputFileSystem|undefined} fs a file system
- * @param {string} rootPath the root path
- * @param {string} targetPath the target path
- * @returns {string} location of targetPath relative to rootPath
- */
- const relative = (fs, rootPath, targetPath) => {
- if (fs && fs.relative) {
- return fs.relative(rootPath, targetPath);
- } else if (path.posix.isAbsolute(rootPath)) {
- return path.posix.relative(rootPath, targetPath);
- } else if (path.win32.isAbsolute(rootPath)) {
- return path.win32.relative(rootPath, targetPath);
- } else {
- throw new Error(
- `${rootPath} is neither a posix nor a windows path, and there is no 'relative' method defined in the file system`
- );
- }
- };
- exports.relative = relative;
- /**
- * @param {InputFileSystem|OutputFileSystem|undefined} fs a file system
- * @param {string} rootPath a path
- * @param {string} filename a filename
- * @returns {string} the joined path
- */
- const join = (fs, rootPath, filename) => {
- if (fs && fs.join) {
- return fs.join(rootPath, filename);
- } else if (path.posix.isAbsolute(rootPath)) {
- return path.posix.join(rootPath, filename);
- } else if (path.win32.isAbsolute(rootPath)) {
- return path.win32.join(rootPath, filename);
- } else {
- throw new Error(
- `${rootPath} is neither a posix nor a windows path, and there is no 'join' method defined in the file system`
- );
- }
- };
- exports.join = join;
- /**
- * @param {InputFileSystem|OutputFileSystem|undefined} fs a file system
- * @param {string} absPath an absolute path
- * @returns {string} the parent directory of the absolute path
- */
- const dirname = (fs, absPath) => {
- if (fs && fs.dirname) {
- return fs.dirname(absPath);
- } else if (path.posix.isAbsolute(absPath)) {
- return path.posix.dirname(absPath);
- } else if (path.win32.isAbsolute(absPath)) {
- return path.win32.dirname(absPath);
- } else {
- throw new Error(
- `${absPath} is neither a posix nor a windows path, and there is no 'dirname' method defined in the file system`
- );
- }
- };
- exports.dirname = dirname;
- /**
- * @param {OutputFileSystem} fs a file system
- * @param {string} p an absolute path
- * @param {function(Error=): void} callback callback function for the error
- * @returns {void}
- */
- const mkdirp = (fs, p, callback) => {
- fs.mkdir(p, err => {
- if (err) {
- if (err.code === "ENOENT") {
- const dir = dirname(fs, p);
- if (dir === p) {
- callback(err);
- return;
- }
- mkdirp(fs, dir, err => {
- if (err) {
- callback(err);
- return;
- }
- fs.mkdir(p, err => {
- if (err) {
- if (err.code === "EEXIST") {
- callback();
- return;
- }
- callback(err);
- return;
- }
- callback();
- });
- });
- return;
- } else if (err.code === "EEXIST") {
- callback();
- return;
- }
- callback(err);
- return;
- }
- callback();
- });
- };
- exports.mkdirp = mkdirp;
- /**
- * @param {IntermediateFileSystem} fs a file system
- * @param {string} p an absolute path
- * @returns {void}
- */
- const mkdirpSync = (fs, p) => {
- try {
- fs.mkdirSync(p);
- } catch (err) {
- if (err) {
- if (/** @type {NodeJS.ErrnoException} */ (err).code === "ENOENT") {
- const dir = dirname(fs, p);
- if (dir === p) {
- throw err;
- }
- mkdirpSync(fs, dir);
- fs.mkdirSync(p);
- return;
- } else if (/** @type {NodeJS.ErrnoException} */ (err).code === "EEXIST") {
- return;
- }
- throw err;
- }
- }
- };
- exports.mkdirpSync = mkdirpSync;
- /**
- * @param {InputFileSystem} fs a file system
- * @param {string} p an absolute path
- * @param {ReadJsonCallback} callback callback
- * @returns {void}
- */
- const readJson = (fs, p, callback) => {
- if ("readJson" in fs)
- return /** @type {NonNullable<InputFileSystem["readJson"]>} */ (
- fs.readJson
- )(p, callback);
- fs.readFile(p, (err, buf) => {
- if (err) return callback(err);
- let data;
- try {
- data = JSON.parse(/** @type {Buffer} */ (buf).toString("utf-8"));
- } catch (e) {
- return callback(/** @type {Error} */ (e));
- }
- return callback(null, data);
- });
- };
- exports.readJson = readJson;
- /**
- * @param {InputFileSystem} fs a file system
- * @param {string} p an absolute path
- * @param {function(NodeJS.ErrnoException | Error | null, (IStats | string)=): void} callback callback
- * @returns {void}
- */
- const lstatReadlinkAbsolute = (fs, p, callback) => {
- let i = 3;
- const doReadLink = () => {
- fs.readlink(p, (err, target) => {
- if (err && --i > 0) {
- // It might was just changed from symlink to file
- // we retry 2 times to catch this case before throwing the error
- return doStat();
- }
- if (err || !target) return doStat();
- const value = target.toString();
- callback(null, join(fs, dirname(fs, p), value));
- });
- };
- const doStat = () => {
- if ("lstat" in fs) {
- return /** @type {NonNullable<InputFileSystem["lstat"]>} */ (fs.lstat)(
- p,
- (err, stats) => {
- if (err) return callback(err);
- if (/** @type {IStats} */ (stats).isSymbolicLink()) {
- return doReadLink();
- }
- callback(null, stats);
- }
- );
- } else {
- return fs.stat(p, callback);
- }
- };
- if ("lstat" in fs) return doStat();
- doReadLink();
- };
- exports.lstatReadlinkAbsolute = lstatReadlinkAbsolute;
|