是否有一种简单的方法可以将目录中的所有文件移动到其父目录,然后删除目录?
我正在进行zip解压缩,源zip包含一个名为archive
的根文件夹,所以当我提取时我得到extract_path/archive/
,但我想提取{{archive
的内容。 1}}直接发送到extract_path
。
我认为这将是简单的重命名,但以下是抛出“有一个文件的方式”错误消息。
fs.renameSync(extractPath + "/archive", extractPath)
答案 0 :(得分:4)
使用mv npm模块。 mv首先尝试fs.rename,如果失败,则使用copy然后取消链接:
mv('source/dir', 'dest/a/b/c/dir', {mkdirp: true}, function(err) {
// done. it first created all the necessary directories, and then
// tried fs.rename, then falls back to using ncp to copy the dir
// to dest and then rimraf to remove the source dir
});
或产生子进程:
var spawn = require('child_process').spawn,
mv = spawn('mv', ['/dir1/dir2/*','dir1/']);
答案 1 :(得分:3)
所选答案不起作用:
var mv = require('mv');
var extractPath = 'E:\\tmp\\dir';
mv(extractPath + "\\sub", extractPath, {mkdirp: true}, console.error);
错误:
{ Error: EPERM: operation not permitted, rename 'E:\tmp\dir\sub' -> 'E:\tmp\dir'
at Error (native)
errno: -4048,
code: 'EPERM',
syscall: 'rename',
path: 'E:\\tmp\\dir\\sub',
dest: 'E:\\tmp\\dir' }
使用fs-extra而不是mv:
var fs = require('fs-extra');
var extractPath = 'E:\\tmp\\dir';
fs.move(extractPath + "\\sub", extractPath, console.error);
移动前我的文件结构如下:
E:\tmp\dir
> sub
> doc.txt
移动后就像这样:
E:\tmp\dir
> doc.txt
更新:
虽然以上适用于Windows,但在Linux上即使使用fs-extra也会出现相同的错误。以下是对此的手动修复,通过将子目录的每个子项单独移动到父项。如果子移动失败,则它会将任何其他成功移动恢复到子目录中的原始位置。
var fs = require('fs-extra')
var Promise = require('promise');
var path = require('path');
var promiseAllWait = function(promises) {
// this is the same as Promise.all(), except that it will wait for all promises to fulfill before rejecting
var all_promises = [];
for(var i_promise=0; i_promise < promises.length; i_promise++) {
all_promises.push(
promises[i_promise]
.then(function(res) {
return { res: res };
}).catch(function(err) {
return { err: err };
})
);
}
return Promise.all(all_promises)
.then(function(results) {
return new Promise(function(resolve, reject) {
var is_failure = false;
var i_result;
for(i_result=0; i_result < results.length; i_result++) {
if (results[i_result].err) {
is_failure = true;
break;
} else {
results[i_result] = results[i_result].res;
}
}
if (is_failure) {
reject( results[i_result].err );
} else {
resolve(results);
}
});
});
};
var movePromiser = function(from, to, records) {
return fs.move(from, to)
.then(function() {
records.push( {from: from, to: to} );
});
};
var moveDir = function(from_dir, to_dir) {
return fs.readdir(from_dir)
.then(function(children) {
return fs.ensureDir(to_dir)
.then(function() {
var move_promises = [];
var moved_records = [];
var child;
for(var i_child=0; i_child < children.length; i_child++) {
child = children[i_child];
move_promises.push(movePromiser(
path.join(from_dir, child),
path.join(to_dir, child),
moved_records
));
}
return promiseAllWait(move_promises)
.catch(function(err) {
var undo_move_promises = [];
for(var i_moved_record=0; i_moved_record < moved_records.length; i_moved_record++) {
undo_move_promises.push( fs.move(moved_records[i_moved_record].to, moved_records[i_moved_record].from) );
}
return promiseAllWait(undo_move_promises)
.then(function() {
throw err;
});
});
}).then(function() {
return fs.rmdir(from_dir);
});
});
};
答案 2 :(得分:0)
没有一个答案对我有用,我深入研究了 mv 的代码并找到了我的解决方案:
我将 folder/subfolder
移至 folder
,因此该文件夹已存在。
mv(oldPath, newPath, {mkdirp: false, clobber: false}, (err) => {
if (err) {
throw err;
}
});
请记住,如果父文件夹中已经存在文件名,它将被子文件夹中的文件覆盖。