使用 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 添加新路由

在 Node/Express 中续集 - “没有这样的表:main.User”错误

在 Node + Express API 中检测到的打开句柄会阻止 Jest 退出

尝试通过 express node.js 提供 ejs 文件时出错

Access-Control-Allow-Headers - Node / Express / KeystoneJs不允许使用标头字段

在 node/express 中启用了 CORS,但得到“对预检请求的响应未通过访问控制检查”