使用promises在Node.js + Express中处理错误

时间:2014-10-20 16:33:41

标签: node.js express error-handling mongoose promise

使用Node.js + Express(4)+ Mongoose(使用promises而不是回调),我无法理清如何整理我的错误处理。

我得到的(相当简化)是:

app.get('/xxx/:id', function(request, response) {
    Xxx.findById(request.params.id).exec()
        .then(function(xxx) {
            if (xxx == null) throw Error('Xxx '+request.params.id+' not found');
            response.send('Found xxx '+request.params.id);
        })
        .then(null, function(error) { // promise rejected
            switch (error.name) {
                case 'Error':
                    response.status(404).send(error.message); // xxx not found
                    break;
                case 'CastError':
                    response.status(404).send('Invalid id '+request.params.id);
                    break;
                default:
                    response.status(500).send(error.message);
                    break;
            }
        });
});

此处,在“承诺被拒绝”部分的切换中,Error是我为自己找不到的潜在有效ID而导致的错误,CastError强制转换为Mongoose针对无效ID抛出了对象失败,例如可以通过将throw Error()误导为throw Err()来触发500错误(导致 ReferenceError:Err未定义)。

但是像这样,我的每条路线都有这个非常笨拙的开关来处理不同的错误。

如何集中处理错误?可以将交换机隐藏在一些中间件中吗?

(我希望我可以在'承诺拒绝'区块中使用throw error;重新投掷,但我无法使其工作。

1 个答案:

答案 0 :(得分:6)

我会创建中间件来处理错误。将next()用于404s。和next(err)其他错误。

app.get('/xxx/:id', function(req, res, next) {
  Xxx.findById(req.params.id).exec()
    .then(function(xxx) {
      if (xxx == null) return next(); // Not found
      return res.send('Found xxx '+request.params.id);
    })
    .then(null, function(err) {
      return next(err);
    });
});

404处理程序

app.use(function(req, res) {
  return res.send('404');
});

错误处理程序

app.use(function(err, req, res) {
  switch (err.name) {
    case 'CastError':
      res.status(400); // Bad Request
      return res.send('400');
    default:
      res.status(500); // Internal server error
      return res.send('500');
  }
});

您可以通过发送类似以下的json响应来进一步改进:

return res.json({
  status: 'OK',
  result: someResult
});

return res.json({
  status: 'error',
  message: err
});