tempy
Version:
Get a random temporary file or directory path
117 lines (89 loc) • 3.79 kB
JavaScript
import fs from 'node:fs';
import fsPromises from 'node:fs/promises';
import path from 'node:path';
import stream from 'node:stream';
import {promisify} from 'node:util';
import uniqueString from 'unique-string';
import tempDir from 'temp-dir';
import {isStream} from 'is-stream';
const pipeline = promisify(stream.pipeline); // TODO: Use `node:stream/promises` when targeting Node.js 16.
// TODO: Use `is-safe-filename` when targeting Node.js 20.
function assertSafePathComponent(pathComponent) {
if (typeof pathComponent !== 'string') {
throw new TypeError(`Expected a string, got ${typeof pathComponent}`);
}
const trimmed = pathComponent.trim();
const isSafe = trimmed !== ''
&& trimmed !== '.'
&& trimmed !== '..'
&& !pathComponent.includes('/')
&& !pathComponent.includes('\\')
&& !pathComponent.includes('\0');
if (!isSafe) {
throw new Error(`Unsafe path component: ${JSON.stringify(pathComponent)}`);
}
}
function resolveParentDirectory(parentDirectory, rootDirectory) {
assertSafePathComponent(parentDirectory);
const base = rootDirectory ?? tempDir;
const resolved = path.join(base, parentDirectory);
fs.mkdirSync(resolved, {recursive: true});
return resolved;
}
const getPath = (prefix = '', {parentDirectory, rootDirectory} = {}) => {
if (prefix) {
assertSafePathComponent(prefix);
}
if (rootDirectory !== undefined) {
if (!path.isAbsolute(rootDirectory)) {
throw new Error('The `rootDirectory` option must be an absolute path');
}
fs.mkdirSync(rootDirectory, {recursive: true});
}
const parent = parentDirectory
? resolveParentDirectory(parentDirectory, rootDirectory)
: (rootDirectory ?? tempDir);
return path.join(parent, prefix + uniqueString());
};
const writeStream = async (filePath, data) => pipeline(data, fs.createWriteStream(filePath));
async function runTask(temporaryPath, callback) {
try {
return await callback(temporaryPath);
} finally {
// Retry to handle EBUSY/EPERM on Windows (antivirus, indexing, CI).
await fsPromises.rm(temporaryPath, {recursive: true, force: true, maxRetries: 10, retryDelay: 100});
}
}
export function temporaryFile({name, extension, parentDirectory, rootDirectory} = {}) {
if (name !== undefined && name !== null) {
if (extension !== undefined && extension !== null) {
throw new Error('The `name` and `extension` options are mutually exclusive');
}
assertSafePathComponent(name);
return path.join(temporaryDirectory({parentDirectory, rootDirectory}), name);
}
if (extension !== undefined && extension !== null) {
assertSafePathComponent(extension);
}
return getPath('', {parentDirectory, rootDirectory}) + (extension === undefined || extension === null ? '' : '.' + extension.replace(/^\./, ''));
}
export const temporaryFileTask = async (callback, options) => runTask(temporaryFile(options), callback);
export function temporaryDirectory({prefix = '', parentDirectory, rootDirectory} = {}) {
const directory = getPath(prefix, {parentDirectory, rootDirectory});
fs.mkdirSync(directory, {recursive: true});
return directory;
}
export const temporaryDirectoryTask = async (callback, options) => runTask(temporaryDirectory(options), callback);
export async function temporaryWrite(fileContent, options) {
const filename = temporaryFile(options);
const write = isStream(fileContent) ? writeStream : fsPromises.writeFile;
await write(filename, fileContent);
return filename;
}
export const temporaryWriteTask = async (fileContent, callback, options) => runTask(await temporaryWrite(fileContent, options), callback);
export function temporaryWriteSync(fileContent, options) {
const filename = temporaryFile(options);
fs.writeFileSync(filename, fileContent);
return filename;
}
export {default as rootTemporaryDirectory} from 'temp-dir';