UNPKG

dd-trace

Version:

Datadog APM tracing client for JavaScript

117 lines (98 loc) 3.99 kB
'use strict' const { inspect } = require('node:util') const { channel } = require('dc-polyfill') const request = require('../common/request') const { logIntegrations, logAgentError } = require('../../startup-log') const runtimeMetrics = require('../../runtime_metrics') const log = require('../../log') const tracerVersion = require('../../../../../package.json').version const BaseWriter = require('../common/writer') const propagationHash = require('../../propagation-hash') const METRIC_PREFIX = 'datadog.tracer.node.exporter.agent' const firstFlushChannel = channel('dd-trace:exporter:first-flush') class AgentWriter extends BaseWriter { constructor (...args) { super({ ...args[0], beforeFirstFlush: () => firstFlushChannel.publish(), }) const { prioritySampler, lookup, protocolVersion, headers, config = {} } = args[0] const AgentEncoder = getEncoder(protocolVersion) this._prioritySampler = prioritySampler this._lookup = lookup this._protocolVersion = protocolVersion this._headers = headers this._config = config this._encoder = new AgentEncoder(this) } _sendPayload (data, count, done) { runtimeMetrics.increment(`${METRIC_PREFIX}.requests`, true) const { _headers, _lookup, _protocolVersion, _url } = this makeRequest(_protocolVersion, data, count, _url, _headers, _lookup, (err, res, status, headers) => { if (status) { runtimeMetrics.increment(`${METRIC_PREFIX}.responses`, true) runtimeMetrics.increment(`${METRIC_PREFIX}.responses.by.status`, `status:${status}`, true) } else if (err) { runtimeMetrics.increment(`${METRIC_PREFIX}.errors`, true) runtimeMetrics.increment(`${METRIC_PREFIX}.errors.by.name`, `name:${err.name}`, true) if (err.code) { runtimeMetrics.increment(`${METRIC_PREFIX}.errors.by.code`, `code:${err.code}`, true) } } if (err) { log.errorWithoutTelemetry('Error sending payload to the agent (status code: %s)', err.status, err) done() return } log.debug('Response from the agent: %s', res) // Capture container tags hash from agent response headers // The hash is sent by the agent only when Datadog-Container-ID is present in the request // (Datadog-Container-ID is automatically injected by docker.inject() in exporters/common/request.js) if (headers) { const containerTagsHash = headers['Datadog-Container-Tags-Hash'] if (containerTagsHash) { propagationHash.updateContainerTagsHash(containerTagsHash) } } try { this._prioritySampler.update(JSON.parse(res).rate_by_service) } catch (e) { log.error('Error updating prioritySampler rates', e) runtimeMetrics.increment(`${METRIC_PREFIX}.errors`, true) runtimeMetrics.increment(`${METRIC_PREFIX}.errors.by.name`, `name:${e.name}`, true) } done() }) } } function getEncoder (protocolVersion) { return protocolVersion === '0.5' ? require('../../encode/0.5').AgentEncoder : require('../../encode/0.4').AgentEncoder } function makeRequest (version, data, count, url, headers, lookup, cb) { const options = { path: `/v${version}/traces`, method: 'PUT', headers: { ...headers, 'Content-Type': 'application/msgpack', 'Datadog-Meta-Tracer-Version': tracerVersion, 'X-Datadog-Trace-Count': String(count), 'Datadog-Meta-Lang': 'nodejs', 'Datadog-Meta-Lang-Version': process.version, 'Datadog-Meta-Lang-Interpreter': process.versions.bun ? 'JavaScriptCore' : 'v8', }, lookup, url, } log.debug('Request to the agent: %j', options) request(data, options, (err, res, status, headers) => { logIntegrations() if (status !== 404 && status !== 200 && err) { logAgentError({ status, message: err.message ?? inspect(err) }) } cb(err, res, status, headers) }) } module.exports = AgentWriter