@envelop/prometheus
Version:
This plugin tracks the complete execution flow, and reports metrics using Prometheus tracing (based on `prom-client`).
311 lines (310 loc) • 11.3 kB
JavaScript
Object.defineProperty(exports, "__esModule", { value: true });
exports.shouldTraceFieldResolver = shouldTraceFieldResolver;
exports.createFillLabelFnParams = createFillLabelFnParams;
exports.registerHistogram = registerHistogram;
exports.createHistogram = createHistogram;
exports.registerSummary = registerSummary;
exports.createSummary = createSummary;
exports.registerCounter = registerCounter;
exports.createCounter = createCounter;
exports.getHistogramFromConfig = getHistogramFromConfig;
exports.getSummaryFromConfig = getSummaryFromConfig;
exports.getCounterFromConfig = getCounterFromConfig;
exports.extractDeprecatedFields = extractDeprecatedFields;
exports.labelExists = labelExists;
exports.filterFillParamsFnParams = filterFillParamsFnParams;
exports.instrumentRegistry = instrumentRegistry;
const graphql_1 = require("graphql");
const prom_client_1 = require("prom-client");
const histograms = new WeakMap();
const summaries = new WeakMap();
const counters = new WeakMap();
function shouldTraceFieldResolver(info, whitelist) {
if (!whitelist) {
return true;
}
const parentType = info.parentType.name;
const fieldName = info.fieldName;
const coordinate = `${parentType}.${fieldName}`;
return whitelist.includes(coordinate) || whitelist.includes(`${parentType}.*`);
}
function getOperation(document) {
return document.definitions[0];
}
function createFillLabelFnParams(parseResult, context, filterParams) {
if (parseResult === null) {
return null;
}
if (parseResult instanceof Error) {
return null;
}
const operation = getOperation(parseResult);
return filterParams({
document: parseResult,
operationName: context?.params?.operationName || operation.name?.value || 'Anonymous',
operationType: operation.operation,
});
}
function registerHistogram(registry, conf) {
if (!histograms.has(registry)) {
histograms.set(registry, new Map());
}
const registryHistograms = histograms.get(registry);
if (!registryHistograms.has(conf.name)) {
conf.registers = [registry];
registryHistograms.set(conf.name, new prom_client_1.Histogram(conf));
}
return registryHistograms.get(conf.name);
}
/**
* Histogram metric factory allowing to define custom metrics with advanced configuration.
* @param options
* @returns
*/
function createHistogram(options) {
return {
histogram: registerHistogram(options.registry, options.histogram),
fillLabelsFn: options.fillLabelsFn,
phases: options.phases,
shouldObserve: options.shouldObserve ?? (() => true),
};
}
function registerSummary(registry, conf) {
if (!summaries.has(registry)) {
summaries.set(registry, new Map());
}
const registrySummaries = summaries.get(registry);
if (!registrySummaries.has(conf.name)) {
conf.registers = [registry];
registrySummaries.set(conf.name, new prom_client_1.Summary(conf));
}
return registrySummaries.get(conf.name);
}
/**
* Summary metric factory allowing to define custom metrics with advanced configuration.
* @param options
* @returns
*/
function createSummary(options) {
return {
summary: registerSummary(options.registry, options.summary),
fillLabelsFn: options.fillLabelsFn,
phases: options.phases,
shouldObserve: options.shouldObserve ?? (() => true),
};
}
/**
* Counter metric factory allowing to define custom metrics with advanced configuration.
* @param options
* @returns
*/
function registerCounter(registry, conf) {
if (!counters.has(registry)) {
counters.set(registry, new Map());
}
const registryCounters = counters.get(registry);
if (!registryCounters.has(conf.name)) {
conf.registers = [registry];
registryCounters.set(conf.name, new prom_client_1.Counter(conf));
}
return registryCounters.get(conf.name);
}
function createCounter(options) {
return {
counter: registerCounter(options.registry, options.counter),
fillLabelsFn: options.fillLabelsFn,
phases: options.phases,
shouldObserve: options.shouldObserve,
};
}
function getHistogramFromConfig(config, phase, availablePhases, histogram, fillLabelsFn = params => ({
operationName: params.operationName,
operationType: params.operationType,
})) {
const metric = config.metrics[phase];
if (!metric) {
return undefined;
}
let phases = availablePhases;
if (Array.isArray(metric)) {
if (metric.length === 0) {
throw TypeError(`Bad value provided for 'metrics.${phase.toString()}': the array must contain at least one element`);
}
else if (isBucketsList(metric)) {
histogram.buckets = metric;
}
else if (isPhasesList(metric)) {
phases = filterAvailablePhases(metric, availablePhases);
}
else {
throw TypeError(`Bad value provided for 'metrics.${phase.toString()}': the array must contain only numbers (buckets) or string (phases)`);
}
}
else if (typeof metric === 'object') {
const customMetric = metric;
if (!customMetric.phases) {
customMetric.phases = availablePhases;
}
if (!customMetric.shouldObserve) {
customMetric.shouldObserve = () => true;
}
return customMetric;
}
return createHistogram({
registry: config.registry || prom_client_1.register,
histogram: {
name: typeof metric === 'string' ? metric : phase,
...histogram,
labelNames: (histogram.labelNames ?? ['operationType', 'operationName']).filter(label => labelExists(config, label)),
},
fillLabelsFn: (...args) => filterFillParamsFnParams(config, fillLabelsFn(...args)),
phases,
shouldObserve: () => true,
});
}
function getSummaryFromConfig(config, phase, availablePhases, summary, fillLabelsFn = params => filterFillParamsFnParams(config, {
operationName: params.operationName,
operationType: params.operationType,
})) {
const metric = config.metrics[phase];
if (!metric) {
return undefined;
}
let phases = availablePhases;
if (Array.isArray(metric)) {
if (metric.length === 0) {
throw TypeError(`Bad value provided for 'metrics.${phase.toString()}': the array must contain at least one element`);
}
else if (isPhasesList(metric)) {
phases = filterAvailablePhases(metric, availablePhases);
}
else {
throw new TypeError(`Bad value provided for 'metrics.${phase.toString()}': the array must contain only strings (phases)`);
}
}
else if (typeof metric === 'object') {
const customMetric = metric;
if (!customMetric.phases) {
customMetric.phases = availablePhases;
}
if (!customMetric.shouldObserve) {
customMetric.shouldObserve = () => true;
}
return customMetric;
}
return createSummary({
registry: config.registry || prom_client_1.register,
summary: {
name: typeof metric === 'string' ? metric : phase,
labelNames: ['operationType', 'operationName'].filter(label => labelExists(config, label)),
...summary,
},
fillLabelsFn,
phases,
shouldObserve: () => true,
});
}
function getCounterFromConfig(config, phase, availablePhases, counter, fillLabelsFn = params => filterFillParamsFnParams(config, {
operationName: params.operationName,
operationType: params.operationType,
})) {
const metric = config.metrics[phase];
let phases = availablePhases;
if (!metric) {
return undefined;
}
if (Array.isArray(metric)) {
if (metric.length === 0) {
throw TypeError(`Bad value provided for 'metrics.${phase.toString()}': the array must contain at least one element`);
}
else if (isPhasesList(metric)) {
phases = filterAvailablePhases(metric, availablePhases);
}
else {
throw new TypeError(`Bad value provided for 'metrics.${phase.toString()}': the array must contain only strings (phases)`);
}
}
else if (typeof metric === 'object') {
const customMetric = metric;
if (!customMetric.phases) {
customMetric.phases = availablePhases;
}
if (!customMetric.shouldObserve) {
customMetric.shouldObserve = () => true;
}
return customMetric;
}
return createCounter({
registry: config.registry || prom_client_1.register,
counter: {
name: typeof metric === 'string' ? metric : phase,
labelNames: ['operationType', 'operationName'].filter(label => labelExists(config, label)),
...counter,
},
fillLabelsFn,
phases,
shouldObserve: () => true,
});
}
function extractDeprecatedFields(node, typeInfo) {
const found = [];
(0, graphql_1.visit)(node, (0, graphql_1.visitWithTypeInfo)(typeInfo, {
Argument: () => {
const argument = typeInfo.getArgument();
const field = typeInfo.getFieldDef();
if (field &&
argument &&
(argument.deprecationReason != null || argument.isDeprecated)) {
found.push({
fieldName: argument.name,
// the GraphQLArgument type doesn't contain context regarding the mutation the argument was passed to
// however, when visiting an argument, typeInfo.getFieldDef returns the mutation
typeName: field.name, // this is the mutation name
});
}
},
Field: () => {
const field = typeInfo.getFieldDef();
if (field && (field.deprecationReason != null || field.isDeprecated)) {
found.push({
fieldName: field.name,
typeName: typeInfo.getParentType().name || '',
});
}
},
}));
return found;
}
function labelExists(config, label) {
const labelFlag = config.labels?.[label];
if (labelFlag == null) {
return true;
}
return !!labelFlag;
}
function filterFillParamsFnParams(config, params) {
return Object.fromEntries(Object.entries(params).filter(([key]) => labelExists(config, key)));
}
const clearRegistry = new WeakMap();
function instrumentRegistry(registry) {
if (!clearRegistry.has(registry)) {
clearRegistry.set(registry, registry.clear.bind(registry));
}
registry.clear = () => {
histograms.delete(registry);
summaries.delete(registry);
counters.delete(registry);
clearRegistry.get(registry)();
};
return registry;
}
function isBucketsList(list) {
return list.every(item => typeof item === 'number');
}
function isPhasesList(list) {
return list.every(item => typeof item === 'string');
}
function filterAvailablePhases(phases, availablePhases) {
return availablePhases.filter(phase => phases.includes(phase));
}
;