Прерывистая ошибка EPERM при переименовании папки

#node.js #file-permissions #fs #multer

#node.js #файл-разрешения #фс #мультер

Вопрос:

Я использую Multer для загрузки изображений в мой fs. Мультер не позволяет вам динамически устанавливать позицию в 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:UsersAndreaDesktopLinkedinLikeSocialNetworkroutesusers.js:558:19 )
при f (C:UsersAndreaDesktopLinkedinLikeSocialNetworknode_modulesonceonce.js:25:25 )
в Глобусе. (C:UsersAndreaDesktopLinkedinLikeSocialNetworknode_modulesglobglob.js:151:7 )
в Glob.emit (events.js:189:13)
в Glob._finish (C:UsersAndreaDesktopLinkedinLikeSocialNetworknode_modulesglobglob.js:197:8 )
по
завершении (C:UsersAndreaDesktopLinkedinLikeSocialNetworknode_modulesglobglob.js:182:14 ) в глобусе._processReaddir2 (C:UsersAndreaDesktopLinkedinLikeSocialNetworknode_modulesglobglob.js:434:12 ) в C:UsersAndreaDesktopLinkedinLikeSocialNetworknode_modulesglobglob.js:371:17

при повторном (C:UsersAndreaDesktopLinkedinLikeSocialNetworknode_modulesinflightinflight.js:31:16 )
при f (C:UsersAndreaDesktopLinkedinLikeSocialNetworknode_modulesonceonce.js:25:25 )
в Glob._readdirEntries (C:UsersAndreaDesktopLinkedinLikeSocialNetworknode_modulesglobglob.js:578:10 ) в C:UsersAndreaDesktopLinkedinLikeSocialNetworknode_modulesglobglob.js:555:12
при переходе $ readdir $ cb (C:UsersAndreaDesktopLinkedinLikeSocialNetworknode_modulesgraceful-fsgraceful-fs.js:162:14 )
в FSReqWrap.args [как завершенный] (fs.js:140:20)

Ответ №1:

Даже если я не получил ответа, я думаю, что пришел к решению: я думаю, что проблема в этом. Когда вы используете метод синхронизации, вы, вероятно, думаете, что узел также будет синхронизирован с операционной системой при удалении и переименовании вашей папки внутри файловой системы. Но похоже, что это не так: он вызовет операционную систему с некоторой функцией ядра, но затем управление fs выполняется ОС, и узел не ожидает возврата функции, даже если вы используете синхронный режим. Например, может случиться так, что все операции ввода-вывода будут буферизованы, а затем будут выполняться все вместе для повышения производительности ОС. Правильной идеей было бы очистить кэш ОС после того, как вы выполнили операцию fs. Однако, используя 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---------------------------"
                  );
              });
            }
          }
        });
      }
    }
  });
});