Passport.js-오류 : 사용자를 세션으로 직렬화하지 못했습니다.


180

Passport.js 모듈 및 Express.js에 문제가 있습니다.

이것은 내 코드이며 첫 번째 시도에 하드 코딩 된 로그인을 사용하고 싶습니다.

나는 항상 메시지를 받는다 :

많은 것을 검색하고 stackoverflow에서 일부 게시물을 찾았지만 실패하지 않았습니다.

Error: failed to serialize user into session
    at pass (c:\Development\private\aortmann\bootstrap_blog\node_modules\passport\lib\passport\index.js:275:19)

내 코드는 다음과 같습니다.

'use strict';

var express = require('express');
var path = require('path');
var fs = require('fs');
var passport = require('passport');
var LocalStrategy = require('passport-local').Strategy;
var nodemailer = require('nodemailer');

var app = express();

module.exports = function setupBlog(mailTransport, database){
var config = JSON.parse(fs.readFileSync('./blog.config'));

app.set('view options', {layout: false});

app.use(express.static(path.join(__dirname, '../', 'resources', 'html')));


app.use(express.bodyParser());
app.use(express.cookieParser());
app.use(express.session({ secret: 'secret' }));
app.use(passport.initialize());
app.use(passport.session());


app.get('/blog/:blogTitle', function(req, res) {
  var blogTitle = req.params.blogTitle;
  if(blogTitle === 'newest'){
    database.getLatestBlogPost(function(post) {
      res.send(post);
    });
  } else {
    database.getBlogPostByTitle(blogTitle, function(blogPost) {
      res.send(blogPost);
    });
  }
});

passport.use(new LocalStrategy(function(username, password, done) {
  // database.login(username, password, done);
  if (username === 'admin' && password === 'admin') {
    console.log('in');
    done(null, { username: username });
  } else {
    done(null, false);
  }
}));

app.post('/login', passport.authenticate('local', {
  successRedirect: '/accessed',
  failureRedirect: '/access'
}));





app.listen(8080);
console.log('Blog is running on port 8080');

}();

감사.

답변:


363

passport.serializeUser및을 구현하지 않은 것 같습니다 passport.deserializeUser. 이것을 추가하십시오 :

passport.serializeUser(function(user, done) {
  done(null, user);
});

passport.deserializeUser(function(user, done) {
  done(null, user);
});

2
나를 위해 일했다. 다른 곳에 쓰여진 'id'부분을 사용할 필요가없는 이유는 무엇입니까?
schlenger

9
@schlenger 그것은 직렬화를 구현하는 방법에 달려 있습니다. 때때로 사용자 ID로 직렬화합니다. 즉, serializeUser함수는 세션에 사용자 ID 만 저장하고 deserializeUser해당 ID를 사용하여 데이터베이스에서 사용자 데이터를 검색합니다 (예 :). 이것은 세션 스토리지가 실제 사용자 데이터 자체를 포함하지 않도록하기위한 것입니다.
robertklep

@robertklep 댓글 +1 나는 항상 사용자 정보를 직렬화하지 않지만 ID 만 (개인적으로 부풀어 오르거나 부당한 이유로) 만 피합니다.
electblake

2
@robertklep 세션에 저장하고 싶지 않습니다. JWT를 사용하고 있습니다. serializeUser / deserializeUser가 필요합니까? 나는 세션을 원하지 않습니다.
Internial

@Internial은 필요한지 확실하지 않지만 테스트하기가 쉽습니다.
robertklep

44

세션을 사용하지 않기로 결정한 경우 세션을 false로 설정할 수 있습니다

app.post('/login', passport.authenticate('local', {
  successRedirect: '/accessed',
  failureRedirect: '/access',
  session: false
}));

그러나 다른 모든 엔드 포인트에서 세션을 끄지는 않습니다 (또한 세션을 원하지 않습니다)
jayarjo

17

Passportjs 설정의 일부, 특히 다음 두 가지 방법을 놓친 것 같습니다.

passport.serializeUser(function(user, done) {
    done(null, user._id);
    // if you use Model.id as your idAttribute maybe you'd want
    // done(null, user.id);
});

passport.deserializeUser(function(id, done) {
  User.findById(id, function(err, user) {
    done(err, user);
  });
});

나는 ._id대 에 대해 조금 추가 .id했지만이 발췌 문장은 문서 의 구성 섹션 에서 가져온 것 입니다. 또 다른 행운과 행운을 빕니다 :)


2

세션을 사용하고 여전히 값을 "직렬화"하는 작동하지만 여전히 게으른 방법입니다.

var user_cache = {};

passport.serializeUser(function(user, next) {
  let id = user._id;
  user_cache[id] = user;
  next(null, id);
});

passport.deserializeUser(function(id, next) {
  next(null, user_cache[id]);
});

또는 이상한 오류가 발생하면 스스로에게 물어보십시오. "사용자 개체에 '_id'를 설정합니까?" -대부분의 경우 당신은하지 않습니다. 따라서 올바른 속성을 키로 사용하십시오.


0

serializeUser 및 deserializeUser와 함께 Promise 사용 :

passport.serializeUser((user, done) => {
  done(null, user.id);
});

passport.deserializeUser((id, done) => {
  // console.log(`id: ${id}`);
  User.findById(id)
    .then((user) => {
      done(null, user);
    })
    .catch((error) => {
      console.log(`Error: ${error}`);
    });
});

이 문제를 해결하는 방법에 대한 전체 코드 예제는 github repo 를 참조하십시오 .


-1

passport.use ( 'local-login'...) / 또는 / ( 'local-singup'...)

오류가 발생하면 "false"를 반환해야합니다. 오류 {return done (null, req.flash ( 'megsign', 'Username already exists! megsign ','사용자 이름이 이미 존재합니다 #! # '));}

당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.