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

问题描述 投票:0回答:4

问题

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

使用案例

我正在进行 zip 提取,源 zip 包含一个名为

archive
的根文件夹,因此当我提取时,我得到
extract_path/archive/
,但我想将
archive
的内容直接提取到
extract_path
.

我以为这只是简单的重命名,但下面抛出了“有文件挡住了”错误消息。

fs.renameSync(extractPath + "/archive", extractPath)
node.js mv fs
4个回答
7
投票

所选答案不起作用:

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

5
投票

使用 mv npm 模块。 mv 首先尝试 fs.rename,如果失败,则使用复制然后取消链接:

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/']);

2
投票

没有一个答案对我有用,我深入研究了 mv 的代码并找到了我的解决方案:

我将

folder/subfolder
移至
folder
,因此该文件夹已经存在。

mv(oldPath, newPath, {mkdirp: false, clobber: false}, (err) => {
    if (err) {
        throw err;
    }
});

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


0
投票

mv
软件包并不是最新的并且不再维护。

我建议:

async function moveFolder(oldPath, newPath) {
  console.log(`Try moving ${oldPath} to ${newPath}`)
  if (!fs.existsSync(oldPath)) {
    console.log(`${oldPath} does not exist`)
    return
  }
  if (!fs.existsSync(newPath)) {
    console.log(`Moving ${oldPath} to ${newPath}`)
    fs.renameSync(oldPath, newPath)
    return
  }
  const files = fs.readdirSync(oldPath)
  for (var key in files) {
    const file = files[key]
    const oldFilePath = path.join(oldPath, file)
    const newFilePath = path.join(newPath, file)
    console.log(`Moving ${oldFilePath} to ${newFilePath}`)
    fs.renameSync(oldFilePath, newFilePath)
  })
}
© www.soinside.com 2019 - 2024. All rights reserved.