使用node.js将目录中的所有文件移动到父级

时间:2014-10-10 03:51:22

标签: node.js mv fs

问题

是否有一种简单的方法可以将目录中的所有文件移动到其父目录,然后删除目录?

用例

我正在进行zip解压缩,源zip包含一个名为archive的根文件夹,所以当我提取时我得到extract_path/archive/,但我想提取{{archive的内容。 1}}直接发送到extract_path

我认为这将是简单的重命名,但以下是抛出“有一个文件的方式”错误消息。

fs.renameSync(extractPath + "/archive", extractPath)

3 个答案:

答案 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;
    }
});

请记住,如果父文件夹中已经存在文件名,它将被子文件夹中的文件覆盖。