mirror of
https://github.com/hcengineering/platform.git
synced 2025-04-12 03:06:11 +00:00
175 lines
4.6 KiB
TypeScript
175 lines
4.6 KiB
TypeScript
//
|
|
// Copyright © 2024 Hardcore Engineering Inc.
|
|
//
|
|
// Licensed under the Eclipse Public License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License. You may
|
|
// obtain a copy of the License at https://www.eclipse.org/legal/epl-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
//
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
//
|
|
|
|
import { type Ref, concatLink, getCurrentAccount } from '@hcengineering/core'
|
|
import { type Person, type PersonAccount } from '@hcengineering/contact'
|
|
import { getMetadata } from '@hcengineering/platform'
|
|
import presence from '@hcengineering/presence'
|
|
import presentation from '@hcengineering/presentation'
|
|
import { type Unsubscriber, get } from 'svelte/store'
|
|
|
|
import { myPresence, onPersonUpdate, onPersonLeave } from './store'
|
|
import { type RoomPresence } from './types'
|
|
|
|
interface Message {
|
|
id: Ref<Person>
|
|
type: 'update' | 'remove'
|
|
presence?: RoomPresence[]
|
|
lastUpdate?: number
|
|
}
|
|
|
|
export class PresenceClient implements Disposable {
|
|
private ws: WebSocket | null = null
|
|
private closed = false
|
|
private reconnectTimeout: number | undefined
|
|
private readonly reconnectInterval = 1000
|
|
|
|
private presence: RoomPresence[]
|
|
private readonly myPresenceUnsub: Unsubscriber
|
|
|
|
constructor (private readonly url: string | URL) {
|
|
this.presence = get(myPresence)
|
|
this.myPresenceUnsub = myPresence.subscribe((presence) => {
|
|
this.handlePresenceChanged(presence)
|
|
})
|
|
|
|
this.connect()
|
|
}
|
|
|
|
close (): void {
|
|
this.closed = true
|
|
clearTimeout(this.reconnectTimeout)
|
|
|
|
this.myPresenceUnsub()
|
|
|
|
if (this.ws !== null) {
|
|
this.ws.close()
|
|
this.ws = null
|
|
}
|
|
}
|
|
|
|
private connect (): void {
|
|
try {
|
|
const ws = new WebSocket(this.url)
|
|
this.ws = ws
|
|
|
|
ws.onopen = () => {
|
|
if (this.ws !== ws) {
|
|
return
|
|
}
|
|
|
|
this.handleConnect()
|
|
}
|
|
|
|
ws.onclose = (event: CloseEvent) => {
|
|
if (this.ws !== ws) {
|
|
ws.close()
|
|
return
|
|
}
|
|
|
|
this.reconnect()
|
|
}
|
|
|
|
ws.onmessage = (event: MessageEvent) => {
|
|
if (this.closed || this.ws !== ws) {
|
|
return
|
|
}
|
|
|
|
this.handleMessage(event.data)
|
|
}
|
|
|
|
ws.onerror = (event: Event) => {
|
|
if (this.ws !== ws) {
|
|
return
|
|
}
|
|
|
|
console.log('client websocket error', event)
|
|
}
|
|
} catch (err: any) {
|
|
this.reconnect()
|
|
}
|
|
}
|
|
|
|
private reconnect (): void {
|
|
clearTimeout(this.reconnectTimeout)
|
|
|
|
if (!this.closed) {
|
|
this.reconnectTimeout = window.setTimeout(() => {
|
|
this.connect()
|
|
}, this.reconnectInterval)
|
|
}
|
|
}
|
|
|
|
private handleConnect (): void {
|
|
const me = getCurrentAccount() as PersonAccount
|
|
this.sendPresence(me.person, this.presence)
|
|
}
|
|
|
|
private handleMessage (data: string): void {
|
|
try {
|
|
const message = JSON.parse(data) as Message
|
|
if (message.type === 'update' && message.presence !== undefined) {
|
|
onPersonUpdate(message.id, message.presence ?? [])
|
|
} else if (message.type === 'remove') {
|
|
onPersonLeave(message.id)
|
|
} else {
|
|
console.warn('Unknown message type', message)
|
|
}
|
|
} catch (err: any) {
|
|
console.error('Error parsing message', err, data)
|
|
}
|
|
}
|
|
|
|
private handlePresenceChanged (presence: RoomPresence[]): void {
|
|
const me = getCurrentAccount() as PersonAccount
|
|
this.presence = presence
|
|
this.sendPresence(me.person, this.presence)
|
|
}
|
|
|
|
private sendPresence (person: Ref<Person>, presence: RoomPresence[]): void {
|
|
if (!this.closed && this.ws !== null && this.ws.readyState === WebSocket.OPEN) {
|
|
const message: Message = { id: person, type: 'update', presence }
|
|
this.ws.send(JSON.stringify(message))
|
|
}
|
|
}
|
|
|
|
[Symbol.dispose] (): void {
|
|
this.close()
|
|
}
|
|
}
|
|
|
|
export function connect (): PresenceClient | undefined {
|
|
const workspaceId = getMetadata(presentation.metadata.WorkspaceId)
|
|
if (workspaceId === undefined) {
|
|
console.warn('Workspace ID is not defined')
|
|
return undefined
|
|
}
|
|
|
|
const token = getMetadata(presentation.metadata.Token)
|
|
|
|
const presenceUrl = getMetadata(presence.metadata.PresenceUrl)
|
|
if (presenceUrl === undefined || presenceUrl === '') {
|
|
console.warn('Presence URL is not defined')
|
|
return undefined
|
|
}
|
|
|
|
const url = new URL(concatLink(presenceUrl, workspaceId))
|
|
if (token !== undefined) {
|
|
url.searchParams.set('token', token)
|
|
}
|
|
|
|
return new PresenceClient(url)
|
|
}
|