177 lines
7.1 KiB
JavaScript
177 lines
7.1 KiB
JavaScript
/*
|
|
* Copyright The OpenTelemetry 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
|
|
*
|
|
* https://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 { AggregatorKind, } from './types';
|
|
import { DataPointType, } from '../export/MetricData';
|
|
import { InstrumentType } from '../InstrumentDescriptor';
|
|
import { binarySearchUB } from '../utils';
|
|
function createNewEmptyCheckpoint(boundaries) {
|
|
const counts = boundaries.map(() => 0);
|
|
counts.push(0);
|
|
return {
|
|
buckets: {
|
|
boundaries,
|
|
counts,
|
|
},
|
|
sum: 0,
|
|
count: 0,
|
|
hasMinMax: false,
|
|
min: Infinity,
|
|
max: -Infinity,
|
|
};
|
|
}
|
|
export class HistogramAccumulation {
|
|
constructor(startTime, _boundaries, _recordMinMax = true, _current = createNewEmptyCheckpoint(_boundaries)) {
|
|
this.startTime = startTime;
|
|
this._boundaries = _boundaries;
|
|
this._recordMinMax = _recordMinMax;
|
|
this._current = _current;
|
|
}
|
|
record(value) {
|
|
// NaN does not fall into any bucket, is not zero and should not be counted,
|
|
// NaN is never greater than max nor less than min, therefore return as there's nothing for us to do.
|
|
if (Number.isNaN(value)) {
|
|
return;
|
|
}
|
|
this._current.count += 1;
|
|
this._current.sum += value;
|
|
if (this._recordMinMax) {
|
|
this._current.min = Math.min(value, this._current.min);
|
|
this._current.max = Math.max(value, this._current.max);
|
|
this._current.hasMinMax = true;
|
|
}
|
|
const idx = binarySearchUB(this._boundaries, value);
|
|
this._current.buckets.counts[idx] += 1;
|
|
}
|
|
setStartTime(startTime) {
|
|
this.startTime = startTime;
|
|
}
|
|
toPointValue() {
|
|
return this._current;
|
|
}
|
|
}
|
|
/**
|
|
* Basic aggregator which observes events and counts them in pre-defined buckets
|
|
* and provides the total sum and count of all observations.
|
|
*/
|
|
export class HistogramAggregator {
|
|
/**
|
|
* @param _boundaries sorted upper bounds of recorded values.
|
|
* @param _recordMinMax If set to true, min and max will be recorded. Otherwise, min and max will not be recorded.
|
|
*/
|
|
constructor(_boundaries, _recordMinMax) {
|
|
this._boundaries = _boundaries;
|
|
this._recordMinMax = _recordMinMax;
|
|
this.kind = AggregatorKind.HISTOGRAM;
|
|
}
|
|
createAccumulation(startTime) {
|
|
return new HistogramAccumulation(startTime, this._boundaries, this._recordMinMax);
|
|
}
|
|
/**
|
|
* Return the result of the merge of two histogram accumulations. As long as one Aggregator
|
|
* instance produces all Accumulations with constant boundaries we don't need to worry about
|
|
* merging accumulations with different boundaries.
|
|
*/
|
|
merge(previous, delta) {
|
|
const previousValue = previous.toPointValue();
|
|
const deltaValue = delta.toPointValue();
|
|
const previousCounts = previousValue.buckets.counts;
|
|
const deltaCounts = deltaValue.buckets.counts;
|
|
const mergedCounts = new Array(previousCounts.length);
|
|
for (let idx = 0; idx < previousCounts.length; idx++) {
|
|
mergedCounts[idx] = previousCounts[idx] + deltaCounts[idx];
|
|
}
|
|
let min = Infinity;
|
|
let max = -Infinity;
|
|
if (this._recordMinMax) {
|
|
if (previousValue.hasMinMax && deltaValue.hasMinMax) {
|
|
min = Math.min(previousValue.min, deltaValue.min);
|
|
max = Math.max(previousValue.max, deltaValue.max);
|
|
}
|
|
else if (previousValue.hasMinMax) {
|
|
min = previousValue.min;
|
|
max = previousValue.max;
|
|
}
|
|
else if (deltaValue.hasMinMax) {
|
|
min = deltaValue.min;
|
|
max = deltaValue.max;
|
|
}
|
|
}
|
|
return new HistogramAccumulation(previous.startTime, previousValue.buckets.boundaries, this._recordMinMax, {
|
|
buckets: {
|
|
boundaries: previousValue.buckets.boundaries,
|
|
counts: mergedCounts,
|
|
},
|
|
count: previousValue.count + deltaValue.count,
|
|
sum: previousValue.sum + deltaValue.sum,
|
|
hasMinMax: this._recordMinMax &&
|
|
(previousValue.hasMinMax || deltaValue.hasMinMax),
|
|
min: min,
|
|
max: max,
|
|
});
|
|
}
|
|
/**
|
|
* Returns a new DELTA aggregation by comparing two cumulative measurements.
|
|
*/
|
|
diff(previous, current) {
|
|
const previousValue = previous.toPointValue();
|
|
const currentValue = current.toPointValue();
|
|
const previousCounts = previousValue.buckets.counts;
|
|
const currentCounts = currentValue.buckets.counts;
|
|
const diffedCounts = new Array(previousCounts.length);
|
|
for (let idx = 0; idx < previousCounts.length; idx++) {
|
|
diffedCounts[idx] = currentCounts[idx] - previousCounts[idx];
|
|
}
|
|
return new HistogramAccumulation(current.startTime, previousValue.buckets.boundaries, this._recordMinMax, {
|
|
buckets: {
|
|
boundaries: previousValue.buckets.boundaries,
|
|
counts: diffedCounts,
|
|
},
|
|
count: currentValue.count - previousValue.count,
|
|
sum: currentValue.sum - previousValue.sum,
|
|
hasMinMax: false,
|
|
min: Infinity,
|
|
max: -Infinity,
|
|
});
|
|
}
|
|
toMetricData(descriptor, aggregationTemporality, accumulationByAttributes, endTime) {
|
|
return {
|
|
descriptor,
|
|
aggregationTemporality,
|
|
dataPointType: DataPointType.HISTOGRAM,
|
|
dataPoints: accumulationByAttributes.map(([attributes, accumulation]) => {
|
|
const pointValue = accumulation.toPointValue();
|
|
// determine if instrument allows negative values.
|
|
const allowsNegativeValues = descriptor.type === InstrumentType.GAUGE ||
|
|
descriptor.type === InstrumentType.UP_DOWN_COUNTER ||
|
|
descriptor.type === InstrumentType.OBSERVABLE_GAUGE ||
|
|
descriptor.type === InstrumentType.OBSERVABLE_UP_DOWN_COUNTER;
|
|
return {
|
|
attributes,
|
|
startTime: accumulation.startTime,
|
|
endTime,
|
|
value: {
|
|
min: pointValue.hasMinMax ? pointValue.min : undefined,
|
|
max: pointValue.hasMinMax ? pointValue.max : undefined,
|
|
sum: !allowsNegativeValues ? pointValue.sum : undefined,
|
|
buckets: pointValue.buckets,
|
|
count: pointValue.count,
|
|
},
|
|
};
|
|
}),
|
|
};
|
|
}
|
|
}
|
|
//# sourceMappingURL=Histogram.js.map
|