dd-trace
Version:
Datadog APM tracing client for JavaScript
139 lines (111 loc) • 3.69 kB
JavaScript
const request = require('./request')
const { startupLog } = require('../../startup-log')
const metrics = require('../../metrics')
const log = require('../../log')
const tracerVersion = require('../../../lib/version')
const METRIC_PREFIX = 'datadog.tracer.node.exporter.agent'
class Writer {
constructor ({ url, prioritySampler, lookup, protocolVersion }) {
const AgentEncoder = getEncoder(protocolVersion)
this._url = url
this._prioritySampler = prioritySampler
this._lookup = lookup
this._protocolVersion = protocolVersion
this._encoderForVersion = new AgentEncoder(this)
}
append (spans) {
log.debug(() => `Encoding trace: ${JSON.stringify(spans)}`)
this._encode(spans)
}
_sendPayload (data, count, done) {
metrics.increment(`${METRIC_PREFIX}.requests`, true)
makeRequest(this._protocolVersion, data, count, this._url, this._lookup, true, (err, res, status) => {
if (status) {
metrics.increment(`${METRIC_PREFIX}.responses`, true)
metrics.increment(`${METRIC_PREFIX}.responses.by.status`, `status:${status}`, true)
} else if (err) {
metrics.increment(`${METRIC_PREFIX}.errors`, true)
metrics.increment(`${METRIC_PREFIX}.errors.by.name`, `name:${err.name}`, true)
if (err.code) {
metrics.increment(`${METRIC_PREFIX}.errors.by.code`, `code:${err.code}`, true)
}
}
startupLog({ agentError: err })
if (err) {
log.error(err)
done()
return
}
log.debug(`Response from the agent: ${res}`)
try {
this._prioritySampler.update(JSON.parse(res).rate_by_service)
} catch (e) {
log.error(e)
metrics.increment(`${METRIC_PREFIX}.errors`, true)
metrics.increment(`${METRIC_PREFIX}.errors.by.name`, `name:${e.name}`, true)
}
done()
})
}
setUrl (url) {
this._url = url
}
_encode (trace) {
this._encoderForVersion.encode(trace)
}
flush (done = () => {}) {
const count = this._encoderForVersion.count()
if (count > 0) {
const payload = this._encoderForVersion.makePayload()
this._sendPayload(payload, count, done)
} else {
done()
}
}
}
function setHeader (headers, key, value) {
if (value) {
headers[key] = value
}
}
function getEncoder (protocolVersion) {
if (protocolVersion === '0.5') {
return require('../../encode/0.5').AgentEncoder
} else {
return require('../../encode/0.4').AgentEncoder
}
}
function makeRequest (version, data, count, url, lookup, needsStartupLog, cb) {
const options = {
path: `/v${version}/traces`,
method: 'PUT',
headers: {
'Content-Type': 'application/msgpack',
'Datadog-Meta-Tracer-Version': tracerVersion,
'X-Datadog-Trace-Count': String(count)
},
lookup
}
setHeader(options.headers, 'Datadog-Meta-Lang', 'nodejs')
setHeader(options.headers, 'Datadog-Meta-Lang-Version', process.version)
setHeader(options.headers, 'Datadog-Meta-Lang-Interpreter', process.jsEngine || 'v8')
if (url.protocol === 'unix:') {
options.socketPath = url.pathname
} else {
options.protocol = url.protocol
options.hostname = url.hostname
options.port = url.port
}
log.debug(() => `Request to the agent: ${JSON.stringify(options)}`)
request(Object.assign({ data }, options), (err, res, status) => {
if (needsStartupLog) {
// Note that logging will only happen once, regardless of how many times this is called.
startupLog({
agentError: status !== 404 && status !== 200 ? err : undefined
})
}
cb(err, res, status)
})
}
module.exports = Writer