node.js - 多次调用 PassportJS 中间件

标签 node.js express passport.js sequelize.js

似乎每当我调用 Passport 的 ensureAuthenticated 中间件时,它都会导致 passport.deserializeUser 函数被额外调用 6-7 次。我不确定这是我的应用程序通过 Express、Sequelize 的结构,还是导入 Passport 的方式。出于这个原因,我将布置一些文件,希望找到它误入歧途的地方。

我是这样安排一切的

application/
  auth/
  models/
  node-modules/
  public/
  routes/
  views/
  app.js

我的假设是因为中间件不是单例,和/或因为我的路由设置很奇怪。 注意:我关注了this关于设置单例 Sequelize 方法的指南。

./app.js

// .. imports

app.set('models', require('./models')); // singleton ORM (my assumption)
// .. session stuff
app.use(passport.initialize());
app.use(passport.session());
app.use(require('./auth'));
// .. etc

app.use('/', require('./routes')); // routing style possible issue?

// .. create server

./auth/index.js

module.exports = function () {

    var express   = require('express')
      , passport  = require('passport')
      , Local     = require('passport-local').Strategy
      , app       = express();


    passport.use(new Local(
        function(username, password, done) {
            app.get('models').User.find({
                where: {
                    username: username,
                    password: password
                }
            }).done(function (err, user) {
                if (err) {
                    return done(err);
                }
                if (!user) {
                    return done(null, false, { message: 'Invalid login' });
                }
                return done(null, user);
            });
        }
    ));

    passport.serializeUser(function(user, done) {
        done(null, user.id);
    });

    passport.deserializeUser(function(id, done) {
        app.get('models').User.find(id).done(function(err, user) {
            done(err, user);
        });
    });


    return app;
}();

./auth/middleware.js

exports.check = function(req, res, next) {
    if (req.isAuthenticated()) {
        return next();
    }
    res.redirect('/login')
};

exports.is = function(role) {
    return function (req, res, next) {
        if (req.usertypes[req.user.type] === role) next();
        else res.redirect('back');
    }
};

./routes/index.js

module.exports = function () {

    var express  = require('express')
      , app      = express();


    app.get('/', function (req, res) {
        if (!req.user) res.redirect('/login');
        else res.redirect('/' + req.usertypes[req.user.type]);
    });

    app.use('/admin',    require('./admin'));
    app.use('/another1', require('./another1')); // yadda
    app.use('/another2', require('./another2')); // yadda
    app.use('/login',    require('./login'));

    app.get('/logout', function(req, res){
        req.logout();
        res.redirect('/');
    });


    return app;
}();

最后./routes/admin.js

module.exports = function () {

    var express = require('express')
      , auth    = require('../auth/middleware')
      , admin   = express();


    // auth.check seems to be what's firing the multiple queries:
    // Executing: SELECT * FROM `users` WHERE `users`.`id`=1 LIMIT 1;
    // 6 times from the looks of it.

    admin.get('/', auth.check, auth.is('admin'), function (req, res) {
        res.render('admin', {
            username: 'req.user.username'
        });
    });

    admin.get('/users.json', auth.check, auth.is('admin'), function (req, res) {
        res.contentType('application/json');
        admin.get('models').User.findAll().done(function (err, users) {
            if (users.length === 0) {
                // handle
            } else {
                res.send(JSON.stringify(users));
            }
        });
    });

    admin.post('/adduser', auth.check, auth.is('admin'), function (req, res) {
        var post = req.body;
        admin.get('models').User.create(post).done(function (err, user) {
            if (!err) {
                res.send(JSON.stringify({success: true, users: user}));
            } else {
                res.send(JSON.stringify({success: false, message: err}));
            }
        });
    });


    return admin;
}();

我知道这是一些代码,但我觉得它非常非常简单。任何指导将不胜感激。

最佳答案

这是因为您在使用静态文件之前使用了 passportJS session 中间件。因此,您的所有静态文件调用(如 <img src="..."> )都通过 session 中间件并调用 deserializeUser() .

解决方案

之后使用你的 session 中间件app.use(express.static(...))在您的 app.js 文件中。

查看 jaredhandson 的这个 GitHub 问题答案以获取更多详细信息:https://github.com/jaredhanson/passport/issues/14#issuecomment-4863459

关于node.js - 多次调用 PassportJS 中间件,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/17498316/

相关文章:

node.js - 什么是对 Express 4 路由器上的某些路由进行身份验证的更好方法?

Node.js HTTP2 服务器错误 : socket hang up

node.js - docker env 变量显示未定义

node.js - 收到错误 PayloadTooLargeError : request entity too large in case of Using express. Router() 后调用

node.js - 将使用node-gd生成的图像发送到客户端

javascript - 如何让 React-Router 呈现服务器的交互式输出?

javascript - 本地 Passport 身份验证策略提供临时 header

javascript - node.js 中的 UDP 套接字

javascript - 喜欢通过 API 在 Facebook 上发帖

node.js - 使用 React Native 和 API 后端进行身份验证