使用 Node/Express 不断收到“发送后无法设置标题”
Posted
技术标签:
【中文标题】使用 Node/Express 不断收到“发送后无法设置标题”【英文标题】:Keep getting "Can't set headers after they are sent" using Node/Express 【发布时间】:2018-10-26 20:54:21 【问题描述】:在构建 Node/Express API 时,我不断收到“发送后无法设置标头”。
问题是在响应发送到任何地方后我没有设置标题。我总是调用 res.status(xxx).json() 来关闭任何条件。
路线
const router = require('express').Router();
router.get('/password/validate/:hash', PasswordController.validate);
router.post('/password/update', PasswordController.update);
控制器
这是发生错误的地方。我正在专门调用 validate 请求。
// Import node packages
const mongoose = require('mongoose');
const Password = require('../models/password');
const User = require('../models/user');
const bcrypt = require('bcryptjs');
const moment = require('moment');
const string = require('../middleware/string_functions')
exports.update = (req, res, next) =>
User.findOne( email: req.body.email )
.exec()
.then(user =>
if (!user)
res.status(401).json(
message: 'Cannot retrieve account'
)
const expiry = moment().add(30, 'seconds');
const unique_string = string.generate_random(32);
const url_hash = string.base64_encode(unique_string +':'+ user._id);
bcrypt.hash(unique_string, 10, (err, hash) =>
if (err)
res.status(500).json(
error: err.message
)
const query = user_id: user._id
const newData =
hash,
expiry
Password.findOneAndUpdate(query, newData, upsert: true, new: true )
.exec()
.then(request =>
res.status(201).json(
message: 'success',
url: 'localhost:8081/users/password/validate/' + url_hash,
data: request
)
)
.catch(err =>
res.status(500).json(
error: err.message
)
)
)
)
.catch(err =>
res.status(500).json(
error: err.message
)
)
exports.validate = (req, res, next) =>
if (!req.params.hash)
res.status(500).json(
error: 'Missing hash'
)
const data = string.base64_decode(req.params.hash).split(':');
console.log(data)
Password.findOne( user_id: data[1] )
.exec()
.then(request =>
if (!request)
res.status(404).json(
message: 'Change request not found or expired'
)
bcrypt.compare( data[0], request.hash, (err, result) =>
if (err)
res.status(500).json(
error: err.message
)
if (result)
if (moment().isAfter(request.expiry))
res.status(401).json(
message: 'Time has expired'
)
res.status(200).json(
message: 'Hash validation successful'
)
res.status(500).json(
error: 'Something went wrong'
)
)
)
.catch(err =>
res.status(500).json(
error: err.message
)
)
控制台错误
_http_outgoing.js:494
throw new Error('Can\'t set headers after they are sent.');
^
Error: Can't set headers after they are sent.
at validateHeader (_http_outgoing.js:494:11)
at ServerResponse.setHeader (_http_outgoing.js:501:3)
at ServerResponse.header (/Users/chrislloyd/Development/Projects/happy-hour-api/node_modules/express/lib/response.js:767:10)
at ServerResponse.send (/Users/chrislloyd/Development/Projects/happy-hour-api/node_modules/express/lib/response.js:170:12)
at ServerResponse.json (/Users/chrislloyd/Development/Projects/happy-hour-api/node_modules/express/lib/response.js:267:15)
at bcrypt.compare (/Users/chrislloyd/Development/Projects/happy-hour-api/api/controllers/passwords.js:83:22)
at /Users/chrislloyd/Development/Projects/happy-hour-api/node_modules/bcryptjs/dist/bcrypt.js:297:21
at /Users/chrislloyd/Development/Projects/happy-hour-api/node_modules/bcryptjs/dist/bcrypt.js:1353:21
at Immediate.next [as _onImmediate] (/Users/chrislloyd/Development/Projects/happy-hour-api/node_modules/bcryptjs/dist/bcrypt.js:1233:21)
at runCallback (timers.js:789:20)
at tryOnImmediate (timers.js:751:5)
at processImmediate [as _immediateCallback] (timers.js:722:5)
更新示例
exports.update = (req, res, next) =>
// Check if hash value exists
if (!req.params.hash)
res.status(500).json(
error: 'Missing hash value'
);
return;
// Check if password and confirmation are the same
if (req.body.password != req.body.passwordConfirmation)
res.status(401).json(
message: 'Password confirmation does not match'
);
return;
// Decode and split hash and user id into array
const data = string.base64_decode(req.params.hash).split(':');
// Find record that contains user id
Password.findOne( user_id: data[1] )
.exec()
.then(request =>
console.log(request)
// Throw 404 error if record is not found
if (!request)
return res.status(404).json(
message: 'Password change request doest not exist or timed out'
);
// Check if change request has expired
if (moment().isAfter(request.expiry))
res.status(401).json(
message: 'Password change request expired',
request:
request: 'http://localhost:3001/users/password/request'
);
// Delete expired record
Password.remove( _id: request._id )
.exec()
.catch(err =>
res.status(500).json(
error: err.message
);
);
return;
// Compare hash value from encoded string to encrypted hash value in database
console.log(mongoose.Types.ObjectId(request.user_id))
bcrypt.compare( data[0], request.hash, (err, result) =>
// Bcrypt error performing comparison
if (err)
res.status(500).json(
error: err.message
);
return;
// Check if result is true
if (result)
// Find user record matching request.user_id and update password
User.findOneAndUpdate( _id: mongoose.Types.ObjectId(request.user_id) , $set: password: req.body.password , new: true, (err, user) =>
console.log(user)
// Error finding and updating user record
if (err)
res.status(500).json(
error: err.message
);
return;
// If returned user account is not null
if (user)
res.status(200).json(
message: 'Password updated',
user
);
return;
// Could not find user record
res.status(404).json(
message: 'Could not find user account to update'
);
return;
)
// Catch all error
res.status(500).json(
error: 'Something went wrong'
);
return;
)
)
.catch(err =>
res.status(500).json(
error: err.message
);
return;
);
【问题讨论】:
错误在验证函数中bcrypt.compare
之后...如果error
如果result
然后再res.json
!!!
【参考方案1】:
当您对同一请求发送多个响应时会导致该特定错误。
您会看到,一旦您执行res.status(...).json(...)
,您的函数就会返回并停止执行。它不是。 res.json()
只是一个普通的函数调用。它根本不会改变函数中的控制流(除非它抛出异常)。对res.json()
的成功调用会执行,然后您的函数会继续执行后面的代码行。
您需要的是每次发送响应后的return
语句(如果您的函数中有任何其他代码可以执行并发送另一个响应),以便您的函数不会继续执行并发送另一个响应,或者您可以将您的响应括在 if/else
语句中,这样您就不会发送多个响应。
这是一个固定版本,添加了 5 条 return
语句,以防止您在发送响应后执行其余代码,并防止您对同一请求发送多个响应。每个添加都用==> added
注释:
// Import node packages
const mongoose = require('mongoose');
const Password = require('../models/password');
const User = require('../models/user');
const bcrypt = require('bcryptjs');
const moment = require('moment');
const string = require('../middleware/string_functions')
exports.update = (req, res, next) =>
User.findOne( email: req.body.email )
.exec()
.then(user =>
if (!user)
res.status(401).json(
message: 'Cannot retrieve account'
)
return; // <== added
const expiry = moment().add(30, 'seconds');
const unique_string = string.generate_random(32);
const url_hash = string.base64_encode(unique_string +':'+ user._id);
bcrypt.hash(unique_string, 10, (err, hash) =>
if (err)
res.status(500).json(
error: err.message
)
return; // <== added
const query = user_id: user._id
const newData =
hash,
expiry
Password.findOneAndUpdate(query, newData, upsert: true, new: true )
.exec()
.then(request =>
res.status(201).json(
message: 'success',
url: 'localhost:8081/users/password/validate/' + url_hash,
data: request
)
)
.catch(err =>
res.status(500).json(
error: err.message
)
)
)
)
.catch(err =>
res.status(500).json(
error: err.message
)
)
exports.validate = (req, res, next) =>
if (!req.params.hash)
res.status(500).json(
error: 'Missing hash'
)
const data = string.base64_decode(req.params.hash).split(':');
console.log(data)
Password.findOne( user_id: data[1] )
.exec()
.then(request =>
if (!request)
res.status(404).json(
message: 'Change request not found or expired'
)
return; // <== added
bcrypt.compare( data[0], request.hash, (err, result) =>
if (err)
res.status(500).json(
error: err.message
)
return; // <== added
if (result)
if (moment().isAfter(request.expiry))
res.status(401).json(
message: 'Time has expired'
)
res.status(200).json(
message: 'Hash validation successful'
)
return; // <== added
res.status(500).json(
error: 'Something went wrong'
)
)
)
.catch(err =>
res.status(500).json(
error: err.message
)
)
【讨论】:
其实res.json
是route的return语句...没有返回代码执行?
@AshishChoudhary - res.json()
只是一个常规函数调用。它不会导致您的函数返回或停止执行任何其他函数调用,例如console.log("hi")
。它确实会导致发送响应,但您的函数在调用该函数后继续执行得很好。如果你想让你的函数返回,你需要一个return
语句,或者你需要将所有res.xxx()
调用括在if/else
中,这样你就不能执行多个。
@AshishChoudhary - 在我的回答中添加了关于这一点的进一步说明。
这太棒了!非常感谢你的解释。它与我得到的另一个错误一致,即未处理的承诺。通过不返回流程继续触发并且不处理代码中的其他承诺。 (我猜)。
@jfriend00 我又遇到了这个标题问题:P .. 这次我按照你的例子并确保我返回了我所有的 res 语句。我在上面添加了另一个代码示例,您可以快速浏览一下吗?【参考方案2】:
res
对象本身不会停止程序的执行。如果您喜欢使用Guard Clauses instead of Nested Conditions,则必须使用return
替换语句如下:
if (err)
res.status(500).json(
error: err.message
)
有了这个:
if (err)
res.status(500).json(
error: err.message
);
return; // return statement added
【讨论】:
其实res.json是route的return语句...没有返回代码执行? 你只是告诉节点我必须响应它,但是如果没有明确的 return 语句,你的代码仍然会运行,所以你可以在响应用户后做任何你想做的事情,除非再次响应。 您也可以查看此答案以获取更多信息***.com/a/16180550/6111342 欢迎您。如果此回复有帮助,请将其标记为解决方案,以便其他用户也可以轻松获得帮助 你和 jfriend00 都答对了。我把解决方案给了 jfriend00 因为他们是第一个回答的。感谢您提供附加链接!以上是关于使用 Node/Express 不断收到“发送后无法设置标题”的主要内容,如果未能解决你的问题,请参考以下文章
在 Node/Express 中续集 - “没有这样的表:main.User”错误
在 Node + Express API 中检测到的打开句柄会阻止 Jest 退出
尝试通过 express node.js 提供 ejs 文件时出错
Access-Control-Allow-Headers - Node / Express / KeystoneJs不允许使用标头字段