UNPKG

@c8y/ngx-components

Version:

Angular modules for Cumulocity IoT applications

193 lines 39.6 kB
import { Injectable } from '@angular/core'; import { from, interval, merge } from 'rxjs'; import { buffer, map, mergeMap, tap, throttleTime } from 'rxjs/operators'; import { AlarmRealtimeService, EventRealtimeService, MeasurementRealtimeService } from '@c8y/ngx-components'; import { EchartsOptionsService } from './echarts-options.service'; import * as i0 from "@angular/core"; import * as i1 from "@c8y/ngx-components"; import * as i2 from "./echarts-options.service"; export class ChartRealtimeService { constructor(measurementRealtime, alarmRealtimeService, eventRealtimeService, echartsOptionsService) { this.measurementRealtime = measurementRealtime; this.alarmRealtimeService = alarmRealtimeService; this.eventRealtimeService = eventRealtimeService; this.echartsOptionsService = echartsOptionsService; this.INTERVAL = 1000; this.MIN_REALTIME_TIMEOUT = 250; this.MAX_REALTIME_TIMEOUT = 5_000; this.lastMeasurements = new Map(); } startRealtime(echartsInstance, datapoints, timeRange, datapointOutOfSyncCallback, timeRangeChangedCallback, alarmOrEventConfig = [], displayOptions) { this.echartsInstance = echartsInstance; this.currentTimeRange = { dateFrom: new Date(timeRange.dateFrom), dateTo: new Date(timeRange.dateTo) }; const activeAlarmsOrEvents = alarmOrEventConfig.filter(alarmOrEvent => alarmOrEvent.__active && !alarmOrEvent.__hidden); const uniqueAlarmOrEventTargets = Array.from(new Set(activeAlarmsOrEvents.map(aOrE => aOrE.__target.id))); const allAlarmsAndEvents$ = from(uniqueAlarmOrEventTargets).pipe(mergeMap(targetId => { const alarmsRealtime$ = this.alarmRealtimeService.onAll$(targetId); const eventsRealtime$ = this.eventRealtimeService.onAll$(targetId); return merge(alarmsRealtime$, eventsRealtime$).pipe(map(realtimeMessage => realtimeMessage.data)); })); const measurementsForDatapoints = datapoints.map(dp => { const source$ = this.measurementRealtime.onCreateOfSpecificMeasurement$(dp.fragment, dp.series, dp.__target?.id); return source$.pipe(map((measurement) => ({ datapoint: dp, measurement }))); }); const updateThrottleTime = this.getRealtimeUpdateThrottleTime(timeRange); const measurement$ = merge(...measurementsForDatapoints); const bufferReset$ = merge(measurement$.pipe(throttleTime(updateThrottleTime)), interval(this.INTERVAL).pipe(tap(() => { this.currentTimeRange = { dateFrom: new Date((this.currentTimeRange?.dateFrom?.valueOf() || 0) + this.INTERVAL), dateTo: new Date((this.currentTimeRange?.dateTo?.valueOf() || 0) + this.INTERVAL) }; timeRangeChangedCallback(this.currentTimeRange); }), throttleTime(updateThrottleTime))).pipe(throttleTime(this.MIN_REALTIME_TIMEOUT)); this.realtimeSubscriptionMeasurements = measurement$ .pipe(buffer(bufferReset$)) .subscribe(measurements => { this.updateChartInstance(measurements, null, displayOptions, datapointOutOfSyncCallback); // Store the last measurements measurements.forEach(measurement => { this.lastMeasurements.set(measurement.datapoint.fragment, measurement); }); }); this.realtimeSubscriptionAlarmsEvents = allAlarmsAndEvents$ .pipe(map(alarmOrEvent => { const foundAlarmOrEvent = activeAlarmsOrEvents.find(aOrE => { return aOrE.filters.type === alarmOrEvent.type; }); if (foundAlarmOrEvent) { alarmOrEvent['color'] = foundAlarmOrEvent.color; alarmOrEvent['selectedDatapoint'] = foundAlarmOrEvent.selectedDatapoint; } if (foundAlarmOrEvent) { const fragment = alarmOrEvent['selectedDatapoint'].fragment; if (this.lastMeasurements.has(fragment)) { const lastMeasurement = this.lastMeasurements.get(fragment); this.updateChartInstance([lastMeasurement], alarmOrEvent, displayOptions, datapointOutOfSyncCallback); } } return foundAlarmOrEvent ? alarmOrEvent : null; })) .subscribe(); } stopRealtime() { this.realtimeSubscriptionMeasurements?.unsubscribe(); this.realtimeSubscriptionAlarmsEvents?.unsubscribe(); } removeValuesBeforeTimeRange(series) { const firstValidValueByDateIndex = series.data.findIndex(([dateString, _]) => { return new Date(dateString) >= (this.currentTimeRange?.dateFrom || new Date()); }); if (firstValidValueByDateIndex > 1) { // we need one value before dateFrom for chart lines to be extended to the left edge of the graph series.data = series.data.slice(firstValidValueByDateIndex - 1); } return series.data; } getRealtimeUpdateThrottleTime(timeRange) { const timeRangeInMs = new Date(timeRange.dateTo).valueOf() - new Date(timeRange.dateFrom).valueOf(); const calculatedThrottleTime = Math.round(timeRangeInMs / 1000); if (calculatedThrottleTime < this.MIN_REALTIME_TIMEOUT) { return this.MIN_REALTIME_TIMEOUT; } else if (calculatedThrottleTime > this.MAX_REALTIME_TIMEOUT) { return this.MAX_REALTIME_TIMEOUT; } return calculatedThrottleTime; } updateChartInstance(receivedMeasurements, alarmOrEvent, displayOptions, datapointOutOfSyncCallback) { const isEvent = (item) => !('severity' in item); const isAlarm = (item) => 'severity' in item; const seriesDataToUpdate = new Map(); receivedMeasurements.forEach(({ datapoint, measurement }) => { if (!seriesDataToUpdate.has(datapoint)) { seriesDataToUpdate.set(datapoint, []); } seriesDataToUpdate.get(datapoint)?.push(measurement); }); let allDataSeries = this.echartsInstance?.getOption()['series']; seriesDataToUpdate.forEach((measurements, datapoint) => { const newValues = measurements.map(m => [ m.time, m[datapoint.fragment][datapoint.series].value ]); const datapointId = datapoint.__target?.id + datapoint.fragment + datapoint.series; const seriesMatchingDatapoint = allDataSeries.find(s => s['datapointId'] === datapointId); if (!seriesMatchingDatapoint) { return; } const seriesDataToUpdate = seriesMatchingDatapoint['data']; seriesDataToUpdate.push(...newValues); seriesMatchingDatapoint['data'] = this.removeValuesBeforeTimeRange(seriesMatchingDatapoint); if (alarmOrEvent) { const renderType = datapoint.renderType || 'min'; const dp = { ...datapoint, values: seriesMatchingDatapoint['data'] }; if (isEvent(alarmOrEvent)) { // if event series with the same id already exists, return const eventExists = allDataSeries.some(series => series['data'].some(data => data[0] === alarmOrEvent.creationTime)); if (eventExists) { return; } const newEventSeries = this.echartsOptionsService.getAlarmOrEventSeries(dp, renderType, false, [alarmOrEvent], 'event', displayOptions, alarmOrEvent.creationTime, null, true); allDataSeries.push(...newEventSeries); } else if (isAlarm(alarmOrEvent)) { const alarmExists = allDataSeries.some((series) => { const seriesData = series['data']; return seriesData.some((data) => data[0] === alarmOrEvent.creationTime); }); if (alarmExists) { const alarmSeries = allDataSeries.filter((series) => { const seriesData = series['data']; return seriesData.some((data) => data[0] === alarmOrEvent.creationTime); }); // remove all matching alarm series which are in the array alarmSeries.forEach(series => { allDataSeries = allDataSeries.filter(s => s['id'] !== series['id']); }); const newAlarmSeries = this.echartsOptionsService.getAlarmOrEventSeries(dp, renderType, false, [alarmOrEvent], 'alarm', displayOptions, alarmOrEvent.creationTime, null, true); allDataSeries.push(...newAlarmSeries); } else { const newAlarmSeries = this.echartsOptionsService.getAlarmOrEventSeries(dp, renderType, false, [alarmOrEvent], 'alarm', displayOptions, alarmOrEvent.id, null, true); allDataSeries.push(...newAlarmSeries); } } } this.checkForValuesAfterTimeRange(seriesMatchingDatapoint['data'], datapoint, datapointOutOfSyncCallback); }); this.echartsInstance?.setOption({ dataZoom: [ { type: 'inside', startValue: this.currentTimeRange?.dateFrom.valueOf(), endValue: this.currentTimeRange?.dateTo.valueOf() } ], xAxis: { max: this.currentTimeRange?.dateTo }, series: allDataSeries }); } checkForValuesAfterTimeRange(data, datapoint, datapointOutOfSyncCallback) { const now = new Date(); const valueAfterNowExists = data.some(([dateString, _]) => { return new Date(dateString).valueOf() > now.valueOf(); }); if (valueAfterNowExists) { datapointOutOfSyncCallback(datapoint); } } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ChartRealtimeService, deps: [{ token: i1.MeasurementRealtimeService }, { token: i1.AlarmRealtimeService }, { token: i1.EventRealtimeService }, { token: i2.EchartsOptionsService }], target: i0.ɵɵFactoryTarget.Injectable }); } static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ChartRealtimeService }); } } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ChartRealtimeService, decorators: [{ type: Injectable }], ctorParameters: () => [{ type: i1.MeasurementRealtimeService }, { type: i1.AlarmRealtimeService }, { type: i1.EventRealtimeService }, { type: i2.EchartsOptionsService }] }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"chart-realtime.service.js","sourceRoot":"","sources":["../../../../echart/services/chart-realtime.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { from, interval, merge, Observable, Subscription } from 'rxjs';\nimport { IAlarm, IEvent, IMeasurement } from '@c8y/client';\nimport { buffer, map, mergeMap, tap, throttleTime } from 'rxjs/operators';\nimport {\n  AlarmOrEventExtended,\n  DatapointChartRenderType,\n  DatapointRealtimeMeasurements,\n  DatapointsGraphKPIDetails,\n  DatapointsGraphWidgetConfig,\n  DpWithValues,\n  SeriesValue\n} from '../models';\nimport {\n  AlarmRealtimeService,\n  EventRealtimeService,\n  MeasurementRealtimeService,\n  RealtimeMessage\n} from '@c8y/ngx-components';\nimport type { ECharts, SeriesOption } from 'echarts';\nimport { EchartsOptionsService } from './echarts-options.service';\nimport { CustomSeriesOptions } from '../models/chart.model';\n\ntype Milliseconds = number;\n\n@Injectable()\nexport class ChartRealtimeService {\n  private INTERVAL: Milliseconds = 1000;\n  private MIN_REALTIME_TIMEOUT: Milliseconds = 250;\n  private MAX_REALTIME_TIMEOUT: Milliseconds = 5_000;\n  private realtimeSubscriptionMeasurements!: Subscription;\n  private realtimeSubscriptionAlarmsEvents!: Subscription;\n  private echartsInstance: ECharts | undefined;\n  private currentTimeRange: { dateFrom: Date; dateTo: Date } | undefined;\n  private lastMeasurements: Map<string, DatapointRealtimeMeasurements> = new Map();\n\n  constructor(\n    private measurementRealtime: MeasurementRealtimeService,\n    private alarmRealtimeService: AlarmRealtimeService,\n    private eventRealtimeService: EventRealtimeService,\n    private echartsOptionsService: EchartsOptionsService\n  ) {}\n\n  startRealtime(\n    echartsInstance: ECharts,\n    datapoints: DatapointsGraphKPIDetails[],\n    timeRange: { dateFrom: string; dateTo: string },\n    datapointOutOfSyncCallback: (dp: DatapointsGraphKPIDetails) => void,\n    timeRangeChangedCallback: (\n      timeRange: Pick<DatapointsGraphWidgetConfig, 'dateFrom' | 'dateTo'>\n    ) => void,\n    alarmOrEventConfig: AlarmOrEventExtended[] = [],\n    displayOptions: { displayMarkedLine: boolean; displayMarkedPoint: boolean }\n  ) {\n    this.echartsInstance = echartsInstance;\n    this.currentTimeRange = {\n      dateFrom: new Date(timeRange.dateFrom),\n      dateTo: new Date(timeRange.dateTo)\n    };\n\n    const activeAlarmsOrEvents = alarmOrEventConfig.filter(\n      alarmOrEvent => alarmOrEvent.__active && !alarmOrEvent.__hidden\n    );\n    const uniqueAlarmOrEventTargets = Array.from(\n      new Set(activeAlarmsOrEvents.map(aOrE => aOrE.__target.id))\n    );\n\n    const allAlarmsAndEvents$: Observable<IAlarm | IEvent> = from(uniqueAlarmOrEventTargets).pipe(\n      mergeMap(targetId => {\n        const alarmsRealtime$: Observable<RealtimeMessage<IAlarm>> =\n          this.alarmRealtimeService.onAll$(targetId);\n        const eventsRealtime$: Observable<RealtimeMessage<IEvent>> =\n          this.eventRealtimeService.onAll$(targetId);\n        return merge(alarmsRealtime$, eventsRealtime$).pipe(\n          map(realtimeMessage => realtimeMessage.data as IAlarm | IEvent)\n        );\n      })\n    );\n\n    const measurementsForDatapoints: Observable<DatapointRealtimeMeasurements>[] = datapoints.map(\n      dp => {\n        const source$: Observable<IMeasurement> =\n          this.measurementRealtime.onCreateOfSpecificMeasurement$(\n            dp.fragment,\n            dp.series,\n            dp.__target?.id\n          );\n        return source$.pipe(map((measurement: IMeasurement) => ({ datapoint: dp, measurement })));\n      }\n    );\n    const updateThrottleTime: Milliseconds = this.getRealtimeUpdateThrottleTime(timeRange);\n    const measurement$ = merge(...measurementsForDatapoints);\n    const bufferReset$ = merge(\n      measurement$.pipe(throttleTime(updateThrottleTime)),\n      interval(this.INTERVAL).pipe(\n        tap(() => {\n          this.currentTimeRange = {\n            dateFrom: new Date((this.currentTimeRange?.dateFrom?.valueOf() || 0) + this.INTERVAL),\n            dateTo: new Date((this.currentTimeRange?.dateTo?.valueOf() || 0) + this.INTERVAL)\n          };\n          timeRangeChangedCallback(this.currentTimeRange);\n        }),\n        throttleTime(updateThrottleTime)\n      )\n    ).pipe(throttleTime(this.MIN_REALTIME_TIMEOUT));\n\n    this.realtimeSubscriptionMeasurements = measurement$\n      .pipe(buffer(bufferReset$))\n      .subscribe(measurements => {\n        this.updateChartInstance(measurements, null, displayOptions, datapointOutOfSyncCallback);\n\n        // Store the last measurements\n        measurements.forEach(measurement => {\n          this.lastMeasurements.set(measurement.datapoint.fragment, measurement);\n        });\n      });\n\n    this.realtimeSubscriptionAlarmsEvents = allAlarmsAndEvents$\n      .pipe(\n        map(alarmOrEvent => {\n          const foundAlarmOrEvent = activeAlarmsOrEvents.find(aOrE => {\n            return aOrE.filters.type === alarmOrEvent.type;\n          });\n          if (foundAlarmOrEvent) {\n            alarmOrEvent['color'] = foundAlarmOrEvent.color;\n            alarmOrEvent['selectedDatapoint'] = foundAlarmOrEvent.selectedDatapoint;\n          }\n\n          if (foundAlarmOrEvent) {\n            const fragment = alarmOrEvent['selectedDatapoint'].fragment;\n            if (this.lastMeasurements.has(fragment)) {\n              const lastMeasurement = this.lastMeasurements.get(fragment);\n\n              this.updateChartInstance(\n                [lastMeasurement],\n                alarmOrEvent,\n                displayOptions,\n                datapointOutOfSyncCallback\n              );\n            }\n          }\n\n          return foundAlarmOrEvent ? alarmOrEvent : null;\n        })\n      )\n      .subscribe();\n  }\n\n  stopRealtime() {\n    this.realtimeSubscriptionMeasurements?.unsubscribe();\n    this.realtimeSubscriptionAlarmsEvents?.unsubscribe();\n  }\n\n  private removeValuesBeforeTimeRange(series: SeriesOption): SeriesValue[] {\n    const firstValidValueByDateIndex = (series.data as SeriesValue[]).findIndex(\n      ([dateString, _]) => {\n        return new Date(dateString) >= (this.currentTimeRange?.dateFrom || new Date());\n      }\n    );\n    if (firstValidValueByDateIndex > 1) {\n      // we need one value before dateFrom for chart lines to be extended to the left edge of the graph\n      series.data = (series.data as SeriesValue[]).slice(firstValidValueByDateIndex - 1);\n    }\n    return series.data as SeriesValue[];\n  }\n\n  private getRealtimeUpdateThrottleTime(timeRange: {\n    dateFrom: string;\n    dateTo: string;\n  }): Milliseconds {\n    const timeRangeInMs =\n      new Date(timeRange.dateTo).valueOf() - new Date(timeRange.dateFrom).valueOf();\n    const calculatedThrottleTime = Math.round(timeRangeInMs / 1000);\n    if (calculatedThrottleTime < this.MIN_REALTIME_TIMEOUT) {\n      return this.MIN_REALTIME_TIMEOUT;\n    } else if (calculatedThrottleTime > this.MAX_REALTIME_TIMEOUT) {\n      return this.MAX_REALTIME_TIMEOUT;\n    }\n    return calculatedThrottleTime;\n  }\n\n  private updateChartInstance(\n    receivedMeasurements: DatapointRealtimeMeasurements[],\n    alarmOrEvent: IAlarm | IEvent | null,\n    displayOptions: { displayMarkedLine: boolean; displayMarkedPoint: boolean },\n    datapointOutOfSyncCallback: (dp: DatapointsGraphKPIDetails) => void\n  ) {\n    const isEvent = (item: IAlarm | IEvent): item is IEvent => !('severity' in item);\n    const isAlarm = (item: IAlarm | IEvent): item is IAlarm => 'severity' in item;\n\n    const seriesDataToUpdate = new Map<DatapointsGraphKPIDetails, IMeasurement[]>();\n\n    receivedMeasurements.forEach(({ datapoint, measurement }) => {\n      if (!seriesDataToUpdate.has(datapoint)) {\n        seriesDataToUpdate.set(datapoint, []);\n      }\n      seriesDataToUpdate.get(datapoint)?.push(measurement);\n    });\n\n    let allDataSeries = this.echartsInstance?.getOption()['series'] as CustomSeriesOptions[];\n\n    seriesDataToUpdate.forEach((measurements, datapoint) => {\n      const newValues: SeriesValue[] = measurements.map(m => [\n        m.time as string,\n        m[datapoint.fragment][datapoint.series].value\n      ]);\n      const datapointId = datapoint.__target?.id + datapoint.fragment + datapoint.series;\n      const seriesMatchingDatapoint = allDataSeries.find(s => s['datapointId'] === datapointId);\n      if (!seriesMatchingDatapoint) {\n        return;\n      }\n      const seriesDataToUpdate = seriesMatchingDatapoint['data'] as SeriesValue[];\n      seriesDataToUpdate.push(...newValues);\n\n      seriesMatchingDatapoint['data'] = this.removeValuesBeforeTimeRange(seriesMatchingDatapoint);\n\n      if (alarmOrEvent) {\n        const renderType: DatapointChartRenderType = datapoint.renderType || 'min';\n        const dp: DpWithValues = {\n          ...datapoint,\n          values: seriesMatchingDatapoint['data'] as {\n            [date: string]: { min: number; max: number }[];\n          }\n        };\n\n        if (isEvent(alarmOrEvent)) {\n          // if event series with the same id already exists, return\n          const eventExists = allDataSeries.some(series =>\n            (series['data'] as string[][]).some(\n              data => data[0] === (alarmOrEvent as IEvent).creationTime\n            )\n          );\n          if (eventExists) {\n            return;\n          }\n          const newEventSeries = this.echartsOptionsService.getAlarmOrEventSeries(\n            dp,\n            renderType,\n            false,\n            [alarmOrEvent],\n            'event',\n            displayOptions,\n            alarmOrEvent.creationTime,\n            null,\n            true\n          );\n          allDataSeries.push(...newEventSeries);\n        } else if (isAlarm(alarmOrEvent)) {\n          const alarmExists = allDataSeries.some((series: CustomSeriesOptions) => {\n            const seriesData = series['data'] as SeriesValue[];\n            return seriesData.some(\n              (data: SeriesValue) => data[0] === (alarmOrEvent as IEvent).creationTime\n            );\n          });\n          if (alarmExists) {\n            const alarmSeries = allDataSeries.filter((series: CustomSeriesOptions) => {\n              const seriesData = series['data'] as SeriesValue[];\n              return seriesData.some(\n                (data: SeriesValue) => data[0] === (alarmOrEvent as IAlarm).creationTime\n              );\n            });\n            // remove all matching alarm series which are in the array\n            alarmSeries.forEach(series => {\n              allDataSeries = allDataSeries.filter(s => s['id'] !== series['id']);\n            });\n\n            const newAlarmSeries = this.echartsOptionsService.getAlarmOrEventSeries(\n              dp,\n              renderType,\n              false,\n              [alarmOrEvent],\n              'alarm',\n              displayOptions,\n              (alarmOrEvent as IAlarm).creationTime,\n              null,\n              true\n            );\n            allDataSeries.push(...newAlarmSeries);\n          } else {\n            const newAlarmSeries = this.echartsOptionsService.getAlarmOrEventSeries(\n              dp,\n              renderType,\n              false,\n              [alarmOrEvent],\n              'alarm',\n              displayOptions,\n              (alarmOrEvent as IEvent).id,\n              null,\n              true\n            );\n            allDataSeries.push(...newAlarmSeries);\n          }\n        }\n      }\n\n      this.checkForValuesAfterTimeRange(\n        seriesMatchingDatapoint['data'] as SeriesValue[],\n        datapoint,\n        datapointOutOfSyncCallback\n      );\n    });\n\n    this.echartsInstance?.setOption({\n      dataZoom: [\n        {\n          type: 'inside',\n          startValue: this.currentTimeRange?.dateFrom.valueOf(),\n          endValue: this.currentTimeRange?.dateTo.valueOf()\n        }\n      ],\n      xAxis: {\n        max: this.currentTimeRange?.dateTo\n      },\n      series: allDataSeries\n    });\n  }\n\n  private checkForValuesAfterTimeRange(\n    data: SeriesValue[],\n    datapoint: DatapointsGraphKPIDetails,\n    datapointOutOfSyncCallback: (dp: DatapointsGraphKPIDetails) => void\n  ) {\n    const now = new Date();\n    const valueAfterNowExists = data.some(([dateString, _]: SeriesValue) => {\n      return new Date(dateString).valueOf() > now.valueOf();\n    });\n    if (valueAfterNowExists) {\n      datapointOutOfSyncCallback(datapoint);\n    }\n  }\n}\n"]}