/** * @fileoverview The main file for the humanfs package. * @author Nicholas C. Zakas */ /* global URL, TextDecoder, TextEncoder */ //----------------------------------------------------------------------------- // Types //----------------------------------------------------------------------------- /** @typedef {import("@humanfs/types").HfsImpl} HfsImpl */ /** @typedef {import("@humanfs/types").HfsDirectoryEntry} HfsDirectoryEntry */ /** @typedef {import("@humanfs/types").HfsWalkEntry} HfsWalkEntry */ //----------------------------------------------------------------------------- // Helpers //----------------------------------------------------------------------------- const decoder = new TextDecoder(); const encoder = new TextEncoder(); /** * Error to represent when a method is missing on an impl. */ export class NoSuchMethodError extends Error { /** * Creates a new instance. * @param {string} methodName The name of the method that was missing. */ constructor(methodName) { super(`Method "${methodName}" does not exist on impl.`); } } /** * Error to represent when a method is not supported on an impl. This happens * when a method on `Hfs` is called with one name and the corresponding method * on the impl has a different name. (Example: `text()` and `bytes()`.) */ export class MethodNotSupportedError extends Error { /** * Creates a new instance. * @param {string} methodName The name of the method that was missing. */ constructor(methodName) { super(`Method "${methodName}" is not supported on this impl.`); } } /** * Error to represent when an impl is already set. */ export class ImplAlreadySetError extends Error { /** * Creates a new instance. */ constructor() { super(`Implementation already set.`); } } /** * Asserts that the given path is a valid file path. * @param {any} fileOrDirPath The path to check. * @returns {void} * @throws {TypeError} When the path is not a non-empty string. */ function assertValidFileOrDirPath(fileOrDirPath) { if ( !fileOrDirPath || (!(fileOrDirPath instanceof URL) && typeof fileOrDirPath !== "string") ) { throw new TypeError("Path must be a non-empty string or URL."); } } /** * Asserts that the given file contents are valid. * @param {any} contents The contents to check. * @returns {void} * @throws {TypeError} When the contents are not a string or ArrayBuffer. */ function assertValidFileContents(contents) { if ( typeof contents !== "string" && !(contents instanceof ArrayBuffer) && !ArrayBuffer.isView(contents) ) { throw new TypeError( "File contents must be a string, ArrayBuffer, or ArrayBuffer view.", ); } } /** * Converts the given contents to Uint8Array. * @param {any} contents The data to convert. * @returns {Uint8Array} The converted Uint8Array. * @throws {TypeError} When the contents are not a string or ArrayBuffer. */ function toUint8Array(contents) { if (contents instanceof Uint8Array) { return contents; } if (typeof contents === "string") { return encoder.encode(contents); } if (contents instanceof ArrayBuffer) { return new Uint8Array(contents); } if (ArrayBuffer.isView(contents)) { const bytes = contents.buffer.slice( contents.byteOffset, contents.byteOffset + contents.byteLength, ); return new Uint8Array(bytes); } throw new TypeError( "Invalid contents type. Expected string or ArrayBuffer.", ); } //----------------------------------------------------------------------------- // Exports //----------------------------------------------------------------------------- /** * A class representing a log entry. */ export class LogEntry { /** * The type of log entry. * @type {string} */ type; /** * The data associated with the log entry. * @type {any} */ data; /** * The time at which the log entry was created. * @type {number} */ timestamp = Date.now(); /** * Creates a new instance. * @param {string} type The type of log entry. * @param {any} [data] The data associated with the log entry. */ constructor(type, data) { this.type = type; this.data = data; } } /** * A class representing a file system utility library. * @implements {HfsImpl} */ export class Hfs { /** * The base implementation for this instance. * @type {HfsImpl} */ #baseImpl; /** * The current implementation for this instance. * @type {HfsImpl} */ #impl; /** * A map of log names to their corresponding entries. * @type {Map>} */ #logs = new Map(); /** * Creates a new instance. * @param {object} options The options for the instance. * @param {HfsImpl} options.impl The implementation to use. */ constructor({ impl }) { this.#baseImpl = impl; this.#impl = impl; } /** * Logs an entry onto all currently open logs. * @param {string} methodName The name of the method being called. * @param {...*} args The arguments to the method. * @returns {void} */ #log(methodName, ...args) { for (const logs of this.#logs.values()) { logs.push(new LogEntry("call", { methodName, args })); } } /** * Starts a new log with the given name. * @param {string} name The name of the log to start; * @returns {void} * @throws {Error} When the log already exists. * @throws {TypeError} When the name is not a non-empty string. */ logStart(name) { if (!name || typeof name !== "string") { throw new TypeError("Log name must be a non-empty string."); } if (this.#logs.has(name)) { throw new Error(`Log "${name}" already exists.`); } this.#logs.set(name, []); } /** * Ends a log with the given name and returns the entries. * @param {string} name The name of the log to end. * @returns {Array} The entries in the log. * @throws {Error} When the log does not exist. */ logEnd(name) { if (this.#logs.has(name)) { const logs = this.#logs.get(name); this.#logs.delete(name); return logs; } throw new Error(`Log "${name}" does not exist.`); } /** * Determines if the current implementation is the base implementation. * @returns {boolean} True if the current implementation is the base implementation. */ isBaseImpl() { return this.#impl === this.#baseImpl; } /** * Sets the implementation for this instance. * @param {object} impl The implementation to use. * @returns {void} */ setImpl(impl) { this.#log("implSet", impl); if (this.#impl !== this.#baseImpl) { throw new ImplAlreadySetError(); } this.#impl = impl; } /** * Resets the implementation for this instance back to its original. * @returns {void} */ resetImpl() { this.#log("implReset"); this.#impl = this.#baseImpl; } /** * Asserts that the given method exists on the current implementation. * @param {string} methodName The name of the method to check. * @returns {void} * @throws {NoSuchMethodError} When the method does not exist on the current implementation. */ #assertImplMethod(methodName) { if (typeof this.#impl[methodName] !== "function") { throw new NoSuchMethodError(methodName); } } /** * Asserts that the given method exists on the current implementation, and if not, * throws an error with a different method name. * @param {string} methodName The name of the method to check. * @param {string} targetMethodName The name of the method that should be reported * as an error when methodName does not exist. * @returns {void} * @throws {NoSuchMethodError} When the method does not exist on the current implementation. */ #assertImplMethodAlt(methodName, targetMethodName) { if (typeof this.#impl[methodName] !== "function") { throw new MethodNotSupportedError(targetMethodName); } } /** * Calls the given method on the current implementation. * @param {string} methodName The name of the method to call. * @param {...any} args The arguments to the method. * @returns {any} The return value from the method. * @throws {NoSuchMethodError} When the method does not exist on the current implementation. */ #callImplMethod(methodName, ...args) { this.#log(methodName, ...args); this.#assertImplMethod(methodName); return this.#impl[methodName](...args); } /** * Calls the given method on the current implementation and doesn't log the call. * @param {string} methodName The name of the method to call. * @param {...any} args The arguments to the method. * @returns {any} The return value from the method. * @throws {NoSuchMethodError} When the method does not exist on the current implementation. */ #callImplMethodWithoutLog(methodName, ...args) { this.#assertImplMethod(methodName); return this.#impl[methodName](...args); } /** * Calls the given method on the current implementation but logs a different method name. * @param {string} methodName The name of the method to call. * @param {string} targetMethodName The name of the method to log. * @param {...any} args The arguments to the method. * @returns {any} The return value from the method. * @throws {NoSuchMethodError} When the method does not exist on the current implementation. */ #callImplMethodAlt(methodName, targetMethodName, ...args) { this.#log(targetMethodName, ...args); this.#assertImplMethodAlt(methodName, targetMethodName); return this.#impl[methodName](...args); } /** * Reads the given file and returns the contents as text. Assumes UTF-8 encoding. * @param {string|URL} filePath The file to read. * @returns {Promise} The contents of the file. * @throws {NoSuchMethodError} When the method does not exist on the current implementation. * @throws {TypeError} When the file path is not a non-empty string. */ async text(filePath) { assertValidFileOrDirPath(filePath); const result = await this.#callImplMethodAlt("bytes", "text", filePath); return result ? decoder.decode(result) : undefined; } /** * Reads the given file and returns the contents as JSON. Assumes UTF-8 encoding. * @param {string|URL} filePath The file to read. * @returns {Promise} The contents of the file as JSON. * @throws {NoSuchMethodError} When the method does not exist on the current implementation. * @throws {SyntaxError} When the file contents are not valid JSON. * @throws {TypeError} When the file path is not a non-empty string. */ async json(filePath) { assertValidFileOrDirPath(filePath); const result = await this.#callImplMethodAlt("bytes", "json", filePath); return result ? JSON.parse(decoder.decode(result)) : undefined; } /** * Reads the given file and returns the contents as an ArrayBuffer. * @param {string|URL} filePath The file to read. * @returns {Promise} The contents of the file as an ArrayBuffer. * @throws {NoSuchMethodError} When the method does not exist on the current implementation. * @throws {TypeError} When the file path is not a non-empty string. * @deprecated Use bytes() instead. */ async arrayBuffer(filePath) { assertValidFileOrDirPath(filePath); const result = await this.#callImplMethodAlt( "bytes", "arrayBuffer", filePath, ); return result?.buffer; } /** * Reads the given file and returns the contents as an Uint8Array. * @param {string|URL} filePath The file to read. * @returns {Promise} The contents of the file as an Uint8Array. * @throws {NoSuchMethodError} When the method does not exist on the current implementation. * @throws {TypeError} When the file path is not a non-empty string. */ async bytes(filePath) { assertValidFileOrDirPath(filePath); return this.#callImplMethod("bytes", filePath); } /** * Writes the given data to the given file. Creates any necessary directories along the way. * If the data is a string, UTF-8 encoding is used. * @param {string|URL} filePath The file to write. * @param {string|ArrayBuffer|ArrayBufferView} contents The data to write. * @returns {Promise} A promise that resolves when the file is written. * @throws {NoSuchMethodError} When the method does not exist on the current implementation. * @throws {TypeError} When the file path is not a non-empty string. */ async write(filePath, contents) { assertValidFileOrDirPath(filePath); assertValidFileContents(contents); this.#log("write", filePath, contents); let value = toUint8Array(contents); return this.#callImplMethodWithoutLog("write", filePath, value); } /** * Appends the given data to the given file. Creates any necessary directories along the way. * If the data is a string, UTF-8 encoding is used. * @param {string|URL} filePath The file to append to. * @param {string|ArrayBuffer|ArrayBufferView} contents The data to append. * @returns {Promise} A promise that resolves when the file is appended to. * @throws {NoSuchMethodError} When the method does not exist on the current implementation. * @throws {TypeError} When the file path is not a non-empty string. * @throws {TypeError} When the file contents are not a string or ArrayBuffer. * @throws {Error} When the file cannot be appended to. */ async append(filePath, contents) { assertValidFileOrDirPath(filePath); assertValidFileContents(contents); this.#log("append", filePath, contents); let value = toUint8Array(contents); return this.#callImplMethodWithoutLog("append", filePath, value); } /** * Determines if the given file exists. * @param {string|URL} filePath The file to check. * @returns {Promise} True if the file exists. * @throws {NoSuchMethodError} When the method does not exist on the current implementation. * @throws {TypeError} When the file path is not a non-empty string. */ async isFile(filePath) { assertValidFileOrDirPath(filePath); return this.#callImplMethod("isFile", filePath); } /** * Determines if the given directory exists. * @param {string|URL} dirPath The directory to check. * @returns {Promise} True if the directory exists. * @throws {NoSuchMethodError} When the method does not exist on the current implementation. * @throws {TypeError} When the directory path is not a non-empty string. */ async isDirectory(dirPath) { assertValidFileOrDirPath(dirPath); return this.#callImplMethod("isDirectory", dirPath); } /** * Creates the given directory. * @param {string|URL} dirPath The directory to create. * @returns {Promise} A promise that resolves when the directory is created. * @throws {NoSuchMethodError} When the method does not exist on the current implementation. * @throws {TypeError} When the directory path is not a non-empty string. */ async createDirectory(dirPath) { assertValidFileOrDirPath(dirPath); return this.#callImplMethod("createDirectory", dirPath); } /** * Deletes the given file or empty directory. * @param {string|URL} filePath The file to delete. * @returns {Promise} A promise that resolves when the file or * directory is deleted, true if the file or directory is deleted, false * if the file or directory does not exist. * @throws {NoSuchMethodError} When the method does not exist on the current implementation. * @throws {TypeError} When the file path is not a non-empty string. */ async delete(filePath) { assertValidFileOrDirPath(filePath); return this.#callImplMethod("delete", filePath); } /** * Deletes the given file or directory recursively. * @param {string|URL} dirPath The directory to delete. * @returns {Promise} A promise that resolves when the file or * directory is deleted, true if the file or directory is deleted, false * if the file or directory does not exist. * @throws {NoSuchMethodError} When the method does not exist on the current implementation. * @throws {TypeError} When the directory path is not a non-empty string. */ async deleteAll(dirPath) { assertValidFileOrDirPath(dirPath); return this.#callImplMethod("deleteAll", dirPath); } /** * Returns a list of directory entries for the given path. * @param {string|URL} dirPath The path to the directory to read. * @returns {AsyncIterable} A promise that resolves with the * directory entries. * @throws {TypeError} If the directory path is not a string or URL. * @throws {Error} If the directory cannot be read. */ async *list(dirPath) { assertValidFileOrDirPath(dirPath); yield* await this.#callImplMethod("list", dirPath); } /** * Walks a directory using a depth-first traversal and returns the entries * from the traversal. * @param {string|URL} dirPath The path to the directory to walk. * @param {Object} [options] The options for the walk. * @param {(entry:HfsWalkEntry) => Promise|boolean} [options.directoryFilter] A filter function to determine * if a directory's entries should be included in the walk. * @param {(entry:HfsWalkEntry) => Promise|boolean} [options.entryFilter] A filter function to determine if * an entry should be included in the walk. * @returns {AsyncIterable} A promise that resolves with the * directory entries. * @throws {TypeError} If the directory path is not a string or URL. * @throws {Error} If the directory cannot be read. */ async *walk( dirPath, { directoryFilter = () => true, entryFilter = () => true } = {}, ) { assertValidFileOrDirPath(dirPath); this.#log("walk", dirPath, { directoryFilter, entryFilter }); // inner function for recursion without additional logging const walk = async function* ( dirPath, { directoryFilter, entryFilter, parentPath = "", depth = 1 }, ) { let dirEntries; try { dirEntries = await this.#callImplMethodWithoutLog( "list", dirPath, ); } catch (error) { // if the directory does not exist then return an empty array if (error.code === "ENOENT") { return; } // otherwise, rethrow the error throw error; } for await (const listEntry of dirEntries) { const walkEntry = { path: listEntry.name, depth, ...listEntry, }; if (parentPath) { walkEntry.path = `${parentPath}/${walkEntry.path}`; } // first emit the entry but only if the entry filter returns true let shouldEmitEntry = entryFilter(walkEntry); if (shouldEmitEntry.then) { shouldEmitEntry = await shouldEmitEntry; } if (shouldEmitEntry) { yield walkEntry; } // if it's a directory then yield the entry and walk the directory if (listEntry.isDirectory) { // if the directory filter returns false, skip the directory let shouldWalkDirectory = directoryFilter(walkEntry); if (shouldWalkDirectory.then) { shouldWalkDirectory = await shouldWalkDirectory; } if (!shouldWalkDirectory) { continue; } // make sure there's a trailing slash on the directory path before appending const directoryPath = dirPath instanceof URL ? new URL( listEntry.name, dirPath.href.endsWith("/") ? dirPath.href : `${dirPath.href}/`, ) : `${dirPath.endsWith("/") ? dirPath : `${dirPath}/`}${listEntry.name}`; yield* walk(directoryPath, { directoryFilter, entryFilter, parentPath: walkEntry.path, depth: depth + 1, }); } } }.bind(this); yield* walk(dirPath, { directoryFilter, entryFilter }); } /** * Returns the size of the given file. * @param {string|URL} filePath The path to the file to read. * @returns {Promise} A promise that resolves with the size of the file. * @throws {TypeError} If the file path is not a string or URL. * @throws {Error} If the file cannot be read. */ async size(filePath) { assertValidFileOrDirPath(filePath); return this.#callImplMethod("size", filePath); } /** * Returns the last modified timestamp of the given file or directory. * @param {string|URL} fileOrDirPath The path to the file or directory. * @returns {Promise} A promise that resolves with the last modified date * or undefined if the file or directory does not exist. * @throws {TypeError} If the path is not a string or URL. */ async lastModified(fileOrDirPath) { assertValidFileOrDirPath(fileOrDirPath); return this.#callImplMethod("lastModified", fileOrDirPath); } /** * Copys a file from one location to another. * @param {string|URL} source The path to the file to copy. * @param {string|URL} destination The path to the new file. * @returns {Promise} A promise that resolves when the file is copied. * @throws {TypeError} If the file path is not a string or URL. * @throws {Error} If the file cannot be copied. */ async copy(source, destination) { assertValidFileOrDirPath(source); assertValidFileOrDirPath(destination); return this.#callImplMethod("copy", source, destination); } /** * Copies a file or directory from one location to another. * @param {string|URL} source The path to the file or directory to copy. * @param {string|URL} destination The path to copy the file or directory to. * @returns {Promise} A promise that resolves when the file or directory is * copied. * @throws {TypeError} If the directory path is not a string or URL. * @throws {Error} If the directory cannot be copied. */ async copyAll(source, destination) { assertValidFileOrDirPath(source); assertValidFileOrDirPath(destination); return this.#callImplMethod("copyAll", source, destination); } /** * Moves a file from the source path to the destination path. * @param {string|URL} source The location of the file to move. * @param {string|URL} destination The destination of the file to move. * @returns {Promise} A promise that resolves when the move is complete. * @throws {TypeError} If the file or directory paths are not strings. * @throws {Error} If the file or directory cannot be moved. */ async move(source, destination) { assertValidFileOrDirPath(source); assertValidFileOrDirPath(destination); return this.#callImplMethod("move", source, destination); } /** * Moves a file or directory from one location to another. * @param {string|URL} source The path to the file or directory to move. * @param {string|URL} destination The path to move the file or directory to. * @returns {Promise} A promise that resolves when the file or directory is * moved. * @throws {TypeError} If the source is not a string or URL. * @throws {TypeError} If the destination is not a string or URL. * @throws {Error} If the file or directory cannot be moved. */ async moveAll(source, destination) { assertValidFileOrDirPath(source); assertValidFileOrDirPath(destination); return this.#callImplMethod("moveAll", source, destination); } }