newrelic
Version:
New Relic agent
361 lines (309 loc) • 11.4 kB
JavaScript
/*
* 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