navidrome/ui/src/eventStream.js
Deluan 7106add40c
Removed event.type from SSE, as it was causing the browser to hang.
Needs more investigation, but for now, back to the simple message format
2026-01-02 19:20:16 +00:00

89 lines
2.1 KiB
JavaScript

import { baseUrl } from './utils'
import throttle from 'lodash.throttle'
import { processEvent, serverDown } from './actions'
import { httpClient } from './dataProvider'
import { REST_URL } from './consts'
const defaultIntervalCheck = 20000
const reconnectIntervalCheck = 2000
let currentIntervalCheck = reconnectIntervalCheck
let es = null
let dispatch = null
let timeout = null
const getEventStream = async () => {
if (!es) {
// Call `keepalive` to refresh the jwt token
await httpClient(`${REST_URL}/keepalive/eventSource`)
es = new EventSource(
baseUrl(`${REST_URL}/events?jwt=${localStorage.getItem('token')}`)
)
}
return es
}
// Reestablish the event stream after 20 secs of inactivity
const setTimeout = (value) => {
currentIntervalCheck = value
if (timeout) {
window.clearTimeout(timeout)
}
timeout = window.setTimeout(async () => {
if (es) {
es.close()
}
es = null
await startEventStream()
}, currentIntervalCheck)
}
const stopEventStream = () => {
if (es) {
es.close()
}
es = null
if (timeout) {
window.clearTimeout(timeout)
}
timeout = null
}
const setDispatch = (dispatchFunc) => {
dispatch = dispatchFunc
}
const eventHandler = throttle(
(event) => {
const data = JSON.parse(event.data)
if (data.name !== 'keepAlive') {
dispatch(processEvent(data.name, data))
}
setTimeout(defaultIntervalCheck) // Reset timeout on every received message
},
100,
{ trailing: true }
)
const startEventStream = async () => {
setTimeout(currentIntervalCheck)
if (!localStorage.getItem('token')) {
console.log('Cannot create a unauthenticated EventSource connection')
return Promise.reject()
}
return getEventStream()
.then((newStream) => {
newStream.onmessage = eventHandler
newStream.onerror = (e) => {
console.log('EventStream error', e)
setTimeout(reconnectIntervalCheck)
dispatch(serverDown())
}
return newStream
})
.catch((e) => {
console.log(`Error connecting to server:`, e)
})
}
export { setDispatch, startEventStream, stopEventStream }