node.js - 发送 Node 后无法设置 header

标签 node.js mongodb express mongoose

我有以下密码重置表单的路线,该表单有一个密码和密码确认字段。

router.post('/users/reset/:token', (req, res, next) => {
  if(req.body.password === req.body['password-confirm']) {
    req.flash('error', 'Passwords do not match!');
    res.redirect('/users/forgot');
  }

  User.findOne({
    resetPasswordToken: req.params.token,
    resetPasswordExpires: { $gt: Date.now() }
  }, function(err, user) {
    if(!user) {
      req.flash('error', ' Password reset is invalid or has expired');
      res.redirect(302, '/login');
    }

    const setPassword = promisify(user.setPassword, user);
    setPassword(req.body.password);
    user.resetPasswordToken = undefined;
    user.resetPasswordExpires = undefined;
    const updatedUser = user.save();

    user.save((saveError, updatedUser) => {
      // Check if saveError is present here and handle appropriately
      req.login(updatedUser, loginError => {
        req.flash('success_msg', 'Your password has been reset successfully! You are now logged in!');
        res.redirect('/dashboard' + req.user);
      })
    });
  });
});

当我填写表单时,出现以下错误

Fri Jan 26 2018 11:28:55 GMT+0000 (GMT): GET /users/reset/6e2574bfa532e0d13af7fae61114308f9a683767
Mongoose: users.findOne({ resetPasswordExpires: { '$gt': new Date("Fri, 26 Jan 2018 11:28:55 GMT") }, resetPasswordToken: '6e2574bfa532e0d13af7fae61114308f

9a683767' }, { fields: {} })
Fri Jan 26 2018 11:28:55 GMT+0000 (GMT): GET /favicon.ico
Fri Jan 26 2018 11:29:02 GMT+0000 (GMT): POST /users/reset/6e2574bfa532e0d13af7fae61114308f9a683767
Mongoose: users.findOne({ resetPasswordExpires: { '$gt': new Date("Fri, 26 Jan 2018 11:29:02 GMT") }, resetPasswordToken: '6e2574bfa532e0d13af7fae61114308f
9a683767' }, { fields: {} })
Mongoose: users.update({ _id: ObjectId("5a5c6740b9e210087e098fd6") }, { '$unset': { resetPasswordExpires: 1, resetPasswordToken: 1 } })
Mongoose: users.update({ _id: ObjectId("5a5c6740b9e210087e098fd6") }, { '$unset': { resetPasswordExpires: 1, resetPasswordToken: 1 } })
(node:1451) UnhandledPromiseRejectionWarning: TypeError: Cannot read property 'apply' of undefined
    at /Users/benbagley/Code/poetry-out-loud/node_modules/es6-promisify/dist/promisify.js:75:41
    at new Promise (<anonymous>)
    at /Users/benbagley/Code/poetry-out-loud/node_modules/es6-promisify/dist/promisify.js:54:20
    at /Users/benbagley/Code/poetry-out-loud/routes/users.js:321:5
    at model.Query.<anonymous> (/Users/benbagley/Code/poetry-out-loud/node_modules/mongoose/lib/model.js:4056:16)
    at /Users/benbagley/Code/poetry-out-loud/node_modules/kareem/index.js:273:21
    at /Users/benbagley/Code/poetry-out-loud/node_modules/kareem/index.js:131:16
    at process._tickCallback (internal/process/next_tick.js:150:11)
(node:1451) UnhandledPromiseRejectionWarning: Unhandled promise rejection. This error originated either by throwing inside of an async function without a c
atch block, or by rejecting a promise which was not handled with .catch(). (rejection id: 1)
(node:1451) [DEP0018] DeprecationWarning: Unhandled promise rejections are deprecated. In the future, promise rejections that are not handled will terminat
e the Node.js process with a non-zero exit code.
Fri Jan 26 2018 11:29:02 GMT+0000 (GMT): GET /users/forgot
events.js:136
      throw er; // Unhandled 'error' event
      ^

Error [ERR_HTTP_HEADERS_SENT]: Cannot set headers after they are sent to the client
    at validateHeader (_http_outgoing.js:503:11)
    at ServerResponse.setHeader (_http_outgoing.js:510:3)
    at ServerResponse.header (/Users/benbagley/Code/poetry-out-loud/node_modules/express/lib/response.js:767:10)
    at ServerResponse.location (/Users/benbagley/Code/poetry-out-loud/node_modules/express/lib/response.js:884:15)
    at ServerResponse.redirect (/Users/benbagley/Code/poetry-out-loud/node_modules/express/lib/response.js:922:18)
    at req.login.loginError (/Users/benbagley/Code/poetry-out-loud/routes/users.js:332:13)
    at /Users/benbagley/Code/poetry-out-loud/node_modules/passport/lib/http/request.js:51:48
    at /Users/benbagley/Code/poetry-out-loud/node_modules/passport/lib/sessionmanager.js:16:14
    at pass (/Users/benbagley/Code/poetry-out-loud/node_modules/passport/lib/authenticator.js:297:14)
    at Authenticator.serializeUser (/Users/benbagley/Code/poetry-out-loud/node_modules/passport/lib/authenticator.js:299:5)
    at SessionManager.logIn (/Users/benbagley/Code/poetry-out-loud/node_modules/passport/lib/sessionmanager.js:14:8)
    at IncomingMessage.req.login.req.logIn (/Users/benbagley/Code/poetry-out-loud/node_modules/passport/lib/http/request.js:50:33)
    at user.save (/Users/benbagley/Code/poetry-out-loud/routes/users.js:330:11)
    at /Users/benbagley/Code/poetry-out-loud/node_modules/mongoose/lib/model.js:4056:16
    at /Users/benbagley/Code/poetry-out-loud/node_modules/mongoose/lib/services/model/applyHooks.js:170:20
    at process._tickCallback (internal/process/next_tick.js:150:11)

我对 Node 还是个新手,所以任何重构技巧或对此的修复都将非常感激,我在这个问题上被困了几天,不确定我做错了什么。

谢谢。

最佳答案

第一件事是,当 Node 多次发送 API 调用的响应时,会发生此错误。其次,发送响应时最好使用 return,例如 return res.json(<OBJECT>);

在您的代码中,您正在检查密码,例如 req.body.password === req.body['password-confirm'] 它应该像 req.body.password !== req.body['password-confirm'] 这可能会导致 Node 发送多个响应

关于node.js - 发送 Node 后无法设置 header ,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/48462086/

相关文章:

java - Spring Data MongoDB 身份验证错误

MongoDB Mongoose 使用 Map 查找

mongodb - 将 $near 与 $maxDistance 和 $and 一起使用可以充当 MongoDB 中的过滤器吗?

javascript - Req.isAuthenticated 变为 false

node.js - 在 Node.js 和浏览器之间共享一个类

javascript - 使用 Typescript+VSCode 调试 Node.js 异步/等待

node.js - RxJS if-else 就像管道过滤

javascript - 无法通过 'findById' 和 save() 更新文档

javascript - 无法将 .pem 文件导入 Express

javascript - 我使用 axios 发送的数据在配置中返回,而不是数据