web-dev-qa-db-ja.com

User.findOrCreate関数は何をしていて、パスポートでいつ呼び出されますか?

この機能に関するドキュメントが見つからないため、正しく機能させることができません。その関数はいつ呼び出されますか、何をしていて、最初のパラメーターとして何を取っていますか?パスポートからアクセストークンを取得しようとしていますが、どうしてもアクセスできません。

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? 
       });

  }
));

パスポートからアクセストークンを取得するにはどうすればよいですか?

42
Tommz

User.findOrCreateは、ユーザーをFacebook IDで検索するために、またはユーザーが存在しない場合に作成するために必要なすべての関数を表す作成関数です。最初の問題は、コールバックURLがルートに移動するだけであるため、おそらくその機能にアクセスできないことです。

コールバックURLはhttp://localhost:3000/auth/facebook/callback

そして、そのURLを処理します。

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

この時点で認証は完了です。 accessTokenが返されます-「これは、アプリが特定の人のFacebookデータを読み取り、変更、または書き込むためにAPIを呼び出すたびに必要です」。ユーザーのアクセストークンを保存するテーブルにこれを保存する必要があります。 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);
            }
        });
    }
));

個人的には、「メンバーシップ」テーブルを使用して、ユーザーごとに複数のアカウントを追跡します(したがって、複数のアカウントで認証できます)。これは、実際にそのアクセストークンを保存する場所です。私はこれをユーザー表に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')
67
MikeSmithDev

findOrCreateを使用する場合は、npmパッケージ mongoose-findorcreate 、または supergoose を試してください。

例えば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);
        });
    }
));
3
Vinnie James