UNPKG

@sentry/core

Version:
274 lines (236 loc) 8.6 kB
Object.defineProperty(exports, '__esModule', { value: true }); const utils = require('@sentry/utils'); const index = require('../asyncContext/index.js'); const carrier = require('../carrier.js'); const currentScopes = require('../currentScopes.js'); const metricSummary = require('../metrics/metric-summary.js'); const semanticAttributes = require('../semanticAttributes.js'); const spanstatus = require('../tracing/spanstatus.js'); const spanOnScope = require('./spanOnScope.js'); // These are aligned with OpenTelemetry trace flags const TRACE_FLAG_NONE = 0x0; const TRACE_FLAG_SAMPLED = 0x1; /** * Convert a span to a trace context, which can be sent as the `trace` context in an event. * By default, this will only include trace_id, span_id & parent_span_id. * If `includeAllData` is true, it will also include data, op, status & origin. */ function spanToTransactionTraceContext(span) { const { spanId: span_id, traceId: trace_id } = span.spanContext(); const { data, op, parent_span_id, status, origin } = spanToJSON(span); return utils.dropUndefinedKeys({ parent_span_id, span_id, trace_id, data, op, status, origin, }); } /** * Convert a span to a trace context, which can be sent as the `trace` context in a non-transaction event. */ function spanToTraceContext(span) { const { spanId: span_id, traceId: trace_id } = span.spanContext(); const { parent_span_id } = spanToJSON(span); return utils.dropUndefinedKeys({ parent_span_id, span_id, trace_id }); } /** * Convert a Span to a Sentry trace header. */ function spanToTraceHeader(span) { const { traceId, spanId } = span.spanContext(); const sampled = spanIsSampled(span); return utils.generateSentryTraceHeader(traceId, spanId, sampled); } /** * Convert a span time input intp a timestamp in seconds. */ function spanTimeInputToSeconds(input) { if (typeof input === 'number') { return ensureTimestampInSeconds(input); } if (Array.isArray(input)) { // See {@link HrTime} for the array-based time format return input[0] + input[1] / 1e9; } if (input instanceof Date) { return ensureTimestampInSeconds(input.getTime()); } return utils.timestampInSeconds(); } /** * Converts a timestamp to second, if it was in milliseconds, or keeps it as second. */ function ensureTimestampInSeconds(timestamp) { const isMs = timestamp > 9999999999; return isMs ? timestamp / 1000 : timestamp; } /** * Convert a span to a JSON representation. */ // Note: Because of this, we currently have a circular type dependency (which we opted out of in package.json). // This is not avoidable as we need `spanToJSON` in `spanUtils.ts`, which in turn is needed by `span.ts` for backwards compatibility. // And `spanToJSON` needs the Span class from `span.ts` to check here. function spanToJSON(span) { if (spanIsSentrySpan(span)) { return span.getSpanJSON(); } try { const { spanId: span_id, traceId: trace_id } = span.spanContext(); // Handle a span from @opentelemetry/sdk-base-trace's `Span` class if (spanIsOpenTelemetrySdkTraceBaseSpan(span)) { const { attributes, startTime, name, endTime, parentSpanId, status } = span; return utils.dropUndefinedKeys({ span_id, trace_id, data: attributes, description: name, parent_span_id: parentSpanId, start_timestamp: spanTimeInputToSeconds(startTime), // This is [0,0] by default in OTEL, in which case we want to interpret this as no end time timestamp: spanTimeInputToSeconds(endTime) || undefined, status: getStatusMessage(status), op: attributes[semanticAttributes.SEMANTIC_ATTRIBUTE_SENTRY_OP], origin: attributes[semanticAttributes.SEMANTIC_ATTRIBUTE_SENTRY_ORIGIN] , _metrics_summary: metricSummary.getMetricSummaryJsonForSpan(span), }); } // Finally, at least we have `spanContext()`.... return { span_id, trace_id, }; } catch (e) { return {}; } } function spanIsOpenTelemetrySdkTraceBaseSpan(span) { const castSpan = span ; return !!castSpan.attributes && !!castSpan.startTime && !!castSpan.name && !!castSpan.endTime && !!castSpan.status; } /** Exported only for tests. */ /** * Sadly, due to circular dependency checks we cannot actually import the Span class here and check for instanceof. * :( So instead we approximate this by checking if it has the `getSpanJSON` method. */ function spanIsSentrySpan(span) { return typeof (span ).getSpanJSON === 'function'; } /** * Returns true if a span is sampled. * In most cases, you should just use `span.isRecording()` instead. * However, this has a slightly different semantic, as it also returns false if the span is finished. * So in the case where this distinction is important, use this method. */ function spanIsSampled(span) { // We align our trace flags with the ones OpenTelemetry use // So we also check for sampled the same way they do. const { traceFlags } = span.spanContext(); return traceFlags === TRACE_FLAG_SAMPLED; } /** Get the status message to use for a JSON representation of a span. */ function getStatusMessage(status) { if (!status || status.code === spanstatus.SPAN_STATUS_UNSET) { return undefined; } if (status.code === spanstatus.SPAN_STATUS_OK) { return 'ok'; } return status.message || 'unknown_error'; } const CHILD_SPANS_FIELD = '_sentryChildSpans'; const ROOT_SPAN_FIELD = '_sentryRootSpan'; /** * Adds an opaque child span reference to a span. */ function addChildSpanToSpan(span, childSpan) { // We store the root span reference on the child span // We need this for `getRootSpan()` to work const rootSpan = span[ROOT_SPAN_FIELD] || span; utils.addNonEnumerableProperty(childSpan , ROOT_SPAN_FIELD, rootSpan); // We store a list of child spans on the parent span // We need this for `getSpanDescendants()` to work if (span[CHILD_SPANS_FIELD] && span[CHILD_SPANS_FIELD].size < 1000) { span[CHILD_SPANS_FIELD].add(childSpan); } else { utils.addNonEnumerableProperty(span, CHILD_SPANS_FIELD, new Set([childSpan])); } } /** This is only used internally by Idle Spans. */ function removeChildSpanFromSpan(span, childSpan) { if (span[CHILD_SPANS_FIELD]) { span[CHILD_SPANS_FIELD].delete(childSpan); } } /** * Returns an array of the given span and all of its descendants. */ function getSpanDescendants(span) { const resultSet = new Set(); function addSpanChildren(span) { // This exit condition is required to not infinitely loop in case of a circular dependency. if (resultSet.has(span)) { return; // We want to ignore unsampled spans (e.g. non recording spans) } else if (spanIsSampled(span)) { resultSet.add(span); const childSpans = span[CHILD_SPANS_FIELD] ? Array.from(span[CHILD_SPANS_FIELD]) : []; for (const childSpan of childSpans) { addSpanChildren(childSpan); } } } addSpanChildren(span); return Array.from(resultSet); } /** * Returns the root span of a given span. */ function getRootSpan(span) { return span[ROOT_SPAN_FIELD] || span; } /** * Returns the currently active span. */ function getActiveSpan() { const carrier$1 = carrier.getMainCarrier(); const acs = index.getAsyncContextStrategy(carrier$1); if (acs.getActiveSpan) { return acs.getActiveSpan(); } return spanOnScope._getSpanForScope(currentScopes.getCurrentScope()); } /** * Updates the metric summary on the currently active span */ function updateMetricSummaryOnActiveSpan( metricType, sanitizedName, value, unit, tags, bucketKey, ) { const span = getActiveSpan(); if (span) { metricSummary.updateMetricSummaryOnSpan(span, metricType, sanitizedName, value, unit, tags, bucketKey); } } exports.TRACE_FLAG_NONE = TRACE_FLAG_NONE; exports.TRACE_FLAG_SAMPLED = TRACE_FLAG_SAMPLED; exports.addChildSpanToSpan = addChildSpanToSpan; exports.getActiveSpan = getActiveSpan; exports.getRootSpan = getRootSpan; exports.getSpanDescendants = getSpanDescendants; exports.getStatusMessage = getStatusMessage; exports.removeChildSpanFromSpan = removeChildSpanFromSpan; exports.spanIsSampled = spanIsSampled; exports.spanTimeInputToSeconds = spanTimeInputToSeconds; exports.spanToJSON = spanToJSON; exports.spanToTraceContext = spanToTraceContext; exports.spanToTraceHeader = spanToTraceHeader; exports.spanToTransactionTraceContext = spanToTransactionTraceContext; exports.updateMetricSummaryOnActiveSpan = updateMetricSummaryOnActiveSpan; //# sourceMappingURL=spanUtils.js.map