2022-05-11 23:31:31 -07:00
import { MessageType , SocketEvent } from '../interfaces/socket-events' ;
2022-05-02 22:13:36 -07:00
2022-05-11 23:31:31 -07:00
export interface SocketMessage {
2022-05-03 14:17:05 -07:00
type : MessageType ;
2022-05-02 17:45:22 -07:00
data : any ;
}
export default class WebsocketService {
websocket : WebSocket ;
accessToken : string ;
2023-03-13 15:23:14 -07:00
host : string ;
2022-05-02 17:45:22 -07:00
path : string ;
websocketReconnectTimer : ReturnType < typeof setTimeout > ;
2022-10-18 16:39:49 -07:00
isShutdown = false ;
backOff = 1000 ;
2022-05-11 23:31:31 -07:00
handleMessage ? : ( message : SocketEvent ) = > void ;
2022-05-02 22:13:36 -07:00
2023-03-13 15:23:14 -07:00
socketConnected : ( ) = > void ;
socketDisconnected : ( ) = > void ;
2022-10-18 20:40:57 -07:00
constructor ( accessToken , path , host ) {
2022-05-02 17:45:22 -07:00
this . accessToken = accessToken ;
2022-05-02 22:13:36 -07:00
this . path = path ;
2022-10-18 16:39:49 -07:00
this . websocketReconnectTimer = null ;
2022-10-18 20:40:57 -07:00
this . isShutdown = false ;
2023-03-13 15:23:14 -07:00
this . host = host ;
2022-05-02 17:45:22 -07:00
2022-10-18 16:39:49 -07:00
this . createAndConnect = this . createAndConnect . bind ( this ) ;
this . shutdown = this . shutdown . bind ( this ) ;
2022-05-02 17:45:22 -07:00
2023-03-13 15:23:14 -07:00
this . createAndConnect ( ) ;
2022-05-02 17:45:22 -07:00
}
2023-03-13 15:23:14 -07:00
createAndConnect() {
if ( ! this . host ) {
return ;
}
const url = new URL ( this . host ) ;
2022-10-09 21:16:46 -07:00
url . protocol = window . location . protocol === 'https:' ? 'wss:' : 'ws:' ;
url . pathname = '/ws' ;
url . port = window . location . port === '3000' ? '8080' : window . location . port ;
2022-05-02 17:45:22 -07:00
url . searchParams . append ( 'accessToken' , this . accessToken ) ;
2022-05-28 18:43:28 -07:00
console . debug ( 'connecting to ' , url . toString ( ) ) ;
2022-05-02 17:45:22 -07:00
const ws = new WebSocket ( url . toString ( ) ) ;
ws . onopen = this . onOpen . bind ( this ) ;
ws . onerror = this . onError . bind ( this ) ;
ws . onmessage = this . onMessage . bind ( this ) ;
this . websocket = ws ;
}
onOpen() {
if ( this . websocketReconnectTimer ) {
clearTimeout ( this . websocketReconnectTimer ) ;
}
2023-03-13 15:23:14 -07:00
this . socketConnected ( ) ;
2022-05-02 17:45:22 -07:00
}
// On ws error just close the socket and let it re-connect again for now.
2023-03-13 15:23:14 -07:00
onError() {
handleNetworkingError ( ) ;
this . socketDisconnected ( ) ;
2022-05-02 17:45:22 -07:00
this . websocket . close ( ) ;
2022-10-18 16:39:49 -07:00
if ( ! this . isShutdown ) {
this . scheduleReconnect ( ) ;
}
}
scheduleReconnect() {
if ( this . websocketReconnectTimer ) {
clearTimeout ( this . websocketReconnectTimer ) ;
}
this . backOff *= 2 ;
this . websocketReconnectTimer = setTimeout (
this . createAndConnect ,
5000 + Math . min ( this . backOff , 10 _000 ) ,
) ;
}
shutdown() {
this . isShutdown = true ;
this . websocket . close ( ) ;
2022-05-02 17:45:22 -07:00
}
/ *
onMessage is fired when an inbound object comes across the websocket .
If the message is of type ` PING ` we send a ` PONG ` back and do not
pass it along to listeners .
* /
onMessage ( e : SocketMessage ) {
// Optimization where multiple events can be sent within a
// single websocket message. So split them if needed.
const messages = e . data . split ( '\n' ) ;
2022-05-26 13:52:04 -07:00
let socketEvent : SocketEvent ;
2022-05-02 17:45:22 -07:00
// eslint-disable-next-line no-plusplus
for ( let i = 0 ; i < messages . length ; i ++ ) {
try {
2022-05-26 13:52:04 -07:00
socketEvent = JSON . parse ( messages [ i ] ) ;
2022-05-02 22:13:36 -07:00
if ( this . handleMessage ) {
2022-05-26 13:52:04 -07:00
this . handleMessage ( socketEvent ) ;
2022-05-02 22:13:36 -07:00
}
2022-10-09 21:16:46 -07:00
} catch ( err ) {
console . error ( err , err . data ) ;
2022-05-02 17:45:22 -07:00
return ;
}
2022-05-26 13:52:04 -07:00
if ( ! socketEvent . type ) {
console . error ( 'No type provided' , socketEvent ) ;
2022-05-02 17:45:22 -07:00
return ;
}
// Send PONGs
2022-05-26 13:52:04 -07:00
if ( socketEvent . type === MessageType . PING ) {
2022-05-02 17:45:22 -07:00
this . sendPong ( ) ;
return ;
}
}
}
2022-05-26 13:52:04 -07:00
isConnected ( ) : boolean {
return this . websocket ? . readyState === this . websocket ? . OPEN ;
}
2022-05-02 17:45:22 -07:00
// Outbound: Other components can pass an object to `send`.
2022-05-26 13:52:04 -07:00
send ( socketEvent : any ) {
2022-05-02 17:45:22 -07:00
// Sanity check that what we're sending is a valid type.
2022-05-26 13:52:04 -07:00
if ( ! socketEvent . type || ! MessageType [ socketEvent . type ] ) {
console . warn ( ` Outbound message: Unknown socket message type: " ${ socketEvent . type } " sent. ` ) ;
2022-05-02 17:45:22 -07:00
}
2022-05-26 13:52:04 -07:00
const messageJSON = JSON . stringify ( socketEvent ) ;
2022-05-02 17:45:22 -07:00
this . websocket . send ( messageJSON ) ;
}
// Reply to a PING as a keep alive.
sendPong() {
2022-05-03 14:17:05 -07:00
const pong = { type : MessageType . PONG } ;
2022-05-02 17:45:22 -07:00
this . send ( pong ) ;
}
}
2023-03-13 15:23:14 -07:00
function handleNetworkingError() {
2022-05-02 17:45:22 -07:00
console . error (
2023-03-13 15:23:14 -07:00
` Chat has been disconnected and is likely not working for you. It's possible you were removed from chat. If this is a server configuration issue, visit troubleshooting steps to resolve. https://owncast.online/docs/troubleshooting/#chat-is-disabled ` ,
2022-05-02 17:45:22 -07:00
) ;
2022-05-02 22:13:36 -07:00
}