mirror of https://github.com/renovatebot/renovate
172 lines
5.0 KiB
TypeScript
172 lines
5.0 KiB
TypeScript
import { ClientRequest } from 'node:http';
|
|
import type {
|
|
Context,
|
|
Span,
|
|
SpanOptions,
|
|
Tracer,
|
|
TracerProvider,
|
|
} from '@opentelemetry/api';
|
|
import * as api from '@opentelemetry/api';
|
|
import { ProxyTracerProvider, SpanStatusCode } from '@opentelemetry/api';
|
|
import { AsyncLocalStorageContextManager } from '@opentelemetry/context-async-hooks';
|
|
import { OTLPTraceExporter } from '@opentelemetry/exporter-trace-otlp-http';
|
|
import type { Instrumentation } from '@opentelemetry/instrumentation';
|
|
import { registerInstrumentations } from '@opentelemetry/instrumentation';
|
|
import { BunyanInstrumentation } from '@opentelemetry/instrumentation-bunyan';
|
|
import { HttpInstrumentation } from '@opentelemetry/instrumentation-http';
|
|
import { Resource } from '@opentelemetry/resources';
|
|
import {
|
|
BatchSpanProcessor,
|
|
ConsoleSpanExporter,
|
|
SimpleSpanProcessor,
|
|
} from '@opentelemetry/sdk-trace-base';
|
|
import { NodeTracerProvider } from '@opentelemetry/sdk-trace-node';
|
|
import {
|
|
ATTR_SERVICE_NAME,
|
|
ATTR_SERVICE_VERSION,
|
|
} from '@opentelemetry/semantic-conventions';
|
|
import { ATTR_SERVICE_NAMESPACE } from '@opentelemetry/semantic-conventions/incubating';
|
|
import { pkg } from '../expose.cjs';
|
|
import {
|
|
isTraceDebuggingEnabled,
|
|
isTraceSendingEnabled,
|
|
isTracingEnabled,
|
|
massageThrowable,
|
|
} from './utils';
|
|
|
|
let instrumentations: Instrumentation[] = [];
|
|
|
|
init();
|
|
|
|
export function init(): void {
|
|
if (!isTracingEnabled()) {
|
|
return;
|
|
}
|
|
|
|
const traceProvider = new NodeTracerProvider({
|
|
resource: new Resource({
|
|
// https://github.com/open-telemetry/opentelemetry-specification/blob/main/specification/resource/semantic_conventions/README.md#semantic-attributes-with-sdk-provided-default-value
|
|
[ATTR_SERVICE_NAME]: process.env.OTEL_SERVICE_NAME ?? 'renovate',
|
|
[ATTR_SERVICE_NAMESPACE]:
|
|
process.env.OTEL_SERVICE_NAMESPACE ?? 'renovatebot.com',
|
|
[ATTR_SERVICE_VERSION]: process.env.OTEL_SERVICE_VERSION ?? pkg.version,
|
|
}),
|
|
});
|
|
|
|
// add processors
|
|
if (isTraceDebuggingEnabled()) {
|
|
traceProvider.addSpanProcessor(
|
|
new SimpleSpanProcessor(new ConsoleSpanExporter()),
|
|
);
|
|
}
|
|
|
|
// OTEL specification environment variable
|
|
if (isTraceSendingEnabled()) {
|
|
const exporter = new OTLPTraceExporter();
|
|
traceProvider.addSpanProcessor(new BatchSpanProcessor(exporter));
|
|
}
|
|
|
|
const contextManager = new AsyncLocalStorageContextManager();
|
|
traceProvider.register({
|
|
contextManager,
|
|
});
|
|
|
|
instrumentations = [
|
|
new HttpInstrumentation({
|
|
applyCustomAttributesOnSpan: /* istanbul ignore next */ (
|
|
span,
|
|
request,
|
|
response,
|
|
) => {
|
|
// ignore 404 errors when the branch protection of Github could not be found. This is expected if no rules are configured
|
|
if (
|
|
request instanceof ClientRequest &&
|
|
request.host === `api.github.com` &&
|
|
request.path.endsWith(`/protection`) &&
|
|
response.statusCode === 404
|
|
) {
|
|
span.setStatus({ code: SpanStatusCode.OK });
|
|
}
|
|
},
|
|
}),
|
|
new BunyanInstrumentation(),
|
|
];
|
|
registerInstrumentations({
|
|
instrumentations,
|
|
});
|
|
}
|
|
|
|
/* istanbul ignore next */
|
|
|
|
// https://github.com/open-telemetry/opentelemetry-js-api/issues/34
|
|
export async function shutdown(): Promise<void> {
|
|
const traceProvider = getTracerProvider();
|
|
if (traceProvider instanceof NodeTracerProvider) {
|
|
await traceProvider.shutdown();
|
|
} else if (traceProvider instanceof ProxyTracerProvider) {
|
|
const delegateProvider = traceProvider.getDelegate();
|
|
if (delegateProvider instanceof NodeTracerProvider) {
|
|
await delegateProvider.shutdown();
|
|
}
|
|
}
|
|
}
|
|
|
|
/* istanbul ignore next */
|
|
export function disableInstrumentations(): void {
|
|
for (const instrumentation of instrumentations) {
|
|
instrumentation.disable();
|
|
}
|
|
}
|
|
|
|
export function getTracerProvider(): TracerProvider {
|
|
return api.trace.getTracerProvider();
|
|
}
|
|
|
|
function getTracer(): Tracer {
|
|
return getTracerProvider().getTracer('renovate');
|
|
}
|
|
|
|
export function instrument<F extends () => ReturnType<F>>(
|
|
name: string,
|
|
fn: F,
|
|
): ReturnType<F>;
|
|
export function instrument<F extends () => ReturnType<F>>(
|
|
name: string,
|
|
fn: F,
|
|
options: SpanOptions,
|
|
): ReturnType<F>;
|
|
export function instrument<F extends () => ReturnType<F>>(
|
|
name: string,
|
|
fn: F,
|
|
options: SpanOptions = {},
|
|
context: Context = api.context.active(),
|
|
): ReturnType<F> {
|
|
return getTracer().startActiveSpan(name, options, context, (span: Span) => {
|
|
try {
|
|
const ret = fn();
|
|
if (ret instanceof Promise) {
|
|
return ret
|
|
.catch((e) => {
|
|
span.recordException(e);
|
|
span.setStatus({
|
|
code: SpanStatusCode.ERROR,
|
|
message: massageThrowable(e),
|
|
});
|
|
throw e;
|
|
})
|
|
.finally(() => span.end()) as ReturnType<F>;
|
|
}
|
|
span.end();
|
|
return ret;
|
|
} catch (e) {
|
|
span.recordException(e);
|
|
span.setStatus({
|
|
code: SpanStatusCode.ERROR,
|
|
message: massageThrowable(e),
|
|
});
|
|
span.end();
|
|
throw e;
|
|
}
|
|
});
|
|
}
|