Skip to content
Snippets Groups Projects
websocket.service.ts 1.97 KiB
Newer Older
  • Learn to ignore specific revisions
  • paflov's avatar
    paflov committed
    /* eslint-disable @typescript-eslint/no-explicit-any */
    import { webSocket, WebSocketSubject } from 'rxjs/webSocket';
    import { BehaviorSubject, Observable, Subscription } from 'rxjs';
    import { map, share } from 'rxjs/operators';
    import { WebSocketMessage } from 'rxjs/internal/observable/dom/WebSocketSubject';
    
    
    interface WsMessage {
      event: string;
      data: any;
    }
    
    export class WebsocketService {
    
    paf's avatar
    paf committed
      protected wsUrl = '';
    
      private wsSubject$: WebSocketSubject<any>;
    
    paf's avatar
    paf committed
      wsConnected$ = new BehaviorSubject<boolean>(null);
    
      private wsSubscription: Subscription;
    
    paf's avatar
    paf committed
      connect(): void {
    
        if (!this.wsSubject$) {
          this.wsSubject$ = webSocket({
    
            deserializer(event: MessageEvent): any {
              return JSON.parse(event.data);
            },
            serializer(value: any): WebSocketMessage {
              return JSON.stringify(value);
            },
            openObserver: {
              next: () => {
    
                this.wsConnected$.next(true);
    
          this.wsSubscription = this.wsSubject$.subscribe(
    
    paflov's avatar
    paflov committed
            () => {},
            () => {
              this.closeConnection();
            },
            () => {
              this.closeConnection();
            }
    
      protected closeConnection(): void {
        this.wsConnected$.next(false);
    
        if (this.wsSubscription) {
          this.wsSubscription.unsubscribe();
        }
    
        if (this.wsSubject$) {
          this.wsSubject$.complete();
          this.wsSubject$ = null;
        }
    
    paf's avatar
    paf committed
      send(event: string, data: any): void {
    
        if (!this.wsSubject$) {
          this.connect();
    
    paflov's avatar
    paflov committed
        this.wsSubject$.next({ event, data });
    
    paf's avatar
    paf committed
      getChannel<T>(channelName: string): Observable<T> {
    
    paflov's avatar
    paflov committed
          this.connect();
    
    paflov's avatar
    paflov committed
          .multiplex(
            () => ({ event: `subscribe:${channelName}` }),
            () => ({ event: `unsubscribe:${channelName}` }),
            message => (message.event === channelName)
          )
          .pipe(map((event: WsMessage): T => event.data))
          .pipe(share());