การใช้ PassportJS จะส่งผ่านฟิลด์แบบฟอร์มเพิ่มเติมไปยังกลยุทธ์การพิสูจน์ตัวตนภายในได้อย่างไร


99

ฉันใช้ passportJS และฉันต้องการที่จะจัดหามากกว่าเพียงreq.body.usernameและreq.body.passwordกับกลยุทธ์การตรวจสอบของฉัน (หนังสือเดินทางท้องถิ่น)

ฉันมีเขตข้อมูลแบบฟอร์มที่ 3: username, passwordและfoo

ฉันจะเข้าถึงได้อย่างไรreq.body.fooจากกลยุทธ์ในพื้นที่ของฉันซึ่งมีลักษณะดังนี้:

passport.use(new LocalStrategy(
  {usernameField: 'email'},
    function(email, password, done) {
      User.findOne({ email: email }, function(err, user) {
        if (err) { return done(err); }
        if (!user) {
          return done(null, false, { message: 'Unknown user' });
        }
        if (password != 1212) {
          return done(null, false, { message: 'Invalid password' });
        }
        console.log('I just wanna see foo! ' + req.body.foo); // this fails!
        return done(null, user, aToken);

      });
    }
));

ฉันเรียกสิ่งนี้ภายในเส้นทางของฉัน (ไม่ใช่มิดเดิลแวร์ของเส้นทาง) ดังนี้:

  app.post('/api/auth', function(req, res, next) {
    passport.authenticate('local', {session:false}, function(err, user, token_record) {
      if (err) { return next(err) }
      res.json({access_token:token_record.access_token});
   })(req, res, next);

  });

คำตอบ:


184

มีpassReqToCallbackตัวเลือกที่คุณสามารถเปิดใช้งานได้ดังนี้:

passport.use(new LocalStrategy(
  {usernameField: 'email', passReqToCallback: true},
  function(req, email, password, done) {
    // now you can check req.body.foo
  }
));

เมื่อใด set reqจะกลายเป็นอาร์กิวเมนต์แรกในการยืนยันการโทรกลับและคุณสามารถตรวจสอบได้ตามที่คุณต้องการ


8
ใช้งานได้ดีขอบคุณ อยู่passReqToCallbackในคู่มือหรือไม่? ไม่เห็นบอกเลย
k00k

2
ยัง. ฉันตามไม่ทันในการเพิ่มคุณสมบัติ / ตัวเลือกใหม่ให้กับคำแนะนำ
Jared Hanson

ขอบคุณช่วยฉันแก้ปัญหาได้มาก
binarygiant

9
สิ่งนี้มีประโยชน์มากสำหรับการใช้การตรวจสอบสิทธิ์แบบผู้เช่าหลายราย / ผู้เช่าหลายราย แต่ไม่ปรากฏในการค้นหา Google ของฉัน หวังว่าความคิดเห็นของฉันจะช่วยให้ผู้คนพบคำตอบนี้ในอนาคต
Kris Dahl

สิ่งนี้มีประโยชน์มาก แต่เป็นไปได้ไหมที่จะตั้งค่าusernameFieldตามเงื่อนไข? ฉันมีสองทางเลือก: หนึ่งคืออีเมลและอีกทางหนึ่งคือหมายเลขโทรศัพท์ และตารางเข้าสู่ระบบประกอบด้วยสองฟิลด์นี้พร้อมกับรหัสผ่าน
Mathew John

1

ในกรณีทั่วไปเราจำเป็นต้องมี 2 ตัวเลือกสำหรับการเข้าสู่ระบบ

  • ด้วยอีเมล
  • ด้วยมือถือ

มันง่ายมากที่เราสามารถใช้ชื่อผู้ใช้ทั่วไปและแบบสอบถาม $ หรือสองตัวเลือกฉันโพสต์ตัวอย่างต่อไปนี้หากบางคนมีคำถามเดียวกัน

เราสามารถใช้ 'passReqToCallback' เป็นตัวเลือกที่ดีที่สุดได้เช่นกันขอบคุณ @Jared Hanson

passport.use(new LocalStrategy({
    usernameField: 'username', passReqToCallback: true
}, async (req, username, password, done) => {
    try {
        //find user with email or mobile
        const user = await Users.findOne({ $or: [{ email: username }, { mobile: username }] });

        //if not handle it
        if (!user) {
            return done(null, {
                status: false,
                message: "That e-mail address or mobile doesn't have an associated user account. Are you sure you've registered?"
            });
        }

        //match password
        const isMatch = await user.isValidPassword(password);
        debugger
        if (!isMatch) {
            return done(null, {
                status: false,
                message: "Invalid username and password."
            })
        }

        //otherwise return user
        done(null, {
            status: true,
            data: user
        });
    } catch (error) {
        done(error, {
            status: false,
            message: error
        });
    }
}));
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.