@envelop/prometheus
Version:
This plugin tracks the complete execution flow, and reports metrics using Prometheus tracing (based on `prom-client`).
460 lines (459 loc) • 22 kB
JavaScript
Object.defineProperty(exports, "__esModule", { value: true });
exports.usePrometheus = exports.execStartTimeMap = exports.fillLabelsFnParamsMap = exports.getSummaryFromConfig = exports.getHistogramFromConfig = exports.getCounterFromConfig = exports.createSummary = exports.createHistogram = exports.createCounter = void 0;
/* eslint-disable @typescript-eslint/no-non-null-asserted-optional-chain */
const graphql_1 = require("graphql");
const prom_client_1 = require("prom-client");
const core_1 = require("@envelop/core");
const on_resolve_1 = require("@envelop/on-resolve");
const utils_js_1 = require("./utils.js");
Object.defineProperty(exports, "createCounter", { enumerable: true, get: function () { return utils_js_1.createCounter; } });
Object.defineProperty(exports, "createHistogram", { enumerable: true, get: function () { return utils_js_1.createHistogram; } });
Object.defineProperty(exports, "createSummary", { enumerable: true, get: function () { return utils_js_1.createSummary; } });
Object.defineProperty(exports, "getCounterFromConfig", { enumerable: true, get: function () { return utils_js_1.getCounterFromConfig; } });
Object.defineProperty(exports, "getHistogramFromConfig", { enumerable: true, get: function () { return utils_js_1.getHistogramFromConfig; } });
Object.defineProperty(exports, "getSummaryFromConfig", { enumerable: true, get: function () { return utils_js_1.getSummaryFromConfig; } });
exports.fillLabelsFnParamsMap = new WeakMap();
exports.execStartTimeMap = new WeakMap();
const usePrometheus = (config) => {
config.registry = (0, utils_js_1.instrumentRegistry)(config.registry || prom_client_1.register);
const phasesToHook = {
parse: [],
validate: [],
context: [],
execute: {
end: [],
result: [],
},
subscribe: {
end: [],
result: [],
error: [],
},
pluginInit: [],
enveloped: [],
schema: [],
};
const parseHistogram = (0, utils_js_1.getHistogramFromConfig)(config, 'graphql_envelop_phase_parse', ['parse'], {
help: 'Time spent on running GraphQL "parse" function',
});
if (parseHistogram) {
phasesToHook.parse.push({
shouldHandle: parseHistogram.shouldObserve,
handler: ({ params, context, totalTime }) => {
parseHistogram.histogram.observe(parseHistogram.fillLabelsFn(params, context), totalTime);
},
});
}
const validateHistogram = (0, utils_js_1.getHistogramFromConfig)(config, 'graphql_envelop_phase_validate', ['validate'], {
help: 'Time spent on running GraphQL "validate" function',
});
if (validateHistogram) {
phasesToHook.validate.push({
shouldHandle: validateHistogram.shouldObserve,
handler: ({ params, context, totalTime }) => {
const labels = validateHistogram.fillLabelsFn(params, context);
validateHistogram.histogram.observe(labels, totalTime);
},
});
}
const contextBuildingHistogram = (0, utils_js_1.getHistogramFromConfig)(config, 'graphql_envelop_phase_context', ['context'], {
help: 'Time spent on building the GraphQL context',
});
if (contextBuildingHistogram) {
phasesToHook.context.push({
shouldHandle: contextBuildingHistogram.shouldObserve,
handler: ({ params, context, totalTime }) => {
const labels = contextBuildingHistogram.fillLabelsFn(params, context);
contextBuildingHistogram.histogram.observe(labels, totalTime);
},
});
}
const executeHistogram = (0, utils_js_1.getHistogramFromConfig)(config, 'graphql_envelop_phase_execute', ['execute'], {
help: 'Time spent on running the GraphQL "execute" function',
});
if (executeHistogram) {
phasesToHook.execute.end.push({
shouldHandle: executeHistogram.shouldObserve,
handler: ({ params, context, totalTime }) => {
const labels = executeHistogram.fillLabelsFn(params, context);
executeHistogram.histogram.observe(labels, totalTime);
},
});
}
const subscribeHistogram = (0, utils_js_1.getHistogramFromConfig)(config, 'graphql_envelop_phase_subscribe', ['subscribe'], {
help: 'Time spent on running the GraphQL "subscribe" function',
});
if (subscribeHistogram) {
phasesToHook.subscribe.end.push({
shouldHandle: subscribeHistogram.shouldObserve,
handler: ({ params, context, totalTime }) => {
const labels = subscribeHistogram.fillLabelsFn(params, context);
subscribeHistogram.histogram.observe(labels, totalTime);
},
});
}
const resolversHistogram = (0, utils_js_1.getHistogramFromConfig)(config, 'graphql_envelop_execute_resolver', ['execute', 'subscribe'], {
help: 'Time spent on running the GraphQL resolvers',
labelNames: ['operationType', 'operationName', 'fieldName', 'typeName', 'returnType'],
}, params => (0, utils_js_1.filterFillParamsFnParams)(config, {
operationName: params.operationName,
operationType: params.operationType,
fieldName: params.info?.fieldName,
typeName: params.info?.parentType.name,
returnType: params.info?.returnType.toString(),
}));
if (resolversHistogram) {
phasesToHook.pluginInit.push(({ addPlugin }) => {
addPlugin((0, on_resolve_1.useOnResolve)(({ info, context }) => {
const phase = info.operation.operation === 'subscription' ? 'subscribe' : 'execute';
if (!resolversHistogram.phases?.includes(phase) ||
!(0, utils_js_1.shouldTraceFieldResolver)(info, config.resolversWhitelist)) {
return undefined;
}
const fillLabelsFnParams = exports.fillLabelsFnParamsMap.get(context);
const paramsCtx = { ...fillLabelsFnParams, info };
if (!resolversHistogram.shouldObserve(paramsCtx, context)) {
return undefined;
}
const startTime = Date.now();
return () => {
const totalTime = (Date.now() - startTime) / 1000;
resolversHistogram.histogram.observe(resolversHistogram.fillLabelsFn(paramsCtx, context), totalTime);
};
}));
});
}
const requestTotalHistogram = (0, utils_js_1.getHistogramFromConfig)(config, 'graphql_envelop_request_duration', ['execute', 'subscribe'], {
help: 'Time spent on running the GraphQL operation from parse to execute',
});
if (requestTotalHistogram) {
const handler = {
shouldHandle: requestTotalHistogram.shouldObserve,
handler: ({ params, context, totalTime }) => {
const labels = requestTotalHistogram.fillLabelsFn(params, context);
requestTotalHistogram.histogram.observe(labels, totalTime);
},
};
for (const phase of requestTotalHistogram.phases) {
phasesToHook[phase].end.push(handler);
}
}
const requestSummary = (0, utils_js_1.getSummaryFromConfig)(config, 'graphql_envelop_request_time_summary', ['execute', 'subscribe'], {
help: 'Summary to measure the time to complete GraphQL operations',
});
if (requestSummary) {
phasesToHook.enveloped.push(({ context }) => {
if (!exports.execStartTimeMap.has(context)) {
exports.execStartTimeMap.set(context, Date.now());
}
});
const handler = {
shouldHandle: (params, context) => requestSummary.shouldObserve(params, context) && exports.execStartTimeMap.has(context),
handler: ({ params, context }) => {
const execStartTime = exports.execStartTimeMap.get(context);
const summaryTime = (Date.now() - execStartTime) / 1000;
const labels = requestSummary.fillLabelsFn(params, context);
requestSummary.summary.observe(labels, summaryTime);
},
};
for (const phase of requestSummary.phases) {
phasesToHook[phase].end.push(handler);
}
}
const errorsCounter = (0, utils_js_1.getCounterFromConfig)(config, 'graphql_envelop_error_result', ['parse', 'validate', 'context', 'execute', 'subscribe'], {
help: 'Counts the amount of errors reported from all phases',
labelNames: ['operationType', 'operationName', 'path', 'phase'],
}, params => {
const labels = {
operationName: params.operationName,
operationType: params.operationType,
phase: params.errorPhase,
};
if (params.error?.path) {
labels.path = params.error.path?.join('.');
}
return (0, utils_js_1.filterFillParamsFnParams)(config, labels);
});
if (errorsCounter) {
['parse', 'validate']
.filter(phase => errorsCounter.phases.includes(phase))
.forEach(phase => {
phasesToHook[phase].push({
shouldHandle: (params, context) => !!params.errorPhase && errorsCounter.shouldObserve(params, context),
handler: ({ params, context }) => {
const labels = errorsCounter.fillLabelsFn(params, context);
errorsCounter?.counter.labels(labels).inc();
},
});
});
['execute', 'subscribe']
.filter(phase => errorsCounter.phases.includes(phase))
.forEach(phase => {
phasesToHook[phase].result.push({
shouldHandle: errorsCounter.shouldObserve,
handler: ({ result, params, context }) => {
if (!result.errors?.length) {
return;
}
for (const error of result.errors) {
const labelParams = { ...params, errorPhase: 'execute', error };
if (errorsCounter.shouldObserve(labelParams, context)) {
errorsCounter.counter
.labels(errorsCounter.fillLabelsFn(labelParams, context))
.inc();
}
}
},
});
});
if (errorsCounter.phases.includes('subscribe')) {
phasesToHook.subscribe.error.push({
shouldHandle: errorsCounter.shouldObserve,
handler: ({ params, context, error }) => {
const labels = errorsCounter.fillLabelsFn(params, context);
errorsCounter.counter.labels(labels).inc();
},
});
}
if (errorsCounter.phases.includes('context')) {
phasesToHook.pluginInit.push(({ registerContextErrorHandler }) => {
registerContextErrorHandler(({ context, error }) => {
const fillLabelsFnParams = exports.fillLabelsFnParamsMap.get(context);
// FIXME: unsafe cast here, but it's ok, fillabelfn is doing duck typing anyway
const params = {
error: error,
errorPhase: 'context',
...fillLabelsFnParams,
};
if (errorsCounter.shouldObserve(params, context)) {
errorsCounter.counter.labels(errorsCounter?.fillLabelsFn(params, context)).inc();
}
});
});
}
}
const reqCounter = (0, utils_js_1.getCounterFromConfig)(config, 'graphql_envelop_request', ['execute', 'subscribe'], {
help: 'Counts the amount of GraphQL requests executed through Envelop',
});
if (reqCounter) {
const handler = {
shouldHandle: reqCounter.shouldObserve,
handler: ({ params, context }) => {
reqCounter.counter.labels(reqCounter.fillLabelsFn(params, context)).inc();
},
};
for (const phase of reqCounter.phases) {
phasesToHook[phase].end.push(handler);
}
}
const deprecationCounter = (0, utils_js_1.getCounterFromConfig)(config, 'graphql_envelop_deprecated_field', ['parse'], {
help: 'Counts the amount of deprecated fields used in selection sets',
labelNames: ['operationType', 'operationName', 'fieldName', 'typeName'],
}, params => (0, utils_js_1.filterFillParamsFnParams)(config, {
operationName: params.operationName,
operationType: params.operationType,
fieldName: params.deprecationInfo?.fieldName,
typeName: params.deprecationInfo?.typeName,
}));
if (deprecationCounter) {
let typeInfo = null;
phasesToHook.schema.push(({ schema }) => {
typeInfo = new graphql_1.TypeInfo(schema);
});
phasesToHook.parse.push({
shouldHandle: (params, context) =>
// If parse error happens, we can't explore the query document
!!typeInfo && !params.errorPhase && deprecationCounter.shouldObserve(params, context),
handler: ({ params, context }) => {
const deprecatedFields = (0, utils_js_1.extractDeprecatedFields)(params.document, typeInfo);
for (const depField of deprecatedFields) {
const deprecationLabelParams = {
...params,
deprecationInfo: depField,
};
if (deprecationCounter.shouldObserve(deprecationLabelParams, context)) {
deprecationCounter.counter
.labels(deprecationCounter.fillLabelsFn(deprecationLabelParams, context))
.inc();
}
}
},
});
}
const schemaChangeCounter = (0, utils_js_1.getCounterFromConfig)(config, 'graphql_envelop_schema_change', ['schema'], {
help: 'Counts the amount of schema changes',
labelNames: [],
}, () => ({}));
if (schemaChangeCounter) {
const countedSchemas = new WeakSet();
phasesToHook.schema.push(({ schema }) => {
if (schemaChangeCounter?.shouldObserve({}, null) && !countedSchemas.has(schema)) {
schemaChangeCounter.counter.inc();
countedSchemas.add(schema);
}
});
}
const onParse = ({ context, params }) => {
if (config.skipIntrospection && (0, core_1.isIntrospectionOperationString)(params.source)) {
return;
}
const startTime = Date.now();
return params => {
const fillLabelsFnParams = (0, utils_js_1.createFillLabelFnParams)(params.result, context, params => (0, utils_js_1.filterFillParamsFnParams)(config, params));
exports.fillLabelsFnParamsMap.set(context, fillLabelsFnParams);
const args = {
context,
totalTime: (Date.now() - startTime) / 1000,
params: fillLabelsFnParams ?? { error: params.result, errorPhase: 'parse' },
};
phasesToHook.parse
.filter(({ shouldHandle }) => shouldHandle(args.params, context))
.forEach(({ handler }) => handler(args));
};
};
const onValidate = ({ context }) => {
const fillLabelsFnParams = exports.fillLabelsFnParamsMap.get(context);
if (!fillLabelsFnParams) {
return undefined;
}
const startTime = Date.now();
return ({ valid }) => {
const args = {
params: valid ? fillLabelsFnParams : { ...fillLabelsFnParams, errorPhase: 'validate' },
context,
totalTime: (Date.now() - startTime) / 1000,
};
phasesToHook.validate
.filter(({ shouldHandle }) => shouldHandle(args.params, context))
.forEach(({ handler }) => handler(args));
// TODO: we should probably iterate over validation errors to report each error.
};
};
const onContextBuilding = ({ context }) => {
const fillLabelsFnParams = exports.fillLabelsFnParamsMap.get(context);
if (!fillLabelsFnParams) {
return;
}
const startTime = Date.now();
const args = {
context,
params: fillLabelsFnParams,
totalTime: (Date.now() - startTime) / 1000,
};
phasesToHook.context
.filter(({ shouldHandle }) => shouldHandle(fillLabelsFnParams, context))
.forEach(({ handler }) => handler(args));
};
const onExecute = ({ args: { contextValue: context } }) => {
const fillLabelsFnParams = exports.fillLabelsFnParamsMap.get(context);
if (!fillLabelsFnParams) {
return;
}
const endHandlers = phasesToHook.execute.end.filter(({ shouldHandle }) => shouldHandle(fillLabelsFnParams, context));
const resultHandlers = phasesToHook.execute.result.filter(({ shouldHandle }) => shouldHandle(fillLabelsFnParams, context));
if (endHandlers.length + resultHandlers.length === 0) {
return undefined;
}
const startTime = Date.now();
function handleResult({ result }) {
const totalTime = (Date.now() - startTime) / 1000;
const args = { params: fillLabelsFnParams, context, totalTime, result };
resultHandlers.forEach(({ handler }) => handler(args));
}
const handleEnd = () => {
const totalTime = (Date.now() - startTime) / 1000;
const args = { params: fillLabelsFnParams, context, totalTime };
endHandlers.forEach(({ handler }) => handler(args));
};
return {
onExecuteDone: ({ result }) => {
if ((0, core_1.isAsyncIterable)(result)) {
return {
onNext: resultHandlers.length ? handleResult : undefined,
onEnd: endHandlers.length ? handleEnd : undefined,
};
}
else {
handleResult({ result });
handleEnd();
return undefined;
}
},
};
};
const onSubscribe = ({ args: { contextValue: context } }) => {
const fillLabelsFnParams = exports.fillLabelsFnParamsMap.get(context);
if (!fillLabelsFnParams) {
return;
}
const endHandlers = phasesToHook.subscribe.end.filter(({ shouldHandle }) => shouldHandle(fillLabelsFnParams, context));
const resultHandlers = phasesToHook.subscribe.result.filter(({ shouldHandle }) => shouldHandle(fillLabelsFnParams, context));
const errorHandlers = phasesToHook.subscribe.error.filter(({ shouldHandle }) => shouldHandle(fillLabelsFnParams, context));
if (endHandlers.length + resultHandlers.length + errorHandlers.length === 0) {
return undefined;
}
const startTime = Date.now();
function handleResult({ result }) {
const totalTime = (Date.now() - startTime) / 1000;
const args = { params: fillLabelsFnParams, context, totalTime, result };
resultHandlers.forEach(({ handler }) => handler(args));
}
const handleEnd = () => {
const totalTime = (Date.now() - startTime) / 1000;
const args = { params: fillLabelsFnParams, context, totalTime };
endHandlers.forEach(({ handler }) => handler(args));
};
const handleError = ({ error }) => {
const totalTime = (Date.now() - startTime) / 1000;
const args = { params: fillLabelsFnParams, context, totalTime, error };
errorHandlers.forEach(({ handler }) => handler(args));
};
return {
onSubscribeResult: ({ result }) => {
if ((0, core_1.isAsyncIterable)(result)) {
return {
onNext: resultHandlers.length ? handleResult : undefined,
onEnd: endHandlers.length ? handleEnd : undefined,
};
}
else {
handleResult({ result });
handleEnd();
return undefined;
}
},
onSubscribeError: errorHandlers.length ? handleError : undefined,
};
};
const onPluginInit = payload => {
for (const handler of phasesToHook.pluginInit) {
handler(payload);
}
};
const onEnveloped = payload => {
for (const handler of phasesToHook.enveloped) {
handler(payload);
}
};
const onSchemaChange = payload => {
for (const handler of phasesToHook.schema) {
handler(payload);
}
};
return {
onParse, // onParse is required, because it sets up the label params WeakMap
onSchemaChange: phasesToHook.schema.length ? onSchemaChange : undefined,
onPluginInit: phasesToHook.pluginInit.length ? onPluginInit : undefined,
onEnveloped: phasesToHook.enveloped.length ? onEnveloped : undefined,
onValidate: phasesToHook.validate.length ? onValidate : undefined,
onContextBuilding: phasesToHook.context.length ? onContextBuilding : undefined,
onExecute: phasesToHook.execute.end.length + phasesToHook.execute.result.length ? onExecute : undefined,
onSubscribe: phasesToHook.subscribe.end.length +
phasesToHook.subscribe.result.length +
phasesToHook.subscribe.error.length
? onSubscribe
: undefined,
};
};
exports.usePrometheus = usePrometheus;
;