@graphql-yoga/plugin-prometheus
Version:
Prometheus plugin for GraphQL Yoga.
110 lines (109 loc) • 4.63 kB
JavaScript
Object.defineProperty(exports, "__esModule", { value: true });
exports.usePrometheus = exports.createSummary = exports.createHistogram = exports.createCounter = void 0;
const prometheus_1 = require("@envelop/prometheus");
Object.defineProperty(exports, "createCounter", { enumerable: true, get: function () { return prometheus_1.createCounter; } });
Object.defineProperty(exports, "createHistogram", { enumerable: true, get: function () { return prometheus_1.createHistogram; } });
Object.defineProperty(exports, "createSummary", { enumerable: true, get: function () { return prometheus_1.createSummary; } });
const graphql_1 = require("graphql");
const prom_client_1 = require("prom-client");
function headersToObj(headers) {
const obj = {};
headers.forEach((value, key) => {
obj[key] = value;
});
return obj;
}
function usePrometheus(options) {
const endpoint = options.endpoint || '/metrics';
const registry = options.registry || prom_client_1.register;
let httpHistogram;
if (options.http) {
const labelNames = [
'url',
'method',
'statusCode',
'statusText',
'operationName',
'operationType',
];
if (options.httpRequestHeaders) {
labelNames.push('requestHeaders');
}
if (options.httpResponseHeaders) {
labelNames.push('responseHeaders');
}
httpHistogram =
typeof options.http === 'object'
? options.http
: (0, prometheus_1.createHistogram)({
histogram: new prom_client_1.Histogram({
name: 'graphql_yoga_http_duration',
help: 'Time spent on HTTP connection',
labelNames,
registers: [registry],
}),
fillLabelsFn(params, { request, response }) {
const labels = {
operationName: params.operationName || 'Anonymous',
url: request.url,
method: request.method,
statusCode: response.status,
statusText: response.statusText,
};
if (params?.operationType) {
labels.operationType = params.operationType;
}
if (options.httpRequestHeaders) {
labels.requestHeaders = JSON.stringify(headersToObj(request.headers));
}
if (options.httpResponseHeaders) {
labels.responseHeaders = JSON.stringify(headersToObj(response.headers));
}
return labels;
},
});
}
const startByRequest = new WeakMap();
const paramsByRequest = new WeakMap();
return {
onPluginInit({ addPlugin }) {
addPlugin((0, prometheus_1.usePrometheus)({ ...options, registry }));
},
async onRequest({ request, url, fetchAPI, endResponse }) {
startByRequest.set(request, Date.now());
if (url.pathname === endpoint) {
const metrics = await registry.metrics();
const response = new fetchAPI.Response(metrics, {
headers: {
'Content-Type': registry.contentType,
},
});
endResponse(response);
}
},
onExecute({ args }) {
const operationAST = (0, graphql_1.getOperationAST)(args.document, args.operationName);
const operationType = operationAST?.operation;
const operationName = operationAST?.name?.value;
paramsByRequest.set(args.contextValue.request, {
document: args.document,
operationName,
operationType,
});
},
onResponse({ request, response, serverContext }) {
const start = startByRequest.get(request);
if (start) {
const duration = Date.now() - start;
const params = paramsByRequest.get(request);
httpHistogram?.histogram.observe(httpHistogram.fillLabelsFn(params || {}, {
...serverContext,
request,
response,
}), duration);
}
},
};
}
exports.usePrometheus = usePrometheus;
;