Інтеграція Socket.IO для real-time комунікації на сайті

Наша компанія займається розробкою, підтримкою та обслуговуванням сайтів будь-якої складності. Від простих односторінкових сайтів до масштабних кластерних систем, побудованих на мікро сервісах. Досвід розробників підтверджено сертифікатами від вендорів.
Розробка та обслуговування будь-яких видів сайтів:
Інформаційні сайти або веб-програми
Сайти візитки, landing page, корпоративні сайти, онлайн каталоги, квіз, промо-сайти, блоги, ресурси новин, інформаційні портали, форуми, агрегатори
Сайти або веб-програми електронної комерції
Інтернет-магазини, B2B-портали, маркетплейси, онлайн-обмінники, кешбек-сайти, біржі, дропшиппінг-платформи, парсери товарів
Веб-програми для управління бізнес-процесами
CRM-системи, ERP-системи, корпоративні портали, системи управління виробництвом, парсери інформації
Сайти або веб-програми електронних послуг
Дошки оголошень, онлайн-школи, онлайн-кінотеатри, конструктори сайтів, портали надання електронних послуг, відеохостинги, тематичні портали

Це лише деякі з технічних типів сайтів, з якими ми працюємо, і кожен із них може мати свої специфічні особливості та функціональність, а також бути адаптованим під конкретні потреби та цілі клієнта.

Пропоновані послуги
Показано 1 з 1 послугУсі 2065 послуг
Інтеграція Socket.IO для real-time комунікації на сайті
Середня
~3-5 робочих днів
Часті питання
Наші компетенції:
Етапи розробки
Останні роботи
  • image_website-b2b-advance_0.png
    Розробка сайту компанії B2B ADVANCE
    1262
  • image_web-applications_feedme_466_0.webp
    Розробка веб-додатків для компанії FEEDME
    1171
  • image_websites_belfingroup_462_0.webp
    Розробка веб-сайту для компанії БЕЛФІНГРУП
    874
  • image_ecommerce_furnoro_435_0.webp
    Розробка інтернет магазину для компанії FURNORO
    1094
  • image_crm_enviok_479_0.webp
    Розробка веб-додатків для компанії Enviok
    831
  • image_bitrix-bitrix-24-1c_fixper_448_0.png
    Розробка веб-сайту для компанії ФІКСПЕР
    851

Інтеграція Socket.IO для real-time комунікації на сайті

Socket.IO — бібліотека для двунаправленої комунікації між сервером та клієнтом. На відміну від чистого WebSocket, Socket.IO забезпечує автоматичне переподключення, fallback на long-polling, кімнати та neymspaces.

Встановлення та базова настройка

npm install socket.io       # сервер
npm install socket.io-client # клієнт

Сервер (Node.js + Express)

import { createServer } from 'http';
import { Server } from 'socket.io';
import express from 'express';

const app = express();
const httpServer = createServer(app);

const io = new Server(httpServer, {
  cors: {
    origin: process.env.CLIENT_URL,
    credentials: true
  },
  pingTimeout: 60000,
  pingInterval: 25000
});

// Middleware аутентифікації
io.use(async (socket, next) => {
  const token = socket.handshake.auth.token;
  try {
    const payload = jwt.verify(token, process.env.JWT_SECRET);
    socket.data.userId = payload.sub;
    socket.data.user = await userRepo.findById(payload.sub);
    next();
  } catch {
    next(new Error('Authentication error'));
  }
});

io.on('connection', (socket) => {
  const userId = socket.data.userId;
  console.log(`User ${userId} connected: ${socket.id}`);

  // Приєднатися до персональної кімнати
  socket.join(`user:${userId}`);

  // Обробники подій
  socket.on('join:room', async ({ roomId }) => {
    const canJoin = await checkRoomAccess(userId, roomId);
    if (!canJoin) {
      socket.emit('error', { message: 'Access denied' });
      return;
    }
    socket.join(`room:${roomId}`);
    socket.to(`room:${roomId}`).emit('user:joined', {
      userId, user: socket.data.user
    });
  });

  socket.on('message:send', async ({ roomId, content }) => {
    const message = await messageRepo.create({ roomId, userId, content });
    io.to(`room:${roomId}`).emit('message:new', message);
  });

  socket.on('disconnect', () => {
    console.log(`User ${userId} disconnected`);
    // Повідомляємо кімнати про вихід
    io.emit('user:offline', { userId });
  });
});

httpServer.listen(3000);

Клієнт (React)

// hooks/useSocket.ts
import { useEffect, useRef } from 'react';
import { io, Socket } from 'socket.io-client';
import { useAuthStore } from '@/stores/auth';

export function useSocket(): Socket | null {
  const socketRef = useRef<Socket | null>(null);
  const token = useAuthStore(s => s.token);

  useEffect(() => {
    if (!token) return;

    const socket = io(process.env.NEXT_PUBLIC_WS_URL, {
      auth: { token },
      reconnection: true,
      reconnectionDelay: 1000,
      reconnectionAttempts: 5,
      transports: ['websocket', 'polling']
    });

    socket.on('connect', () => console.log('Socket connected'));
    socket.on('connect_error', (err) => console.error('Connection error:', err));

    socketRef.current = socket;

    return () => {
      socket.disconnect();
      socketRef.current = null;
    };
  }, [token]);

  return socketRef.current;
}

// Компонент чату
function ChatRoom({ roomId }) {
  const socket = useSocket();
  const [messages, setMessages] = useState([]);

  useEffect(() => {
    if (!socket) return;

    socket.emit('join:room', { roomId });

    socket.on('message:new', (message) => {
      setMessages(prev => [...prev, message]);
    });

    return () => {
      socket.off('message:new');
      socket.emit('leave:room', { roomId });
    };
  }, [socket, roomId]);

  const sendMessage = (content: string) => {
    socket?.emit('message:send', { roomId, content });
  };

  return <ChatUI messages={messages} onSend={sendMessage} />;
}

Масштабування через Redis Adapter

При кількох Node.js процесах Socket.IO потребує Redis для синхронізації:

import { createAdapter } from '@socket.io/redis-adapter';
import { createClient } from 'redis';

const pubClient = createClient({ url: process.env.REDIS_URL });
const subClient = pubClient.duplicate();

await Promise.all([pubClient.connect(), subClient.connect()]);

io.adapter(createAdapter(pubClient, subClient));

// Тепер io.to('room:123').emit() працює across всіх Node.js процесів

Повідомлення з інших сервісів

// З backend-сервісу відправити повідомлення конкретному користувачу
// через Redis pub/sub (без прямого доступу до Socket.IO сервера)
const publisher = createClient({ url: process.env.REDIS_URL });

await publisher.publish('socket:notify', JSON.stringify({
  room: `user:${userId}`,
  event: 'order:status_changed',
  data: { orderId, newStatus: 'shipped' }
}));

Часові рамки

Socket.IO сервер + клієнтський хук + чат-кімнати + Redis adapter: 1–2 тижні.