TS + Decorators to get Metrics with Node and Telegraf
$ npm install besiktning
slf is a peer-dependency:
$ npm install slf
Initialize besiktning
:
import { Collector, telegrafFactory } from 'besiktning';
const telegraf = telegrafFactory({
uri: process.env.NODE_TELEGRAF_URI || 'udp://:8094',
bufferSize: parseInt(process.env.NODE_TELEGRAF_BUFFER_SIZE, 10) || 1,
flushInterval: parseInt(process.env.NODE_TELEGRAF_FLUSH_INTERVAL, 10 * 1000) || -1,
prefix: 'myMeasurementPrefix'
});
Collector.set(telegraf);
The API provides four method decorators for performing measurements:
withExceptionMeter
: counts number of exceptions thrown by the target functionwithGauge
: collects return value of the target functionwithMeter
: counts number of invocations of the target functionwithTimer
: collects execution time of the target function
The decorators work with regular functions as well.
Each decorator accepts an object with structure based on the data model of InfluxDB:
type FieldValue = NonNullable<number | bigint | string | boolean>;
type Dynamic<T> = T | ((...args: any) => T);
type Dictionary<T> = { [key: string]: T };
interface DecoratorPayload {
measurement: Dynamic<string>;
key: Dynamic<string>;
tags?: Dynamic<Dictionary<string>>;
apply?: (value: FieldValue) => FieldValue;
}
Measure number of exceptions:
class Vehicle {
@withExceptionMeter({
measurement: 'vehicle',
key: 'exception',
tags: { brand: 'myBrand' }
})
start(): void {
// ...
throw new Error('Vehicle failed to start.')
// ...
}
}
Measure number of method invocations:
class Store {
@withMeter({
measurement: 'store',
key: 'membership_count',
tags: person => ({
age_group: getAgeGroup(person.age),
region: getRegion(person.address)
})
})
registerMembership(person: Person): void {
// ...
}
}
Measure return value of a method:
class Store {
@withGauge({
measurement: 'store',
key: 'income',
tags: item => item
})
sell(item: Item): number {
// ...
}
}
Measure execution time of a method:
class Fibonacci {
@withTimer({
measurement: 'fibonacci',
key: getKey()
})
next(): number {
// ...
}
}
The decorators can also wrap regular functions:
let i = 0
const counter = () => i++;
const gaugedCounter = withGauge({
measurement: 'regular_function',
key: 'count'
})(counter)
gaugedCounter();
The codebase is tested extensively, and the test cases may serve as further examples.