Я использую node с express + mongoose и пытаюсь использовать password.js с спокойным api.
Я продолжаю получать это исключение после успешной аутентификации (я вижу URL-адрес обратного вызова в браузере):
/Users/naorye/dev/naorye/myproj/node_modules/mongoose/lib/utils.js:419
throw err;
^
Error: passport.initialize() middleware not in use
at IncomingMessage.req.login.req.logIn (/Users/naorye/dev/naorye/myproj/node_modules/passport/lib/passport/http/request.js:30:30)
at Context.module.exports.delegate.success (/Users/naorye/dev/naorye/myproj/node_modules/passport/lib/passport/middleware/authenticate.js:194:13)
at Context.actions.success (/Users/naorye/dev/naorye/myproj/node_modules/passport/lib/passport/context/http/actions.js:21:25)
at verified (/Users/naorye/dev/naorye/myproj/node_modules/passport-facebook/node_modules/passport-oauth/lib/passport-oauth/strategies/oauth2.js:133:18)
at Promise.module.exports.passport.use.GitHubStrategy.clientID (/Users/naorye/dev/naorye/myproj/config/passport.js:91:24)
at Promise.onResolve (/Users/naorye/dev/naorye/myproj/node_modules/mongoose/node_modules/mpromise/lib/promise.js:162:8)
at Promise.EventEmitter.emit (events.js:96:17)
at Promise.emit (/Users/naorye/dev/naorye/myproj/node_modules/mongoose/node_modules/mpromise/lib/promise.js:79:38)
at Promise.fulfill (/Users/naorye/dev/naorye/myproj/node_modules/mongoose/node_modules/mpromise/lib/promise.js:92:20)
at /Users/naorye/dev/naorye/myproj/node_modules/mongoose/lib/query.js:1822:13
Прочитал, что надо ставить app.use(passport.initialize());
и app.use(passport.session());
раньше app.use(app.router);
и вот что сделал. Вот мой express.js, который регистрирует промежуточное ПО:
var express = require('express'),
mongoStore = require('connect-mongo')(express),
flash = require('connect-flash'),
helpers = require('view-helpers');
module.exports = function (app, config, passport) {
app.set('showStackError', true);
// should be placed before express.static
app.use(express.compress({
filter: function (req, res) {
return /json|text|javascript|css/.test(res.getHeader('Content-Type'));
},
level: 9
}));
app.use(express.favicon());
app.use(express.static(config.root + '/public'));
app.use(express.logger('dev'));
// set views path, template engine and default layout
app.set('views', config.root + '/app/views');
app.set('view engine', 'jade');
app.configure(function () {
// use passport session
app.use(passport.initialize());
app.use(passport.session());
// dynamic helpers
app.use(helpers(config.app.name));
// cookieParser should be above session
app.use(express.cookieParser());
// bodyParser should be above methodOverride
app.use(express.bodyParser());
app.use(express.methodOverride());
// express/mongo session storage
app.use(express.session({
secret: 'linkit',
store: new mongoStore({
url: config.db,
collection : 'sessions'
})
}));
// connect flash for flash messages
app.use(flash());
// routes should be at the last
app.use(app.router);
// assume "not found" in the error msgs
// is a 404. this is somewhat silly, but
// valid, you can do whatever you like, set
// properties, use instanceof etc.
app.use(function(err, req, res, next){
// treat as 404
if (~err.message.indexOf('not found')) {
return next();
}
// log it
console.error(err.stack);
// error page
res.status(500).render('500', { error: err.stack });
});
// assume 404 since no middleware responded
app.use(function(req, res, next){
res.status(404).render('404', {
url: req.originalUrl,
error: 'Not found'
});
});
});
};
Что случилось?
ОБНОВЛЕНИЕ Согласно @Peter Lyons, я изменил порядок конфигураций на следующий, но все равно получил ту же ошибку:
var express = require('express'),
mongoStore = require('connect-mongo')(express),
flash = require('connect-flash'),
helpers = require('view-helpers');
module.exports = function (app, config, passport) {
app.set('showStackError', true);
// should be placed before express.static
app.use(express.compress({
filter: function (req, res) {
return /json|text|javascript|css/.test(res.getHeader('Content-Type'));
},
level: 9
}));
app.use(express.favicon());
app.use(express.static(config.root + '/public'));
app.use(express.logger('dev'));
// set views path, template engine and default layout
app.set('views', config.root + '/app/views');
app.set('view engine', 'jade');
app.configure(function () {
// dynamic helpers
app.use(helpers(config.app.name));
// cookieParser should be above session
app.use(express.cookieParser());
// bodyParser should be above methodOverride
app.use(express.bodyParser());
app.use(express.methodOverride());
// express/mongo session storage
app.use(express.session({
secret: 'linkit',
store: new mongoStore({
url: config.db,
collection : 'sessions'
})
}));
// connect flash for flash messages
app.use(flash());
// use passport session
app.use(passport.initialize());
app.use(passport.session());
// routes should be at the last
app.use(app.router);
// assume "not found" in the error msgs
// is a 404. this is somewhat silly, but
// valid, you can do whatever you like, set
// properties, use instanceof etc.
app.use(function(err, req, res, next){
// treat as 404
if (~err.message.indexOf('not found')) {
return next();
}
// log it
console.error(err.stack);
// error page
res.status(500).render('500', { error: err.stack });
});
// assume 404 since no middleware responded
app.use(function(req, res, next){
res.status(404).render('404', {
url: req.originalUrl,
error: 'Not found'
});
});
});
};
Ответы:
Следуйте примеру, чтобы избежать ада вышедшего из строя промежуточного программного обеспечения, в которое так легко войти. Прямо из документации. Обратите внимание на то, что у вас это не совсем так.
Документы
Вы
источник
app.get
,app.post
и т.д.? Это приведет к тому, что маршрутизатор будет добавлен в стек раньше, чем вы предполагаете. Покажите нам ВСЕ откровенный код, начиная с момента, когда вы вызываетеexpress()
функцию для получения вашегоapp
объекта. Это мое второе предположение.В моем случае (то же сообщение об ошибке) я вообще забыл добавить инициализацию паспорта:
ОБНОВЛЕНИЕ: работает только до экспресс-версии 3, версия 4 больше не поддерживает app.configure ()
источник
В моем случае ошибка была связана с тем , что я пытался обещать
req.login
без привязкиthis
кreq
, поэтому при вызове функции она не могла найтиpassport
настройки. Решение является обязательнымreq.login.bind(req)
перед передачей,promisify
если вы используете Node v8.источник
function({ login })
передачаreq
первого аргумента. Ваше решение сработало для меня, спасибоthis
работает в Javascript. Если вы не вызываете функцию как объектный метод, тогдаthis
будетundefined
(илиwindow
в браузере)Function.prototype.call
,Function.prototype.apply
какthis
работает в Javascript и принципы, лежащие в основе прототипного наследования, вы продвинетесь до уровня Javascript Guru в процессе :)util.promisify(req.login.bind(req));
Мне также помогло размещение маршрутов ПОСЛЕ конфигурации файлов cookie :
источник
Ответ Питера Лайона помог мне решить эту проблему, но я решил ее немного по-другому.
Взгляните на мой репозиторий GitHub, чтобы увидеть весь код, а не только фрагмент кода здесь.
источник
В моем случае (такое же сообщение об ошибке) я разрабатывал собственную стратегию, и мне не нужно использовать сеанс . Я просто забыл добавить промежуточное ПО для
session: false
маршрутаauthenticate
.источник
Поместите
app.use(passport.initialize())
промежуточное ПО перед промежуточным ПО,app.router
и оно работает как шармисточник