/// import kbn from 'app/core/utils/kbn'; import _ from 'lodash'; function matchSeriesOverride(aliasOrRegex, seriesAlias) { if (!aliasOrRegex) { return false; } if (aliasOrRegex[0] === '/') { var regex = kbn.stringToJsRegex(aliasOrRegex); return seriesAlias.match(regex) != null; } return aliasOrRegex === seriesAlias; } function translateFillOption(fill) { return fill === 0 ? 0.001 : fill/10; } export default class TimeSeries { datapoints: any; id: string; label: string; alias: string; color: string; valueFormater: any; stats: any; legend: boolean; allIsNull: boolean; allIsZero: boolean; decimals: number; scaledDecimals: number; hasMsResolution: boolean; isOutsideRange: boolean; lines: any; dashes: any; bars: any; points: any; yaxis: any; zindex: any; stack: any; nullPointMode: any; fillBelowTo: any; transform: any; flotpairs: any; unit: any; constructor(opts) { this.datapoints = opts.datapoints; this.label = opts.alias; this.id = opts.alias; this.alias = opts.alias; this.color = opts.color; this.valueFormater = kbn.valueFormats.none; this.stats = {}; this.legend = true; this.unit = opts.unit; this.hasMsResolution = this.isMsResolutionNeeded(); } applySeriesOverrides(overrides) { this.lines = {}; this.dashes = { dashLength: [] }; this.points = {}; this.bars = {}; this.yaxis = 1; this.zindex = 0; this.nullPointMode = null; delete this.stack; for (var i = 0; i < overrides.length; i++) { var override = overrides[i]; if (!matchSeriesOverride(override.alias, this.alias)) { continue; } if (override.lines !== void 0) { this.lines.show = override.lines; } if (override.dashes !== void 0) { this.dashes.show = override.dashes; this.lines.lineWidth = 0; } if (override.points !== void 0) { this.points.show = override.points; } if (override.bars !== void 0) { this.bars.show = override.bars; } if (override.fill !== void 0) { this.lines.fill = translateFillOption(override.fill); } if (override.stack !== void 0) { this.stack = override.stack; } if (override.linewidth !== void 0) { this.lines.lineWidth = override.linewidth; this.dashes.lineWidth = override.linewidth; } if (override.nullPointMode !== void 0) { this.nullPointMode = override.nullPointMode; } if (override.pointradius !== void 0) { this.points.radius = override.pointradius; } if (override.steppedLine !== void 0) { this.lines.steps = override.steppedLine; } if (override.zindex !== void 0) { this.zindex = override.zindex; } if (override.fillBelowTo !== void 0) { this.fillBelowTo = override.fillBelowTo; } if (override.color !== void 0) { this.color = override.color; } if (override.transform !== void 0) { this.transform = override.transform; } if (override.legend !== void 0) { this.legend = override.legend; } if (override.yaxis !== void 0) { this.yaxis = override.yaxis; } } } getFlotPairs(fillStyle) { var result = []; this.stats.total = 0; this.stats.max = -Number.MAX_VALUE; this.stats.min = Number.MAX_VALUE; this.stats.logmin = Number.MAX_VALUE; this.stats.avg = null; this.stats.current = null; this.stats.first = null; this.stats.delta = 0; this.stats.diff = null; this.stats.range = null; this.stats.timeStep = Number.MAX_VALUE; this.allIsNull = true; this.allIsZero = true; var ignoreNulls = fillStyle === 'connected'; var nullAsZero = fillStyle === 'null as zero'; var currentTime; var currentValue; var nonNulls = 0; var previousTime; var previousValue = 0; var previousDeltaUp = true; for (var i = 0; i < this.datapoints.length; i++) { currentValue = this.datapoints[i][0]; currentTime = this.datapoints[i][1]; // Due to missing values we could have different timeStep all along the series // so we have to find the minimum one (could occur with aggregators such as ZimSum) if (previousTime !== undefined) { let timeStep = currentTime - previousTime; if (timeStep < this.stats.timeStep) { this.stats.timeStep = timeStep; } } previousTime = currentTime; if (currentValue === null) { if (ignoreNulls) { continue; } if (nullAsZero) { currentValue = 0; } } if (currentValue !== null) { if (_.isNumber(currentValue)) { this.stats.total += currentValue; this.allIsNull = false; nonNulls++; } if (currentValue > this.stats.max) { this.stats.max = currentValue; } if (currentValue < this.stats.min) { this.stats.min = currentValue; } if (this.stats.first === null){ this.stats.first = currentValue; }else{ if (previousValue > currentValue) { // counter reset previousDeltaUp = false; if (i === this.datapoints.length-1) { // reset on last this.stats.delta += currentValue; } }else{ if (previousDeltaUp) { this.stats.delta += currentValue - previousValue; // normal increment } else { this.stats.delta += currentValue; // account for counter reset } previousDeltaUp = true; } } previousValue = currentValue; if (currentValue < this.stats.logmin && currentValue > 0) { this.stats.logmin = currentValue; } } if (currentValue !== 0) { this.allIsZero = false; } result.push([currentTime, currentValue]); } if (this.stats.max === -Number.MAX_VALUE) { this.stats.max = null; } if (this.stats.min === Number.MAX_VALUE) { this.stats.min = null; } if (result.length) { this.stats.avg = (this.stats.total / nonNulls); this.stats.current = result[result.length-1][1]; if (this.stats.current === null && result.length > 1) { this.stats.current = result[result.length-2][1]; } } if (this.stats.max !== null && this.stats.min !== null) { this.stats.range = this.stats.max - this.stats.min; } if (this.stats.current !== null && this.stats.first !== null) { this.stats.diff = this.stats.current - this.stats.first; } this.stats.count = result.length; return result; } updateLegendValues(formater, decimals, scaledDecimals) { this.valueFormater = formater; this.decimals = decimals; this.scaledDecimals = scaledDecimals; } formatValue(value) { return this.valueFormater(value, this.decimals, this.scaledDecimals); } isMsResolutionNeeded() { for (var i = 0; i < this.datapoints.length; i++) { if (this.datapoints[i][1] !== null) { var timestamp = this.datapoints[i][1].toString(); if (timestamp.length === 13 && (timestamp % 1000) !== 0) { return true; } } } return false; } hideFromLegend(options) { if (options.hideEmpty && this.allIsNull) { return true; } // ignore series excluded via override if (!this.legend) { return true; } // ignore zero series if (options.hideZero && this.allIsZero) { return true; } return false; } }