req.user 未定义 - 节点 + 快递 + 护照-facebook

Posted

技术标签:

【中文标题】req.user 未定义 - 节点 + 快递 + 护照-facebook【英文标题】:req.user undefined - node + express + passport-facebook 【发布时间】:2012-11-29 05:53:29 【问题描述】:

我正在尝试让护照与我的节点快递服务器一起工作。我可以使用 Facebook 登录并在我的数据库中找到正确的用户;但是,当我重定向 req.user 时始终未定义。这是我的服务器代码:

var express = require('express'),
path = require('path'),
http = require('http'),
passport = require('passport'),
FacebookStrategy = require('passport-facebook').Strategy,
user = require('./routes/users');


var app = express();

passport.use(new FacebookStrategy(
    clientID: "HIDDEN",
    clientSecret: "HIDDEN",
    callbackURL: "http://MYURL.com/auth/facebook/callback",
    passReqToCallback: true
 ,
  function(req, accessToken, refreshToken, profile, done) 
    user.findOrCreate(profile, function(err, user)       
      if (err)  return done(err);       
      done(null, user);
    );    
  
));

passport.serializeUser(function(user, done)   
  done(null, user._id);
);

passport.deserializeUser(function(id, done) 
  user.findById(id, function(err, user) 
    done(err, user);
  );
);

app.configure(function ()     
app.set('port', process.env.PORT || 3000);
    app.use(express.static(path.join(__dirname, 'public')));
    app.use(express.logger('dev')); /* 'default', 'short', 'tiny', 'dev' */
    app.use(express.bodyParser());
    app.use(express.cookieParser());
    app.use(express.session( secret: 'foobar' ));   
    app.use(passport.initialize());
    app.use(passport.session()); 
    app.use(app.router);    
);

app.get('/auth/user', function (req, res) 
    console.log(req);
);

app.get('/auth/facebook', passport.authenticate('facebook'));
app.get('/auth/facebook/callback', 
  passport.authenticate('facebook',  successRedirect: '/',
                                      failureRedirect: '/login' ));


http.createServer(app).listen(app.get('port'), function () 
    console.log("Express server listening on port " + app.get('port'));
);

我转到 /auth/facebook,它会找到正确的用户并将我重定向到 /。但后来我转到 /auth/user 并且 req.user 未定义,我的会话显示:

cookies:  'connect.sid': 's:JFdQkihKQQyR4q70q7h2zWFt.VS+te0pT0z/Gtwg7w5B33naCvA/ckKMk60SFenObxUU' ,
  signedCookies: ,
  url: '/auth/user',
  method: 'GET',
  sessionStore:
    sessions:
       '5sk3Txa2vs5sYhvtdYwGaUZx': '"cookie":"originalMaxAge":null,"expires":null,"httpOnly":true,"path":"/","passport":"user":"50c527c9c6cb41860b000001"',
        'Au6m0hAj/3warKOGNSWw0yu2': '"cookie":"originalMaxAge":null,"expires":null,"httpOnly":true,"path":"/","passport":',
        JFdQkihKQQyR4q70q7h2zWFt: '"cookie":"originalMaxAge":null,"expires":null,"httpOnly":true,"path":"/","passport":' ,
     generate: [Function],
     _events:  disconnect: [Function], connect: [Function]  ,
  sessionID: 'JFdQkihKQQyR4q70q7h2zWFt',

是否与我的 sessionID 与设置护照用户的会话不匹配有关?

更新

所以我确定 sessionID 不匹配是因为我在 c9.io 上运行我的代码,它实际上有两个 URL。当我使用正确的 URL 并转到 /auth/user 时,我的 sessionID 与带有护照用户集的会话匹配,我可以在日志中看到我的 deserializeUser 找到了正确的用户对象。但是,req.user 在此之后仍然未定义。

Trying to find user with id: 50c527c9c6cb41860b000001
 sessions:
    yoOUOxyXZ0SmutA0t5xUr6nI: '"cookie":"originalMaxAge":null,"expires":null,"httpOnly":true,"path":"/","passport":"user":"50c527c9c6cb41860b000001"',
     goZpmK3y3tOfn660hRbz2hSa: '"cookie":"originalMaxAge":null,"expires":null,"httpOnly":true,"path":"/","passport":',
     'O5Sz1GuZqUO8aOw4Vm/hriuC': '"cookie":"originalMaxAge":null,"expires":null,"httpOnly":true,"path":"/","passport":' ,
  generate: [Function],
  _events:  disconnect: [Function], connect: [Function]  
sessionID: yoOUOxyXZ0SmutA0t5xUr6nI
req.user: undefined
 cookie:
    path: '/',
     _expires: null,
     originalMaxAge: null,
     httpOnly: true ,
  passport:  

更新2

我发现了问题所在。它在我的 user.findByID 函数中:

exports.findById = function(id, callback)     
    console.log('Trying to find user with id: ' + id);
    db.collection('users').findOne('_id':id, function(err, user)            
            callback(err, user);
    );
;

改为:

exports.findById = function(id, callback)     
    console.log('Trying to find user with id: ' + id);
    db.collection('users').findOne('_id':new BSON.ObjectID(id), function(err, user)            
            callback(err, user);
    );
;

【问题讨论】:

另一个注意事项:注意快速路线的定义顺序 - 当您定义 app.get 或任何其他路线时,这很重要。由于您可以在护照中间件之前定义用户路由,因此在此路由中也不会有用户数据。 (这不是您的具体情况,但症状非常相似)。 现在可以使用了吗?如果是这样,请将其作为答案发布 一个方便的工具是将你的节点复制并粘贴到 JSHint jshint.com 你也可以运行一个 grunt 任务让 JSHint 监视你的代码。 使用 mongoosejs 可以避免这种事情。 【参考方案1】:

正如你在更新中所说的那样,user._id 变量的格式不正确。为了避免这种情况,并且以后在其他请求和方法中必须处理这种格式,我建议您在注册时生成一个新的用户 ID。

你可以使用这个模块:

node-uuid

var uuid = require('node-uuid');

function findOrCreate(profile, callback) 
  // save new profile
  profile.uid = uuid.v1().replace(/\-/g, '');

【讨论】:

以上是关于req.user 未定义 - 节点 + 快递 + 护照-facebook的主要内容,如果未能解决你的问题,请参考以下文章

req.user 注册后未定义

Node + Express + Passport + Mongoose:req.user 未定义

Passport JS req.user 未定义

使用 google-passport-oauth 登录后 req.user 未定义

Passport + NodeJs + Express 得到“req.user”未定义

req.user.displayname 未定义 Nodejs + passport = google oauth