9

所以我试图让 Sessions 在我的 socket.on('connection', ...) 中工作的其他模块。

无论如何,我已经尝试使用两个模块“ connect-redis ”和“ session.socket.io ”,它们都有类似的问题。

在我的代码中,我有 2 个 redis 存储(socketio.RedisStore 和 require('connect-redis')(express)),现在这个程序运行良好,但是因为 express 和 socket.io 需要共享会话数据,我想知道是否此设置将正确使用会话吗?会话存储是否需要与 express/socketio 相同的对象?对我来说有点灰色地带,因为 2 个 RedisStore 将在后台使用相同的数据库?

我已经尝试在这两个地方使用 socket.io redisStore 或 connect-redis redisStore,但是 socket.io 不喜欢 connect-redis redisStore 并且 express 不喜欢 socketio.redisStore。

如果我使用 connect-redis RedisStore,那么 socket.io/lib/manager.js 会抱怨:this.store.subscribe(... TypeError Object # has no method 'subscribe'

如果我使用 socketio.RedisStore 然后 express/node_modules/connect/lib/middleware/session.js 抱怨: TypeError: Object # has no method 'get'

*注意我宁愿让 session.socket.io 插件工作,但是当我用那个插件做同样的设置时,表达(也)抱怨:TypeError: Object # has no method 'get'

那么我可以使用 2 个不同的 RedisStores 进行会话,还是我需要以某种方式让其中一个或另一个同时为两者工作,如果是的话,关于如何修复的任何想法?

我当前的代码如下所示:

var
    CONST = {
        port: 80,
        sessionKey: 'your secret sauce'
    };

var 
    redis = require('redis');

var 
    express = require('express'),
    socketio = require('socket.io'),
    RedisStore = require('connect-redis')(express);

var 
    redisStore = new RedisStore(),
    socketStore = new socketio.RedisStore();

var 
    app = express(),
    server = require('http').createServer(app),
    io = socketio.listen(server);

app.configure(function(){
    app.use(express.cookieParser( CONST.sessionKey ));
    app.use(express.session({ secret: CONST.sessionKey, store: redisStore }));
    app.use(express.static(__dirname + '/test'));
    app.get('/', function (req, res) {res.sendfile(__dirname + '/test/' + 'index.htm');});
});

io.configure(function(){
    io.set('log level', 1);
    io.enable('browser client minification');
    io.enable('browser client etag');
    io.enable('browser client gzip');
    io.set('store', socketStore);
});

io.sockets.on('connection', function(socket){
    socket.emit('message', 'Test 1 from server')
});

server.listen( CONST.port );

console.log('running...');
4

2 回答 2

4

在内部io.configure,您必须将套接字与 http 会话链接起来。

这是一段提取cookie的代码(这是使用socket.io with xhr-polling,我不知道这是否适用于websocket,尽管我怀疑它会起作用)。

var cookie = require('cookie');
var connect = require('connect');

var sessionStore = new RedisStore({
  client: redis // the redis client
});

socketio.set('authorization', function(data, cb) {
  if (data.headers.cookie) {
    var sessionCookie = cookie.parse(data.headers.cookie);
    var sessionID = connect.utils.parseSignedCookie(sessionCookie['connect.sid'], secret);
    sessionStore.get(sessionID, function(err, session) {
      if (err || !session) {
        cb('Error', false);
      } else {
        data.session = session;
        data.sessionID = sessionID;
        cb(null, true);
      }
    });
  } else {
    cb('No cookie', false);
  }
});

然后您可以使用以下命令访问会话:

socket.on("selector", function(data, reply) {
  var session = this.handshake.session;
  ...
}

这还有一个额外的好处,它检查是否存在有效会话,因此只有您登录的用户才能使用套接字。不过,您可以使用不同的逻辑。

于 2013-03-02T01:31:15.583 回答
4

查看您的最后一条注释(将无法使用 redis 在多个进程中共享其状态)我遇到了同样的问题并找到了解决方案:

var express = require("express.io");
var swig = require('swig');
var redis = require('redis');
var RedisStore = require('connect-redis')(express);

workers = function() {
    var app = express().http().io();

    app.use(express.cookieParser());
    app.use(express.session({
        secret: 'very cool secretcode',
        store: new RedisStore({ client: redis.createClient() })
    }));

    app.io.set('store', new express.io.RedisStore({
        redisPub: redis.createClient(),
        redisSub: redis.createClient(),
        redisClient: redis.createClient()
    }));


    app.get('/', function(req, res) {
        res.render('index');
    });

    app.listen(3000);


    app.io.route('ready', function(req){
        //setup session stuff, use session stuff, etc.  Or make new routes
    });
};

cluster = require('cluster');
numCPUs = require('os').cpus().length;

if (cluster.isMaster)
{
    for (var i = 0; i < numCPUs; i++)
    {
        cluster.fork();
    }
}
else
{
    workers();
}
于 2013-11-16T03:46:20.280 回答