- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我已经奋斗了几天,并且取得了一些不错的进展,但我无法让我的 session 内容正常工作。
我已成功使用 Passport 通过 Facebook 进行身份验证。当我单击“使用 FB 登录”按钮时, session 已完美加载,并且我的 req.user 对象已准备就绪,但在我看来,我只需要做一次。
我知道 Passport 保存有 cookie。我查了一下,发现它有效。
我试图检测用户是否已经在加载索引页面时登录,但是当我加载页面时,req.user对象始终为空,并且我的passport.deserializeUser方法永远不会被调用来加载它(如果我点击登录 FB 按钮,它就会被调用)。
所以我的问题是,如何告诉页面加载时的 Passport 检查 cookie 并加载用户 session (如果有)?
<小时/>更新 - 好的,对于那些以后发现这个问题的人,我只是想回顾一下我在这里学到的东西(感谢那些发表评论的人)。希望它能对其他人有所帮助。
我已经习惯了 .NET cookie,无论服务器如何,它都会存在。 Node.js 和 Passport 不能在同一前提下工作。默认情况下,node.js 使用内存存储来保留其 session 。当您在终端/命令行中关闭 Node 时(每次更改服务器代码时都必须执行此操作),内存存储 cookie 信息将被重置,因此与其绑定(bind)的任何 cookie 都毫无意义。
为了解决这个问题,我安装了 Redis ( http://cook.coredump.me/post/18886668039/brew-install-redis ) 并将其连接为我的商店 ( http://www.hacksparrow.com/use-redisstore-instead-of-memorystore-express-js-in-production.html )。
这将在 Azure 上运行,这是我计划的生产服务器,所以一切都很好。
<小时/>好的,这是一些代码。我不确定要放置哪些部分...
这是 server.js
/**
* Module dependencies.
*/
var express = require('express')
, app = express()
, partials = require('express-partials')
, http = require('http')
, server = http.createServer(app)
, io = require('socket.io').listen(server)
, routes = require('./routes')
// facebook
, passport = require('passport')
, facebookStrategy = require('passport-facebook').Strategy
// custom stuff
, urlCommand = require('./middleware/UrlCommand')
, azureCommand = require('./middleware/AzureCommand')
, userCommand = require('./middleware/UserCommand');
// Ports
var port = process.env.port;
if (isNaN(port))
port = 3000;
server.listen(port);
//allows the use of Layouts
app.use(partials());
passport.serializeUser(function(user, done) {
done(null, user.RowKey);
});
passport.deserializeUser(function (id, done) {
console.log("deserialize");
userCommand.findByID(id, function (err, user) {
done(err, user);
});
});
// Configuration
app.configure(function () {
app.set('views', __dirname + '/views');
app.set('view engine', 'jade');
app.use(express.cookieParser());
app.use(express.bodyParser());
app.use(express.session({ secret: 'SECRET!' }));
app.use(express.methodOverride());
app.use(passport.initialize());
app.use(passport.session());
app.use(app.router);
app.use(express.static(__dirname + '/public'));
});
app.configure('development', function(){
app.use(express.errorHandler({ dumpExceptions: true, showStack: true }));
});
app.configure('production', function(){
app.use(express.errorHandler());
});
// Facebook
passport.use(new facebookStrategy({
clientID: CLIENT,
clientSecret: "SECRET",
callbackURL: "http://localhost:3000/auth/facebook/callback" //DEV
},
function (accessToken, refreshToken, profile, done) {
userCommand.findOrCreate(profile.id, profile.name.givenName, profile.name.familyName, profile.emails[0].value, accessToken, function (error, user) {
return done(null, user);
});
}
));
// Routes
app.get('/', routes.index);
app.get('/auth/facebook', passport.authenticate('facebook', { scope: 'email' }));
app.get('/auth/facebook/callback', passport.authenticate('facebook', { successRedirect: '/',
failureRedirect: '/login' }));
// Sockets
io.sockets.on('connection', function (socket) {
// when the client emits 'sendURL', this listens and executes
socket.on('sendURL', function (data) {
console.log('sendURL called: ' + data);
urlCommand.AddURL(data);
// we tell the client to execute 'processURL'
io.sockets.emit('urlComplete', data);
});
});
console.log("Express server listening on port %d in %s mode", port, app.settings.env);
index.js
exports.index = function (req, res) {
console.log(req.user); // ALWAYS NULL
res.render('index', { title: 'Express' })
};
用户命令
var azureCommand = require('../middleware/AzureCommand');
var tableService = azureCommand.CreateTableService();
function findByID(id, callback) {
console.log('FindByID');
tableService.queryEntity('user', 'user', id, function (error, entity) {
console.log('Found him: ' + entity.Email);
callback(error, entity);
});
}
function findOrCreate(id, first, last, email, accessToken, callback) {
var user = {
PartitionKey: 'user'
, RowKey: id
, First: first
, Last: last
, Email: email
, AccessToken: accessToken
}
tableService.insertEntity('user', user, function (error) {
callback(null, user);
});
}
exports.findByID = findByID;
exports.findOrCreate = findOrCreate;
这是我输出 session 时输出日志显示的内容...
node server.js
info - socket.io started
Express server listening on port 3000 in development mode
{ cookie:
{ path: '/',
_expires: null,
originalMaxAge: null,
httpOnly: true },
passport: {}
}
debug - served static content /socket.io.js
最佳答案
问题出在您的 serializeUser
和 deserializeUser
函数中。
正如您所注意到的,当您单击 FBlogin 按钮 deserializeUser
时,这是第一次也是唯一一次调用 - 实际上它是使用之前返回的 id
进行调用的通过serializeUser
函数。如果此时无法通过提供的 id
找到用户,则 passport.user
将不会保存到 session 中。
因此,在以下请求中,根本不会调用 passport.deserializeUser
,因为 express.session
不会填充 req.session
> 带有 Passport 的用户 ID。
总结:您需要检查您的 serializeUser
返回的 id
是否可以被您的 deserializeUser 理解和反序列化
.
仅供引用:经过身份验证的用户的正确 req.user
对象应如下所示:
{
cookie: { path: '/', _expires: null, originalMaxAge: null, httpOnly: true },
passport: { user: 51772291b3367f0000000001 }
}
关于node.js - 为什么我的 req.user 在加载时始终为空?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/16185164/
当我使用路径文件上的快捷方式在文件之间移动时,似乎我不仅仅是在文件之间移动。 我使用>转到一个文件,在该文件中我更改光标的位置并执行某些操作,然后按 gf noremap 关于vim 通过快捷方式直
我正在尝试使用 Pong P. Chu 的书来学习 Verilog。我有一个关于如何评估和实现始终 block 的问题。作者代码中的风格让我感到困惑。 在此示例中,他编写了一个具有两个输出寄存器“y1
我正在尝试制作一个聊天应用程序,因此我需要它始终接收服务器信息。因此,当请求完成时,在: http.onreadystatechange=function(){ 我再次调用该函数,因此: reques
当您在 always block 敏感度列表中使用通配符 @* 时,我对什么被视为输入有点困惑。例如,在下面的示例中,哪些信号被解释为导致 always block 被重新评估的输入? 据我了解,cl
我有一个充当调试器的程序。我为线程设置了一个 hw bp,将 dr0 设置为我希望 bp 所在的地址,将 dr7 设置为 1,因为我希望 bp 在每次执行该地址时生成一个事件。 它有效,但现在的问题是
如何每次都以管理员身份在 Windows 上运行 git bash。 操作系统 - Windows 10 家庭版 64 位 最佳答案 我在 Google 上找到了这个结果: 将 Git Bash 设置
使用 accept() 时或 getpeername() , sockaddr_storage总是有 ss_family=AF_INET6 : struct sockaddr_storage addr
我在 Cordova 方面还有另一个问题。我想在 Cordova 7.1.0 中使用插件“cordova.custom.plugins.exitapp”和“cordova-plugins-printe
我试图让模块通过 ISE 12.4 中的语法检查,但它给了我一个我不明白的错误。首先是代码片段: parameter ROWBITS = 4; reg [ROWBITS-1:0] temp; genv
我正在使用Cordova开发适用于iOS的应用程序,其中包括地理位置功能(我使用官方插件https://github.com/apache/cordova-plugin-geolocation)。我在
我想知道是否有可能只在敏感列表中的多个信号一起变化时才执行 always block 。 例如,假设我有一个信号“in”和另一个“posedge clk”。我希望在两个信号都发生变化时执行 alway
我需要实现一种算法来访问数据库来检查最后一个元素,以便计算新的元素。当然,第一次这是不可能的,因为数据库是空的,我得到 IndexOutOfBoundsException) index 0 reque
我正在利用我在网上找到的画廊系统,根据鼠标图像的接近程度,它会按比例增长。 链接:Gallery 好吧,我调整了代码以响应(如您所见正在 build 中)并且没有明显的问题。我的问题在更改分辨率时开始
我正在创建一个 kiosk 应用程序,我想确保它无论如何始终位于其他 Windows 应用程序和 Windows 任务栏之上。 我已经阻止了 Windows 键盘命令(alt-tab 等),但仍有可能
我即将开始一个新的 React 项目,并尝试利用我以前的知识来创建一些关于我如何构建应用程序的规则。 有些事情我认为是真的: Redux 保存整个应用程序的“主要”数据 如果需要跨应用程序共享,Red
当你打开 VS Code 时,终端默认是在底部打开的。您可以单击该图标将其向右移动。我想知道是否有办法将右侧打开设置为默认值。 谢谢。 最佳答案 是的 - 在 v1.20 中引入了设置 workb
我有一个Events表,其中包含各种类型的事件。我只关心其中一种类型。因此,我编写的每个查询都以开头 Events.objects.filter(event_type="the_type").\
我在单例中创建了一个Timer,并且我一直在努力解决为什么Timer没有触发。我查看了这里的帖子,但没有找到我认为可以直接回答我的问题的帖子。 class ConnectionStateMonitor
我在 TableViewController 中显示了一组项目。它们在 TVC 中正确显示。下面的代码会继续,但它只会继续到我的 MKMapItem 数组的 indexPath 0,而不是被单击的单元
我的 VC 是这样的: var coins = 50 // coins override func viewDidLoad() { super.viewDidLoad() if(SKP
我是一名优秀的程序员,十分优秀!