Editar
La versión NodeJS 10.12.0
ha agregado un soporte nativo para ambos mkdir
y mkdirSync
para crear un directorio recursivamente con la recursive: true
opción de la siguiente manera:
fs.mkdirSync(targetDir, { recursive: true });
Y si lo prefieres fs Promises API
, puedes escribir
fs.promises.mkdir(targetDir, { recursive: true });
Respuesta original
¡Cree directorios recursivamente si no existen! ( Cero dependencias )
const fs = require('fs');
const path = require('path');
function mkDirByPathSync(targetDir, { isRelativeToScript = false } = {}) {
const sep = path.sep;
const initDir = path.isAbsolute(targetDir) ? sep : '';
const baseDir = isRelativeToScript ? __dirname : '.';
return targetDir.split(sep).reduce((parentDir, childDir) => {
const curDir = path.resolve(baseDir, parentDir, childDir);
try {
fs.mkdirSync(curDir);
} catch (err) {
if (err.code === 'EEXIST') { // curDir already exists!
return curDir;
}
// To avoid `EISDIR` error on Mac and `EACCES`-->`ENOENT` and `EPERM` on Windows.
if (err.code === 'ENOENT') { // Throw the original parentDir error on curDir `ENOENT` failure.
throw new Error(`EACCES: permission denied, mkdir '${parentDir}'`);
}
const caughtErr = ['EACCES', 'EPERM', 'EISDIR'].indexOf(err.code) > -1;
if (!caughtErr || caughtErr && curDir === path.resolve(targetDir)) {
throw err; // Throw if it's just the last created dir.
}
}
return curDir;
}, initDir);
}
Uso
// Default, make directories relative to current working directory.
mkDirByPathSync('path/to/dir');
// Make directories relative to the current script.
mkDirByPathSync('path/to/dir', {isRelativeToScript: true});
// Make directories with an absolute path.
mkDirByPathSync('/path/to/dir');
Manifestación
¡Intentalo!
Explicaciones
- [ACTUALIZACIÓN] Esta solución maneja errores específicos de la plataforma como
EISDIR
para Mac EPERM
yEACCES
para Windows. Gracias a todos los comentarios de informes de @PediT., @JohnQ, @ deed02392, @robyoder y @Almenon.
- Esta solución maneja tanto relativos como absolutos. rutas . Gracias al comentario de @john.
- En el caso de rutas relativas, los directorios de destino se crearán (resolverán) en el directorio de trabajo actual. Para resolverlos en relación con el directorio de script actual, pase
{isRelativeToScript: true}
.
- Uso
path.sep
y path.resolve()
, no solo /
concatenación, para evitar problemas entre plataformas.
- El uso
fs.mkdirSync
y manejo del error con try/catch
si se tiran para manejar las condiciones de carrera: otro proceso puede agregar el archivo entre las llamadas a fs.existsSync()
y fs.mkdirSync()
y provoca una excepción.
- La otra manera de lograr que se podría comprobar si un archivo existe, entonces la creación de ella, es decir,
if (!fs.existsSync(curDir) fs.mkdirSync(curDir);
. Pero este es un antipatrón que deja el código vulnerable a las condiciones de carrera. Gracias al comentario de @GershomMaes sobre la verificación de existencia del directorio.
- Requiere Nodo v6 y posterior para soportar la desestructuración. (Si tiene problemas para implementar esta solución con versiones antiguas de Node, solo déjeme un comentario)