Editar
A versão do NodeJS 10.12.0
adicionou um suporte nativo para ambos mkdir
e mkdirSync
para criar um diretório recursivamente com a recursive: true
opção como a seguir:
fs.mkdirSync(targetDir, { recursive: true });
E se você preferir fs Promises API
, você pode escrever
fs.promises.mkdir(targetDir, { recursive: true });
Resposta original
Crie diretórios recursivamente se eles não existirem! ( Zero dependências )
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');
Demo
Tente!
Explicações
- [UPDATE] Esta solução lida com erros específicos da plataforma, como
EISDIR
no Mac EPERM
e EACCES
no Windows. Obrigado a todos os comentários de @PediT., @JohnQ, @ deed02392, @robyoder e @Almenon.
- Esta solução lida com caminhos relativos e absolutos . Graças ao comentário @john.
- No caso de caminhos relativos, os diretórios de destino serão criados (resolvidos) no diretório de trabalho atual. Para resolvê-los em relação ao diretório de script atual, passe
{isRelativeToScript: true}
.
- Usando
path.sep
e path.resolve()
não apenas /
concatenação, para evitar problemas entre plataformas.
- Usando
fs.mkdirSync
e manusear o erro com try/catch
se jogado para lidar com as condições de corrida: outro processo pode adicionar o arquivo entre as chamadas para fs.existsSync()
e fs.mkdirSync()
e causa uma exceção.
- A outra maneira de conseguir isso seria verificar se existe um arquivo e criá-lo, ou seja
if (!fs.existsSync(curDir) fs.mkdirSync(curDir);
,. Mas esse é um antipadrão que deixa o código vulnerável às condições de corrida. Graças ao comentário de @GershomMaes sobre a verificação de existência do diretório.
- Requer o Nó v6 e mais recente para suportar a desestruturação. (Se você tiver problemas para implementar esta solução com versões antigas do Node, deixe-me um comentário)