UNPKG

@graphql-yoga/plugin-prometheus

Version:
110 lines (109 loc) 4.63 kB
"use strict"; 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;