import { Injectable } from '@angular/core'; import { webSocket, WebSocketSubject } from 'rxjs/webSocket'; import { WebsocketResponse } from '../interfaces/websocket.interface'; import { StateService } from './state.service'; import { Block, Transaction } from '../interfaces/electrs.interface'; import { Subscription } from 'rxjs'; const WEB_SOCKET_PROTOCOL = (document.location.protocol === 'https:') ? 'wss:' : 'ws:'; const WEB_SOCKET_URL = WEB_SOCKET_PROTOCOL + '//' + document.location.hostname + ':' + document.location.port + '/ws'; @Injectable({ providedIn: 'root' }) export class WebsocketService { private websocketSubject: WebSocketSubject = webSocket(WEB_SOCKET_URL); private goneOffline = false; private lastWant: string[] | null = null; private trackingTxId: string | null = null; private trackingAddress: string | null = null; private latestGitCommit = ''; private onlineCheckTimeout: number; private onlineCheckTimeoutTwo: number; private connectionCheckTimeout: number; private subscription: Subscription; constructor( private stateService: StateService, ) { this.startSubscription(); } startSubscription(retrying = false) { this.connectionCheckTimeout = window.setTimeout(() => { console.log('WebSocket failed to connect, force closing, trying to reconnect'); this.websocketSubject.complete(); this.subscription.unsubscribe(); this.goOffline(true); }, 10000); if (retrying) { if (this.stateService.connectionState$.value === 2) { return; } this.stateService.connectionState$.next(1); } this.websocketSubject.next({'action': 'init'}); this.subscription = this.websocketSubject .subscribe((response: WebsocketResponse) => { if (response.blocks && response.blocks.length) { const blocks = response.blocks; blocks.forEach((block: Block) => { if (block.height > this.stateService.latestBlockHeight) { this.stateService.latestBlockHeight = block.height; this.stateService.blocks$.next(block); } }); } if (response.block) { if (response.block.height > this.stateService.latestBlockHeight) { this.stateService.latestBlockHeight = response.block.height; this.stateService.blocks$.next(response.block); } if (response.txConfirmed) { this.trackingTxId = null; this.stateService.txConfirmed$.next(response.block); } } if (response.conversions) { this.stateService.conversions$.next(response.conversions); } if (response['mempool-blocks']) { this.stateService.mempoolBlocks$.next(response['mempool-blocks']); } if (response['git-commit']) { if (!this.latestGitCommit) { this.latestGitCommit = response['git-commit']; } else { if (this.latestGitCommit !== response['git-commit']) { setTimeout(() => { window.location.reload(); }, Math.floor(Math.random() * 60000) + 1000); } } } if (response['address-transactions']) { response['address-transactions'].forEach((addressTransaction: Transaction) => { this.stateService.mempoolTransactions$.next(addressTransaction); }); } if (response['address-block-transactions']) { response['address-block-transactions'].forEach((addressTransaction: Transaction) => { this.stateService.blockTransactions$.next(addressTransaction); }); } if (response['live-2h-chart']) { this.stateService.live2Chart$.next(response['live-2h-chart']); } if (response.mempoolInfo) { this.stateService.mempoolStats$.next({ memPoolInfo: response.mempoolInfo, vBytesPerSecond: response.vBytesPerSecond, }); } if (this.goneOffline === true) { this.goneOffline = false; if (this.lastWant) { this.want(this.lastWant); } if (this.trackingTxId) { this.startTrackTransaction(this.trackingTxId); } if (this.trackingAddress) { this.startTrackTransaction(this.trackingAddress); } this.stateService.connectionState$.next(2); } if (this.stateService.connectionState$.value === 1) { this.stateService.connectionState$.next(2); } clearTimeout(this.connectionCheckTimeout); this.startOnlineCheck(); }, (err: Error) => { console.log(err); console.log('WebSocket error, trying to reconnect in 10 seconds'); this.goOffline(); }); } startTrackTransaction(txId: string) { this.websocketSubject.next({ 'track-tx': txId }); this.trackingTxId = txId; } startTrackAddress(address: string) { this.websocketSubject.next({ 'track-address': address }); this.trackingAddress = address; } fetchStatistics(historicalDate: string) { this.websocketSubject.next({ historicalDate }); } want(data: string[]) { this.websocketSubject.next({action: 'want', data: data}); this.lastWant = data; } goOffline(instant = false) { this.goneOffline = true; this.stateService.connectionState$.next(0); window.setTimeout(() => { this.startSubscription(true); }, instant ? 10 : 10000); } startOnlineCheck() { clearTimeout(this.onlineCheckTimeout); clearTimeout(this.onlineCheckTimeoutTwo); this.onlineCheckTimeout = window.setTimeout(() => { this.websocketSubject.next({action: 'ping'}); this.onlineCheckTimeoutTwo = window.setTimeout(() => { if (!this.goneOffline) { console.log('WebSocket response timeout, force closing, trying to reconnect in 10 seconds'); this.websocketSubject.complete(); this.subscription.unsubscribe(); this.goOffline(); } }, 1000); }, 10000); } }