我需要将文件写入以下路径:
fs.writeFile('/folder1/folder2/file.txt', 'content', function () {…});
但'/folder1/folder2'
路径可能不存在。因此,我得到以下错误:
message=ENOENT,打开/folder1/folder2/file.txt
如何将内容写入该路径?
发布于 2013-05-01 19:52:00
在节点v10中,这是内置于fs.mkdir函数中的,我们可以将该函数与path.dirname结合使用
var fs = require('fs');
var getDirName = require('path').dirname;
function writeFile(path, contents, cb) {
fs.mkdir(getDirName(path), { recursive: true}, function (err) {
if (err) return cb(err);
fs.writeFile(path, contents, cb);
});
}
对于旧版本,您可以使用mkdirp
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);
});
}
如果整个路径已经存在,则mkdirp
是noop。否则,它会为您创建所有缺少的目录。
此模块执行您想要的操作:https://npmjs.org/package/writefile。在谷歌上搜索"writefile mkdirp“时得到的。此模块返回promise,而不是接受回调,因此请务必先阅读promise的一些介绍。它实际上可能会让你的事情变得复杂。
我给出的函数在任何情况下都有效。
发布于 2017-02-04 01:13:21
我发现最简单的方法是使用fs-extra模块中的()方法。
与writeFile几乎相同(即覆盖),只是如果父目录不存在,则会创建父目录。选项是您要传递给fs.writeFile()的内容。
示例:
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!
});
});
如今,它还承诺提供开箱即用的支持!
发布于 2016-11-19 06:31:14
编辑
NodeJS version 10.12.0
添加了对mkdir
和mkdirSync
的本机支持,以使用recursive: true
选项递归创建父控制器,如下所示:
fs.mkdirSync(targetDir, { recursive: true });
如果你更喜欢fs Promises API
,你可以写
fs.promises.mkdir(targetDir, { recursive: true });
原始答案
如果父目录不存在,则递归地创建它们!(零依赖)
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);
}
用法
// 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');
演示
解释
EPERM
的EISDIR
以及适用于Windows.EACCES
解决方案可同时处理相对和绝对路径。{isRelativeToScript: true}
.path.sep
和path.resolve()
,而不仅仅是/
连接,以避免跨平台issues.fs.mkdirSync
,并在抛出竞争条件时使用try/catch
处理错误:另一个进程可能会在对fs.existsSync()
和fs.mkdirSync()
的调用之间添加文件并导致异常。if (!fs.existsSync(curDir) fs.mkdirSync(curDir);
。但是这是一种反模式,使得代码容易受到竞争conditions.的攻击。
https://stackoverflow.com/questions/16316330
复制相似问题