mongoose-findorcreate在使用passport-facebook

时间:2016-12-09 16:55:08

标签: node.js mongoose passport.js passport-facebook

我正在尝试使用passportJS设置用户身份验证。我决定使用这个名为mongoose-findorcreate的npm包而不是编写一个findOrCreate函数来自己使用护照。

所以现在我正在测试我的应用程序,我可以按预期使用我的facebook凭据登录,但问题是我在我的数据库中获得了多个条目以进行相同的facebook登录。

这是我的用户模型:

let mongoose = require('mongoose');
let Schema = mongoose.Schema;

let findOrCreate = require('mongoose-findorcreate');

// User schema definition
let UserSchema = new Schema(
  {
    provider: { type: String, required: true },
    id: { type: String, default: 'unknown' },
    displayName: { type: String, default: 'unknown' },
    name: { type: Object },
    emails: { type: Array },
    photos: { type: Array },
    createdAt: { type: Date, default: Date.now },
    updatedAt: { type: Date, default: Date.now },
  }
);

UserSchema.plugin(findOrCreate);

// Sets the updatedAt parameter equal to the current time
UserSchema.pre('save', (next) => {
  now = new Date();
  if(!this.updatedAt) {
    this.updatedAt = now;
  }
  next();
});

module.exports = mongoose.model('user', UserSchema);

这些是我的路线:

let passport = require('passport');
let FacebookStrategy = require('passport-facebook').Strategy;

let User = require('../models/user');

passport.use(new FacebookStrategy({
    clientID: 12345,
    clientSecret: 'supersecretsecret',
    callbackURL: 'http://localhost:1337/auth/facebook/callback'
  },
  (accessToken, refreshToken, profile, done) => {
    User.findOrCreate( profile, (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);
  });
});

function redirectFB() {
  return passport.authenticate('facebook');
}

function callbackFB() {
  return passport.authenticate('facebook', {
    successRedirect: '/word',
    failureRedirect: '/'
  })
}

module.exports = { redirectFB, callbackFB };

重复用户的示例:

{  
   "_id":ObjectId("584ade8915644f1186fa8a96"),
   "provider":"facebook",
   "updatedAt":   ISODate("2016-12-09T16:40:41.921   Z"),
   "createdAt":   ISODate("2016-12-09T16:40:41.921   Z"),
   "photos":[  ],
   "emails":[  ],
   "displayName":"Miha Šušteršič",
   "id":"10154914634234238",
   "__v":0
}{  
   "_id":ObjectId("584adea3b8adfb11948ed1d6"),
   "provider":"facebook",
   "updatedAt":   ISODate("2016-12-09T16:41:07.626   Z"),
   "createdAt":   ISODate("2016-12-09T16:41:07.625   Z"),
   "photos":[  ],
   "emails":[  ],
   "displayName":"Miha Šušteršič",
   "id":"10154914634234238",
   "__v":0
}

我无法弄清楚这是否是我的代码的问题(将个人资料传递给findOrCreate功能。

2 个答案:

答案 0 :(得分:1)

findOrCreate无效的原因是findOrCreate查询将整个profile对象传递给查询条件。

如果你看这里:https://github.com/drudge/mongoose-findorcreate/blob/master/index.js#L22

根据您从Facebook返回的数据类型,除非配置文件对象与您的架构完全匹配,否则您的查询将始终返回空白。

你在这里可以做的是:

User.findOrCreate({ provider: profile.provider, id: profile.id, .... }, (err, user) => {
  if (err) {
    return done(err);
  }
  done(null, user);
})

代替profile传递{ provider: profile.provider, id: profile.id, .... }

当然,修复{ provider: profile.provider, id: profile.id, .... }以与您的架构定义匹配。

答案 1 :(得分:0)

发现问题 - 我需要以正确的格式将用户数据传递给findOrCreate函数。

User.findOrCreate( profile._json, (err, user) => {
      if (err) {
        return done(err);
      }
      done(null, user);
    })