11 changed files with 641 additions and 10 deletions
@ -0,0 +1,309 @@ |
|||||
|
///
|
||||
|
/// Copyright © 2016-2019 The Thingsboard Authors
|
||||
|
///
|
||||
|
/// Licensed under the Apache 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
|
||||
|
///
|
||||
|
/// http://www.apache.org/licenses/LICENSE-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 { Inject, Injectable } from '@angular/core'; |
||||
|
import { |
||||
|
AttributesSubscriptionCmd, |
||||
|
GetHistoryCmd, |
||||
|
SubscriptionCmd, |
||||
|
SubscriptionUpdate, |
||||
|
SubscriptionUpdateMsg, |
||||
|
TelemetryFeature, |
||||
|
TelemetryPluginCmdsWrapper, |
||||
|
TelemetryService, |
||||
|
TelemetrySubscriber, |
||||
|
TimeseriesSubscriptionCmd |
||||
|
} from '@app/shared/models/telemetry/telemetry.models'; |
||||
|
import { select, Store } from '@ngrx/store'; |
||||
|
import { AppState } from '@core/core.state'; |
||||
|
import { AuthService } from '@core/auth/auth.service'; |
||||
|
import { selectIsAuthenticated } from '@core/auth/auth.selectors'; |
||||
|
import { WINDOW } from '@core/services/window.service'; |
||||
|
import { webSocket, WebSocketSubject } from 'rxjs/webSocket'; |
||||
|
import { ActionNotificationShow } from '@core/notification/notification.actions'; |
||||
|
import Timeout = NodeJS.Timeout; |
||||
|
|
||||
|
const RECONNECT_INTERVAL = 2000; |
||||
|
const WS_IDLE_TIMEOUT = 90000; |
||||
|
const MAX_PUBLISH_COMMANDS = 10; |
||||
|
|
||||
|
@Injectable({ |
||||
|
providedIn: 'root' |
||||
|
}) |
||||
|
export class TelemetryWebsocketService implements TelemetryService { |
||||
|
|
||||
|
isActive = false; |
||||
|
isOpening = false; |
||||
|
isOpened = false; |
||||
|
isReconnect = false; |
||||
|
|
||||
|
socketCloseTimer: Timeout; |
||||
|
reconnectTimer: Timeout; |
||||
|
|
||||
|
lastCmdId = 0; |
||||
|
subscribersCount = 0; |
||||
|
subscribersMap = new Map<number, TelemetrySubscriber>(); |
||||
|
|
||||
|
reconnectSubscribers = new Set<TelemetrySubscriber>(); |
||||
|
|
||||
|
cmdsWrapper = new TelemetryPluginCmdsWrapper(); |
||||
|
telemetryUri: string; |
||||
|
|
||||
|
dataStream: WebSocketSubject<TelemetryPluginCmdsWrapper | SubscriptionUpdateMsg>; |
||||
|
|
||||
|
constructor(private store: Store<AppState>, |
||||
|
private authService: AuthService, |
||||
|
@Inject(WINDOW) private window: Window) { |
||||
|
this.store.pipe(select(selectIsAuthenticated)).subscribe( |
||||
|
(authenticated: boolean) => { |
||||
|
if (!authenticated) { |
||||
|
this.reset(true); |
||||
|
} |
||||
|
} |
||||
|
); |
||||
|
|
||||
|
let port = this.window.location.port; |
||||
|
if (this.window.location.protocol === 'https:') { |
||||
|
if (!port) { |
||||
|
port = '443'; |
||||
|
} |
||||
|
this.telemetryUri = 'wss:'; |
||||
|
} else { |
||||
|
if (!port) { |
||||
|
port = '80'; |
||||
|
} |
||||
|
this.telemetryUri = 'ws:'; |
||||
|
} |
||||
|
this.telemetryUri += `//${this.window.location.hostname}:${port}/api/ws/plugins/telemetry`; |
||||
|
} |
||||
|
|
||||
|
public subscribe(subscriber: TelemetrySubscriber) { |
||||
|
this.isActive = true; |
||||
|
subscriber.subscriptionCommands.forEach( |
||||
|
(subscriptionCommand) => { |
||||
|
const cmdId = this.nextCmdId(); |
||||
|
this.subscribersMap.set(cmdId, subscriber); |
||||
|
subscriptionCommand.cmdId = cmdId; |
||||
|
if (subscriptionCommand instanceof SubscriptionCmd) { |
||||
|
if (subscriptionCommand.getType() === TelemetryFeature.TIMESERIES) { |
||||
|
this.cmdsWrapper.tsSubCmds.push(subscriptionCommand as TimeseriesSubscriptionCmd); |
||||
|
} else { |
||||
|
this.cmdsWrapper.attrSubCmds.push(subscriptionCommand as AttributesSubscriptionCmd); |
||||
|
} |
||||
|
} else if (subscriptionCommand instanceof GetHistoryCmd) { |
||||
|
this.cmdsWrapper.historyCmds.push(subscriptionCommand); |
||||
|
} |
||||
|
} |
||||
|
); |
||||
|
this.subscribersCount++; |
||||
|
this.publishCommands(); |
||||
|
} |
||||
|
|
||||
|
public unsubscribe(subscriber: TelemetrySubscriber) { |
||||
|
if (this.isActive) { |
||||
|
subscriber.subscriptionCommands.forEach( |
||||
|
(subscriptionCommand) => { |
||||
|
if (subscriptionCommand instanceof SubscriptionCmd) { |
||||
|
subscriptionCommand.unsubscribe = true; |
||||
|
if (subscriptionCommand.getType() === TelemetryFeature.TIMESERIES) { |
||||
|
this.cmdsWrapper.tsSubCmds.push(subscriptionCommand as TimeseriesSubscriptionCmd); |
||||
|
} else { |
||||
|
this.cmdsWrapper.attrSubCmds.push(subscriptionCommand as AttributesSubscriptionCmd); |
||||
|
} |
||||
|
} |
||||
|
const cmdId = subscriptionCommand.cmdId; |
||||
|
if (cmdId) { |
||||
|
this.subscribersMap.delete(cmdId); |
||||
|
} |
||||
|
} |
||||
|
); |
||||
|
this.reconnectSubscribers.delete(subscriber); |
||||
|
this.subscribersCount--; |
||||
|
this.publishCommands(); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
private nextCmdId(): number { |
||||
|
this.lastCmdId++; |
||||
|
return this.lastCmdId; |
||||
|
} |
||||
|
|
||||
|
private publishCommands() { |
||||
|
while (this.isOpened && this.cmdsWrapper.hasCommands()) { |
||||
|
this.dataStream.next(this.cmdsWrapper.preparePublishCommands(MAX_PUBLISH_COMMANDS)); |
||||
|
this.checkToClose(); |
||||
|
} |
||||
|
this.tryOpenSocket(); |
||||
|
} |
||||
|
|
||||
|
private checkToClose() { |
||||
|
if (this.subscribersCount === 0 && this.isOpened) { |
||||
|
if (!this.socketCloseTimer) { |
||||
|
this.socketCloseTimer = setTimeout( |
||||
|
() => this.closeSocket(), WS_IDLE_TIMEOUT); |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
private reset(close: boolean) { |
||||
|
if (this.socketCloseTimer) { |
||||
|
clearTimeout(this.socketCloseTimer); |
||||
|
this.socketCloseTimer = null; |
||||
|
} |
||||
|
this.lastCmdId = 0; |
||||
|
this.subscribersMap.clear(); |
||||
|
this.subscribersCount = 0; |
||||
|
this.cmdsWrapper.clear(); |
||||
|
if (close) { |
||||
|
this.closeSocket(); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
private closeSocket() { |
||||
|
this.isActive = false; |
||||
|
if (this.isOpened) { |
||||
|
this.dataStream.unsubscribe(); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
private tryOpenSocket() { |
||||
|
if (this.isActive) { |
||||
|
if (!this.isOpened && !this.isOpening) { |
||||
|
this.isOpening = true; |
||||
|
if (AuthService.isJwtTokenValid()) { |
||||
|
this.openSocket(AuthService.getJwtToken()); |
||||
|
} else { |
||||
|
this.authService.refreshJwtToken().subscribe(() => { |
||||
|
this.openSocket(AuthService.getJwtToken()); |
||||
|
}, |
||||
|
() => { |
||||
|
this.isOpening = false; |
||||
|
this.authService.logout(true); |
||||
|
} |
||||
|
); |
||||
|
} |
||||
|
} |
||||
|
if (this.socketCloseTimer) { |
||||
|
clearTimeout(this.socketCloseTimer); |
||||
|
this.socketCloseTimer = null; |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
private openSocket(token: string) { |
||||
|
const uri = `${this.telemetryUri}?token=${token}`; |
||||
|
this.dataStream = webSocket( |
||||
|
{ |
||||
|
url: uri, |
||||
|
openObserver: { |
||||
|
next: (e: Event) => { |
||||
|
this.onOpen(); |
||||
|
} |
||||
|
}, |
||||
|
closeObserver: { |
||||
|
next: (e: CloseEvent) => { |
||||
|
this.onClose(e); |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
); |
||||
|
|
||||
|
this.dataStream.subscribe((message) => { |
||||
|
this.onMessage(message as SubscriptionUpdateMsg); |
||||
|
}, |
||||
|
(error) => { |
||||
|
this.onError(error); |
||||
|
}); |
||||
|
} |
||||
|
|
||||
|
private onOpen() { |
||||
|
this.isOpening = false; |
||||
|
this.isOpened = true; |
||||
|
if (this.reconnectTimer) { |
||||
|
clearTimeout(this.reconnectTimer); |
||||
|
this.reconnectTimer = null; |
||||
|
} |
||||
|
if (this.isReconnect) { |
||||
|
this.isReconnect = false; |
||||
|
this.reconnectSubscribers.forEach( |
||||
|
(reconnectSubscriber) => { |
||||
|
reconnectSubscriber.onReconnected(); |
||||
|
this.subscribe(reconnectSubscriber); |
||||
|
} |
||||
|
); |
||||
|
this.reconnectSubscribers.clear(); |
||||
|
} else { |
||||
|
this.publishCommands(); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
private onMessage(message: SubscriptionUpdateMsg) { |
||||
|
if (message.errorCode) { |
||||
|
this.showWsError(message.errorCode, message.errorMsg); |
||||
|
} else if (message.subscriptionId) { |
||||
|
const subscriber = this.subscribersMap.get(message.subscriptionId); |
||||
|
if (subscriber) { |
||||
|
subscriber.onData(new SubscriptionUpdate(message)); |
||||
|
} |
||||
|
} |
||||
|
this.checkToClose(); |
||||
|
} |
||||
|
|
||||
|
private onError(errorEvent) { |
||||
|
if (errorEvent) { |
||||
|
console.warn('WebSocket error event', errorEvent); |
||||
|
} |
||||
|
this.isOpening = false; |
||||
|
} |
||||
|
|
||||
|
private onClose(closeEvent: CloseEvent) { |
||||
|
if (closeEvent && closeEvent.code > 1000 && closeEvent.code !== 1006) { |
||||
|
this.showWsError(closeEvent.code, closeEvent.reason); |
||||
|
} |
||||
|
this.isOpening = false; |
||||
|
this.isOpened = false; |
||||
|
if (this.isActive) { |
||||
|
if (!this.isReconnect) { |
||||
|
this.reconnectSubscribers.clear(); |
||||
|
this.subscribersMap.forEach( |
||||
|
(subscriber) => { |
||||
|
this.reconnectSubscribers.add(subscriber); |
||||
|
} |
||||
|
); |
||||
|
this.reset(false); |
||||
|
this.isReconnect = true; |
||||
|
} |
||||
|
if (this.reconnectTimer) { |
||||
|
clearTimeout(this.reconnectTimer); |
||||
|
} |
||||
|
this.reconnectTimer = setTimeout(() => this.tryOpenSocket(), RECONNECT_INTERVAL); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
private showWsError(errorCode: number, errorMsg: string) { |
||||
|
let message = 'WebSocket Error: '; |
||||
|
if (errorMsg) { |
||||
|
message += errorMsg; |
||||
|
} else { |
||||
|
message += `error code - ${errorCode}.`; |
||||
|
} |
||||
|
this.store.dispatch(new ActionNotificationShow( |
||||
|
{ |
||||
|
message, type: 'error' |
||||
|
})); |
||||
|
} |
||||
|
|
||||
|
} |
||||
Loading…
Reference in new issue