2021-08-27 14:32:44 +02:00
|
|
|
import { Server as HTTPServer } from 'http'
|
|
|
|
import { Namespace, Server as SocketServer, Socket } from 'socket.io'
|
|
|
|
import { isIdValid } from '@server/helpers/custom-validators/misc'
|
2020-09-25 10:04:21 +02:00
|
|
|
import { MVideo } from '@server/types/models'
|
2020-06-18 10:45:25 +02:00
|
|
|
import { UserNotificationModelForApi } from '@server/types/models/user'
|
2020-09-25 10:04:21 +02:00
|
|
|
import { LiveVideoEventPayload, LiveVideoEventType } from '@shared/models'
|
|
|
|
import { logger } from '../helpers/logger'
|
|
|
|
import { authenticateSocket } from '../middlewares'
|
2018-12-26 10:36:24 +01:00
|
|
|
|
|
|
|
class PeerTubeSocket {
|
|
|
|
|
|
|
|
private static instance: PeerTubeSocket
|
|
|
|
|
2021-08-27 14:32:44 +02:00
|
|
|
private userNotificationSockets: { [ userId: number ]: Socket[] } = {}
|
|
|
|
private liveVideosNamespace: Namespace
|
2018-12-26 10:36:24 +01:00
|
|
|
|
|
|
|
private constructor () {}
|
|
|
|
|
2021-08-27 14:32:44 +02:00
|
|
|
init (server: HTTPServer) {
|
|
|
|
const io = new SocketServer(server)
|
2018-12-26 10:36:24 +01:00
|
|
|
|
|
|
|
io.of('/user-notifications')
|
|
|
|
.use(authenticateSocket)
|
|
|
|
.on('connection', socket => {
|
2021-03-03 15:22:38 +01:00
|
|
|
const userId = socket.handshake.auth.user.id
|
2018-12-26 10:36:24 +01:00
|
|
|
|
|
|
|
logger.debug('User %d connected on the notification system.', userId)
|
|
|
|
|
2019-08-22 10:33:22 +02:00
|
|
|
if (!this.userNotificationSockets[userId]) this.userNotificationSockets[userId] = []
|
|
|
|
|
|
|
|
this.userNotificationSockets[userId].push(socket)
|
2018-12-26 10:36:24 +01:00
|
|
|
|
|
|
|
socket.on('disconnect', () => {
|
|
|
|
logger.debug('User %d disconnected from SocketIO notifications.', userId)
|
|
|
|
|
2019-08-22 10:33:22 +02:00
|
|
|
this.userNotificationSockets[userId] = this.userNotificationSockets[userId].filter(s => s !== socket)
|
2018-12-26 10:36:24 +01:00
|
|
|
})
|
|
|
|
})
|
2020-09-25 10:04:21 +02:00
|
|
|
|
|
|
|
this.liveVideosNamespace = io.of('/live-videos')
|
|
|
|
.on('connection', socket => {
|
2020-11-04 15:31:32 +01:00
|
|
|
socket.on('subscribe', ({ videoId }) => {
|
|
|
|
if (!isIdValid(videoId)) return
|
|
|
|
|
2021-04-12 17:00:21 +02:00
|
|
|
/* eslint-disable @typescript-eslint/no-floating-promises */
|
2020-11-04 15:31:32 +01:00
|
|
|
socket.join(videoId)
|
|
|
|
})
|
|
|
|
|
|
|
|
socket.on('unsubscribe', ({ videoId }) => {
|
|
|
|
if (!isIdValid(videoId)) return
|
|
|
|
|
2021-04-12 17:00:21 +02:00
|
|
|
/* eslint-disable @typescript-eslint/no-floating-promises */
|
2020-11-04 15:31:32 +01:00
|
|
|
socket.leave(videoId)
|
|
|
|
})
|
2020-09-25 10:04:21 +02:00
|
|
|
})
|
2018-12-26 10:36:24 +01:00
|
|
|
}
|
|
|
|
|
2019-08-15 11:53:26 +02:00
|
|
|
sendNotification (userId: number, notification: UserNotificationModelForApi) {
|
2019-08-22 10:33:22 +02:00
|
|
|
const sockets = this.userNotificationSockets[userId]
|
|
|
|
if (!sockets) return
|
2018-12-26 10:36:24 +01:00
|
|
|
|
2020-09-25 10:04:21 +02:00
|
|
|
logger.debug('Sending user notification to user %d.', userId)
|
|
|
|
|
2019-08-23 09:05:30 +02:00
|
|
|
const notificationMessage = notification.toFormattedJSON()
|
2019-08-22 10:33:22 +02:00
|
|
|
for (const socket of sockets) {
|
2019-08-23 09:05:30 +02:00
|
|
|
socket.emit('new-notification', notificationMessage)
|
2019-08-22 10:33:22 +02:00
|
|
|
}
|
2018-12-26 10:36:24 +01:00
|
|
|
}
|
|
|
|
|
2020-09-25 10:04:21 +02:00
|
|
|
sendVideoLiveNewState (video: MVideo) {
|
|
|
|
const data: LiveVideoEventPayload = { state: video.state }
|
|
|
|
const type: LiveVideoEventType = 'state-change'
|
|
|
|
|
2020-12-09 15:00:02 +01:00
|
|
|
logger.debug('Sending video live new state notification of %s.', video.url, { state: video.state })
|
|
|
|
|
|
|
|
this.liveVideosNamespace
|
|
|
|
.in(video.id)
|
|
|
|
.emit(type, data)
|
|
|
|
}
|
|
|
|
|
|
|
|
sendVideoViewsUpdate (video: MVideo) {
|
|
|
|
const data: LiveVideoEventPayload = { views: video.views }
|
|
|
|
const type: LiveVideoEventType = 'views-change'
|
|
|
|
|
|
|
|
logger.debug('Sending video live views update notification of %s.', video.url, { views: video.views })
|
2020-09-25 10:04:21 +02:00
|
|
|
|
|
|
|
this.liveVideosNamespace
|
|
|
|
.in(video.id)
|
|
|
|
.emit(type, data)
|
|
|
|
}
|
|
|
|
|
2018-12-26 10:36:24 +01:00
|
|
|
static get Instance () {
|
|
|
|
return this.instance || (this.instance = new this())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// ---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
export {
|
|
|
|
PeerTubeSocket
|
|
|
|
}
|