我正在尝试使用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
功能。
答案 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);
})