如何在 Node.js expressjs 的异步对象方法中处理未处理的承诺拒绝?

Posted

技术标签:

【中文标题】如何在 Node.js expressjs 的异步对象方法中处理未处理的承诺拒绝?【英文标题】:How to Handle Unhandled promise rejection in async object method in Node.js expressjs? 【发布时间】:2017-09-19 17:25:40 【问题描述】:

我使用对象内部的异步函数在 express.js 中发送响应

控制器代码:

module.exports = 

    async signUpEmail(req, res) 

        /**
         * @description Parameters from body
         * @param string firstName - First Name
         * @inner
         */  

        const firstName = req.body.firstName;

        res.send( success: name );
        throw new Error(); // purposely Done
    

问题:

由于在我的情况下 signUpEmail 方法是异步的,因此无论我的异步方法抛出什么,它都会被拒绝Error。(故意放在那里)

所以让这个记录在控制台中。

(node:13537) UnhandledPromiseRejectionWarning: Unhandled promise rejection (rejection id: 1): Error
(node:13537) DeprecationWarning: Unhandled promise rejections are deprecated. In the future, promise rejections that are not handled will terminate the Node.js process with a non-zero exit code.

所以我应该从我调用它的路线来处理它。

路由器代码

    const routes = require('express').Router();
const SignUpController = require('../controllers/signUpController')

// /signup
routes.post('/', SignUpController.signUpEmail);

module.exports = routes;

有些像这样SignUpController.signUpEmail().then(…); 但是因为我没有在路由中调用函数,所以我只是通过。 如何有效地做到这一点

PS:请不要提出太复杂的解决方案。我是 JS 的初学者,正在学习中。

我没有使用可链接的路由处理程序,因为我想创建模块化、可安装的路由处理程序。

Official Doc Example

【问题讨论】:

只需传递一个函数:routes.pos('/', (...args) => SignUpController.signUpEmail().then(…)) @Bergi 现在得到这个 TypeError: Cannot read property 'body' of undefined。似乎传播没有通过 req ..tried console.log 变得未定义 猜你会想要 …signUpEmail(...args)… 或任何你想要的函数 @Bergi 是的,明白了.. 如果您能提供一点启发,为什么它的工作方式特别是 (...args) => 是如何可用的,这是什么?因为如果我这样做`routes.post('/', console.log(args));` 我得到这个ReferenceError: args is not defined 会很有帮助 仅供参考,async/await 不是 ES7 的一部分,它是 ES2017 的一部分。 【参考方案1】:

在你的路由中,你需要添加一个包装器来捕获抛出的错误:

let wrapper = fn => (...args) => fn(...args).catch(args[2]);

// /signup
routes.post('/', wrapper(SignUpController.signUpEmail));

使用此方法,您可以使用***错误捕获器,并且不需要在路由中使用内部 try catch 块,除非您需要根据上下文使用它们。

使用错误捕获中间件来实现这一点,如下所示:

// last middleware in chain

app.use(function(err, req, res, next) 
    // handle your errors
);

现在在您的路由中,您可以抛出错误,它们将被该中间件捕获。我喜欢在这个中间件中抛出自定义错误并处理它们的响应和日志记录。

旁白:异步等待模式非常适合编写异步代码,便于人类以同步方式阅读。请记住,一旦你去异步,你应该保持异步!强烈建议使用 Bluebird 等承诺库并在 nodeback 库上调用 .promisifyAll

编辑:Source - Asynchronous Error Handling in Express with Promises, Generators and ES7

【讨论】:

第一个有很大帮助并理解它,但是对于使用中间件,我们不是已经将 next(err) 传递给它以某种方式在这里工作吗? let wrapper = fn => (...args) => fn(...args).catch(args[2]); 从包装器内部的函数中捕获抛出的错误并将其作为路由的下一个返回,然后由您的错误处理中间件拾取 这能回答你的问题吗?

以上是关于如何在 Node.js expressjs 的异步对象方法中处理未处理的承诺拒绝?的主要内容,如果未能解决你的问题,请参考以下文章

将参数传递给 Node.js 异步瀑布

如何使用 Express JS/Railway JS (Node.JS) 生成 JSON

Node.js 和 expressjs paypal ipn 监听器给出错误 500

我应该在每次使用 ExpressJS/Node 更改时重新启动应用程序服务器吗? [复制]

ExpressJS包的用途?

ExpressJS Node.js Angular 错误:获取 OAuth2 API 授权令牌时出现 getaddrinfo ENOTFOUND