javascript - express.session.MemoryStore 不返回 session ?

标签 javascript node.js session express socket.io

设置 session 并在 PHP 中使用它们真的很容易。但是我的网站需要处理 WebSockets。我在 node.js 中设置 session 时遇到问题。我可以在不使用 session 的情况下轻松推送数据,它可以正常工作,但是当打开多个选项卡时,会创建新的 socket.id 并且之前打开的选项卡将无法正常运行。所以我一直在处理 session 并且在访问 session 存储时遇到了问题,它的日志 session 没有被抓取。我也尝试过 session.load 但没有运气

如何获取 session 对象并以打开其他选项卡不会影响功能并在所有选项卡上将数​​据从服务器推送到客户端的方式使用它?

var express=require('express');
var http = require('http');
var io = require('socket.io');
var cookie = require("cookie");
var connect = require("connect"),
MemoryStore = express.session.MemoryStore,
sessionStore = new MemoryStore();

var app = express();

app.configure(function () {
    app.use(express.cookieParser());
    app.use(express.session({store: sessionStore
        , secret: 'secret'
        , key: 'express.sid'}));
    app.use(function (req, res) {
        res.end('<h2>Hello, your session id is ' + req.sessionID + '</h2>');
    });
});

server = http.createServer(app);
server.listen(3000);

sio = io.listen(server);

var Session = require('connect').middleware.session.Session;

sio.set('authorization', function (data, accept) {
    // check if there's a cookie header
    if (data.headers.cookie) {
        // if there is, parse the cookie
        data.cookie = connect.utils.parseSignedCookies(cookie.parse(data.headers.cookie),'secret');
        // note that you will need to use the same key to grad the
        // session id, as you specified in the Express setup.
        data.sessionID = data.cookie['express.sid'];
        sessionStore.get(data.sessionID, function (err, session) {
            if (err || !session) {
                // if we cannot grab a session, turn down the connection
                console.log("session not grabbed");
                accept('Error', false);
            } else {
                // save the session data and accept the connection
                console.log("session grabbed");
                data.session = session;
                accept(null, true);
            }
        });
    } else {
       // if there isn't, turn down the connection with a message
       // and leave the function.
       return accept('No cookie transmitted.', false);
    }
    // accept the incoming connection
    accept(null, true);
});

sio.sockets.on('connection', function (socket) {
    console.log('A socket with sessionID ' + socket.handshake.sessionID 
        + ' connected!');
});

最佳答案

看看这篇文章:Session-based Authorization with Socket.IO

您的代码运行良好,但需要进行 2 处改进才能完成您想要的工作(从服务器向客户端发送 session 数据):

  1. 它仅在授权期间提取sessionID
  2. 它在连接期间通过此sessionID从存储中提取 session 数据,您可以在此间隔内将数据从服务器发送到客户端。

这是改进后的代码:

var express = require('express');
var connect = require('connect');
var cookie = require('cookie');
var sessionStore = new express.session.MemoryStore();    

var app = express();    

app.use(express.logger('dev'));
app.use(express.cookieParser());
app.use(express.session({store: sessionStore, secret: "secret", key: 'express.sid'}));
// web page
app.use(express.static('public'));    

app.get('/', function(req, res) {
  var body = '';
  if (req.session.views) {
    ++req.session.views;
  } else {
    req.session.views = 1;
    body += '<p>First time visiting? view this page in several browsers :)</p>';
  }
  res.send(body + '<p>viewed <strong>' + req.session.views + '</strong> times.</p>');
});    

var sio = require('socket.io').listen(app.listen(3000));   

sio.set('authorization', function (data, accept) {
    // check if there's a cookie header
    if (data.headers.cookie) {
        // if there is, parse the cookie
        var rawCookies = cookie.parse(data.headers.cookie);
        data.sessionID = connect.utils.parseSignedCookie(rawCookies['express.sid'],'secret');    
        // it checks if the session id is unsigned successfully
        if (data.sessionID == rawCookies['express.sid']) {
            accept('cookie is invalid', false);
        }
    } else {
       // if there isn't, turn down the connection with a message
       // and leave the function.
       return accept('No cookie transmitted.', false);
    }
    // accept the incoming connection
    accept(null, true);
});    

sio.sockets.on('connection', function (socket) {
    //console.log(socket);
    console.log('A socket with sessionID ' + socket.handshake.sessionID + ' connected!');
    // it sets data every 5 seconds
    var handle = setInterval(function() {
        sessionStore.get(socket.handshake.sessionID, function (err, data) {
            if (err || !data) {
                console.log('no session data yet');
            } else {
                socket.emit('views', data);
            }
        });
    }, 5000);    

    socket.on('disconnect', function() {
        clearInterval(handle);
    });
});

然后您可以在 http://localhost:3000/client.htmlpublic/client.html 下有一个客户端页面,以查看从 http://localhost:3000:

<html>
<head>
    <script src="/socket.io/socket.io.js" type="text/javascript"></script>
    <script type="text/javascript">
        tick = io.connect('http://localhost:3000/');
        tick.on('data', function (data) {
            console.log(data);
        });

        tick.on('views', function (data) {
            document.getElementById('views').innerText = data.views;
        });

        tick.on('error', function (reason){
          console.error('Unable to connect Socket.IO', reason);
        });

        tick.on('connect', function (){
          console.info('successfully established a working and authorized connection');
        });
    </script>
</head>
<body>
    Open the browser console to see tick-tocks!
    <p>This session is viewed <b><span id="views"></span></b> times.</p>
</body>

关于javascript - express.session.MemoryStore 不返回 session ?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/18302242/

相关文章:

javascript - JS中基本的 "Alert Box"和 "Multiplying Input with 5"

javascript - 是否可以检查函数是 javascript 函数还是开发人员定义的函数?

node.js - 如何在 laravel 应用程序启动时自动运行 Node 和 Redis 服务器

javascript - MongoDB 聚合框架 - 如何查询平均值

php - $_SESSION ['id'] 为空

javascript - 如何在js中多次使用resize

javascript - 从 React 中的另一个组件访问 const 的方法

javascript - 编写一个可以与带有 axios 输出的 then() 链接的函数

session - 如何更改tomcat集群中的广播IP

javascript - 如何使用 Javascript 和 Jquery Vector Map 设置 session