如何 create/join 在 node js 中使用 ws (Websocket) 包聊天室
How to create/join chat room using ws (Websocket) package in node js
我在服务器端使用 ws 包,我希望客户端到服务器套接字中的 create/join 房间。并在它们不再连接时将它们从创建的房间中移除。
PS: 我不想使用 socketIo。
您可以使用或不使用套接字以这种方式创建用户和房间
const users = [];//It can be collection(noSQL) or table(SQL)
const addUser = ({ id, name, room }) => {
name = name.trim().toLowerCase();
room = room.trim().toLowerCase();
const existingUser = users.find((user) => user.room === room && user.name === name);
if(!name || !room) return { error: 'Username and room are required.' };
if(existingUser) return { error: 'Username is taken.' };
const user = { id, name, room };
users.push(user);
return { user };
}
您可以尝试类似的方法:
const rooms = {};
wss.on("connection", socket => {
const uuid = ...; // create here a uuid for this connection
const leave = room => {
// not present: do nothing
if(! rooms[room][uuid]) return;
// if the one exiting is the last one, destroy the room
if(Object.keys(rooms[room]).length === 1) delete rooms[room];
// otherwise simply leave the room
else delete rooms[room][uuid];
};
socket.on("message", data => {
const { message, meta, room } = data;
if(meta === "join") {
if(! rooms[room]) rooms[room] = {}; // create the room
if(! rooms[room][uuid]) rooms[room][uuid] = socket; // join the room
}
else if(meta === "leave") {
leave(room);
}
else if(! meta) {
// send the message to all in the room
Object.entries(rooms[room]).forEach(([, sock]) => sock.send({ message }));
}
});
socket.on("close", () => {
// for each room, remove the closed socket
Object.keys(rooms).forEach(room => leave(room));
});
});
这只是一个草图:您需要处理离开房间、断开与客户端的连接(离开所有房间)以及当没人再进来时删除房间。
const port = process.env.PORT || 8000
const WebSocket = require('ws');
function noop() {}
function heartbeat() {
this.isAlive = true;
}
const wss = new WebSocket.Server({ port });
var rooms = {};
const paramsExist = (data) =>{
try {
if('meta' in data && 'roomID' in data && 'clientID' in data && 'message' in data){
return true;
}else{
return false;
}
} catch (error) {
return false;
}
}
const roomExist = (roomID) =>{
// check for room is already exist or not
if(roomID in rooms){
return true;
}else{
return false;
}
}
const insideRoomdataExist = (arr,data) =>{
var status = false;
for(var i =0; i<arr.length;i++){
if(data in arr[i]){
status= true;
break;
}
}
return status;
}
const clientExistInRoom = (roomID,ws,clientID) =>{
var status = false;
const data = rooms[roomID];
for(var i =0; i< data.length ;i++){
var temp = data[i];
// if(roomID in temp){
// status=true;
// console.log("hello world");
// }
for(const obj in temp){
// if(ws == temp[obj]){
if(clientID == obj){
status = true;
break;
}
}
}return status;
}
// create room
const createRoom =(data,ws)=>{
try {
var {roomID,clientID} = data;
const status = roomExist(roomID);
if(status){
ws.send(JSON.stringify({
'message':'room already exist',
'status':0
}));
}else{
rooms[roomID] = [];
var obj = {};
obj[clientID] = ws;
rooms[roomID].push(obj);
ws['roomID']=roomID;
ws['clientID']=clientID;
ws['admin']=true;
ws.send(JSON.stringify({
'message':'room created succesfully',
'status':1
}));
}
} catch (error) {
ws.send(JSON.stringify({
'message':'there was some problem in creating a room',
'status':0
}));
}
}
// join room
const joinRoom = (data,ws) => {
try {
var {roomID,clientID} = data;
// check if room exist or not
const roomExist = roomID in rooms;
if(!roomExist){
ws.send(JSON.stringify({
'message':'Check room id',
'status':0
}));
return;
}
// const inRoom = insideRoomdataExist(rooms[roomID],clientID);
const inRoom = clientExistInRoom(roomID,ws,clientID)
if(inRoom){
ws.send(JSON.stringify({
"message":"you are already in a room",
"status":0
}));
}else{
var obj = {};
obj[clientID] = ws;
rooms[roomID].push(obj);
ws['roomID']=roomID
ws['clientID']=clientID;
ws.send(JSON.stringify({
"message":"Joined succesfully",
"status":1
}));
}
} catch (error) {
ws.send(JSON.stringify({
'message':'there was some problem in joining a room',
'status':0
}));
}
}
// send message
const sendMessage = (data,ws,Status=null) => {
try {
var {roomID, message,clientID} = data;
//check whether room exist or not
const roomExist = roomID in rooms;
if(!roomExist){
ws.send(JSON.stringify({
'message':'Check room id',
'status':0
}));
return;
}
// check whether client is in room or not
const clientExist = clientExistInRoom(roomID,ws,clientID);
if(!clientExist){
ws.send(JSON.stringify({
'message':"You are not allowed to send message",
'status':0
}));
return;
}
const obj = rooms[roomID];
for(i=0;i<obj.length;i++){
var temp = obj[i];
for(var innerObject in temp){
var wsClientID = temp[innerObject];
if(ws!==wsClientID){
wsClientID.send(JSON.stringify({
'message':message,
'status':Status?Status:1
}));
}
}
}
} catch (error) {
ws.send(JSON.stringify({
'message':'There was some problem in sending message',
'status':0
}));
}
}
const leaveRoom = (ws,data) => {
try {
const {roomID} = data;
// manual code started------------------------------------------------------------
const roomExist = roomID in rooms;
if(!roomExist){
ws.send(JSON.stringify({
'message':'Check room id',
'status':0
}));
return;
}
if('admin' in ws){
data['message']="Admin left the room.";
sendMessage(data,ws,Status=2);
delete rooms[ws.roomID]
return;
}
else{
// find the index of object
lst_obj = rooms[roomID];
var index = null;
for(let i=0;i<lst_obj.length;i++){
var temp_obj = lst_obj[i];
for(var key in temp_obj){
var temp_inside = temp_obj[key]
if('admin' in temp_inside){
temp_inside.send(JSON.stringify({
'message':'Somebody leave the room',
'status':3
}));
}
if(ws==temp_inside){
index =i;
}
}
}
if(index!=null){
rooms[roomID].splice(index,1);
console.log((rooms[roomID].length));
}
}
} catch (error) {
ws.send(JSON.stringify({
'message':'There was some problem----------------------',
'status':0
}))
}
}
const available_room = (ws) =>{
try {
var available_room_id=[];
for(var i in rooms){
available_room_id.push(parseInt(i));
}
ws.send(JSON.stringify({
"rooms":available_room_id,
"status":4
}))
} catch (error) {
ws.send(JSON.stringify({
'message':'There was some problem----------------------',
'status':0
}))
}
}
wss.on('connection', function connection(ws) {
try {
ws.on('message',(recieveData)=>{
var data = JSON.parse(recieveData);
const error = paramsExist(data);
if(!error){
ws.send(JSON.stringify({
'message':'check params',
'status':0
}));
return;
}
var {roomID,meta} = data;
switch (meta) {
case "create_room":
createRoom(data,ws);
console.log(rooms);
break;
case "join_room":
joinRoom(data,ws);
console.log(rooms);
break;
case "send_message":
sendMessage(data,ws);
console.log(rooms);
break;
case "show_all_rooms":
ws.send(JSON.stringify({
"rooms":[rooms]
}))
break;
default:
ws.send(JSON.stringify({
"message":"Unsupported meta data provided provide valid data",
"status":0
}));
break;
}
})
ws.on('close', function(data) {
leaveRoom(ws,{roomID:ws.roomID,clientID:ws.clientID,message:"Leave request"})
ws.terminate();
});
ws.on('pong', heartbeat);
} catch (error) {
ws.send(JSON.stringify({
"message":"there was some problem",
"status":0
}))
}
});
const interval = setInterval(function ping() {
var a = wss.clients;
wss.clients.forEach(function each(ws) {
if (ws.isAlive === false) {
leaveRoom(ws,{roomID:ws.roomID,clientID:ws.clientID});
ws.terminate();
}
ws.isAlive = false;
ws.ping(noop);
});
}, 50000);
const serverFree = setInterval(()=>{
var removeKey = [];
for(const obj in rooms){
if(rooms[obj].length<1){
removeKey.push(obj);
}
}
for(var i =0; i<removeKey.length;i++){
delete rooms[removeKey[i]];
}
},30000)
您好,上面是使用was进行socket连接的代码。您需要做的就是在元键中从前端提供 JSON 。您需要使用套接字库从 JSON 中的前端发送 4 个参数。
必填键
-meta:"create_room"/"join_room"/"send_message"/"show_all_rooms"
-消息:“任何东西”
-roomID:“创建或加入的roomid”
-clientID:“唯一客户端 ID”
package.json 文件依赖项
{
"name": "nodesocket",
"version": "1.0.0",
"description": "",
"main": "app.js",
"scripts": {
"test": "echo \"Error: no test specified\" && exit 1",
"start": "node src/app.js"
},
"author": "",
"license": "ISC",
"dependencies": {
"cors": "^2.8.5",
"express": "^4.17.1",
"ws": "^7.4.6"
}
}
我建议使用存储频道和 WS 数组的 Map 而不是注册用户房间的数组。
如果你有很多用户,这将是一种更好的发送消息的方法(我为我的公司实现了这些类型的 WS)可以轻松处理 40k 到 50k(对于 WS 的每个节点)并发送数千个每秒消息数。
您可以轻松地广播消息(只需从您的地图中获取一个房间)。
地图是这样的:Map<String,[]> (room, ws array)
.
当用户加入一个新房间时,只需将它添加到房间的数组中,当他离开时将其删除。 (/!\ 不要忘记双向 ping 用户)
要向房间发送消息,只需调用(避免在未使用 map.has(channel)
检查房间是否存在的情况下调用它)。
map.get(room).forEach(ws => ws.send(yourMessage))
它还帮助我进行了线性缩放。这个数组在我的 wss
和 ws
之外,所以当从 REDIS(或其他任何东西)发布消息时,我只是放置一个回调,将消息发送到相应的房间。所以当你广播消息时它不会检查所有用户的房间。
我在服务器端使用 ws 包,我希望客户端到服务器套接字中的 create/join 房间。并在它们不再连接时将它们从创建的房间中移除。 PS: 我不想使用 socketIo。
您可以使用或不使用套接字以这种方式创建用户和房间
const users = [];//It can be collection(noSQL) or table(SQL)
const addUser = ({ id, name, room }) => {
name = name.trim().toLowerCase();
room = room.trim().toLowerCase();
const existingUser = users.find((user) => user.room === room && user.name === name);
if(!name || !room) return { error: 'Username and room are required.' };
if(existingUser) return { error: 'Username is taken.' };
const user = { id, name, room };
users.push(user);
return { user };
}
您可以尝试类似的方法:
const rooms = {};
wss.on("connection", socket => {
const uuid = ...; // create here a uuid for this connection
const leave = room => {
// not present: do nothing
if(! rooms[room][uuid]) return;
// if the one exiting is the last one, destroy the room
if(Object.keys(rooms[room]).length === 1) delete rooms[room];
// otherwise simply leave the room
else delete rooms[room][uuid];
};
socket.on("message", data => {
const { message, meta, room } = data;
if(meta === "join") {
if(! rooms[room]) rooms[room] = {}; // create the room
if(! rooms[room][uuid]) rooms[room][uuid] = socket; // join the room
}
else if(meta === "leave") {
leave(room);
}
else if(! meta) {
// send the message to all in the room
Object.entries(rooms[room]).forEach(([, sock]) => sock.send({ message }));
}
});
socket.on("close", () => {
// for each room, remove the closed socket
Object.keys(rooms).forEach(room => leave(room));
});
});
这只是一个草图:您需要处理离开房间、断开与客户端的连接(离开所有房间)以及当没人再进来时删除房间。
const port = process.env.PORT || 8000
const WebSocket = require('ws');
function noop() {}
function heartbeat() {
this.isAlive = true;
}
const wss = new WebSocket.Server({ port });
var rooms = {};
const paramsExist = (data) =>{
try {
if('meta' in data && 'roomID' in data && 'clientID' in data && 'message' in data){
return true;
}else{
return false;
}
} catch (error) {
return false;
}
}
const roomExist = (roomID) =>{
// check for room is already exist or not
if(roomID in rooms){
return true;
}else{
return false;
}
}
const insideRoomdataExist = (arr,data) =>{
var status = false;
for(var i =0; i<arr.length;i++){
if(data in arr[i]){
status= true;
break;
}
}
return status;
}
const clientExistInRoom = (roomID,ws,clientID) =>{
var status = false;
const data = rooms[roomID];
for(var i =0; i< data.length ;i++){
var temp = data[i];
// if(roomID in temp){
// status=true;
// console.log("hello world");
// }
for(const obj in temp){
// if(ws == temp[obj]){
if(clientID == obj){
status = true;
break;
}
}
}return status;
}
// create room
const createRoom =(data,ws)=>{
try {
var {roomID,clientID} = data;
const status = roomExist(roomID);
if(status){
ws.send(JSON.stringify({
'message':'room already exist',
'status':0
}));
}else{
rooms[roomID] = [];
var obj = {};
obj[clientID] = ws;
rooms[roomID].push(obj);
ws['roomID']=roomID;
ws['clientID']=clientID;
ws['admin']=true;
ws.send(JSON.stringify({
'message':'room created succesfully',
'status':1
}));
}
} catch (error) {
ws.send(JSON.stringify({
'message':'there was some problem in creating a room',
'status':0
}));
}
}
// join room
const joinRoom = (data,ws) => {
try {
var {roomID,clientID} = data;
// check if room exist or not
const roomExist = roomID in rooms;
if(!roomExist){
ws.send(JSON.stringify({
'message':'Check room id',
'status':0
}));
return;
}
// const inRoom = insideRoomdataExist(rooms[roomID],clientID);
const inRoom = clientExistInRoom(roomID,ws,clientID)
if(inRoom){
ws.send(JSON.stringify({
"message":"you are already in a room",
"status":0
}));
}else{
var obj = {};
obj[clientID] = ws;
rooms[roomID].push(obj);
ws['roomID']=roomID
ws['clientID']=clientID;
ws.send(JSON.stringify({
"message":"Joined succesfully",
"status":1
}));
}
} catch (error) {
ws.send(JSON.stringify({
'message':'there was some problem in joining a room',
'status':0
}));
}
}
// send message
const sendMessage = (data,ws,Status=null) => {
try {
var {roomID, message,clientID} = data;
//check whether room exist or not
const roomExist = roomID in rooms;
if(!roomExist){
ws.send(JSON.stringify({
'message':'Check room id',
'status':0
}));
return;
}
// check whether client is in room or not
const clientExist = clientExistInRoom(roomID,ws,clientID);
if(!clientExist){
ws.send(JSON.stringify({
'message':"You are not allowed to send message",
'status':0
}));
return;
}
const obj = rooms[roomID];
for(i=0;i<obj.length;i++){
var temp = obj[i];
for(var innerObject in temp){
var wsClientID = temp[innerObject];
if(ws!==wsClientID){
wsClientID.send(JSON.stringify({
'message':message,
'status':Status?Status:1
}));
}
}
}
} catch (error) {
ws.send(JSON.stringify({
'message':'There was some problem in sending message',
'status':0
}));
}
}
const leaveRoom = (ws,data) => {
try {
const {roomID} = data;
// manual code started------------------------------------------------------------
const roomExist = roomID in rooms;
if(!roomExist){
ws.send(JSON.stringify({
'message':'Check room id',
'status':0
}));
return;
}
if('admin' in ws){
data['message']="Admin left the room.";
sendMessage(data,ws,Status=2);
delete rooms[ws.roomID]
return;
}
else{
// find the index of object
lst_obj = rooms[roomID];
var index = null;
for(let i=0;i<lst_obj.length;i++){
var temp_obj = lst_obj[i];
for(var key in temp_obj){
var temp_inside = temp_obj[key]
if('admin' in temp_inside){
temp_inside.send(JSON.stringify({
'message':'Somebody leave the room',
'status':3
}));
}
if(ws==temp_inside){
index =i;
}
}
}
if(index!=null){
rooms[roomID].splice(index,1);
console.log((rooms[roomID].length));
}
}
} catch (error) {
ws.send(JSON.stringify({
'message':'There was some problem----------------------',
'status':0
}))
}
}
const available_room = (ws) =>{
try {
var available_room_id=[];
for(var i in rooms){
available_room_id.push(parseInt(i));
}
ws.send(JSON.stringify({
"rooms":available_room_id,
"status":4
}))
} catch (error) {
ws.send(JSON.stringify({
'message':'There was some problem----------------------',
'status':0
}))
}
}
wss.on('connection', function connection(ws) {
try {
ws.on('message',(recieveData)=>{
var data = JSON.parse(recieveData);
const error = paramsExist(data);
if(!error){
ws.send(JSON.stringify({
'message':'check params',
'status':0
}));
return;
}
var {roomID,meta} = data;
switch (meta) {
case "create_room":
createRoom(data,ws);
console.log(rooms);
break;
case "join_room":
joinRoom(data,ws);
console.log(rooms);
break;
case "send_message":
sendMessage(data,ws);
console.log(rooms);
break;
case "show_all_rooms":
ws.send(JSON.stringify({
"rooms":[rooms]
}))
break;
default:
ws.send(JSON.stringify({
"message":"Unsupported meta data provided provide valid data",
"status":0
}));
break;
}
})
ws.on('close', function(data) {
leaveRoom(ws,{roomID:ws.roomID,clientID:ws.clientID,message:"Leave request"})
ws.terminate();
});
ws.on('pong', heartbeat);
} catch (error) {
ws.send(JSON.stringify({
"message":"there was some problem",
"status":0
}))
}
});
const interval = setInterval(function ping() {
var a = wss.clients;
wss.clients.forEach(function each(ws) {
if (ws.isAlive === false) {
leaveRoom(ws,{roomID:ws.roomID,clientID:ws.clientID});
ws.terminate();
}
ws.isAlive = false;
ws.ping(noop);
});
}, 50000);
const serverFree = setInterval(()=>{
var removeKey = [];
for(const obj in rooms){
if(rooms[obj].length<1){
removeKey.push(obj);
}
}
for(var i =0; i<removeKey.length;i++){
delete rooms[removeKey[i]];
}
},30000)
您好,上面是使用was进行socket连接的代码。您需要做的就是在元键中从前端提供 JSON 。您需要使用套接字库从 JSON 中的前端发送 4 个参数。
必填键 -meta:"create_room"/"join_room"/"send_message"/"show_all_rooms" -消息:“任何东西” -roomID:“创建或加入的roomid” -clientID:“唯一客户端 ID”
package.json 文件依赖项
{
"name": "nodesocket",
"version": "1.0.0",
"description": "",
"main": "app.js",
"scripts": {
"test": "echo \"Error: no test specified\" && exit 1",
"start": "node src/app.js"
},
"author": "",
"license": "ISC",
"dependencies": {
"cors": "^2.8.5",
"express": "^4.17.1",
"ws": "^7.4.6"
}
}
我建议使用存储频道和 WS 数组的 Map 而不是注册用户房间的数组。
如果你有很多用户,这将是一种更好的发送消息的方法(我为我的公司实现了这些类型的 WS)可以轻松处理 40k 到 50k(对于 WS 的每个节点)并发送数千个每秒消息数。
您可以轻松地广播消息(只需从您的地图中获取一个房间)。
地图是这样的:Map<String,[]> (room, ws array)
.
当用户加入一个新房间时,只需将它添加到房间的数组中,当他离开时将其删除。 (/!\ 不要忘记双向 ping 用户)
要向房间发送消息,只需调用(避免在未使用 map.has(channel)
检查房间是否存在的情况下调用它)。
map.get(room).forEach(ws => ws.send(yourMessage))
它还帮助我进行了线性缩放。这个数组在我的 wss
和 ws
之外,所以当从 REDIS(或其他任何东西)发布消息时,我只是放置一个回调,将消息发送到相应的房间。所以当你广播消息时它不会检查所有用户的房间。