User.findOrCreate的功能是什么以及何时在护照中调用?

时间:2013-12-06 18:28:45

标签: node.js passport.js

我找不到关于此功能的文档,因此我无法使其正常工作。该函数何时被调用,它正在做什么以及它作为第一个参数采取了什么?我正试图从护照获取访问令牌,但无论如何都无法访问它。

passport.use(new FacebookStrategy({
    clientID:   APP_ID,
    clientSecret:   APP_SECRET,
    callbackURL: "http://localhost:3000/",
  },
  function(accessToken, refreshToken, profile, done) {
    User.findOrCreate({// what are these parameters?}, function (err, user) {
        // when is this function called and what is it doing? 
       });

  }
));

如何从护照获取访问令牌?

2 个答案:

答案 0 :(得分:64)

User.findOrCreate是一个组合函数,表示您通过Facebook ID查找用户的任何功能,或者如果用户不存在则创建一个功能。我认为你的第一个问题是你的回调网址只是你的根,所以你可能永远不会去那个功能。

您的回调网址应该像http://localhost:3000/auth/facebook/callback

然后处理该网址:

app.get('/auth/facebook/callback', 
  passport.authenticate('facebook', { failureRedirect: '/login' }),
  function(req, res) {
    res.redirect('/');
  });

此时身份验证已完成。 accessToken会返回给您 - “只要应用程序调用API来代表他们读取,修改或写入特定人员的Facebook数据,就需要这样做”。您应该在一些用于存储用户访问令牌的表中将其保存。 profile是另一个关键变量,因为这是关于用户的信息(哪些信息取决于服务)。

您在该功能中所做的工作取决于您。所以,制作自己的User.findOrCreate。以下是Facebook护照的代码,并附有一些注释来解释它。这假设您正在使用类似MongoDB的东西并且具有User表。在这种情况下,User是您声明的可以与User表接口的任何变量。

//Use facebook strategy
passport.use(new FacebookStrategy({
        clientID: config.facebook.clientID,
        clientSecret: config.facebook.clientSecret,
        callbackURL: config.facebook.callbackURL
    },
    function(accessToken, refreshToken, profile, done) {
        //check user table for anyone with a facebook ID of profile.id
        User.findOne({
            'facebook.id': profile.id 
        }, function(err, user) {
            if (err) {
                return done(err);
            }
            //No user was found... so create a new user with values from Facebook (all the profile. stuff)
            if (!user) {
                user = new User({
                    name: profile.displayName,
                    email: profile.emails[0].value,
                    username: profile.username,
                    provider: 'facebook',
                    //now in the future searching on User.findOne({'facebook.id': profile.id } will match because of this next line
                    facebook: profile._json
                });
                user.save(function(err) {
                    if (err) console.log(err);
                    return done(err, user);
                });
            } else {
                //found user. Return
                return done(err, user);
            }
        });
    }
));

我个人也使用“会员”表来跟踪每个用户的多个帐户(因此他们可以通过多个帐户进行身份验证),因为我通过mongoose进行设置。这实际上是我存储该访问令牌的地方。我更喜欢这个在用户表中有一个facebook列....但这取决于你。

var mongoose = require('mongoose'),
    Schema = mongoose.Schema,
    ObjectId = Schema.ObjectId;

var membershipSchema = new Schema({
    provider:  String,
    providerUserId:  String,
    accessToken: String,
    userId: {type: ObjectId, ref: 'User'},
    dateAdded: {type: Date, default: Date.now}
});

module.exports = mongoose.model('Membership', membershipSchema);

因此,我的User.findOrCreate版本就像这样开始:

function(accessToken, refreshToken, profile, done) {
    Membership.findOne({
        providerUserId: profile.id
    }, function(err,membershipData) {
            //blah blah blah

其中成员资格是上述模型,并被定义为变量:

var Membership =  require('./models/membership.js')

答案 1 :(得分:3)

如果您想使用findOrCreate,请尝试使用npm包mongoose-findorcreatesupergoose

e.g。 mongoose-findorcreate

var mongoose = require('mongoose');
mongoose.connect('mongodb://localhost');

var findOrCreate = require('mongoose-findorcreate')
var Schema = mongoose.Schema;
var UserSchema = new Schema({ facebookId: Number});
UserSchema.plugin(findOrCreate);
var User = mongoose.model('User', UserSchema);

passport.use(new FacebookStrategy({
        clientID: 'clientID',
        clientSecret: 'clientSecret',
        callbackURL: "/auth/facebook/callback"
    },
    function(accessToken, refreshToken, profile, cb) {
        User.findOrCreate({ facebookId: profile.id }, function (err, user) {
          console.log('A new uxer from "%s" was inserted', user.facebookId);
          return cb(err, user);
        });
    }
));