UNPKG

@graphql-yoga/plugin-apollo-usage-report

Version:

Apollo's GraphOS usage report plugin for GraphQL Yoga.

137 lines (136 loc) 5.74 kB
import { google, Report, ReportHeader } from '@apollo/usage-reporting-protobuf'; import { OurReport } from './stats.js'; const DEFAULT_REPORTING_ENDPOINT = 'https://usage-reporting.api.apollographql.com/api/ingress/traces'; export class Reporter { yoga; logger; reportHeaders; options; reportsBySchema = {}; nextSendAfterDelay; sending = []; constructor(options, yoga, logger) { this.yoga = yoga; this.logger = logger; this.options = { ...options, maxBatchDelay: options.maxBatchDelay ?? 20_000, // 20s maxBatchUncompressedSize: options.maxBatchUncompressedSize ?? 4 * 1024 * 1024, // 4mb maxTraceSize: options.maxTraceSize ?? 10 * 1024 * 1024, // 10mb exportTimeout: options.exportTimeout ?? 30_000, // 30s onError: options.onError ?? (err => this.logger.error('Failed to send report', err)), }; this.reportHeaders = { graphRef: getGraphRef(options), hostname: options.hostname ?? getEnvVar('HOSTNAME') ?? '', uname: options.uname ?? '', // TODO: find a cross-platform way to get the uname runtimeVersion: options.runtimeVersion ?? '', agentVersion: options.agentVersion || `graphql-yoga@${yoga.version}`, }; } addTrace(schemaId, options) { const report = this.getReport(schemaId); report.addTrace(options); if (this.options.alwaysSend || report.sizeEstimator.bytes >= this.options.maxBatchUncompressedSize) { return this._sendReport(schemaId); } this.nextSendAfterDelay ||= setTimeout(() => this.flush(), this.options.maxBatchDelay); return; } async flush() { return Promise.allSettled([ ...this.sending, // When flushing, we want to also wait for previous traces to be sent, because it's mostly used for clean up ...Object.keys(this.reportsBySchema).map(schemaId => this._sendReport(schemaId)), ]); } async sendReport(schemaId) { const sending = this._sendReport(schemaId); this.sending.push(sending); sending.finally(() => (this.sending = this.sending?.filter(p => p !== sending))); return sending; } async _sendReport(schemaId) { const { fetchAPI: { fetch, CompressionStream, ReadableStream }, } = this.yoga; const report = this.reportsBySchema[schemaId]; if (!report) { throw new Error(`No report to send for schema ${schemaId}`); } if (this.nextSendAfterDelay != null) { clearTimeout(this.nextSendAfterDelay); this.nextSendAfterDelay = undefined; } delete this.reportsBySchema[schemaId]; report.endTime = dateToProtoTimestamp(new Date()); report.ensureCountsAreIntegers(); const validationError = Report.verify(report); if (validationError) { throw new TypeError(`Invalid report: ${validationError}`); } const { apiKey = getEnvVar('APOLLO_KEY'), endpoint = DEFAULT_REPORTING_ENDPOINT } = this.options; const encodedReport = Report.encode(report).finish(); let lastError; for (let tries = 0; tries < 5; tries++) { try { this.logger.debug(`Sending report (try ${tries}/5)`); const response = await fetch(endpoint, { method: 'POST', headers: { 'content-type': 'application/protobuf', 'content-encoding': 'gzip', // The presence of the api key is already checked at Yoga initialization time 'x-api-key': apiKey, accept: 'application/json', }, body: new ReadableStream({ start(controller) { controller.enqueue(encodedReport); controller.close(); }, }).pipeThrough(new CompressionStream('gzip')), signal: AbortSignal.timeout(this.options.exportTimeout), }); const result = await response.text(); if (response.ok) { this.logger.debug('Report sent:', result); return; } throw result; } catch (err) { lastError = err; this.logger.error('Failed to send report:', err); } } this.options.onError(new Error('Failed to send traces after 5 tries', { cause: lastError })); } getReport(schemaId) { const report = this.reportsBySchema[schemaId]; if (report) { return report; } return (this.reportsBySchema[schemaId] = new OurReport(new ReportHeader({ ...this.reportHeaders, executableSchemaId: schemaId, }))); } } function getGraphRef(options) { const graphRef = options.graphRef || getEnvVar('APOLLO_GRAPH_REF'); if (!graphRef) { throw new Error('Missing GraphRef. Either provide `graphRef` option or `APOLLO_GRAPH_REF` environment variable'); } return graphRef; } export function getEnvVar(name, defaultValue) { return globalThis.process?.env?.[name] || defaultValue || undefined; } // Converts a JS Date into a Timestamp. export function dateToProtoTimestamp(date) { const totalMillis = date.getTime(); const millis = totalMillis % 1000; return new google.protobuf.Timestamp({ seconds: (totalMillis - millis) / 1000, nanos: millis * 1e6, }); }