Демонстрационная версия
Схематичное представление проекта
Введение
Чаты, которыми мы привычно пользуемся, кажутся чем-то загадочным. На самом деле они традиционно разрабатываются с использованием Websocket
. В этой статье кратко изложена основная информация о Websocket
и реализована простая демонстрационная версия чата на несколько человек. После прочтения вы сможете использовать Websocket
для создания чата.
Общие сведения о Websocket
Бэкграунд Websocket
Во многих случаях пользователям требуются сообщения в реальном времени, например для обмена мнениями, доступа к показаниям медицинских приборов и т.д. Традиционное решение заключается в получении актуальных данных на основе опроса. Однако оно не позволяет достичь полной синхронизации онлайн-сообщений. В большинстве случаев запросы оказываются неоправданными, расходуют много трафика и ресурсов сервера. Эта проблема и привела к появлению Websocket
.
Основная концепция Websocket
Websocket
— это протокол полнодуплексной связи на одном TCP-соединении, предоставляемый HTML5. Коммуникационный протокол Websocket
появился в 2008 году и стал международным стандартом в 2011 году.
Websocket
упрощает обмен данными между клиентом и сервером, позволяя серверу активно передавать данные клиенту. В API Websocket
браузеру и серверу необходимо только завершить ”рукопожатие” (подтверждение установления связи), после чего они могут напрямую создавать постоянное соединение и осуществлять двустороннюю передачу данных.
Проблемы совместимости (поддерживается основными браузерами):
Особенности Websocket
- Незначительное потребление ресурсов
После создания соединения, когда происходит обмен данными между сервером и клиентом, заголовок пакета для управления протоколом относительно невелик. Без расширения размер заголовка составляет всего от 2 до 10 байт для передачи контента от сервера к клиенту (в зависимости от длины пакета). Для передачи контента от клиента к серверу этот заголовок необходимо маскировать дополнительными 4 байтами. По сравнению с HTTP-запросами, которые должны каждый раз нести полный заголовок, такие затраты являются значительно низкими.
- Связь в реальном времени
Поскольку протокол является полнодуплексным, сервер может активно отправлять данные клиенту в любое время. По сравнению с HTTP-запросами, серверу нужно ждать, пока клиент инициирует запрос, прежде чем ответить, и задержка значительно меньше. Даже по сравнению с длинными опросами, такими как Comet, в данном случае данные могут доставляться большее количество раз за короткое время.
- Постоянное соединение
В отличие от HTTP, веб-сокету необходимо сначала создать соединение, поэтому он является протоколом состояний. В дальнейшем при обмене данными некоторая информация о состоянии может быть опущена.
- Поддержка двоичной передачи
Вы можете передавать текстовые и двоичные данные. Websocket
определяет двоичные фреймы, которые могут обрабатывать двоичный контент легче, чем HTTP.
- Идентификатором протокола является
ws
(илиwss
при поддержке шифрования), а URL сервера — это URL - Простота реализации
Реализация серверной стороны основана на TCP-протоколе и относительно проста, а также не имеет ограничений по источникам. Клиент может общаться с любым сервером.
- Хорошая совместимость с HTTP-протоколом
Порты по умолчанию также 80 и 443, а во время фазы “рукопожатия” используется HTTP-протокол. Поэтому заблокировать “рукопожатие” нелегко, и оно может проходить через различные HTTP-прокси-серверы.
- Поддержка расширений
Websocket
определяет расширения. Тут можно расширять протоколы и реализовывать субпротоколы, определяемые пользователем. Например, некоторые браузеры поддерживают сжатие.
Начальное “рукопожатие” WebSocket
Каждое соединение веб-сокета начинается с HTTP-запроса, который похож на другие запросы, но содержит специальный заголовок — Upgrade
. Upgrade
указывает на то, что клиент переведет соединение на протокол Websocket
(т.е. обновит соединение).
До “рукопожатия” Websocket
следует протоколу HTTP/1.1.
Запрос, отправляемый клиентом для перехода на Websocket, также называется начальным рукопожатием. После того как клиент отправляет HTTP-запрос на переход, соединение не будет успешным до тех пор, пока сервер не ответит кодом состояния 101, заголовком Upgrade
и Sec WebSocket
Accept. В противном случае соединение не удастся.
Ниже приведены заголовки запросов и соответствующие заголовки копируемого “рукопожатия” WebSocket
:
// Заголовок запроса, отправляемый клиентом
GET wss://www.example.cn/webSocket HTTP/1.1 // Используемый протокол https и соответствующий запрос wss.
Host: www.example.cn
Connection: Upgrade // Сообщение http1.1 с заголовком обновления должно содержать заголовок соединения. Это означает, что любой, кто принимает это сообщение, удаляет домен, указанный в соединении, перед пересылкой сообщения (то есть не пересылает домен обновления).
Upgrade: websocket // Определите домен заголовка протокола преобразования. Если сервер поддерживает его, клиент будет использовать установленное соединение http (tcp).
Sec-WebSocket-Version: 13 // Список версий протокола WebSocket, поддерживаемых клиентом.
Origin: http://example.cn // Origin используется для предотвращения межсайтовых атак. Он помогает браузерам идентифицировать исходный домен.
Sec-WebSocket-Key: afmbhhBRQuwCLmnWDRWHxw== // Случайным образом генерируется клиентом. Сервер будет использовать это поле для сборки другого значения ключа и поместит его в возвращаемую информацию о "рукопожатии". Используется для начального "рукопожатия" между клиентским и серверным веб-сокетами, чтобы избежать межпротокольных атак.
Sec-WebSocket-Protocol: chat, superchat // Сообщает приложению клиента, какие протоколы доступны.
Sec-WebSocket-Extensions: permessage-deflate; client_max_window_bits // permessage-deflate —— Сообщает приложению клиента, какие протоколы доступны. client_max_window_bits —— Договор об использовании сжатия данных при передаче
// Заголовки ответов, отправляемые сервером
HTTP/1.1 101
Server: nginx/1.12.2
Date: Sat, 11 Aug 2018 13:21:27 GMT
Connection: upgrade
Upgrade: websocket
Sec-WebSocket-Accept: sLMyWetYOwus23qJyUD/fa1hztc= // Confirm whether the server understands the websocket protocol
Sec-WebSocket-Protocol: chat
Sec-WebSocket-Extensions: permessage-deflate;client_max_window_bits=15
/**
Этапы создания Sec-WebSocket-Accept的:
(1)Склейка Sec-WebSocket-Key с помощью GUID, определенного в соглашении.
(3)Base64 кодирует строку, сгенерированную в пункте 2
Sec-WebSocket-Accept используется для определения:
(1)Понимает ли сервер протокол Websocket? Если нет, то он не вернет корректный Sec-WebSocket-Accept.
(2)Возвращаемое значение - текущий запрос, а не предыдущий кэш.
*/
Сходства и различия между WebSocket и HTTP
- Идентичные моменты:
- Оба являются протоколами прикладного уровня, основанными на
TCP
. - Оба используют модель
Request/Response
для установления соединений. - Метод обработки ошибок во время установления соединения у них одинаков. На этом этапе
WebSocket
может вернуть тот же код возврата, что иHTTP
.
2. Различия:
HTTP
-протокол основан на методе “запрос/ответ” (Request/Response). Он может осуществлять только одностороннюю передачу (полудуплексная связь), в то время какWebSocket
является полнодуплексной связью.
Half duplex communication
(полудуплексная связь): односторонний поток, сервер не активно передает данные клиенту.
Full duplex communication
(полнодуплексная связь): сервер может активно передавать информацию клиенту, а клиент может также активно передавать информацию серверу. Это полноценный двусторонний равноправный диалог, работающий по технологии server push.
HTTP
не фиксирует данные о поступающих запросах, поэтому запрос закрывается после получения ответа. Преимущество отсутствия фиксации данных о запросах в том, что серверу не нужно хранить соответствующую информацию о сессии. Недостатком является то, что каждый HTTP
-запрос и ответ будут посылать избыточную информацию о запросе.
Между тем, WebSocket
нужно только один раз установить пару сообщений Request/Response
, за которой следует TCP
-соединение, что позволяет избежать избыточной информации заголовков, генерируемой несколькими парами сообщений Request/Response
. Это экономит существенное количество трафика и ресурсов сервера.
Когда WebSocket
устанавливает соединение “рукопожатия”, данные передаются через HTTP
-протокол , но после установления соединения фаза фактической передачи данных не требует участия HTTP
-протокола. HTTP
требуется три ”рукопожатия”.
Данные, передаваемые WebSocket
, представляют собой двоичный поток, основанный на фреймах. Передача данных по HTTP
-протоколу — это передача открытого текста, то есть передача строк.
API Websocket
WebSocket
преобразует HTTP
-протокол в WebSocket
-протокол во время первого “рукопожатия” между клиентом и сервером. После установления соединения следующие сообщения напрямую передаются туда и обратно по методу, определенному интерфейсом WebSocket
.
Экземпляр WebSocket
По сути WebSocket
-протокол является протоколом на основе TCP
. Вызовите конструктор WebSocket
для создания WebSocket
-соединения и возвращения экземпляра объекта WebSocket
.
Протокол WebSocket
определяет две URL-схемы.
- ws — незашифрованный формат.
- wss — шифрованный формат (используется механизм безопасности, принятый в HTTPS для обеспечения безопасности HTTP-соединений).
/**
* @param
* URL: connection target
* protocols(optional):string | string[]. имя протокола или набор имен протоколов
*/
const ws = new WebSocket(URL, protocols)
Чтобы установить соединение WebSocket
, браузер клиента должен сначала отправить HTTP
-запрос на сервер. Этот запрос отличается от обычного HTTP
-запроса и содержит некоторую дополнительную информацию заголовка. Дополнительная информация заголовка “Upgrade: WebSocket” указывает на то, что это HTTP
-запрос на обновление протокола.
Сервер анализирует дополнительную информацию заголовка и затем генерирует ответную информацию для возврата клиенту. Между клиентом и сервером устанавливается WebSocket
-соединение. Обе стороны могут свободно передавать информацию по этому каналу связи. Соединение будет продолжаться до тех пор, пока один из клиентов или сервер активно не закроет соединение.
События Websocket
WebSocket
ориентирован исключительно на события. Вы можете обрабатывать ввод данных и изменения состояния соединения, прослушивая события на объекте WebSocket
. Ниже перечислены четыре события объекта WebSocket
.
onopen
: срабатывает после того, как клиент и сервер устанавливают соединение. Это называется начальным “рукопожатием” между клиентом и сервером. Если получен сигнал open, соединение успешно установлено и связь готова.onmessage
: срабатывает при получении сообщения. Сообщения, отправляемые сервером клиенту, могут включать открытые текстовые сообщения и двоичные данные (сообщенияblob
иArrayBuffer
).onerror
: срабатывает в ответ на неожиданный сбой. После ошибки соединение всегда прерывается.onclose
: срабатывает при закрытии соединения. После закрытия соединения клиент и сервер не смогут отправлять и получать сообщения. Для закрытия соединения можно также активно вызывать методclose()
.
Методы Websocket
send()
: отправка сообщения после успешного соединения и перед закрытием (сообщения могут быть отправлены только после открытия и перед закрытием).close()
: закрытие соединения.
Свойства объектов Websocket
readyState
: атрибут только для чтения, который указывает на состояние соединения веб-сокета. Значения следующие:
Constant property | Value | Status
--------------------------------------------------------------------------------------------------------
Websocket.CONNECTING | 0 | Соединение выполняется, но еще не было успешно установлено.
Websocket.OPEN | 1 | Соединение установлено, и сообщение может быть отправлено в обычном режиме.
Websocket.CLOSING | 2 | Соединение находится в процессе завершения "рукопожатия".
Websocket.CLOSED | 3 | Указывает на то, что соединение было закрыто или не может быть открыто.
bufferedAmount
: свойство только для чтения. Количество текстовых байтов UTF-8, которые были поставлены в очередь посредствомsend()
и ожидают передачи, но еще не были выданы.Protocol
: протокол, используемый во время открытого “рукопожатия”.
Создание простого чата с помощью Websocket
Базовое знакомство с WebSocket
завершено. Теперь воспользуемся полученными знаниями, чтобы вручную реализовать демонстрационную версию чата.
Создание соединения
В нашем случае используем nodejs
для реализации логики на стороне сервера и сторонний модуль связи WebSocket
ws
для создания простого WebSocket
-соединения.
- Сначала необходимо установить
ws
.
npm i ws
2. Сервер. Создайте файл server.js
, установите соединение WebSocket
и запустите службу (здесь используется порт 3000). Используйте сторонний плагин ws
для создания нового экземпляра WebSocket
. После успешного подключения прослушайте событие onmessage
для получения поступившего сообщения, прослушайте событие onclose
для обработки логики соединения/разъединения и используйте метод send
для отправки сообщений клиенту.
const Websocket = require('ws')
const wss = new Websocket.Server({ port: 3000 })
wss.on('connection', function (ws) { // Клиент подключен
ws.on('message', function (message) {
console.log('server receive message: ', message.toString())
})
ws.send('msg from server!')
ws.on('close', function (message) {
console.log('连接断开', message)
})
})
3. Клиент. Здесь Vue
используется для создания базового проекта и соединения на странице home.vue
. Сначала создайте экземпляр ws
, используя нативный WebSocket
, и отследите успешность соединения WebSocket
посредством метода onopen
(когда readyState
равен “1”, соединение выполнено успешно). Затем обработайте логику входа пользователя в чат (open
), используйте onmessage
для получения сообщений сервера и onclose
для отслеживания разъединения и выполните соответствующую обработку. Когда пользователь покидает чат, примените метод close
вручную.
/* home.vue */
this.ws = new WebSocket('ws://localhost:3000')
console.log('before open', this.ws.readyState) // 0
// Прослушивание успешности создания соединения
this.ws.onopen = () => {
console.log('onopen', this.ws.readyState) // 1
this.roomOpen = true
this.ws.send(JSON.stringify({
userId: this.userName,
userName: this.nickname,
roomId: item.roomId,
roomName: item.name,
event: 'login', // Отправить на сервер сообщение о входе в систему с соответствующей информацией о чате и информацией о пользователе
}))
}
// Обратный вызов для полученного сообщения
this.ws.onmessage = (message) => {
console.log('The client receives the message', message)
}
// Получение уведомления о разъединении
this.ws.onclose = () => { // Callback to listen for websocket close
console.log('onclose', this.ws.readyState)
}
// Ручное отключение соединения websocket
close () {
this.ws && this.ws.close()
}
Статистика онлайн-статуса нескольких чатов и пользователей
Логика создания нескольких чатов осуществляется на стороне сервера. Здесь мы различаем несколько чатов в соответствии с roomId
. Сначала создайте массив для хранения roomId
. Каждый раз, когда создается новый чат, добавляйте roomId
в массив. Если вы входите в уже созданный чат, добавьте “1” к количеству людей, находящихся онлайн в этом чате. Фронтенд отвечает за отправку на сервер уведомления о входе и выходе пользователя и соответствующей информации о пользователе, а также за рендеринг информации, необходимой странице, в соответствии с сообщением, отправленным сервером. Код выглядит следующим образом:
home.vue
// home.vue
this.ws.onopen = () => {
this.roomOpen = true
this.ws.send(JSON.stringify({
userId: this.userName,
userName: this.nickname,
roomId: item.roomId,
roomName: item.name,
event: 'login',
}))
}
this.ws.onmessage = (message) => {
const data = JSON.parse(message.data)
this.onlineNum = data.num
if (data.event === 'login') { // Сообщение о том, что другие пользователи заходят в чат
this.msgList.push({
content: `Welcome ${data.userName} to room ${data.roomName}~`,
})
} else if (data.event === 'logout') {
// Сообщение о том, что другие пользователи покидают чат
console.log('logout', data)
this.msgList.push({
content: `${data.userName} Leave the room`,
})
} else { // normal message
const self = this.userId === data.userId
if (self) return
this.msgList.push({
name: data.userName,
self: false,
content: data.content,
})
}
}
serve.js
// server.js
ws.on('message', function (message) {
console.log('server receive message: ', message.toString())
const data = JSON.parse(message.toString())
if (typeof ws.roomId === 'undefined' && data.roomId) {
ws.roomId = data.roomId
if (typeof group[ws.roomId] === 'undefined') {
group[ws.roomId] = 1
} else {
group[ws.roomId]++
}
}
data.num = group[ws.roomId]
wss.clients.forEach(client => {
if (client.readyState === Websocket.OPEN && client.roomId === ws.roomId) {
client.send(JSON.stringify(data))
}
})
})
ws.on('close', function (message) {
// После мониторинга закрытия чата уменьшите количество онлайн-пользователей на 1 и передайте сообщение о выходе из чата другим клиентам, чтобы обновить количество пользователей онлайн на странице
group[ws.roomId]--
wss.clients.forEach(function each (client) {
if (client !== ws && client.readyState === Websocket.OPEN && client.roomId === ws.roomId) {
client.send(JSON.stringify({
...ws.enterInfo,
event: 'logout',
num: group[ws.roomId],
}))
}
})
})
Поддержка heartbeat (“пульсации”)
В сценарии длительного соединения клиент и сервер не всегда находятся на связи. Если две стороны долгое время не общались, неясно, находится ли другая сторона на связи. Поэтому для обеспечения нормального соединения необходимо отправить небольшое контрольное сообщение, чтобы передать другой стороне: “я все еще жив”.
Как показано на изображении выше, на прикладном уровне клиент обычно посылает серверу heartbeat-пакет ping
, а сервер после его получения отвечает pong
. Это указывает на то, что обе стороны “живы”.
Помимо обеспечения нормального соединения, heartbeat-поддержка выполняет следующие функции.
- Если сервер обнаружит, что клиент не отвечает на heartbeat-сигнал, он может активно закрыть канал и перевести его в оффлайн-режим.
- Клиент может переподключиться для получения нового соединения, если обнаружит, что сервер не ответил на heartbeat-сигнал.
Допустим, пользователь ПК регистрируется на хосте с помощью Telnet и TCP/IP. Если в конце дня он просто выключит питание, не выходя из системы, то останется полуоткрытое соединение. Если клиент исчез, оставив на сервере полуоткрытое соединение, а сервер ждет данных от клиента, то сервер будет ждать вечно. Функция live-keeping будет пытаться обнаружить это полуоткрытое соединение на стороне сервера.
Для сервера также очень важно своевременно узнавать о наличии соединения. С одной стороны, ему необходимо своевременно очищать некорректные соединения, чтобы снизить нагрузку. С другой, это является одним из бизнес-требований. Например, если речь идет о копии игры, серверу необходимо своевременно решать проблемы, вызванные отключением игроков.
Ниже приведен код для heartbeat-поддержки:
this.ws.onopen = () => {
if (this.heartbeatTimer !== -1) {
clearInterval(this.heartbeatTimer)
this.heartbeatTimer = -1
}
this.heartbeatTimer = setInterval(() => {
if (this.heartBeatTimeoutJob !== -1) {
clearTimeout(this.heartBeatTimeoutJob)
this.heartBeatTimeoutJob = -1
}
this.heartBeatTimeoutJob = setTimeout(() => {
console.log('heartbeat timeout')
}, 10000)
this.ws.send(JSON.stringify({
event: 'heartBeat',
content: 'ping',
}))
console.log('send ping')
}, 25000)
}
this.ws.onmessage = (message) => {
console.log('onmessage', message)
const data = JSON.parse(message.data)
console.log('message.data: ', data)
if (data.event === 'heartBeat' && data.content === 'pong') {
console.log('receive server pong')
if (this.heartBeatTimeoutJob !== -1) {
clearTimeout(this.heartBeatTimeoutJob)
this.heartBeatTimeoutJob = -1
}
return
}
}
this.ws.onclose = () => {
console.log('onclose', this.ws.readyState)
clearInterval(this.heartbeatTimer)
this.heartbeatTimer = -1
clearTimeout(this.heartBeatTimeoutJob)
this.heartBeatTimeoutJob = -1
}
В home.vue
мы создали таймер heartbeatTimer
для отправки в реальном времени ping
-сообщений на сервер каждый раз после прослушивания onopen
. В то же время внутри heartBeatTimer
мы создали таймер heartBeatTimeoutJob
для проверки того, ответил ли сервер pong
-сообщением. Если pong
-сообщение от сервера не будет получено в течение времени, установленного heartBeatTimeoutJob
, это будет считаться heartbeat-таймаутом. После обнаружения heartbeat-таймаута можно отключиться и подключиться снова.
Обратите внимание, что длительность heartbeatTimer
должна быть больше, чем длительность heartBeatTimeoutJob
. В противном случае таймаут heartBeatTimeoutJob
не наступит. Конкретная продолжительность может быть определена в соответствии с вашими производственными потребностями.
Когда вы получите pong
-сообщение от сервера в обратном вызове onmessage
, вам необходимо очистить и переустановить таймер heartbeat-таймаута heartBeatTimeoutJob
. Следует также отметить, что перед созданием каждого нового таймера необходимо определить, существует ли в текущей среде таймер с таким же именем, и очистить его, чтобы избежать возникновения ошибок при многократном запуске таймера.
Кроме того, таймер должен своевременно очищаться при каждом закрытии WebSocket
-соединения. В противном случае, даже если пользователь вышел из чата, фоновый таймер не прекратит работу, что может привести к утечке памяти и другим непредвиденным проблемам.
server.js
ws.on('message', function (message) {
console.log('server receive message: ', message.toString())
const data = JSON.parse(message.toString())
if (data.event === 'login') {
ws.enterInfo = data
}
if (data.event === 'heartBeat' && data.content === 'ping') {
console.log('receive ping message')
ws.isAlive = true
ws.send(JSON.stringify({
event: 'heartBeat',
content: 'pong',
}))
return
}
if (typeof ws.roomId === 'undefined' && data.roomId) {
ws.roomId = data.roomId
if (typeof group[ws.roomId] === 'undefined') {
group[ws.roomId] = 1
} else {
group[ws.roomId]++
}
}
console.log('groun', group)
data.num = group[ws.roomId]
wss.clients.forEach(client => {
if (client.readyState === Websocket.OPEN && client.roomId === ws.roomId) {
client.send(JSON.stringify(data))
}
})
})
В server.js
сервер также должен иметь таймер, чтобы проверить, истек ли heartbeat-таймаут. Если это подтвердится, он отключится и пересчитает количество людей онлайн.
Сообщение должно быть доставлено
Зачем нужно получать оповещения?
Сообщение Bida используется для обработки некоторых важных сообщений в процессе длительного соединения. Из-за неполадок в сети, на сервере и по другим причинам пользователи могут не получить сообщений.
Несколько ситуаций потери сообщений:
Ситуация 1 на приведенном выше изображении: сообщение теряется, когда клиент отправляет сообщение на сервер. Такая ситуация не может быть исправлена.
Ситуации 2 и 3 на том же изображении: если сообщение успешно отправлено на сервер, но доставить его не удается из-за разрыва сети и переключения, о чем сервер уведомляет пользователя или других пользователей, пользователь-адресат не может получить сообщение. В этом случае пользователи могут получать важные сообщения с помощью подтверждения ACK, чтобы улучшить показатель количества доставленных сообщений.
Использование ACK-механизма для обработки сообщения Bida
ACK-механизм используется для обработки сообщения, которое должно дойти до адресата. Когда клиент получает сообщение, он должен отправить ACK-подтверждение, тем самым уведомляя сервер о получении сообщения. Если сервер не получает от клиента ACK-подтверждение, считается, что клиент сообщение не получил. В таком случае сервер повторно отправляет сообщение, чтобы пользователь гарантированно его получил.
Когда клиенту необходимо получить сообщение, могут возникнуть следующие ситуации при использовании ACK для обработки сообщения.
1. Нормальные условия
- После получения сообщения пользователь отправляет ACK-подтверждение на сервер. Получив уведомление о том, что клиент получил сообщение, сервер не будет больше отправлять это сообщение (как показано на рисунке ниже слева).
- Пользователь не получил сообщение, поэтому не отправил ACK-подтверждение на сервер. Сервер, не получив ACK-подтверждение, повторно отправил сообщение. Когда пользователь получает сообщение, отправка этого сообщения завершается (см. рисунок ниже справа).
2. Сбой
После получения сообщения пользователь отправляет ACK-подтверждение на сервер. Во время процесса отправки связь прерывается. В результате сервер ошибочно считает, что клиент не получил сообщение, повторно передает сообщение, и на стороне клиента отображается несколько повторяющихся сообщений. (Это проблема, вызванная обработкой ACK трафика, связанного с сообщениями. Клиент должен принимать участие в дедупликации сообщений).
Заключение
Websocket
нуждается в определенных доработках — аутентификации чатов, синхронизации сообщений в оффлайн-режиме, роуминге сообщений и т.п. Иными словами, Websocket
нужно развивать, чтобы совершенствовать чат-функции и мгновенные сообщения.
Читайте также:
- 50 вопросов, ответы на которые вы должны знать, прежде чем идти на собеседование по JavaScript
- Создание пользовательских уведомлений с помощью AWS WebSockets
- Как сократить время начальной загрузки веб-приложения
Читайте нас в Telegram, VK и Дзен
Перевод статьи fatfish: How to Build a Multiplayer Chatroom With WebSocket in 10 Minutes