296 lines
10 KiB
TypeScript
296 lines
10 KiB
TypeScript
import { writable } from 'svelte/store'
|
|
import {
|
|
Message,
|
|
CorrelationRequest,
|
|
CorrelationResponse,
|
|
protoMetadata,
|
|
type MessageFns
|
|
} from '$lib/platform_shared/message'
|
|
import * as Messages from '$lib/platform_shared/message'
|
|
|
|
export const MESSAGE_TYPE_TO_KEY = new Map<MessageFns<unknown>, string>()
|
|
export const MESSAGE_TYPE_TO_TAG = new Map<MessageFns<unknown>, number>()
|
|
export const MESSAGE_KEY_TO_TAG = new Map<string, number>()
|
|
export const MESSAGE_TAG_TO_KEY = new Map<number, string>()
|
|
|
|
type CorrelationRequestData = Omit<CorrelationRequest, 'correlationId'>
|
|
type PendingRequest = {
|
|
resolve: (response: CorrelationResponse) => void
|
|
reject: (error: Error) => void
|
|
timeoutId: ReturnType<typeof setTimeout>
|
|
}
|
|
|
|
const MessageType = protoMetadata.fileDescriptor.messageType?.find(
|
|
(msg: { name: string }) => msg.name === 'Message'
|
|
)
|
|
|
|
if (MessageType?.field) {
|
|
for (const field of MessageType.field) {
|
|
if (field.typeName) {
|
|
const messageFns = protoMetadata.references[field.typeName]
|
|
if (messageFns && field.jsonName && field.number) {
|
|
MESSAGE_TYPE_TO_KEY.set(messageFns, field.jsonName)
|
|
MESSAGE_TYPE_TO_TAG.set(messageFns, field.number)
|
|
MESSAGE_KEY_TO_TAG.set(field.jsonName, field.number)
|
|
MESSAGE_TAG_TO_KEY.set(field.number, field.jsonName)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
function getNameFromMessageType<T>(event_type: MessageFns<T>): string {
|
|
const event = MESSAGE_TYPE_TO_KEY.get(event_type as MessageFns<unknown>)
|
|
if (!event) {
|
|
throw new Error(
|
|
"Event type not found in 'Message'. The MessageFns you passed doesn't correspond to any Message field."
|
|
)
|
|
}
|
|
return event
|
|
}
|
|
|
|
function getTagFromMessageType<T>(event_type: MessageFns<T>): number {
|
|
const fieldNumber = MESSAGE_TYPE_TO_TAG.get(event_type as MessageFns<unknown>)
|
|
if (fieldNumber === undefined) {
|
|
throw new Error(
|
|
"Tag not found in 'Message'. The MessageFns you passed doesn't correspond to any Message field."
|
|
)
|
|
}
|
|
return fieldNumber
|
|
}
|
|
|
|
type SocketEvent = 'open' | 'close' | 'error' | 'message' | 'unresponsive'
|
|
|
|
type TaggedMessage = { tag: number; msg: Message }
|
|
|
|
export const decodeMessage = (data: ArrayBuffer): TaggedMessage => {
|
|
const decoded = Message.decode(new Uint8Array(data))
|
|
const values = Object.entries(decoded).filter(([, value]) => value !== undefined)
|
|
if (values.length != 1) {
|
|
throw new Error('Message included either 0 or more than 1 data point')
|
|
}
|
|
const fieldName = values[0][0]
|
|
const tag = MESSAGE_KEY_TO_TAG.get(fieldName)
|
|
if (tag === undefined) {
|
|
throw new Error(`Tag not found for field: ${fieldName}`)
|
|
}
|
|
return { tag: tag, msg: decoded }
|
|
}
|
|
|
|
export const encodeMessage = (data: Message): Uint8Array<ArrayBuffer> => {
|
|
const encoded = Message.encode(data).finish()
|
|
return encoded
|
|
}
|
|
|
|
function createWebSocket() {
|
|
const message_listeners = new Map<number, Set<(data?: unknown) => void>>()
|
|
const event_listeners = new Map<string, Set<(data?: unknown) => void>>()
|
|
const pending_requests = new Map<number, PendingRequest>()
|
|
const queued_requests = new Map<
|
|
string,
|
|
{
|
|
data: CorrelationRequestData
|
|
resolve: (r: CorrelationResponse) => void
|
|
reject: (e: Error) => void
|
|
}
|
|
>()
|
|
const { subscribe, set } = writable(false)
|
|
const reconnectTimeoutTime = 500000
|
|
const requestTimeoutTime = 10000
|
|
let correlationIdCounter = 0
|
|
let unresponsiveTimeoutId: ReturnType<typeof setTimeout>
|
|
let reconnectTimeoutId: ReturnType<typeof setTimeout>
|
|
let ws: WebSocket
|
|
let socketUrl: string | URL
|
|
|
|
function getRequestKey(data: CorrelationRequestData): string {
|
|
return (
|
|
Object.keys(data).find(k => data[k as keyof CorrelationRequestData] !== undefined) ??
|
|
'unknown'
|
|
)
|
|
}
|
|
|
|
function init(url: string | URL) {
|
|
socketUrl = url
|
|
connect()
|
|
}
|
|
|
|
function disconnect(reason: SocketEvent, event?: Event) {
|
|
ws.close()
|
|
set(false)
|
|
clearTimeout(unresponsiveTimeoutId)
|
|
clearTimeout(reconnectTimeoutId)
|
|
event_listeners.get(reason)?.forEach(listener => listener(event))
|
|
reconnectTimeoutId = setTimeout(connect, reconnectTimeoutTime)
|
|
}
|
|
|
|
function connect() {
|
|
ws = new WebSocket(socketUrl)
|
|
ws.binaryType = 'arraybuffer'
|
|
ws.onopen = ev => {
|
|
ping()
|
|
set(true)
|
|
clearTimeout(reconnectTimeoutId)
|
|
resubscribeAll()
|
|
flushQueuedRequests()
|
|
event_listeners.get('open')?.forEach(listener => listener(ev))
|
|
}
|
|
ws.onmessage = frame => {
|
|
resetUnresponsiveCheck()
|
|
const { tag, msg } = decodeMessage(frame.data)
|
|
if (msg.correlationResponse) {
|
|
const pending = pending_requests.get(msg.correlationResponse.correlationId)
|
|
if (pending) {
|
|
clearTimeout(pending.timeoutId)
|
|
pending_requests.delete(msg.correlationResponse.correlationId)
|
|
pending.resolve(msg.correlationResponse)
|
|
}
|
|
return
|
|
}
|
|
if (tag) {
|
|
const key = MESSAGE_TAG_TO_KEY.get(tag)!
|
|
message_listeners
|
|
.get(tag)
|
|
?.forEach(listener => listener(msg[key as keyof typeof msg]))
|
|
}
|
|
}
|
|
ws.onerror = ev => disconnect('error', ev)
|
|
ws.onclose = ev => disconnect('close', ev)
|
|
}
|
|
|
|
function unsubscribe<MT>(event_type: MessageFns<MT>, listener: (data: MT) => void) {
|
|
const tag = getTagFromMessageType(event_type)
|
|
const message_listeners_totag = message_listeners.get(tag)
|
|
if (!message_listeners_totag) return
|
|
|
|
message_listeners_totag?.delete(listener as (data?: unknown) => void)
|
|
if (message_listeners_totag.size == 0) {
|
|
unsubscribeToMessageFromServer(event_type)
|
|
}
|
|
}
|
|
|
|
function unsubscribe_event(event_type: SocketEvent, listener: (data: unknown) => void) {
|
|
const message_listeners_totag = event_listeners.get(event_type)
|
|
if (!message_listeners_totag) return
|
|
|
|
message_listeners_totag?.delete(listener)
|
|
}
|
|
|
|
function resetUnresponsiveCheck() {
|
|
clearTimeout(unresponsiveTimeoutId)
|
|
unresponsiveTimeoutId = setTimeout(() => disconnect('unresponsive'), reconnectTimeoutTime)
|
|
}
|
|
|
|
function sendEvent<T>(event: MessageFns<T>, data: T) {
|
|
if (!ws || ws.readyState !== WebSocket.OPEN) return
|
|
const type = getNameFromMessageType(event)
|
|
const wsm = Message.create() as Record<string, unknown>
|
|
wsm[type] = data
|
|
send(wsm as Message)
|
|
}
|
|
|
|
function unsubscribeToMessageFromServer<T>(event_type: MessageFns<T>) {
|
|
if (!ws || ws.readyState !== WebSocket.OPEN) return
|
|
const unsub_msg = Messages.UnsubscribeNotification.create({
|
|
tag: getTagFromMessageType(event_type)
|
|
})
|
|
send(Message.create({ unsubNotif: unsub_msg }))
|
|
}
|
|
|
|
function subscribeToEvent<T>(event_type: MessageFns<T>) {
|
|
if (!ws || ws.readyState !== WebSocket.OPEN) return
|
|
const sub_msg = Messages.SubscribeNotification.create({
|
|
tag: getTagFromMessageType(event_type)
|
|
})
|
|
send(Message.create({ subNotif: sub_msg }))
|
|
}
|
|
|
|
function resubscribeAll() {
|
|
for (const tag of message_listeners.keys()) {
|
|
const sub_msg = Messages.SubscribeNotification.create({ tag })
|
|
send(Message.create({ subNotif: sub_msg }))
|
|
}
|
|
}
|
|
|
|
function send(data: Message) {
|
|
if (!ws || ws.readyState !== WebSocket.OPEN) return
|
|
const encoded = encodeMessage(data)
|
|
ws.send(encoded)
|
|
}
|
|
|
|
function ping() {
|
|
send(Message.create({ pingmsg: {} }))
|
|
}
|
|
|
|
function sendRequest(
|
|
data: CorrelationRequestData,
|
|
resolve: (r: CorrelationResponse) => void,
|
|
reject: (e: Error) => void
|
|
) {
|
|
const correlationId = ++correlationIdCounter
|
|
const timeoutId = setTimeout(() => {
|
|
pending_requests.delete(correlationId)
|
|
reject(new Error(`Request timeout (id: ${correlationId})`))
|
|
}, requestTimeoutTime)
|
|
|
|
pending_requests.set(correlationId, { resolve, reject, timeoutId })
|
|
|
|
const request = CorrelationRequest.create({ correlationId, ...data })
|
|
send(Message.create({ correlationRequest: request }))
|
|
}
|
|
|
|
function flushQueuedRequests() {
|
|
for (const [, { data, resolve, reject }] of queued_requests) {
|
|
sendRequest(data, resolve, reject)
|
|
}
|
|
queued_requests.clear()
|
|
}
|
|
|
|
return {
|
|
subscribe,
|
|
sendEvent,
|
|
init,
|
|
on: <MT>(event_type: MessageFns<MT>, listener: (data: MT) => void): (() => void) => {
|
|
const tag = getTagFromMessageType(event_type)
|
|
|
|
let message_listeners_totag = message_listeners.get(tag)
|
|
if (!message_listeners_totag) {
|
|
message_listeners_totag = new Set()
|
|
message_listeners.set(tag, message_listeners_totag)
|
|
subscribeToEvent(event_type)
|
|
}
|
|
message_listeners_totag.add(listener as (data: unknown) => void)
|
|
|
|
return () => {
|
|
unsubscribe(event_type, listener)
|
|
}
|
|
},
|
|
onEvent: (event_type: SocketEvent, listener: (data: unknown) => void): (() => void) => {
|
|
let listeners = event_listeners.get(event_type)
|
|
if (!listeners) {
|
|
listeners = new Set()
|
|
event_listeners.set(event_type, listeners)
|
|
}
|
|
listeners.add(listener)
|
|
return () => {
|
|
unsubscribe_event(event_type, listener)
|
|
}
|
|
},
|
|
request: (data: CorrelationRequestData): Promise<CorrelationResponse> => {
|
|
return new Promise((resolve, reject) => {
|
|
if (ws && ws.readyState === WebSocket.OPEN) {
|
|
sendRequest(data, resolve, reject)
|
|
} else {
|
|
const key = getRequestKey(data)
|
|
const existing = queued_requests.get(key)
|
|
if (existing) {
|
|
existing.reject(new Error('Request superseded by newer request'))
|
|
}
|
|
queued_requests.set(key, { data, resolve, reject })
|
|
}
|
|
})
|
|
}
|
|
}
|
|
}
|
|
|
|
export const socket = createWebSocket()
|