Новое соединение приводит к тому, что текущие соединения перестают работать - PullRequest
1 голос
/ 02 августа 2020

Я создаю приложение чата, используя сокет (в котором я новичок) со структурой с несколькими арендаторами и с использованием пространств имен. Вот мой код:

Socket server: index. js

class Server {
    constructor() {
        this.port = process.env.PORT || 3000;
        this.host = process.env.HOST || `localhost`;

        this.app = express();
        this.http = http.Server(this.app);
        this.rootSocket = socketio(this.http);
    }

    run() {
        new socketEvents(this.rootSocket).socketConfig();
        this.app.use(express.static(__dirname + '/uploads'));
        this.http.listen(this.port, this.host, () => {
            console.log(`Listening on ${this.host}:${this.port}`);
        });
    }
}

const app = new Server();
app.run();

socket. js

var redis = require('redis');
var redisConnection = {
    host: process.env.REDIS_HOST,
    password: process.env.REDIS_PASS
};
var sub = redis.createClient(redisConnection);
var pub = redis.createClient(redisConnection);

class Socket {

    constructor(rootSocket) {
        this.rootIo = rootSocket;
    }

    socketEvents() {

        /**
         * Subscribe redis channel
         */
        sub.subscribe('visitorBehaviorApiResponse');
        //TODO: subscribe channel..

        // Listen to redis channel that published from api
        sub.on('message', (channel, data) => {
            data = JSON.parse(data);
            console.log(data);
            const io = this.rootIo.of(data.namespace);
            if (channel === 'visitorBehaviorApiResponse') {
                io.to(data.thread_id).emit('receiveBehavior', data);
                io.to('staff_room').emit('incomingBehavior', data);
            }
        })

        sub.on('error', function (error) {
            console.log('ERROR ' + error)
        })

        var clients = 0;
        this.rootIo.on('connection', (rootSocket) => {
            clients++;
            console.log('root:' + rootSocket.id + ' connected (total ' + clients + ' clients connected)');
            const ns = rootSocket.handshake['query'].namespace;

            // Dynamic namespace for multiple tenants
            if (typeof (ns) === 'string') {
                const splitedUrl = ns.split("/");
                const namespace = splitedUrl[splitedUrl.length - 1];
                const nsio = this.rootIo.of(namespace);
                this.io = nsio;

                this.io.once('connection', (socket) => {
                    var visitors = [];

                    console.log('new ' + socket.id + ' connected');

                    // once a client has connected, we expect to get a ping from them saying what room they want to join
                    socket.on('createChatRoom', function (data) {
                        socket.join(data.thread_id);
                        if (typeof data.is_staff !== 'undefined' && data.is_staff == 1) {
                            socket.join('staff_room');
                        } else {
                            if (visitors.some(e => e.visitor_id === data.visitor_id)) {
                                visitors.forEach(function (visitor) {
                                    if (visitor.visitor_id === data.visitor_id) {
                                        visitor.socket_ids.push(socket.id);
                                    }
                                })
                            } else {
                                data.socket_ids = [];
                                data.socket_ids.push(socket.id);
                                visitors.push(data);
                            }
                            socket.join('visitor_room');
                        }

                        //TODO: push to redis to check conversation type

                    });

                    socket.on('sendMessage', function (data) {
                        console.log(data);
                        pub.publish('chatMessage', JSON.stringify(data));
                        this.io.in(data.thread_id).emit('receiveMessage', data);
                        this.io.in('staff_room').emit('incomingMessage', data);
                        // Notify new message in room
                        data.notify_type = 'default';
                        socket.to(data.thread_id).emit('receiveNotify', data);
                    }.bind(this))

                    socket.on('disconnect', (reason) => {
                        sub.quit();
                        console.log('client ' + socket.id + ' left, ' + reason);
                    });

                    socket.on('error', (error) => {
                        console.log(error);
                    });

                });
            }

            // Root disconnect
            rootSocket.on('disconnect', function () {
               clients--;
               console.log('root:' + rootSocket.id + ' disconnected (total ' + clients + ' clients connected)');
            });
        });
    }

    socketConfig() {
        this.socketEvents();
    }
}
module.exports = Socket;

Клиент:

    const server = 'https://socket-server'
    const connect = function (namespace) {
        return io.connect(namespace, {
           query: 'namespace=' + namespace,
           resource: 'socket.io',
           transports: ['websocket'],
           upgrade: false
        })
    }
    const url_string = window.location.href
    const url = new URL(url_string)
    const parameters = Object.fromEntries(url.searchParams)
    const socket = connect(`${server}/${parameters.shopify_domain}`)
    var handleErrors = (err) => {
      console.error(err);
    }
    
    socket.on('connect_error', err => handleErrors(err))
    socket.on('connect_failed', err => handleErrors(err))
    socket.on('disconnect', err => handleErrors(err))

Проблема, с которой я столкнулся, заключается в том, что когда сервер сокетов получает новое соединение, существующие соединения перестают работать, когда они обновляют страницу для повторного подключения new socket.id.

И когда клиент пространства имен выдает данные, он отправляет их в другие пространства имен, кажется, мой код не работает правильно в пространстве имен.

Не могли бы вы взглянуть на мой код и укажите мне, где я ошибаюсь?

Спасибо

1 Ответ

0 голосов
/ 04 августа 2020
1) Get UserId or accessToken while handshaking(in case of accessToken decrypt it).
 and store userID: socketId(in Redis or in local hashmap) depends upon the requirement .

2) When u are going to emit to particular user fetch the  socketid to that userid from redis or local hashmap
and emit to it.


**io.to(socketId).emit('hey', 'I just met you');**


3) If you are using multiple servers use sticky sessions
 
4) Hope this will help you
...