UNPKG

newrelic

Version:
361 lines (309 loc) 11.4 kB
/* * Copyright 2020 New Relic Corporation. All rights reserved. * SPDX-License-Identifier: Apache-2.0 */ 'use strict' var codec = require('../../util/codec') var Segment = require('./segment') var {Attributes, MAXIMUM_CUSTOM_ATTRIBUTES} = require('../../attributes') var logger = require('../../logger').child({component: 'trace'}) var {DESTINATIONS} = require('../../config/attribute-filter') var FROM_MILLIS = 1e-3 const ATTRIBUTE_SCOPE = 'transaction' const REQUEST_URI_KEY = 'request.uri' const UNKNOWN_URI_PLACEHOLDER = '/Unknown' /** * A Trace holds the root of the Segment graph and produces the final * serialization of the transaction trace. * * @param {Transaction} transaction The transaction bound to the trace. */ function Trace(transaction) { if (!transaction) throw new Error('All traces must be associated with a transaction.') this.transaction = transaction this.root = new Segment(transaction, 'ROOT') this.root.start() this.intrinsics = Object.create(null) this.segmentsSeen = 0 this.totalTimeCache = null this.custom = new Attributes(ATTRIBUTE_SCOPE, MAXIMUM_CUSTOM_ATTRIBUTES) this.attributes = new Attributes(ATTRIBUTE_SCOPE) // sending displayName if set by user var displayName = transaction.agent.config.getDisplayHost() var hostName = transaction.agent.config.getHostnameSafe() if (displayName !== hostName) { this.attributes.addAttribute( DESTINATIONS.TRANS_COMMON, 'host.displayName', displayName ) this.displayName = displayName } this.domain = null } /** * End and close the current trace. Triggers metric recording for trace * segments that support recording. */ Trace.prototype.end = function end() { const segments = [this.root] while (segments.length) { const segment = segments.pop() segment.finalize() const children = segment.getChildren() for (let i = 0; i < children.length; ++i) { segments.push(children[i]) } } } /** * Iterates over the trace tree and generates a span event for each segment. */ Trace.prototype.generateSpanEvents = function generateSpanEvents() { const config = this.transaction.agent.config const infiniteTracingConfigured = Boolean(config.infinite_tracing.trace_observer.host) if ( (infiniteTracingConfigured || this.transaction.sampled) && config.span_events.enabled && config.distributed_tracing.enabled ) { let toProcess = [] // Root segment does not become a span, so we need to process it separately. const spanAggregator = this.transaction.agent.spanEventAggregator const children = this.root.getChildren() if (children.length > 0) { // At the point where these attributes are available, we only have a // root span. Adding attributes to first non-root span here. const attributeMap = { 'host.displayName': this.displayName, 'parent.type': this.transaction.parentType, 'parent.app': this.transaction.parentApp, 'parent.account': this.transaction.parentAcct, 'parent.transportType': this.transaction.parentTransportType, 'parent.transportDuration': this.transaction.parentTransportDuration } for (const [key, value] of Object.entries(attributeMap)) { if (value !== null) { children[0].addSpanAttribute(key, value) } } } for (let i = 0; i < children.length; ++i) { toProcess.push(new DTTraceNode(children[i], this.transaction.parentSpanId, true)) } while (toProcess.length) { let segmentInfo = toProcess.pop() let segment = segmentInfo.segment // Even though at some point we might want to stop adding events because all the priorities // should be the same, we need to count the spans as seen. spanAggregator.addSegment(segment, segmentInfo.parentId, segmentInfo.isRoot) const nodes = segment.getChildren() for (let i = 0; i < nodes.length; ++i) { const node = new DTTraceNode(nodes[i], segment.id) toProcess.push(node) } } } } function DTTraceNode(segment, parentId, isRoot = false) { this.segment = segment this.parentId = parentId this.isRoot = isRoot } /** * Add a child to the list of segments. * * @param {string} childName Name for the new segment. * @returns {Segment} Newly-created Segment. */ Trace.prototype.add = function add(childName, callback) { return this.root.add(childName, callback) } /** * Explicitly set a trace's runtime instead of using it as a stopwatch. * (As a byproduct, stops the timer.) * * @param {int} duration Duration of this particular trace. * @param {int} startTimeInMillis (optional) Start of this trace. */ Trace.prototype.setDurationInMillis = setDurationInMillis function setDurationInMillis(duration, startTimeInMillis) { this.root.setDurationInMillis(duration, startTimeInMillis) } /** * @return {integer} The amount of time the trace took, in milliseconds. */ Trace.prototype.getDurationInMillis = function getDurationInMillis() { return this.root.getDurationInMillis() } /** * Adds given key-value pair to trace's custom attributes, if it passes filtering rules. * * @param {string} key - The attribute name. * @param {string} value - The attribute value. */ Trace.prototype.addCustomAttribute = function addCustomAttribute(key, value) { if (this.custom.has(key)) { logger.debug( 'Potentially changing custom attribute %s from %s to %s.', key, this.custom.attributes[key].value, value ) } this.custom.addAttribute(DESTINATIONS.TRANS_SCOPE, key, value) } /** * The duration of the transaction trace tree that only this level accounts * for. * * @return {integer} The amount of time the trace took, minus any child * traces, in milliseconds. */ Trace.prototype.getExclusiveDurationInMillis = function getExclusiveDurationInMillis() { return this.root.getExclusiveDurationInMillis() } /** * The duration of all segments in a transaction trace. The root is not * accounted for, since it doesn't represent a unit of work. * * @return {integer} The sum of durations for all segments in a trace in * milliseconds */ Trace.prototype.getTotalTimeDurationInMillis = function getTotalTimeDurationInMillis() { if (this.totalTimeCache !== null) return this.totalTimeCache if (this.root.children.length === 0) return 0 var segments = this.root.getChildren() var totalTimeInMillis = 0 while (segments.length !== 0) { var segment = segments.pop() totalTimeInMillis += segment.getExclusiveDurationInMillis() segments = segments.concat(segment.getChildren()) } if (!this.transaction.isActive()) this.totalTimeCache = totalTimeInMillis return totalTimeInMillis } /** * The serializer is asynchronous, so serialization is as well. * * The transaction trace sent to the collector is a nested set of arrays. The * outermost array has the following fields, in order: * * 0: start time of the trace, in milliseconds * 1: duration, in milliseconds * 2: the path, or root metric name * 3: the URL (fragment) for this trace * 4: an array of segment arrays, deflated and then base64 encoded * 5: the guid for this transaction, used to correlate across * transactions * 6: reserved for future use, specified to be null for now * 7: FIXME: RUM2 force persist flag * * In addition, there is a "root node" (not the same as the first child, which * is a node with the special name ROOT and contents otherwise identical to the * top-level segment of the actual trace) with the following fields: * * 0: start time IN SECONDS * 1: a dictionary containing request parameters * 2: a dictionary containing custom parameters (currently not user-modifiable) * 3: the transaction trace segments (including the aforementioned root node) * 4: FIXME: a dictionary containing "parameter groups" with special information * related to this trace * * @param {Function} callback Called after serialization with either * an error (in the first parameter) or * the serialized transaction trace. */ Trace.prototype.generateJSON = function generateJSON(callback) { const serializedTrace = this._serializeTrace() const trace = this if (!this.transaction.agent.config.simple_compression) { codec.encode(serializedTrace, respond) } else { setImmediate(respond, null, serializedTrace) } function respond(err, data) { if (err) { return callback(err, null, null) } return callback(null, trace._generatePayload(data), trace) } } /** * This is the synchronous version of Trace#generateJSON */ Trace.prototype.generateJSONSync = function generateJSONSync() { const serializedTrace = this._serializeTrace() const shouldCompress = !this.transaction.agent.config.simple_compression const data = shouldCompress ? codec.encodeSync(serializedTrace) : serializedTrace return this._generatePayload(data) } /** * Generates the payload used in a trace harvest. * * @private * * @returns {Array} The formatted payload. */ Trace.prototype._generatePayload = function _generatePayload(data) { let syntheticsResourceId = null if (this.transaction.syntheticsData) { syntheticsResourceId = this.transaction.syntheticsData.resourceId } const requestUri = this._getRequestUri() return [ this.root.timer.start, // start this.transaction.getResponseTimeInMillis(), // response time this.transaction.getFullName(), // path requestUri, // request.uri data, // encodedCompressedData this.transaction.id, // guid null, // reserved for future use false, // forcePersist null, // xraySessionId syntheticsResourceId // synthetics resource id ] } /** * Returns the transaction URL if attribute is not exluded globally or * for transaction traces. Returns '/Unknown' if included but not known. * * The URI on a trace is a special attribute. It is included as a positional field, * not as an "agent attribute", to avoid having to decompress on the backend. * But it still needs to be gaited by the same attribute exclusion/inclusion * rules so sensitive information can be removed. */ Trace.prototype._getRequestUri = function _getRequestUri() { const canAddUri = this.attributes.hasValidDestination(DESTINATIONS.TRANS_TRACE, REQUEST_URI_KEY) let requestUri = null // must be null if excluded if (canAddUri) { requestUri = this.transaction.url || UNKNOWN_URI_PLACEHOLDER } return requestUri } /** * Serializes the trace into the expected JSON format to be sent. * * @private * * @returns {Array} Serialized trace data. */ Trace.prototype._serializeTrace = function _serializeTrace() { const attributes = { agentAttributes: this.attributes.get(DESTINATIONS.TRANS_TRACE), userAttributes: this.custom.get(DESTINATIONS.TRANS_TRACE), intrinsics: this.intrinsics } return [ this.root.timer.start * FROM_MILLIS, {}, // moved to agentAttributes { // hint to RPM for how to display this trace's segments nr_flatten_leading: false }, // moved to userAttributes this.root.toJSON(), attributes, [] // FIXME: parameter groups ] } module.exports = Trace