Laravel IO套接字和redis over SSL https连接

时间:2016-08-08 14:36:21

标签: laravel ssl io redis

目前有带有laravel广播的IO插槽,redis工作正常。直到我在域上设置SSL证书。

我在端口3001上运行redis-server。

然后有一个socket.js设置为侦听3000。

我在页面上的JS我通过io('// {{$ listen}}:3000')收听。

如何通过https实现这一目标的任何指导都会很棒。我会用443作为端口吗?

感谢。

我的socket.js

var app = require('express')();
var http = require('http').Server(app);
var io = require('socket.io')(http);
var Redis = require('ioredis');
var redis = new Redis();
redis.subscribe('notifications', function(err, count) {
});
redis.on('message', function(channel, message) {
    console.log('Message Recieved: ' + message);
    message = JSON.parse(message);
    io.emit(channel + ':' + message.event, message.data);
});
http.listen(3000, function(){
    console.log('Listening on Port 3000');
});

1 个答案:

答案 0 :(得分:1)

首先,设置serverOptions对象:

var serverOptions = {
    port: 3000,
    host: 127.0.0.1, //address to your site
    key: '/etc/nginx/ssl/your_site/server.key', //Or whatever the path to your SSL is
    cert: '/etc/nginx/ssl/your_site/server.crt',
    NPNProtocols: ['http/2.0', 'spdy', 'http/1.1', 'http/1.0']
}

对于NPNProtocols,您可能并不关心所有这些内容,但它们仅供参考。

现在只需创建服务器:

var app = require('https').createServer(serverOptions),
    io = require('socket.io')(app);

在他的观点上,这应该是非常好的插件。\

作为旁注,你的东西是完全开放的,任何人都可以在你的网络套接字上收听,所以不应该发送任何私密的东西。如果您需要将数据设为私有,那么您将需要两件事中的一件;

  1. JWT-Auth Token
  2. 之类的东西
  3. Redis队列接口的自定义内容:
  4. 以下是后者的一个例子:

    var SECRET_KEY = '<YOUR_LARAVEL_SECRET_KEY>';
    var laravel_session_parser = {
        ord: function (string) {
            return string.charCodeAt(0);
        },
        decryptSession: function (cookie, secret) {
            if (cookie) {
                var session_cookie = JSON.parse(new Buffer(cookie, 'base64'));
                var iv = new Buffer(session_cookie.iv, 'base64');
                var value = new Buffer(session_cookie.value, 'base64');
                var rijCbc = new mcrypt.MCrypt('rijndael-128', 'cbc');
                rijCbc.open(secret, iv);
                var decrypted = rijCbc.decrypt(value).toString();
                var len = decrypted.length - 1;
                var pad = laravel_session_parser.ord(decrypted.charAt(len));
                return phpunserialize.unserialize(decrypted.substr(0, decrypted.length - pad));
            }
            return null;
        },
        getUidFromObj: function (obj, pattern) {
            var regexp = /login_web_([a-zA-Z0-9]+)/gi;
            if (pattern) {
                regexp = pattern;
            }
            var u_id = null;
            for (var key in obj) {
                var matches_array = key.match(regexp);
                if (matches_array && matches_array.length > 0) {
                    u_id = obj[matches_array[0]];
                    return u_id;
                }
            }
            return u_id;
        },
        getRedisSession: function (s_id, cb) {
            var _sessionId = 'laravel:' + s_id;
            client.get(_sessionId, function (err, session) {
                if (err) {
                    cb && cb(err);
                    return;
                }
                cb && cb(null, session);
            });
        },
        getSessionId: function (session, _callback) {
            var u_id = null,
                err = null;
            try {
                var laravelSession = phpunserialize.unserialize(phpunserialize.unserialize(session));
                u_id = laravel_session_parser.getUidFromObj(laravelSession);
            } catch (err) {
                _callback(err, null);
            }
            _callback(err, u_id);
        },
        ready: function (socket, _callback) {
            if (typeof socket.handshake.headers.cookie === 'string') {
                var cookies = cookie.parse(socket.handshake.headers.cookie);
                var laravel_session = cookies.laravel_session;
                var session_id = laravel_session_parser.decryptSession(laravel_session, SECRET_KEY);
    
                laravel_session_parser.getRedisSession(session_id, function (err, session) {
                    if (!err && session) {
                        laravel_session_parser.getSessionId(session, function (err, user_id) {
                            if (user_id) {
                                _callback(null, session_id, user_id, laravel_session)
                            } else {
                                _callback(new Error('Authentication error'), null);
                            }
                        });
                    } else {
                        _callback(new Error('Authentication error'), null);
                    }
                });
            }
        }
    };
    

    现在,您可以让IO在与socket.io建立连接时获取个人session的实例

    io.on('connection', function (socket) {
        laravel_session_parser.ready(socket, function(err, session_id, user_id, laravel_session) {
            //log out the variables above to see what they provide
        });
    });
    

    注意,我更喜欢在dotenv中使用NodeJS来共享Laravel和Node之间的环境变量。

    然后您可以执行process.env.APP_KEY,而无需担心共享变量。

    另外值得注意的是,上面的脚本不完整且没有生产就绪,它只是用作一个例子。