mirror of
https://gitlab.com/MrFry/mrfrys-node-server
synced 2025-04-01 20:24:18 +02:00
Added chat submodule and socketAuth middleware
This commit is contained in:
parent
a12aadf32d
commit
106bd88f17
2 changed files with 266 additions and 0 deletions
54
src/middlewares/socketAuth.middleware.ts
Normal file
54
src/middlewares/socketAuth.middleware.ts
Normal file
|
@ -0,0 +1,54 @@
|
|||
import logger from '../utils/logger'
|
||||
import dbtools from '../utils/dbtools'
|
||||
import cookie from 'cookie'
|
||||
|
||||
interface Options {
|
||||
userDB: any
|
||||
}
|
||||
|
||||
export default function (options: Options): any {
|
||||
const { userDB } = options
|
||||
|
||||
return function (socket, next) {
|
||||
const cookies = cookie.parse(socket.handshake.headers.cookie)
|
||||
const sessionID = cookies.sessionID
|
||||
|
||||
if (process.env.NS_NOUSER) {
|
||||
next()
|
||||
return
|
||||
}
|
||||
|
||||
if (!sessionID) {
|
||||
next(new Error('Authentication error'))
|
||||
return
|
||||
}
|
||||
|
||||
const user = GetUserBySessionID(userDB, sessionID)
|
||||
|
||||
if (!user) {
|
||||
next(new Error('Authentication error'))
|
||||
return
|
||||
}
|
||||
next()
|
||||
}
|
||||
}
|
||||
|
||||
function GetUserBySessionID(db: any, sessionID: string) {
|
||||
logger.DebugLog(`Getting user from db`, 'auth', 2)
|
||||
|
||||
const session = dbtools.Select(db, 'sessions', {
|
||||
id: sessionID,
|
||||
})[0]
|
||||
|
||||
if (!session) {
|
||||
return
|
||||
}
|
||||
|
||||
const user = dbtools.Select(db, 'users', {
|
||||
id: session.userID,
|
||||
})[0]
|
||||
|
||||
if (user) {
|
||||
return user
|
||||
}
|
||||
}
|
212
src/modules/api/submodules/chat.ts
Normal file
212
src/modules/api/submodules/chat.ts
Normal file
|
@ -0,0 +1,212 @@
|
|||
import { Server as socket } from 'socket.io'
|
||||
|
||||
import dbtools from '../../../utils/dbtools'
|
||||
import logger from '../../../utils/logger'
|
||||
import { Request, SubmoduleData } from '../../../types/basicTypes'
|
||||
import socketAuth from '../../../middlewares/socketAuth.middleware'
|
||||
|
||||
const msgDbPath = './data/dbs/msgs.db'
|
||||
|
||||
function setup(data: SubmoduleData): void {
|
||||
const { app, httpServer, httpsServer, userDB } = data
|
||||
const msgDB = dbtools.GetDB(msgDbPath)
|
||||
|
||||
logger.Log(`Starting Socket.io Server on ${httpsServer ? 'https' : 'http'}`)
|
||||
// https://socket.io/docs/v4/handling-cors/#Configuration
|
||||
const io = new socket(httpsServer || httpServer, {
|
||||
cors: {
|
||||
credentials: true,
|
||||
origin: true,
|
||||
},
|
||||
})
|
||||
|
||||
function chatMessageRead(data) {
|
||||
const { sender, reciever } = data
|
||||
dbtools.runStatement(
|
||||
msgDB,
|
||||
`update msgs
|
||||
set unread = 0
|
||||
where sender = ${sender} and reciever = ${reciever}`,
|
||||
'run'
|
||||
)
|
||||
io.sockets.in(sender).emit('chat message read', {
|
||||
userReadMsg: reciever,
|
||||
})
|
||||
}
|
||||
|
||||
io.use(socketAuth({ userDB: userDB }))
|
||||
|
||||
io.on('connection', (socket) => {
|
||||
// TODO
|
||||
// https://stackoverflow.com/questions/19106861/authorizing-and-handshaking-with-socket-io#19106961
|
||||
// socket.handshake.headers
|
||||
// if (data.id !== user.id) {
|
||||
// socket.disconnect()
|
||||
// socket.emit('connectData', {
|
||||
// success: false,
|
||||
// msg: `You are only authorized to connect as ${user.id}!`
|
||||
// })
|
||||
// return
|
||||
// }
|
||||
|
||||
socket.on('join', function (data) {
|
||||
socket.join(data.id)
|
||||
const userid = data.id
|
||||
let currUser = dbtools.Select(msgDB, 'users', {
|
||||
id: userid,
|
||||
})[0]
|
||||
if (!currUser) {
|
||||
currUser = {
|
||||
id: userid,
|
||||
}
|
||||
dbtools.Insert(msgDB, 'users', currUser)
|
||||
}
|
||||
|
||||
const groups = dbtools
|
||||
.runStatement(
|
||||
msgDB,
|
||||
`select * from
|
||||
(
|
||||
select sender as a
|
||||
from msgs
|
||||
where sender = ${userid} or reciever = ${userid}
|
||||
union
|
||||
select reciever
|
||||
from msgs
|
||||
where sender = ${userid} or reciever = ${userid}
|
||||
)t
|
||||
order by t.a asc`
|
||||
)
|
||||
.reduce((acc, x) => {
|
||||
if (x.a !== userid) acc.push(x.a)
|
||||
return acc
|
||||
}, [])
|
||||
|
||||
socket.emit('prev messages', {
|
||||
prevMsgs: groups.map((to) => {
|
||||
const first = dbtools.runStatement(
|
||||
msgDB,
|
||||
`select * from msgs
|
||||
where sender = ${userid} and reciever = ${to} or
|
||||
sender = ${to} and reciever = ${userid}
|
||||
order by date desc
|
||||
limit 1`
|
||||
)[0]
|
||||
return first
|
||||
}),
|
||||
})
|
||||
|
||||
socket.on('chat message open', (data) => {
|
||||
const { chatPartner } = data
|
||||
|
||||
// TODO: pagination
|
||||
socket.emit(
|
||||
'chat message open',
|
||||
dbtools.runStatement(
|
||||
msgDB,
|
||||
`select * from msgs
|
||||
where sender = ${userid} and reciever = ${chatPartner} or
|
||||
sender = ${chatPartner} and reciever = ${userid}
|
||||
order by date asc`
|
||||
)
|
||||
)
|
||||
|
||||
// Read update
|
||||
chatMessageRead({ sender: chatPartner, reciever: userid })
|
||||
})
|
||||
|
||||
socket.on('chat message read', (data) => {
|
||||
const { chatPartner } = data
|
||||
chatMessageRead({ sender: chatPartner, reciever: userid })
|
||||
})
|
||||
|
||||
socket.on('chat message', (message) => {
|
||||
const { reciever, msg } = message
|
||||
const recieverUser = dbtools.Select(msgDB, 'users', {
|
||||
id: reciever,
|
||||
})[0]
|
||||
if (!recieverUser) {
|
||||
socket.emit('chat message', {
|
||||
success: false,
|
||||
date: new Date().getTime(),
|
||||
sender: reciever,
|
||||
reciever: userid,
|
||||
msg: `A #${reciever} számú felhasználó nem létezik`,
|
||||
})
|
||||
return
|
||||
}
|
||||
|
||||
const msgObj = {
|
||||
sender: parseInt(userid),
|
||||
reciever: parseInt(reciever),
|
||||
msg: msg,
|
||||
date: new Date().getTime(),
|
||||
unread: 1,
|
||||
}
|
||||
dbtools.Insert(msgDB, 'msgs', msgObj)
|
||||
if (userid !== reciever) {
|
||||
io.sockets.in(reciever).emit('chat message', msgObj)
|
||||
}
|
||||
})
|
||||
})
|
||||
|
||||
// socket.on('disconnect', () => {})
|
||||
})
|
||||
|
||||
app.get('/hasNewMsg', (req: Request, res) => {
|
||||
let userid: any = req.query.userid
|
||||
if (!userid || isNaN(userid)) {
|
||||
res.json({
|
||||
success: false,
|
||||
msg: 'Query "userid" (number) is required!',
|
||||
})
|
||||
return
|
||||
}
|
||||
userid = parseInt(userid)
|
||||
|
||||
const groups = dbtools
|
||||
.runStatement(
|
||||
msgDB,
|
||||
`select * from
|
||||
(
|
||||
select sender as a
|
||||
from msgs
|
||||
where sender = ${userid} or reciever = ${userid}
|
||||
union
|
||||
select reciever
|
||||
from msgs
|
||||
where sender = ${userid} or reciever = ${userid}
|
||||
)t
|
||||
order by t.a asc`
|
||||
)
|
||||
.reduce((acc, x) => {
|
||||
if (x.a !== userid) acc.push(x.a)
|
||||
return acc
|
||||
}, [])
|
||||
|
||||
const prevMsgs = groups.map((to) => {
|
||||
const first = dbtools.runStatement(
|
||||
msgDB,
|
||||
`select * from msgs
|
||||
where sender = ${userid} and reciever = ${to} or
|
||||
sender = ${to} and reciever = ${userid}
|
||||
order by date desc
|
||||
limit 1`
|
||||
)[0]
|
||||
return first
|
||||
})
|
||||
|
||||
res.json({
|
||||
unreads: prevMsgs.reduce((acc, msg) => {
|
||||
if (msg && msg.unread === 1 && msg.sender !== userid) {
|
||||
acc.push(msg.sender)
|
||||
}
|
||||
return acc
|
||||
}, []),
|
||||
})
|
||||
})
|
||||
}
|
||||
|
||||
export default {
|
||||
setup: setup,
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue