Node.js(Express)错误处理中间件与路由器

rau*_*aul 4 javascript error-handling node.js express

这是我的应用程序结构:

- app.js
- routes
---- index.js
Run Code Online (Sandbox Code Playgroud)

ExpressJS应用程序为环境developmentproduction环境创建错误处理程序.以下是来自以下内容的代码段app.js:

app.use('/', routes); // routing is handled by index.js in the routes folder

//The following middleware are generated when you create the Express App

// catch 404 and forward to error handler
app.use(function (req, res, next) {
    var err = new Error('Not Found');
    err.status = 404;
    next(err);
});

// development error handler
// will print stacktrace
if (app.get('env') === 'development') {
    app.use(function (err, req, res, next) {
        res.status(err.status || 500);
        res.render('error.ejs', {
            message: err.message,
            error: err
        });
    });
}

// production error handler
// no stacktraces leaked to user
app.use(function (err, req, res, next) {
    res.status(err.status || 500);
    res.render('error', {
        message: err.message,
        error: {}
    });
});
Run Code Online (Sandbox Code Playgroud)

在里面routes/index.js,我处理所有路由:

var router = express.Router();

router.get('/', function (req, res) {
    someAsyncFunction(function(err, result) {
        if (err) throw err; // Handle this error
    }
});

module.exports = router;
Run Code Online (Sandbox Code Playgroud)

我希望将err其传递给其中一个错误处理程序而不是被抛出.我怎样才能做到这一点?

sha*_*nks 7

您必须将它传递给下一个回调,这通常是路由处理程序中的第三个参数

var router = express.Router();

router.get('/', function (req, res, next) {
    someAsyncFunction(function(err, result) {
        if (err) {
            next(err); // Handle this error
        }
    }
});

module.exports = router;
Run Code Online (Sandbox Code Playgroud)

调用next(err)将允许错误被捕获到具有以下签名的链中的中间件:

app.use(function (err, req, res, next){
    // do something about the err
});
Run Code Online (Sandbox Code Playgroud)

参考:http://expressjs.com/en/guide/error-handling.html


c-c*_*vez 7

您还可以创建一个中间件函数来处理所有路由中的错误,而无需到处复制代码,如果您愿意,可以使用箭头函数。

1)创建一个const函数来处理错误。

任何一个:

const handleErrorAsync = func => (req, res, next) => {
    func(req, res, next).catch((error) => next(error));
};
Run Code Online (Sandbox Code Playgroud)

或者

const handleErrorAsync = func => async (req, res, next) => {
    try {
        await func(req, res, next);
    } catch (error) {
        next(error);
    }
};
Run Code Online (Sandbox Code Playgroud)

2)在您的路由器中为每个请求使用它:

var router = express.Router();

router.get('/req1', handleErrorAsync(async (req, res, next) => {
   let result = await someAsyncFunction1();
   if(result){
       // res.send whatever
   }
}));
router.post('/req2', handleErrorAsync(async (req, res, next) => {
    let result = await someAsyncFunction2(req.body.param1);
    if(result){
        // res.send whatever
    }
}));
router.post('/req3', handleErrorAsync(async (req, res, next) => {
    let result = await someAsyncFunction3(req.body.param1, req.body.param2);
    if(result){
        // res.send whatever
    }
}));

module.exports = router;
Run Code Online (Sandbox Code Playgroud)

3)在您的服务器主应用程序句柄错误:

// development error handler
// will print stacktrace
if (app.get('env') === 'development') {
    app.use(function (err, req, res, next) {
        res.status(err.status || 500);
        res.render('error.ejs', {
            message: err.message,
            error: err
        });
    });
}

// production error handler
// no stacktraces leaked to user
app.use(function (err, req, res, next) {
    res.status(err.status || 500);
    res.render('error', {
        message: err.message,
        error: {}
    });
});
Run Code Online (Sandbox Code Playgroud)

这样你就可以在任何路由中重用错误处理功能。此外,如果您的任何函数中存在任何未处理的错误,这也会捕获它们。

尝试捕获从这里获取的错误处理