Muszę zapisać plik do następującej ścieżki:
fs.writeFile('/folder1/folder2/file.txt', 'content', function () {…});
Ale '/folder1/folder2'
ścieżka może nie istnieć. Więc otrzymuję następujący błąd:
message = ENOENT, otwórz /folder1/folder2/file.txt
Jak mogę napisać treść na tej ścieżce?
fs.promises.mkdir(path.dirname('/folder1/folder2/file.txt'), {recursive: true}).then(x => fs.promises.writeFile('/folder1/folder2/file.txt', 'content'))
Odpowiedzi:
Użyj mkdirp w połączeniu z
path.dirname
first.var mkdirp = require('mkdirp'); var fs = require('fs'); var getDirName = require('path').dirname; function writeFile(path, contents, cb) { mkdirp(getDirName(path), function (err) { if (err) return cb(err); fs.writeFile(path, contents, cb); }); }
Jeśli cała ścieżka już istnieje,
mkdirp
jest noop. W przeciwnym razie tworzy dla Ciebie wszystkie brakujące katalogi.Ten moduł robi, co chcesz: https://npmjs.org/package/writefile . Rozumiem, szukając w Google hasła „writefile mkdirp”. Ten moduł zwraca obietnicę zamiast odbierania wywołania zwrotnego, więc najpierw przeczytaj wprowadzenie do obietnic. W rzeczywistości może to skomplikować sprawę.
Funkcja, którą podałem, działa w każdym przypadku.
źródło
fs.promises.mkdir(path.dirname(file), {recursive: true}).then(x => fs.promises.writeFile(file, data))
Uważam, że najłatwiejszym sposobem jest użycie metody outputFile () z modułu fs-extra .
Przykład:
var fs = require('fs-extra'); var file = '/tmp/this/path/does/not/exist/file.txt' fs.outputFile(file, 'hello!', function (err) { console.log(err); // => null fs.readFile(file, 'utf8', function (err, data) { console.log(data); // => hello! }); });
W dzisiejszych czasach ma również obiecujące wsparcie po wyjęciu z pudełka!
źródło
Edytować
Wersja NodeJS
10.12.0
dodała natywną obsługę obumkdir
imkdirSync
rekurencyjne tworzenie nadrzędnego reżysera z następującąrecursive: true
opcją:fs.mkdirSync(targetDir, { recursive: true });
A jeśli wolisz
fs Promises API
, możesz pisaćfs.promises.mkdir(targetDir, { recursive: true });
Oryginalna odpowiedź
Utwórz katalogi nadrzędne rekurencyjnie, jeśli nie istnieją! ( Zero zależności )
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); }
Stosowanie
// 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');
Próbny
Spróbuj!
Wyjaśnienia
EISDIR
dla Mac iEPERM
iEACCES
dla Windows.{isRelativeToScript: true}
.path.sep
ipath.resolve()
, a nie tylko/
konkatenacja, w celu uniknięcia problemów między platformami.fs.mkdirSync
i obsługa błędu za pomocątry/catch
if zgłoszonego w celu obsługi warunków wyścigu: inny proces może dodać plik między wywołaniami dofs.existsSync()
ifs.mkdirSync()
i powoduje wyjątek.if (!fs.existsSync(curDir) fs.mkdirSync(curDir);
. Ale jest to anty-wzorzec, który pozostawia kod podatny na warunki wyścigu.źródło
Być może najprościej można po prostu użyć modułu npm fs-path .
Twój kod wyglądałby wtedy następująco:
var fsPath = require('fs-path'); fsPath.writeFile('/folder1/folder2/file.txt', 'content', function(err){ if(err) { throw err; } else { console.log('wrote a file like DaVinci drew machines'); } });
źródło
Możesz użyć
fs.stat('/folder1/folder2', function(err, stats){ ... });
stats
tofs.Stats
rodzaj obiektu, możesz sprawdzićstats.isDirectory()
. W zależności od zbadaniaerr
istats
można coś zrobićfs.mkdir( ... )
lub wyrzucić błąd.Odniesienie
Aktualizacja: Poprawiono przecinki w kodzie.
źródło
Dzięki node-fs-extra możesz to łatwo zrobić.
Zainstaluj to
Następnie użyj outputfile metodę zamiast writeFileSync
const fs = require('fs-extra'); fs.outputFile('tmp/test.txt', 'Hey there!', err => { if(err) { console.log(err); } else { console.log('The file was saved!'); } })
źródło
Oto moja niestandardowa funkcja do rekurencyjnego tworzenia katalogów (bez zewnętrznych zależności):
var fs = require('fs'); var path = require('path'); var myMkdirSync = function(dir){ if (fs.existsSync(dir)){ return } try{ fs.mkdirSync(dir) }catch(err){ if(err.code == 'ENOENT'){ myMkdirSync(path.dirname(dir)) //create parent dir myMkdirSync(dir) //create dir } } } myMkdirSync(path.dirname(filePath)); var file = fs.createWriteStream(filePath);
źródło
Oto moja funkcja, która działa w Node 10.12.0. Mam nadzieję, że to pomoże.
const fs = require('fs'); function(dir,filename,content){ fs.promises.mkdir(dir, { recursive: true }).catch(error => { console.error('caught exception : ', error.message); }); fs.writeFile(dir+filename, content, function (err) { if (err) throw err; console.info('file saved!'); }); }
źródło
let name = "./new_folder/" + file_name + ".png"; await driver.takeScreenshot().then( function(image, err) { require('mkdirp')(require('path').dirname(name), (err) => { require('fs').writeFile(name, image, 'base64', function(err) { console.log(err); }); }); } );
źródło
Oto część odpowiedzi Myrne Stol podzielona jako osobna odpowiedź:
źródło