异步等待快速中间件无法正常工作

时间:2018-04-12 18:16:42

标签: javascript node.js express async-await

我对Node有点新手,所以要温柔。我正在为我的婚礼创建一个应用程序,它采用上传的访客列表(Excel文件格式)并将其转换为JSON数组,然后我可以使用它来构建关于每个访客的配置文件(饮食要求,rsvp答案等)。

到目前为止,我在主页上有一个表单,允许用户上传.xlsx文件,当提交表单时,用户会再次重定向回主页。

我创建了以下路线:

router.post('/',
    guestsController.upload,
    guestsController.getGuestlist,
    guestsController.displayGuestlist
);

这是我的客人控制器:

const multer = require('multer');
const convertExcel = require('excel-as-json').processFile;

const storage = multer.diskStorage({ //multers disk storage settings
    destination: function (req, file, cb) {
      cb(null, './uploads/')
    },
    filename: function (req, file, cb) {
        var datetimestamp = Date.now();
        cb(null, file.fieldname + '-' + datetimestamp + '.' + file.originalname.split('.')[file.originalname.split('.').length -1])
    }
});

exports.upload = multer({storage: storage}).single('file');

exports.getGuestlist = async (req, res, next) => {

  try {
     await convertExcel(req.file.path, null, null, (err, guestData) => {
         req.guestlist = guestData.map((guestObj) => Object.values(guestObj)[0]);
     });
     console.log(req.guestlist);
     next();
  } catch (e){
    res.json({error_code:1,err_desc:"Corrupted Excel file"});
    next();
  }
};

exports.displayGuestlist = (req, res) => {
    console.log(req.guestlist);
};

目前由于函数的同步特性,displayGuestlist返回undefined到控制台,因为covertExcel还没有完成抓取数据。您可以看到我尝试使用新的async await语法来解决此问题,但遗憾的是它还没有修复它。

我还尝试将displayGuestlist中的日志放入超时函数中,这已经证明这纯粹是一个计时问题。

任何帮助者都会非常感激。

2 个答案:

答案 0 :(得分:7)

看起来convertExcel不是Promise-returns函数,而是使用旧式回调函数。 await不适用于那些,因此它等待Promise.resolve(undefined),因为函数返回undefined,而不是Promise。值得庆幸的是,在Node 8及更高版本中,有一个promisify实用程序可将回调样式函数转换为Promise返回函数,以便可以使用await

const { promisify } = require('util');
const convertExcel = promisify(require('excel-as-json').processFile);

// ...

const guestData = await convertExcel(req.file.path, null, null);
req.guestlist = guestData.map((guestObj) => Object.values(guestObj)[0]);

答案 1 :(得分:0)

您可以将代码封装在承诺中并等待此承诺解决。

exports.getGuestlist = async (req, res, next) => {
let promise = new Promise((resolve, reject) => {
    convertExcel(req.file.path, null, null, (err, guestData) => {
        if(err) reject(err);
        else resolve(guestData);
    });
});
try {
    let guestData = await promise;
    req.guestlist = guestData.map((guestObj) => Object.values(guestObj)[0]); 
    console.log(req.guestlist);
    next();
} catch (e){
    res.json({error_code:1,err_desc:"Corrupted Excel file"});
    next();
    }
};