重命名文件夹时出现间歇性EPERM错误

时间:2019-04-18 09:04:22

标签: node.js file-permissions fs multer


我正在使用Multer在我的fs中上传图像。 Multer不允许您动态设置fs中的位置,因此我总是在同一文件夹中上载,然后使用fs.renamesynch更改文件夹的名称。
我使用的是同步版本,因为我认为我必须等待任务结束,否则,在以下任务中将出现问题,需要与重命名功能同步。
但是,我有间歇性错误,不知道如何解决
这是我的代码:

router.post("/changeprofile", ensureAuthenticated, (req, res) => {
  upload(req, res, err => {
    if (err) {
      res.render("changeprofile", { msg: err });
    } else {
      if (req.file == undefined) {
        res.render("changeprofile", { msg: "Error: No file Selected!" });
      } else {
        res.render("changeprofile", {
          msg: "File Uploaded!",
          file: `uploads/${req.file.filename}`
        });
        fsextra.removeSync("./public/profile" + id);
        if (err) console.log(err);
        else console.log("Deleted old folder");
        glob("./public/uploads/profile.*", (err, matches) => {
          if (err) console.log(err);
          else {
            console.log("renaming folder...");
            fs.renameSync("./public/uploads", "./public/profile/" + id);
            if (err) console.log(err);
            else {
              fs.mkdir("./public/uploads", err => {
                if (err) console.log(err);
                else
                  console.log(
                    "------------------FOLDER RECREATED---------------------------"
                  );
              });
            }
          }
        });
      }
    }
  });
});



这是我得到的错误:
错误:EPERM:不允许该操作,将其重命名为'./public/uploads'->'./public/profile/21'     在Object.renameSync(fs.js:593:3)     在全球(C:\ Users \ Andrea \ Desktop \ LinkedinLikeSocialNetwork \ routes \ users.js:558:19)     在f(C:\ Users \ Andrea \ Desktop \ LinkedinLikeSocialNetwork \ node_modules \ once \ once.js:25:25)     在Glob。 (C:\ Users \ Andrea \ Desktop \ LinkedinLikeSocialNetwork \ node_modules \ glob \ glob.js:151:7)     在Glob.emit(events.js:189:13)     在Glob._finish(C:\ Users \ Andrea \ Desktop \ LinkedinLikeSocialNetwork \ node_modules \ glob \ glob.js:197:8)     完成时(C:\ Users \ Andrea \ Desktop \ LinkedinLikeSocialNetwork \ node_modules \ glob \ glob.js:182:14)     在Glob._processReaddir2(C:\ Users \ Andrea \ Desktop \ LinkedinLikeSocialNetwork \ node_modules \ glob \ glob.js:434:12)     在C:\ Users \ Andrea \ Desktop \ LinkedinLikeSocialNetwork \ node_modules \ glob \ glob.js:371:17     在RES(C:\ Users \ Andrea \ Desktop \ LinkedinLikeSocialNetwork \ node_modules \ inflight \ inflight.js:31:16)     在f(C:\ Users \ Andrea \ Desktop \ LinkedinLikeSocialNetwork \ node_modules \ once \ once.js:25:25)     在Glob._readdirEntries(C:\ Users \ Andrea \ Desktop \ LinkedinLikeSocialNetwork \ node_modules \ glob \ glob.js:578:10)在C:\ Users \ Andrea \ Desktop \ LinkedinLikeSocialNetwork \ node_modules \ glob \ glob.js:555: 12     在go $ readdir $ cb(C:\ Users \ Andrea \ Desktop \ LinkedinLikeSocialNetwork \ node_modules \ graceful-fs \ graceful-fs.js:162:14)     在FSReqWrap.args [按完成](fs.js:140:20)

1 个答案:

答案 0 :(得分:0)

即使我没有得到答案,我也认为我已经找到了解决方案: 我认为问题是这样的。当您使用sync方法时,您可能会认为在删除和重命名文件系统中的文件夹时,该节点也会与操作系统同步。但似乎并非如此:它将使用某些内核函数调用操作系统,但是fs的管理由os进行,并且节点不等待函数的返回,即使您使用的是同步模式。例如,可能发生所有I / O操作都将被缓冲然后一起执行以提高os的性能的情况。 正确的想法是在执行fs操作之后刷新os缓存。但是,我认为使用fs-extra和graceful-fs可以正常工作:




var fs = require("graceful-fs");
const fsextra = require("fs-extra");
fs.gracefulify(realfs);

router.post("/changeprofile", ensureAuthenticated, (req, res) => {
  upload(req, res, err => {
    if (err) {
      res.render("changeprofile", { msg: err });
    } else {
      if (req.file == undefined) {
        res.render("changeprofile", { msg: "Error: No file Selected!" });
      } else {
        res.render("changeprofile", {
          msg: "File Uploaded!",
          file: `uploads/${req.file.filename}`
        });
        var path = "./public/profile/" + id;
        if (fsextra.existsSync(path)){
          fsextra.removeSync(path);
        }
        if (err) console.log(err);
        else console.log("Deleted old folder");
        glob("./public/uploads/profile.*", (err, matches) => {
          if (err) console.log(err);
          else {
            console.log("renaming folder...");
            fs.renameSync("./public/uploads", "./public/profile/" + id);
            if (err) console.log(err);
            else {
              fs.mkdir("./public/uploads", err => {
                if (err) console.log(err);
                else
                  console.log(
                    "------------------FOLDER RECREATED---------------------------"
                  );
              });
            }
          }
        });
      }
    }
  });
});