opentelemetry-js/packages/opentelemetry-exporter-jaeger/src/transform.ts

170 lines
5.1 KiB
TypeScript

/*
* Copyright The OpenTelemetry Authors
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* https://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
import { Link, SpanStatusCode, SpanKind } from '@opentelemetry/api';
import { ReadableSpan } from '@opentelemetry/sdk-trace-base';
import {
hrTimeToMilliseconds,
hrTimeToMicroseconds,
} from '@opentelemetry/core';
import {
ThriftSpan,
Tag,
Log,
ThriftTag,
ThriftLog,
ThriftUtils,
Utils,
ThriftReference,
TagValue,
ThriftReferenceType,
} from './types';
const DEFAULT_FLAGS = 0x1;
/**
* Translate OpenTelemetry ReadableSpan to Jaeger Thrift Span
* @param span Span to be translated
*/
export function spanToThrift(span: ReadableSpan): ThriftSpan {
const traceId = span.spanContext().traceId.padStart(32, '0');
const traceIdHigh = traceId.slice(0, 16);
const traceIdLow = traceId.slice(16);
const parentSpan = span.parentSpanId
? Utils.encodeInt64(span.parentSpanId)
: ThriftUtils.emptyBuffer;
const tags = Object.keys(span.attributes).map(
(name): Tag => ({ key: name, value: toTagValue(span.attributes[name]) })
);
if (span.status.code !== SpanStatusCode.UNSET) {
tags.push({
key: 'otel.status_code',
value: SpanStatusCode[span.status.code],
});
if (span.status.message) {
tags.push({ key: 'otel.status_description', value: span.status.message });
}
}
// Ensure that if SpanStatus.Code is ERROR, that we set the "error" tag on the
// Jaeger span.
if (span.status.code === SpanStatusCode.ERROR) {
tags.push({ key: 'error', value: true });
}
if (span.kind !== undefined && span.kind !== SpanKind.INTERNAL) {
tags.push({ key: 'span.kind', value: SpanKind[span.kind].toLowerCase() });
}
Object.keys(span.resource.attributes).forEach(name =>
tags.push({
key: name,
value: toTagValue(span.resource.attributes[name]),
})
);
if (span.instrumentationLibrary) {
tags.push({
key: 'otel.library.name',
value: toTagValue(span.instrumentationLibrary.name),
});
tags.push({
key: 'otel.library.version',
value: toTagValue(span.instrumentationLibrary.version),
});
}
/* Add droppedAttributesCount as a tag */
if (span.droppedAttributesCount) {
tags.push({
key: 'otel.dropped_attributes_count',
value: toTagValue(span.droppedAttributesCount),
});
}
/* Add droppedEventsCount as a tag */
if (span.droppedEventsCount) {
tags.push({
key: 'otel.dropped_events_count',
value: toTagValue(span.droppedEventsCount),
});
}
/* Add droppedLinksCount as a tag */
if (span.droppedLinksCount) {
tags.push({
key: 'otel.dropped_links_count',
value: toTagValue(span.droppedLinksCount),
});
}
const spanTags: ThriftTag[] = ThriftUtils.getThriftTags(tags);
const logs = span.events.map((event): Log => {
const fields: Tag[] = [{ key: 'event', value: event.name }];
const attrs = event.attributes;
if (attrs) {
Object.keys(attrs).forEach(attr =>
fields.push({ key: attr, value: toTagValue(attrs[attr]) })
);
}
if (event.droppedAttributesCount) {
fields.push({
key: 'otel.event.dropped_attributes_count',
value: event.droppedAttributesCount,
});
}
return { timestamp: hrTimeToMilliseconds(event.time), fields };
});
const spanLogs: ThriftLog[] = ThriftUtils.getThriftLogs(logs);
return {
traceIdLow: Utils.encodeInt64(traceIdLow),
traceIdHigh: Utils.encodeInt64(traceIdHigh),
spanId: Utils.encodeInt64(span.spanContext().spanId),
parentSpanId: parentSpan,
operationName: span.name,
references: spanLinksToThriftRefs(span.links),
flags: span.spanContext().traceFlags || DEFAULT_FLAGS,
startTime: Utils.encodeInt64(hrTimeToMicroseconds(span.startTime)),
duration: Utils.encodeInt64(hrTimeToMicroseconds(span.duration)),
tags: spanTags,
logs: spanLogs,
};
}
/** Translate OpenTelemetry {@link Link}s to Jaeger ThriftReference. */
function spanLinksToThriftRefs(links: Link[]): ThriftReference[] {
return links.map((link): ThriftReference => {
const refType = ThriftReferenceType.FOLLOWS_FROM;
const traceId = link.context.traceId;
const traceIdHigh = Utils.encodeInt64(traceId.slice(0, 16));
const traceIdLow = Utils.encodeInt64(traceId.slice(16));
const spanId = Utils.encodeInt64(link.context.spanId);
return { traceIdLow, traceIdHigh, spanId, refType };
});
}
/** Translate OpenTelemetry attribute value to Jaeger TagValue. */
function toTagValue(value: unknown): TagValue {
const valueType = typeof value;
if (valueType === 'boolean') {
return value as boolean;
} else if (valueType === 'number') {
return value as number;
}
return String(value);
}